diff --git a/src/hotspot/cpu/aarch64/foreign_globals_aarch64.cpp b/src/hotspot/cpu/aarch64/foreign_globals_aarch64.cpp index d2c40478261..6531eb03edc 100644 --- a/src/hotspot/cpu/aarch64/foreign_globals_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/foreign_globals_aarch64.cpp @@ -84,3 +84,8 @@ const BufferLayout ForeignGlobals::parse_buffer_layout_impl(jobject jlayout) con return layout; } + +const CallRegs ForeignGlobals::parse_call_regs_impl(jobject jconv) const { + ShouldNotCallThis(); + return {}; +} diff --git a/src/hotspot/cpu/aarch64/frame_aarch64.cpp b/src/hotspot/cpu/aarch64/frame_aarch64.cpp index 90e25866e20..e025895e392 100644 --- a/src/hotspot/cpu/aarch64/frame_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/frame_aarch64.cpp @@ -359,6 +359,16 @@ frame frame::sender_for_entry_frame(RegisterMap* map) const { return fr; } +JavaFrameAnchor* OptimizedEntryBlob::jfa_for_frame(const frame& frame) const { + ShouldNotCallThis(); + return nullptr; +} + +frame frame::sender_for_optimized_entry_frame(RegisterMap* map) const { + ShouldNotCallThis(); + return {}; +} + //------------------------------------------------------------------------------ // frame::verify_deopt_original_pc // diff --git a/src/hotspot/cpu/aarch64/sharedRuntime_aarch64.cpp b/src/hotspot/cpu/aarch64/sharedRuntime_aarch64.cpp index c48ab639e66..e724a3df036 100644 --- a/src/hotspot/cpu/aarch64/sharedRuntime_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/sharedRuntime_aarch64.cpp @@ -878,7 +878,7 @@ int SharedRuntime::c_calling_convention(const BasicType *sig_bt, // 64 bits items (Aarch64 abi) even though java would only store // 32bits for a parameter. On 32bit it will simply be 32 bits // So this routine will do 32->32 on 32bit and 32->64 on 64bit -static void move32_64(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::move32_64(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { if (src.first()->is_stack()) { if (dst.first()->is_stack()) { // stack to stack @@ -979,7 +979,7 @@ static void object_move(MacroAssembler* masm, } // A float arg may have to do float reg int reg conversion -static void float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { assert(src.first()->is_stack() && dst.first()->is_stack() || src.first()->is_reg() && dst.first()->is_reg(), "Unexpected error"); if (src.first()->is_stack()) { @@ -998,7 +998,7 @@ static void float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { } // A long move -static void long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { if (src.first()->is_stack()) { if (dst.first()->is_stack()) { // stack to stack @@ -1022,7 +1022,7 @@ static void long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { // A double move -static void double_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::double_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { assert(src.first()->is_stack() && dst.first()->is_stack() || src.first()->is_reg() && dst.first()->is_reg(), "Unexpected error"); if (src.first()->is_stack()) { diff --git a/src/hotspot/cpu/aarch64/universalUpcallHandler_aarch64.cpp b/src/hotspot/cpu/aarch64/universalUpcallHandler_aarch64.cpp index 598364d2ec8..41bb34e40ab 100644 --- a/src/hotspot/cpu/aarch64/universalUpcallHandler_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/universalUpcallHandler_aarch64.cpp @@ -99,3 +99,12 @@ address ProgrammableUpcallHandler::generate_upcall_stub(jobject rec, jobject jab return blob->code_begin(); } + +address ProgrammableUpcallHandler::generate_optimized_upcall_stub(jobject mh, Method* entry, jobject jabi, jobject jconv) { + ShouldNotCallThis(); + return nullptr; +} + +bool ProgrammableUpcallHandler::supports_optimized_upcalls() { + return false; +} diff --git a/src/hotspot/cpu/x86/foreign_globals_x86.cpp b/src/hotspot/cpu/x86/foreign_globals_x86.cpp index dcdb05d8421..71546f76168 100644 --- a/src/hotspot/cpu/x86/foreign_globals_x86.cpp +++ b/src/hotspot/cpu/x86/foreign_globals_x86.cpp @@ -87,3 +87,32 @@ const BufferLayout ForeignGlobals::parse_buffer_layout_impl(jobject jlayout) con return layout; } + +const CallRegs ForeignGlobals::parse_call_regs_impl(jobject jconv) const { + oop conv_oop = JNIHandles::resolve_non_null(jconv); + objArrayOop arg_regs_oop = cast(conv_oop->obj_field(CallConvOffsets.arg_regs_offset)); + objArrayOop ret_regs_oop = cast(conv_oop->obj_field(CallConvOffsets.ret_regs_offset)); + + CallRegs result; + result._args_length = arg_regs_oop->length(); + result._arg_regs = NEW_RESOURCE_ARRAY(VMReg, result._args_length); + + result._rets_length = ret_regs_oop->length(); + result._ret_regs = NEW_RESOURCE_ARRAY(VMReg, result._rets_length); + + for (int i = 0; i < result._args_length; i++) { + oop storage = arg_regs_oop->obj_at(i); + jint index = storage->int_field(VMS.index_offset); + jint type = storage->int_field(VMS.type_offset); + result._arg_regs[i] = VMRegImpl::vmStorageToVMReg(type, index); + } + + for (int i = 0; i < result._rets_length; i++) { + oop storage = ret_regs_oop->obj_at(i); + jint index = storage->int_field(VMS.index_offset); + jint type = storage->int_field(VMS.type_offset); + result._ret_regs[i] = VMRegImpl::vmStorageToVMReg(type, index); + } + + return result; +} diff --git a/src/hotspot/cpu/x86/frame_x86.cpp b/src/hotspot/cpu/x86/frame_x86.cpp index 9c5d4afbea6..7c9804974d3 100644 --- a/src/hotspot/cpu/x86/frame_x86.cpp +++ b/src/hotspot/cpu/x86/frame_x86.cpp @@ -102,6 +102,8 @@ bool frame::safe_for_sender(JavaThread *thread) { if (is_entry_frame()) { // an entry frame must have a valid fp. return fp_safe && is_entry_frame_valid(thread); + } else if (is_optimized_entry_frame()) { + return fp_safe; } intptr_t* sender_sp = NULL; @@ -199,6 +201,8 @@ bool frame::safe_for_sender(JavaThread *thread) { address jcw = (address)sender.entry_frame_call_wrapper(); return thread->is_in_stack_range_excl(jcw, (address)sender.fp()); + } else if (sender_blob->is_optimized_entry_blob()) { + return false; } CompiledMethod* nm = sender_blob->as_compiled_method_or_null(); @@ -349,6 +353,32 @@ frame frame::sender_for_entry_frame(RegisterMap* map) const { return fr; } +JavaFrameAnchor* OptimizedEntryBlob::jfa_for_frame(const frame& frame) const { + // need unextended_sp here, since normal sp is wrong for interpreter callees + return reinterpret_cast(reinterpret_cast(frame.unextended_sp()) + in_bytes(jfa_sp_offset())); +} + +frame frame::sender_for_optimized_entry_frame(RegisterMap* map) const { + assert(map != NULL, "map must be set"); + OptimizedEntryBlob* blob = _cb->as_optimized_entry_blob(); + // Java frame called from C; skip all C frames and return top C + // frame of that chunk as the sender + JavaFrameAnchor* jfa = blob->jfa_for_frame(*this); + assert(jfa->last_Java_sp() > sp(), "must be above this frame on stack"); + // Since we are walking the stack now this nested anchor is obviously walkable + // even if it wasn't when it was stacked. + if (!jfa->walkable()) { + // Capture _last_Java_pc (if needed) and mark anchor walkable. + jfa->capture_last_Java_pc(); + } + map->clear(); + assert(map->include_argument_oops(), "should be set by clear"); + vmassert(jfa->last_Java_pc() != NULL, "not walkable"); + frame fr(jfa->last_Java_sp(), jfa->last_Java_fp(), jfa->last_Java_pc()); + + return fr; +} + //------------------------------------------------------------------------------ // frame::verify_deopt_original_pc // @@ -478,8 +508,9 @@ frame frame::sender_raw(RegisterMap* map) const { // update it accordingly map->set_include_argument_oops(false); - if (is_entry_frame()) return sender_for_entry_frame(map); - if (is_interpreted_frame()) return sender_for_interpreter_frame(map); + if (is_entry_frame()) return sender_for_entry_frame(map); + if (is_optimized_entry_frame()) return sender_for_optimized_entry_frame(map); + if (is_interpreted_frame()) return sender_for_interpreter_frame(map); assert(_cb == CodeCache::find_blob(pc()),"Must be the same"); if (_cb != NULL) { diff --git a/src/hotspot/cpu/x86/javaFrameAnchor_x86.hpp b/src/hotspot/cpu/x86/javaFrameAnchor_x86.hpp index bb39c8e513e..6fa2b55033b 100644 --- a/src/hotspot/cpu/x86/javaFrameAnchor_x86.hpp +++ b/src/hotspot/cpu/x86/javaFrameAnchor_x86.hpp @@ -70,8 +70,6 @@ public: address last_Java_pc(void) { return _last_Java_pc; } -private: - static ByteSize last_Java_fp_offset() { return byte_offset_of(JavaFrameAnchor, _last_Java_fp); } public: diff --git a/src/hotspot/cpu/x86/sharedRuntime_x86_32.cpp b/src/hotspot/cpu/x86/sharedRuntime_x86_32.cpp index 87369d5b007..c0525b6997a 100644 --- a/src/hotspot/cpu/x86/sharedRuntime_x86_32.cpp +++ b/src/hotspot/cpu/x86/sharedRuntime_x86_32.cpp @@ -1124,7 +1124,7 @@ static void object_move(MacroAssembler* masm, } // A float arg may have to do float reg int reg conversion -static void float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { assert(!src.second()->is_valid() && !dst.second()->is_valid(), "bad float_move"); // Because of the calling convention we know that src is either a stack location @@ -1142,7 +1142,7 @@ static void float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { } // A long move -static void long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { // The only legal possibility for a long_move VMRegPair is: // 1: two stack slots (possibly unaligned) @@ -1161,7 +1161,7 @@ static void long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { } // A double move -static void double_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::double_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { // The only legal possibilities for a double_move VMRegPair are: // The painful thing here is that like long_move a VMRegPair might be diff --git a/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp b/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp index 244a16ec843..88ce2a744b4 100644 --- a/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp +++ b/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp @@ -1167,7 +1167,7 @@ int SharedRuntime::c_calling_convention(const BasicType *sig_bt, // 64 bits items (x86_32/64 abi) even though java would only store // 32bits for a parameter. On 32bit it will simply be 32 bits // So this routine will do 32->32 on 32bit and 32->64 on 64bit -static void move32_64(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::move32_64(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { if (src.first()->is_stack()) { if (dst.first()->is_stack()) { // stack to stack @@ -1285,7 +1285,7 @@ static void object_move(MacroAssembler* masm, } // A float arg may have to do float reg int reg conversion -static void float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { assert(!src.second()->is_valid() && !dst.second()->is_valid(), "bad float_move"); // The calling conventions assures us that each VMregpair is either @@ -1314,7 +1314,7 @@ static void float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { } // A long move -static void long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { // The calling conventions assures us that each VMregpair is either // all really one physical register or adjacent stack slots. @@ -1339,7 +1339,7 @@ static void long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { } // A double move -static void double_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { +void SharedRuntime::double_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst) { // The calling conventions assures us that each VMregpair is either // all really one physical register or adjacent stack slots. @@ -1448,13 +1448,13 @@ static void unpack_array_argument(MacroAssembler* masm, VMRegPair reg, BasicType // load the length relative to the body. __ movl(tmp_reg, Address(tmp_reg, arrayOopDesc::length_offset_in_bytes() - arrayOopDesc::base_offset_in_bytes(in_elem_type))); - move32_64(masm, tmp, length_arg); + SharedRuntime::move32_64(masm, tmp, length_arg); __ jmpb(done); __ bind(is_null); // Pass zeros __ xorptr(tmp_reg, tmp_reg); move_ptr(masm, tmp, body_arg); - move32_64(masm, tmp, length_arg); + SharedRuntime::move32_64(masm, tmp, length_arg); __ bind(done); __ block_comment("} unpack_array_argument"); @@ -1541,8 +1541,8 @@ class ComputeMoveOrder: public StackObj { GrowableArray edges; public: - ComputeMoveOrder(int total_in_args, VMRegPair* in_regs, int total_c_args, VMRegPair* out_regs, - BasicType* in_sig_bt, GrowableArray& arg_order, VMRegPair tmp_vmreg) { + ComputeMoveOrder(int total_in_args, const VMRegPair* in_regs, int total_c_args, VMRegPair* out_regs, + const BasicType* in_sig_bt, GrowableArray& arg_order, VMRegPair tmp_vmreg) { // Move operations where the dest is the stack can all be // scheduled first since they can't interfere with the other moves. for (int i = total_in_args - 1, c_arg = total_c_args - 1; i >= 0; i--, c_arg--) { @@ -4087,3 +4087,13 @@ void OptoRuntime::generate_exception_blob() { _exception_blob = ExceptionBlob::create(&buffer, oop_maps, SimpleRuntimeFrame::framesize >> 1); } #endif // COMPILER2 + +void SharedRuntime::compute_move_order(const BasicType* in_sig_bt, + int total_in_args, const VMRegPair* in_regs, + int total_out_args, VMRegPair* out_regs, + GrowableArray& arg_order, + VMRegPair tmp_vmreg) { + ComputeMoveOrder order(total_in_args, in_regs, + total_out_args, out_regs, + in_sig_bt, arg_order, tmp_vmreg); +} diff --git a/src/hotspot/cpu/x86/universalUpcallHandler_x86_32.cpp b/src/hotspot/cpu/x86/universalUpcallHandler_x86_32.cpp index 966ab750db0..a251b31ea6a 100644 --- a/src/hotspot/cpu/x86/universalUpcallHandler_x86_32.cpp +++ b/src/hotspot/cpu/x86/universalUpcallHandler_x86_32.cpp @@ -28,3 +28,12 @@ address ProgrammableUpcallHandler::generate_upcall_stub(jobject rec, jobject jab Unimplemented(); return nullptr; } + +address ProgrammableUpcallHandler::generate_optimized_upcall_stub(jobject mh, Method* entry, jobject jabi, jobject jconv) { + ShouldNotCallThis(); + return nullptr; +} + +bool ProgrammableUpcallHandler::supports_optimized_upcalls() { + return false; +} diff --git a/src/hotspot/cpu/x86/universalUpcallHandler_x86_64.cpp b/src/hotspot/cpu/x86/universalUpcallHandler_x86_64.cpp index 0b71f79a17a..59d33b42a55 100644 --- a/src/hotspot/cpu/x86/universalUpcallHandler_x86_64.cpp +++ b/src/hotspot/cpu/x86/universalUpcallHandler_x86_64.cpp @@ -24,8 +24,17 @@ #include "precompiled.hpp" #include "asm/macroAssembler.hpp" #include "code/codeBlob.hpp" +#include "code/codeBlob.hpp" +#include "code/vmreg.inline.hpp" +#include "compiler/disassembler.hpp" +#include "logging/logStream.hpp" #include "memory/resourceArea.hpp" #include "prims/universalUpcallHandler.hpp" +#include "runtime/sharedRuntime.hpp" +#include "runtime/signature.hpp" +#include "runtime/stubRoutines.hpp" +#include "utilities/formatBuffer.hpp" +#include "utilities/globalDefinitions.hpp" #define __ _masm-> @@ -141,3 +150,701 @@ address ProgrammableUpcallHandler::generate_upcall_stub(jobject rec, jobject jab return blob->code_begin(); } + +struct ArgMove { + BasicType bt; + VMRegPair from; + VMRegPair to; + + bool is_identity() const { + return from.first() == to.first() && from.second() == to.second(); + } +}; + +static GrowableArray compute_argument_shuffle(Method* entry, int& out_arg_size_bytes, const CallRegs& conv, BasicType& ret_type) { + assert(entry->is_static(), ""); + + // Fill in the signature array, for the calling-convention call. + const int total_out_args = entry->size_of_parameters(); + assert(total_out_args > 0, "receiver arg "); + + BasicType* out_sig_bt = NEW_RESOURCE_ARRAY(BasicType, total_out_args); + VMRegPair* out_regs = NEW_RESOURCE_ARRAY(VMRegPair, total_out_args); + + { + int i = 0; + SignatureStream ss(entry->signature()); + for (; !ss.at_return_type(); ss.next()) { + out_sig_bt[i++] = ss.type(); // Collect remaining bits of signature + if (ss.type() == T_LONG || ss.type() == T_DOUBLE) + out_sig_bt[i++] = T_VOID; // Longs & doubles take 2 Java slots + } + assert(i == total_out_args, ""); + ret_type = ss.type(); + } + + int out_arg_slots = SharedRuntime::java_calling_convention(out_sig_bt, out_regs, total_out_args); + + const int total_in_args = total_out_args - 1; // skip receiver + BasicType* in_sig_bt = NEW_RESOURCE_ARRAY(BasicType, total_in_args); + VMRegPair* in_regs = NEW_RESOURCE_ARRAY(VMRegPair, total_in_args); + + for (int i = 0; i < total_in_args ; i++ ) { + in_sig_bt[i] = out_sig_bt[i+1]; // skip receiver + } + + // Now figure out where the args must be stored and how much stack space they require. + conv.calling_convention(in_sig_bt, in_regs, total_in_args); + + GrowableArray arg_order(2 * total_in_args); + + VMRegPair tmp_vmreg; + tmp_vmreg.set2(rbx->as_VMReg()); + + // Compute a valid move order, using tmp_vmreg to break any cycles + SharedRuntime::compute_move_order(in_sig_bt, + total_in_args, in_regs, + total_out_args, out_regs, + arg_order, + tmp_vmreg); + + GrowableArray arg_order_vmreg(total_in_args); // conservative + +#ifdef ASSERT + bool reg_destroyed[RegisterImpl::number_of_registers]; + bool freg_destroyed[XMMRegisterImpl::number_of_registers]; + for ( int r = 0 ; r < RegisterImpl::number_of_registers ; r++ ) { + reg_destroyed[r] = false; + } + for ( int f = 0 ; f < XMMRegisterImpl::number_of_registers ; f++ ) { + freg_destroyed[f] = false; + } +#endif // ASSERT + + for (int i = 0; i < arg_order.length(); i += 2) { + int in_arg = arg_order.at(i); + int out_arg = arg_order.at(i + 1); + + assert(in_arg != -1 || out_arg != -1, ""); + BasicType arg_bt = (in_arg != -1 ? in_sig_bt[in_arg] : out_sig_bt[out_arg]); + switch (arg_bt) { + case T_BOOLEAN: + case T_BYTE: + case T_SHORT: + case T_CHAR: + case T_INT: + case T_FLOAT: + break; // process + + case T_LONG: + case T_DOUBLE: + assert(in_arg == -1 || (in_arg + 1 < total_in_args && in_sig_bt[in_arg + 1] == T_VOID), "bad arg list: %d", in_arg); + assert(out_arg == -1 || (out_arg + 1 < total_out_args && out_sig_bt[out_arg + 1] == T_VOID), "bad arg list: %d", out_arg); + break; // process + + case T_VOID: + continue; // skip + + default: + fatal("found in upcall args: %s", type2name(arg_bt)); + } + + ArgMove move; + move.bt = arg_bt; + move.from = (in_arg != -1 ? in_regs[in_arg] : tmp_vmreg); + move.to = (out_arg != -1 ? out_regs[out_arg] : tmp_vmreg); + + if(move.is_identity()) { + continue; // useless move + } + +#ifdef ASSERT + if (in_arg != -1) { + if (in_regs[in_arg].first()->is_Register()) { + assert(!reg_destroyed[in_regs[in_arg].first()->as_Register()->encoding()], "destroyed reg!"); + } else if (in_regs[in_arg].first()->is_XMMRegister()) { + assert(!freg_destroyed[in_regs[in_arg].first()->as_XMMRegister()->encoding()], "destroyed reg!"); + } + } + if (out_arg != -1) { + if (out_regs[out_arg].first()->is_Register()) { + reg_destroyed[out_regs[out_arg].first()->as_Register()->encoding()] = true; + } else if (out_regs[out_arg].first()->is_XMMRegister()) { + freg_destroyed[out_regs[out_arg].first()->as_XMMRegister()->encoding()] = true; + } + } +#endif /* ASSERT */ + + arg_order_vmreg.push(move); + } + + int stack_slots = SharedRuntime::out_preserve_stack_slots() + out_arg_slots; + out_arg_size_bytes = align_up(stack_slots * VMRegImpl::stack_slot_size, StackAlignmentInBytes); + + return arg_order_vmreg; +} + +static const char* null_safe_string(const char* str) { + return str == nullptr ? "NULL" : str; +} + +#ifdef ASSERT +static void print_arg_moves(const GrowableArray& arg_moves, Method* entry) { + LogTarget(Trace, foreign) lt; + if (lt.is_enabled()) { + ResourceMark rm; + LogStream ls(lt); + ls.print_cr("Argument shuffle for %s {", entry->name_and_sig_as_C_string()); + for (int i = 0; i < arg_moves.length(); i++) { + ArgMove arg_mv = arg_moves.at(i); + BasicType arg_bt = arg_mv.bt; + VMRegPair from_vmreg = arg_mv.from; + VMRegPair to_vmreg = arg_mv.to; + + ls.print("Move a %s from (", null_safe_string(type2name(arg_bt))); + from_vmreg.first()->print_on(&ls); + ls.print(","); + from_vmreg.second()->print_on(&ls); + ls.print(") to "); + to_vmreg.first()->print_on(&ls); + ls.print(","); + to_vmreg.second()->print_on(&ls); + ls.print_cr(")"); + } + ls.print_cr("}"); + } +} +#endif + +void save_java_frame_anchor(MacroAssembler* _masm, ByteSize store_offset, Register thread) { + __ block_comment("{ save_java_frame_anchor "); + // upcall->jfa._last_Java_fp = _thread->_anchor._last_Java_fp; + __ movptr(rscratch1, Address(thread, JavaThread::last_Java_fp_offset())); + __ movptr(Address(rsp, store_offset + JavaFrameAnchor::last_Java_fp_offset()), rscratch1); + + // upcall->jfa._last_Java_pc = _thread->_anchor._last_Java_pc; + __ movptr(rscratch1, Address(thread, JavaThread::last_Java_pc_offset())); + __ movptr(Address(rsp, store_offset + JavaFrameAnchor::last_Java_pc_offset()), rscratch1); + + // upcall->jfa._last_Java_sp = _thread->_anchor._last_Java_sp; + __ movptr(rscratch1, Address(thread, JavaThread::last_Java_sp_offset())); + __ movptr(Address(rsp, store_offset + JavaFrameAnchor::last_Java_sp_offset()), rscratch1); + __ block_comment("} save_java_frame_anchor "); +} + +void restore_java_frame_anchor(MacroAssembler* _masm, ByteSize load_offset, Register thread) { + __ block_comment("{ restore_java_frame_anchor "); + // thread->_last_Java_sp = NULL + __ movptr(Address(thread, JavaThread::last_Java_sp_offset()), NULL_WORD); + + // ThreadStateTransition::transition_from_java(_thread, _thread_in_vm); + // __ movl(Address(r15_thread, JavaThread::thread_state_offset()), _thread_in_native_trans); + __ movl(Address(r15_thread, JavaThread::thread_state_offset()), _thread_in_native); + + //_thread->frame_anchor()->copy(&_anchor); +// _thread->_last_Java_fp = upcall->_last_Java_fp; +// _thread->_last_Java_pc = upcall->_last_Java_pc; +// _thread->_last_Java_sp = upcall->_last_Java_sp; + + __ movptr(rscratch1, Address(rsp, load_offset + JavaFrameAnchor::last_Java_fp_offset())); + __ movptr(Address(thread, JavaThread::last_Java_fp_offset()), rscratch1); + + __ movptr(rscratch1, Address(rsp, load_offset + JavaFrameAnchor::last_Java_pc_offset())); + __ movptr(Address(thread, JavaThread::last_Java_pc_offset()), rscratch1); + + __ movptr(rscratch1, Address(rsp, load_offset + JavaFrameAnchor::last_Java_sp_offset())); + __ movptr(Address(thread, JavaThread::last_Java_sp_offset()), rscratch1); + __ block_comment("} restore_java_frame_anchor "); +} + +static void save_native_arguments(MacroAssembler* _masm, const CallRegs& conv, int arg_save_area_offset) { + __ block_comment("{ save_native_args "); + int store_offset = arg_save_area_offset; + for (int i = 0; i < conv._args_length; i++) { + VMReg reg = conv._arg_regs[i]; + if (reg->is_Register()) { + __ movptr(Address(rsp, store_offset), reg->as_Register()); + store_offset += 8; + } else if (reg->is_XMMRegister()) { + // Java API doesn't support vector args + __ movdqu(Address(rsp, store_offset), reg->as_XMMRegister()); + store_offset += 16; + } + // do nothing for stack + } + __ block_comment("} save_native_args "); +} + +static void restore_native_arguments(MacroAssembler* _masm, const CallRegs& conv, int arg_save_area_offset) { + __ block_comment("{ restore_native_args "); + int load_offset = arg_save_area_offset; + for (int i = 0; i < conv._args_length; i++) { + VMReg reg = conv._arg_regs[i]; + if (reg->is_Register()) { + __ movptr(reg->as_Register(), Address(rsp, load_offset)); + load_offset += 8; + } else if (reg->is_XMMRegister()) { + // Java API doesn't support vector args + __ movdqu(reg->as_XMMRegister(), Address(rsp, load_offset)); + load_offset += 16; + } + // do nothing for stack + } + __ block_comment("} restore_native_args "); +} + +static bool is_valid_XMM(XMMRegister reg) { + return reg->is_valid() && (UseAVX >= 3 || (reg->encoding() < 16)); // why is this not covered by is_valid()? +} + +// for callee saved regs, according to the caller's ABI +static int compute_reg_save_area_size(const ABIDescriptor& abi) { + int size = 0; + for (Register reg = as_Register(0); reg->is_valid(); reg = reg->successor()) { + if (reg == rbp || reg == rsp) continue; // saved/restored by prologue/epilogue + if (!abi.is_volatile_reg(reg)) { + size += 8; // bytes + } + } + + for (XMMRegister reg = as_XMMRegister(0); is_valid_XMM(reg); reg = reg->successor()) { + if (!abi.is_volatile_reg(reg)) { + if (UseAVX >= 3) { + size += 64; // bytes + } else if (UseAVX >= 1) { + size += 32; + } else { + size += 16; + } + } + } + +#ifndef _WIN64 + // for mxcsr + size += 8; +#endif + + return size; +} + +static int compute_arg_save_area_size(const CallRegs& conv) { + int result_size = 0; + for (int i = 0; i < conv._args_length; i++) { + VMReg reg = conv._arg_regs[i]; + if (reg->is_Register()) { + result_size += 8; + } else if (reg->is_XMMRegister()) { + // Java API doesn't support vector args + result_size += 16; + } + // do nothing for stack + } + return result_size; +} + +constexpr int MXCSR_MASK = 0xFFC0; // Mask out any pending exceptions + +static void preserve_callee_saved_registers(MacroAssembler* _masm, const ABIDescriptor& abi, int reg_save_area_offset) { + // 1. iterate all registers in the architecture + // - check if they are volatile or not for the given abi + // - if NOT, we need to save it here + // 2. save mxcsr on non-windows platforms + + int offset = reg_save_area_offset; + + __ block_comment("{ preserve_callee_saved_regs "); + for (Register reg = as_Register(0); reg->is_valid(); reg = reg->successor()) { + if (reg == rbp || reg == rsp) continue; // saved/restored by prologue/epilogue + if (!abi.is_volatile_reg(reg)) { + __ movptr(Address(rsp, offset), reg); + offset += 8; + } + } + + for (XMMRegister reg = as_XMMRegister(0); is_valid_XMM(reg); reg = reg->successor()) { + if (!abi.is_volatile_reg(reg)) { + if (UseAVX >= 3) { + __ evmovdqul(Address(rsp, offset), reg, Assembler::AVX_512bit); + offset += 64; + } else if (UseAVX >= 1) { + __ vmovdqu(Address(rsp, offset), reg); + offset += 32; + } else { + __ movdqu(Address(rsp, offset), reg); + offset += 16; + } + } + } + +#ifndef _WIN64 + { + const Address mxcsr_save(rsp, offset); + Label skip_ldmx; + __ stmxcsr(mxcsr_save); + __ movl(rax, mxcsr_save); + __ andl(rax, MXCSR_MASK); // Only check control and mask bits + ExternalAddress mxcsr_std(StubRoutines::addr_mxcsr_std()); + __ cmp32(rax, mxcsr_std); + __ jcc(Assembler::equal, skip_ldmx); + __ ldmxcsr(mxcsr_std); + __ bind(skip_ldmx); + } +#endif + + __ block_comment("} preserve_callee_saved_regs "); +} + +static void restore_callee_saved_registers(MacroAssembler* _masm, const ABIDescriptor& abi, int reg_save_area_offset) { + // 1. iterate all registers in the architecture + // - check if they are volatile or not for the given abi + // - if NOT, we need to restore it here + // 2. restore mxcsr on non-windows platforms + + int offset = reg_save_area_offset; + + __ block_comment("{ restore_callee_saved_regs "); + for (Register reg = as_Register(0); reg->is_valid(); reg = reg->successor()) { + if (reg == rbp || reg == rsp) continue; // saved/restored by prologue/epilogue + if (!abi.is_volatile_reg(reg)) { + __ movptr(reg, Address(rsp, offset)); + offset += 8; + } + } + + for (XMMRegister reg = as_XMMRegister(0); is_valid_XMM(reg); reg = reg->successor()) { + if (!abi.is_volatile_reg(reg)) { + if (UseAVX >= 3) { + __ evmovdqul(reg, Address(rsp, offset), Assembler::AVX_512bit); + offset += 64; + } else if (UseAVX >= 1) { + __ vmovdqu(reg, Address(rsp, offset)); + offset += 32; + } else { + __ movdqu(reg, Address(rsp, offset)); + offset += 16; + } + } + } + +#ifndef _WIN64 + const Address mxcsr_save(rsp, offset); + __ ldmxcsr(mxcsr_save); +#endif + + __ block_comment("} restore_callee_saved_regs "); +} + +static void shuffle_arguments(MacroAssembler* _masm, const GrowableArray& arg_moves) { + for (int i = 0; i < arg_moves.length(); i++) { + ArgMove arg_mv = arg_moves.at(i); + BasicType arg_bt = arg_mv.bt; + VMRegPair from_vmreg = arg_mv.from; + VMRegPair to_vmreg = arg_mv.to; + + assert( + !((from_vmreg.first()->is_Register() && to_vmreg.first()->is_XMMRegister()) + || (from_vmreg.first()->is_XMMRegister() && to_vmreg.first()->is_Register())), + "move between gp and fp reg not supported"); + + __ block_comment(err_msg("bt=%s", null_safe_string(type2name(arg_bt)))); + switch (arg_bt) { + case T_BOOLEAN: + case T_BYTE: + case T_SHORT: + case T_CHAR: + case T_INT: + SharedRuntime::move32_64(_masm, from_vmreg, to_vmreg); + break; + + case T_FLOAT: + SharedRuntime::float_move(_masm, from_vmreg, to_vmreg); + break; + + case T_DOUBLE: + SharedRuntime::double_move(_masm, from_vmreg, to_vmreg); + break; + + case T_LONG : + SharedRuntime::long_move(_masm, from_vmreg, to_vmreg); + break; + + default: + fatal("found in upcall args: %s", type2name(arg_bt)); + } + } +} + +struct AuxiliarySaves { + JavaFrameAnchor jfa; + uintptr_t thread; + bool should_detach; +}; + +address ProgrammableUpcallHandler::generate_optimized_upcall_stub(jobject receiver, Method* entry, jobject jabi, jobject jconv) { + ResourceMark rm; + const ABIDescriptor abi = ForeignGlobals::parse_abi_descriptor(jabi); + const CallRegs conv = ForeignGlobals::parse_call_regs(jconv); + assert(conv._rets_length <= 1, "no multi reg returns"); + CodeBuffer buffer("upcall_stub_linkToNative", /* code_size = */ 1024, /* locs_size = */ 1024); + + int register_size = sizeof(uintptr_t); + int buffer_alignment = xmm_reg_size; + + int out_arg_area = -1; + BasicType ret_type; + GrowableArray arg_moves = compute_argument_shuffle(entry, out_arg_area, conv, ret_type); + assert(out_arg_area != -1, "Should have been set"); + DEBUG_ONLY(print_arg_moves(arg_moves, entry);) + + // out_arg_area (for stack arguments) doubles as shadow space for native calls. + // make sure it is big enough. + if (out_arg_area < frame::arg_reg_save_area_bytes) { + out_arg_area = frame::arg_reg_save_area_bytes; + } + + int reg_save_area_size = compute_reg_save_area_size(abi); + int arg_save_area_size = compute_arg_save_area_size(conv); + // To spill receiver during deopt + int deopt_spill_size = 1 * BytesPerWord; + + int shuffle_area_offset = 0; + int deopt_spill_offset = shuffle_area_offset + out_arg_area; + int arg_save_area_offset = deopt_spill_offset + deopt_spill_size; + int reg_save_area_offset = arg_save_area_offset + arg_save_area_size; + int auxiliary_saves_offset = reg_save_area_offset + reg_save_area_size; + int frame_bottom_offset = auxiliary_saves_offset + sizeof(AuxiliarySaves); + + ByteSize jfa_offset = in_ByteSize(auxiliary_saves_offset) + byte_offset_of(AuxiliarySaves, jfa); + ByteSize thread_offset = in_ByteSize(auxiliary_saves_offset) + byte_offset_of(AuxiliarySaves, thread); + ByteSize should_detach_offset = in_ByteSize(auxiliary_saves_offset) + byte_offset_of(AuxiliarySaves, should_detach); + + int frame_size = frame_bottom_offset; + frame_size = align_up(frame_size, StackAlignmentInBytes); + + // Ok The space we have allocated will look like: + // + // + // FP-> | | + // |---------------------| = frame_bottom_offset = frame_size + // | | + // | AuxiliarySaves | + // |---------------------| = auxiliary_saves_offset + // | | + // | reg_save_area | + // |---------------------| = reg_save_are_offset + // | | + // | arg_save_area | + // |---------------------| = arg_save_are_offset + // | | + // | deopt_spill | + // |---------------------| = deopt_spill_offset + // | | + // SP-> | out_arg_area | needs to be at end for shadow space + // + // + + ////////////////////////////////////////////////////////////////////////////// + + MacroAssembler* _masm = new MacroAssembler(&buffer); + Label call_return; + address start = __ pc(); + __ enter(); // set up frame + if ((abi._stack_alignment_bytes % 16) != 0) { + // stack alignment of caller is not a multiple of 16 + __ andptr(rsp, -StackAlignmentInBytes); // align stack + } + // allocate frame (frame_size is also aligned, so stack is still aligned) + __ subptr(rsp, frame_size); + + // we have to always spill args since we need to do a call to get the thread + // (and maybe attach it). + save_native_arguments(_masm, conv, arg_save_area_offset); + + preserve_callee_saved_registers(_masm, abi, reg_save_area_offset); + + __ block_comment("{ get_thread"); + __ vzeroupper(); + __ lea(c_rarg0, Address(rsp, should_detach_offset)); + // stack already aligned + __ call(RuntimeAddress(CAST_FROM_FN_PTR(address, ProgrammableUpcallHandler::maybe_attach_and_get_thread))); + __ movptr(r15_thread, rax); + __ reinit_heapbase(); + __ movptr(Address(rsp, thread_offset), r15_thread); + __ block_comment("} get_thread"); + + // TODO: + // We expect not to be coming from JNI code, but we might be. + // We should figure out what our stance is on supporting that and then maybe add + // some more handling here for: + // - handle blocks + // - check for active exceptions (and emit an error) + + __ block_comment("{ safepoint poll"); + __ movl(Address(r15_thread, JavaThread::thread_state_offset()), _thread_in_native_trans); + + if (os::is_MP()) { + __ membar(Assembler::Membar_mask_bits( + Assembler::LoadLoad | Assembler::StoreLoad | + Assembler::LoadStore | Assembler::StoreStore)); + } + + // check for safepoint operation in progress and/or pending suspend requests + Label L_after_safepoint_poll; + Label L_safepoint_poll_slow_path; + + __ safepoint_poll(L_safepoint_poll_slow_path, r15_thread, false /* at_return */, false /* in_nmethod */); + + __ cmpl(Address(r15_thread, JavaThread::suspend_flags_offset()), 0); + __ jcc(Assembler::notEqual, L_safepoint_poll_slow_path); + + __ bind(L_after_safepoint_poll); + __ block_comment("} safepoint poll"); + // change thread state + __ movl(Address(r15_thread, JavaThread::thread_state_offset()), _thread_in_Java); + + __ block_comment("{ reguard stack check"); + Label L_reguard; + Label L_after_reguard; + __ cmpl(Address(r15_thread, JavaThread::stack_guard_state_offset()), StackOverflow::stack_guard_yellow_reserved_disabled); + __ jcc(Assembler::equal, L_reguard); + __ bind(L_after_reguard); + __ block_comment("} reguard stack check"); + + __ block_comment("{ argument shuffle"); + // TODO merge these somehow + restore_native_arguments(_masm, conv, arg_save_area_offset); + shuffle_arguments(_masm, arg_moves); + __ block_comment("} argument shuffle"); + + __ block_comment("{ receiver "); + __ movptr(rscratch1, (intptr_t)receiver); + __ resolve_jobject(rscratch1, r15_thread, rscratch2); + __ movptr(j_rarg0, rscratch1); + __ block_comment("} receiver "); + + __ mov_metadata(rbx, entry); + __ movptr(Address(r15_thread, JavaThread::callee_target_offset()), rbx); // just in case callee is deoptimized + __ reinit_heapbase(); + + save_java_frame_anchor(_masm, jfa_offset, r15_thread); + __ reset_last_Java_frame(r15_thread, true); + + __ call(Address(rbx, Method::from_compiled_offset())); + +#ifdef ASSERT + if (conv._rets_length == 1) { // 0 or 1 + VMReg j_expected_result_reg; + switch (ret_type) { + case T_BOOLEAN: + case T_BYTE: + case T_SHORT: + case T_CHAR: + case T_INT: + case T_LONG: + j_expected_result_reg = rax->as_VMReg(); + break; + case T_FLOAT: + case T_DOUBLE: + j_expected_result_reg = xmm0->as_VMReg(); + break; + default: + fatal("unexpected return type: %s", type2name(ret_type)); + } + // No need to move for now, since CallArranger can pick a return type + // that goes in the same reg for both CCs. But, at least assert they are the same + assert(conv._ret_regs[0] == j_expected_result_reg, + "unexpected result register: %s != %s", conv._ret_regs[0]->name(), j_expected_result_reg->name()); + } +#endif + + __ bind(call_return); + + // also sets last Java frame + __ movptr(r15_thread, Address(rsp, thread_offset)); + // TODO corrupted thread pointer causes havoc. Can we verify it here? + restore_java_frame_anchor(_masm, jfa_offset, r15_thread); // also transitions to native state + + __ block_comment("{ maybe_detach_thread"); + Label L_after_detach; + __ cmpb(Address(rsp, should_detach_offset), 0); + __ jcc(Assembler::equal, L_after_detach); + __ vzeroupper(); + __ mov(c_rarg0, r15_thread); + // stack already aligned + __ call(RuntimeAddress(CAST_FROM_FN_PTR(address, ProgrammableUpcallHandler::detach_thread))); + __ reinit_heapbase(); + __ bind(L_after_detach); + __ block_comment("} maybe_detach_thread"); + + restore_callee_saved_registers(_masm, abi, reg_save_area_offset); + + __ leave(); + __ ret(0); + + ////////////////////////////////////////////////////////////////////////////// + + __ block_comment("{ L_safepoint_poll_slow_path"); + __ bind(L_safepoint_poll_slow_path); + __ vzeroupper(); + __ mov(c_rarg0, r15_thread); + // stack already aligned + __ call(RuntimeAddress(CAST_FROM_FN_PTR(address, JavaThread::check_special_condition_for_native_trans))); + __ reinit_heapbase(); + __ jmp(L_after_safepoint_poll); + __ block_comment("} L_safepoint_poll_slow_path"); + + ////////////////////////////////////////////////////////////////////////////// + + __ block_comment("{ L_reguard"); + __ bind(L_reguard); + __ vzeroupper(); + // stack already aligned + __ call(RuntimeAddress(CAST_FROM_FN_PTR(address, SharedRuntime::reguard_yellow_pages))); + __ reinit_heapbase(); + __ jmp(L_after_reguard); + __ block_comment("} L_reguard"); + + ////////////////////////////////////////////////////////////////////////////// + + __ block_comment("{ exception handler"); + + intptr_t exception_handler_offset = __ pc() - start; + + // TODO: this is always the same, can we bypass and call handle_uncaught_exception directly? + + // native caller has no idea how to handle exceptions + // we just crash here. Up to callee to catch exceptions. + __ verify_oop(rax); + __ vzeroupper(); + __ mov(c_rarg0, rax); + __ andptr(rsp, -StackAlignmentInBytes); // align stack as required by ABI + __ subptr(rsp, frame::arg_reg_save_area_bytes); // windows (not really needed) + __ call(RuntimeAddress(CAST_FROM_FN_PTR(address, ProgrammableUpcallHandler::handle_uncaught_exception))); + __ should_not_reach_here(); + + __ block_comment("} exception handler"); + + _masm->flush(); + + +#ifndef PRODUCT + stringStream ss; + ss.print("optimized_upcall_stub_%s", entry->signature()->as_C_string()); + const char* name = _masm->code_string(ss.as_string()); +#else // PRODUCT + const char* name = "optimized_upcall_stub"; +#endif // PRODUCT + + OptimizedEntryBlob* blob = OptimizedEntryBlob::create(name, &buffer, exception_handler_offset, receiver, jfa_offset); + + if (TraceOptimizedUpcallStubs) { + blob->print_on(tty); + Disassembler::decode(blob, tty); + } + + return blob->code_begin(); +} + +bool ProgrammableUpcallHandler::supports_optimized_upcalls() { + return true; +} diff --git a/src/hotspot/share/ci/ciNativeEntryPoint.cpp b/src/hotspot/share/ci/ciNativeEntryPoint.cpp index 5d60fd3c543..963865b9bf0 100644 --- a/src/hotspot/share/ci/ciNativeEntryPoint.cpp +++ b/src/hotspot/share/ci/ciNativeEntryPoint.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -60,11 +60,6 @@ ciNativeEntryPoint::ciNativeEntryPoint(instanceHandle h_i) : ciInstance(h_i), _n _ret_moves = getVMRegArray(CURRENT_ENV->get_object(jdk_internal_invoke_NativeEntryPoint::returnMoves(get_oop()))->as_array()); } -address ciNativeEntryPoint::entry_point() const { - VM_ENTRY_MARK; - return jdk_internal_invoke_NativeEntryPoint::addr(get_oop()); -} - jint ciNativeEntryPoint::shadow_space() const { VM_ENTRY_MARK; return jdk_internal_invoke_NativeEntryPoint::shadow_space(get_oop()); diff --git a/src/hotspot/share/ci/ciNativeEntryPoint.hpp b/src/hotspot/share/ci/ciNativeEntryPoint.hpp index 21d490b9614..65e622cbf4d 100644 --- a/src/hotspot/share/ci/ciNativeEntryPoint.hpp +++ b/src/hotspot/share/ci/ciNativeEntryPoint.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -44,7 +44,6 @@ public: // What kind of ciObject is this? bool is_native_entry_point() const { return true; } - address entry_point() const; jint shadow_space() const; VMReg* argMoves() const; VMReg* returnMoves() const; diff --git a/src/hotspot/share/classfile/javaClasses.cpp b/src/hotspot/share/classfile/javaClasses.cpp index fb6efe264ad..6a067c13c42 100644 --- a/src/hotspot/share/classfile/javaClasses.cpp +++ b/src/hotspot/share/classfile/javaClasses.cpp @@ -3890,7 +3890,6 @@ bool java_lang_invoke_LambdaForm::is_instance(oop obj) { return obj != NULL && is_subclass(obj->klass()); } -int jdk_internal_invoke_NativeEntryPoint::_addr_offset; int jdk_internal_invoke_NativeEntryPoint::_shadow_space_offset; int jdk_internal_invoke_NativeEntryPoint::_argMoves_offset; int jdk_internal_invoke_NativeEntryPoint::_returnMoves_offset; @@ -3899,7 +3898,6 @@ int jdk_internal_invoke_NativeEntryPoint::_method_type_offset; int jdk_internal_invoke_NativeEntryPoint::_name_offset; #define NEP_FIELDS_DO(macro) \ - macro(_addr_offset, k, "addr", long_signature, false); \ macro(_shadow_space_offset, k, "shadowSpace", int_signature, false); \ macro(_argMoves_offset, k, "argMoves", long_array_signature, false); \ macro(_returnMoves_offset, k, "returnMoves", long_array_signature, false); \ @@ -3922,10 +3920,6 @@ void jdk_internal_invoke_NativeEntryPoint::serialize_offsets(SerializeClosure* f } #endif -address jdk_internal_invoke_NativeEntryPoint::addr(oop entry) { - return (address)entry->long_field(_addr_offset); -} - jint jdk_internal_invoke_NativeEntryPoint::shadow_space(oop entry) { return entry->int_field(_shadow_space_offset); } diff --git a/src/hotspot/share/classfile/javaClasses.hpp b/src/hotspot/share/classfile/javaClasses.hpp index 0602f99db23..8be56fb5462 100644 --- a/src/hotspot/share/classfile/javaClasses.hpp +++ b/src/hotspot/share/classfile/javaClasses.hpp @@ -1043,7 +1043,6 @@ class jdk_internal_invoke_NativeEntryPoint: AllStatic { friend class JavaClasses; private: - static int _addr_offset; // type is jlong static int _shadow_space_offset; static int _argMoves_offset; static int _returnMoves_offset; @@ -1057,7 +1056,6 @@ class jdk_internal_invoke_NativeEntryPoint: AllStatic { static void serialize_offsets(SerializeClosure* f) NOT_CDS_RETURN; // Accessors - static address addr(oop entry); static jint shadow_space(oop entry); static oop argMoves(oop entry); static oop returnMoves(oop entry); @@ -1073,7 +1071,6 @@ class jdk_internal_invoke_NativeEntryPoint: AllStatic { static bool is_instance(oop obj); // Accessors for code generation: - static int addr_offset_in_bytes() { return _addr_offset; } static int shadow_space_offset_in_bytes() { return _shadow_space_offset; } static int argMoves_offset_in_bytes() { return _argMoves_offset; } static int returnMoves_offset_in_bytes() { return _returnMoves_offset; } diff --git a/src/hotspot/share/classfile/vmSymbols.hpp b/src/hotspot/share/classfile/vmSymbols.hpp index c2d0e51075b..f8b0b386b6d 100644 --- a/src/hotspot/share/classfile/vmSymbols.hpp +++ b/src/hotspot/share/classfile/vmSymbols.hpp @@ -346,7 +346,7 @@ template(DEFAULT_CONTEXT_name, "DEFAULT_CONTEXT") \ NOT_LP64( do_alias(intptr_signature, int_signature) ) \ LP64_ONLY( do_alias(intptr_signature, long_signature) ) \ - /* Panama Support */ \ + /* Foreign API Support */ \ template(jdk_internal_invoke_NativeEntryPoint, "jdk/internal/invoke/NativeEntryPoint") \ template(jdk_internal_invoke_NativeEntryPoint_signature, "Ljdk/internal/invoke/NativeEntryPoint;") \ template(jdk_incubator_foreign_MemoryAccess, "jdk/incubator/foreign/MemoryAccess") \ diff --git a/src/hotspot/share/code/codeBlob.cpp b/src/hotspot/share/code/codeBlob.cpp index 2652d7c26df..da225bc2b0f 100644 --- a/src/hotspot/share/code/codeBlob.cpp +++ b/src/hotspot/share/code/codeBlob.cpp @@ -709,3 +709,30 @@ void SingletonBlob::print_value_on(outputStream* st) const { void DeoptimizationBlob::print_value_on(outputStream* st) const { st->print_cr("Deoptimization (frame not available)"); } + +// Implementation of OptimizedEntryBlob + +OptimizedEntryBlob::OptimizedEntryBlob(const char* name, int size, CodeBuffer* cb, intptr_t exception_handler_offset, + jobject receiver, ByteSize jfa_sp_offset) : + BufferBlob(name, size, cb), + _exception_handler_offset(exception_handler_offset), + _receiver(receiver), + _jfa_sp_offset(jfa_sp_offset) { + CodeCache::commit(this); +} + +OptimizedEntryBlob* OptimizedEntryBlob::create(const char* name, CodeBuffer* cb, intptr_t exception_handler_offset, + jobject receiver, ByteSize jfa_sp_offset) { + ThreadInVMfromUnknown __tiv; // get to VM state in case we block on CodeCache_lock + + OptimizedEntryBlob* blob = nullptr; + unsigned int size = CodeBlob::allocation_size(cb, sizeof(OptimizedEntryBlob)); + { + MutexLocker mu(CodeCache_lock, Mutex::_no_safepoint_check_flag); + blob = new (size) OptimizedEntryBlob(name, size, cb, exception_handler_offset, receiver, jfa_sp_offset); + } + // Track memory usage statistic after releasing CodeCache_lock + MemoryService::track_code_cache_memory_usage(); + + return blob; +} diff --git a/src/hotspot/share/code/codeBlob.hpp b/src/hotspot/share/code/codeBlob.hpp index 1937f2c7df4..61d33463387 100644 --- a/src/hotspot/share/code/codeBlob.hpp +++ b/src/hotspot/share/code/codeBlob.hpp @@ -58,6 +58,7 @@ struct CodeBlobType { // AdapterBlob : Used to hold C2I/I2C adapters // VtableBlob : Used for holding vtable chunks // MethodHandlesAdapterBlob : Used to hold MethodHandles adapters +// OptimizedEntryBlob : Used for upcalls from native code // RuntimeStub : Call to VM runtime methods // SingletonBlob : Super-class for all blobs that exist in only one instance // DeoptimizationBlob : Used for deoptimization @@ -75,6 +76,8 @@ struct CodeBlobType { class CodeBlobLayout; +class OptimizedEntryBlob; // for as_optimized_entry_blob() +class JavaFrameAnchor; // for EntryBlob::jfa_for_frame class CodeBlob { friend class VMStructs; @@ -136,6 +139,7 @@ public: virtual bool is_vtable_blob() const { return false; } virtual bool is_method_handles_adapter_blob() const { return false; } virtual bool is_compiled() const { return false; } + virtual bool is_optimized_entry_blob() const { return false; } inline bool is_compiled_by_c1() const { return _type == compiler_c1; }; inline bool is_compiled_by_c2() const { return _type == compiler_c2; }; @@ -149,6 +153,7 @@ public: CompiledMethod* as_compiled_method_or_null() { return is_compiled() ? (CompiledMethod*) this : NULL; } CompiledMethod* as_compiled_method() { assert(is_compiled(), "must be compiled"); return (CompiledMethod*) this; } CodeBlob* as_codeblob_or_null() const { return (CodeBlob*) this; } + OptimizedEntryBlob* as_optimized_entry_blob() const { assert(is_optimized_entry_blob(), "must be entry blob"); return (OptimizedEntryBlob*) this; } // Boundaries address header_begin() const { return (address) this; } @@ -379,6 +384,7 @@ class BufferBlob: public RuntimeBlob { friend class AdapterBlob; friend class VtableBlob; friend class MethodHandlesAdapterBlob; + friend class OptimizedEntryBlob; friend class WhiteBox; private: @@ -718,4 +724,33 @@ class SafepointBlob: public SingletonBlob { bool is_safepoint_stub() const { return true; } }; +//---------------------------------------------------------------------------------------------------- + +// For optimized upcall stubs +class OptimizedEntryBlob: public BufferBlob { + private: + intptr_t _exception_handler_offset; + jobject _receiver; + ByteSize _jfa_sp_offset; + + OptimizedEntryBlob(const char* name, int size, CodeBuffer* cb, intptr_t exception_handler_offset, + jobject receiver, ByteSize jfa_sp_offset); + + public: + // Creation + static OptimizedEntryBlob* create(const char* name, CodeBuffer* cb, + intptr_t exception_handler_offset, jobject receiver, + ByteSize jfa_sp_offset); + + address exception_handler() { return code_begin() + _exception_handler_offset; } + jobject receiver() { return _receiver; } + ByteSize jfa_sp_offset() const { return _jfa_sp_offset; } + + // defined in frame_ARCH.cpp + JavaFrameAnchor* jfa_for_frame(const frame& frame) const; + + // Typing + virtual bool is_optimized_entry_blob() const override { return true; } +}; + #endif // SHARE_CODE_CODEBLOB_HPP diff --git a/src/hotspot/share/logging/logTag.hpp b/src/hotspot/share/logging/logTag.hpp index 97f857c1b8e..53cd1d36433 100644 --- a/src/hotspot/share/logging/logTag.hpp +++ b/src/hotspot/share/logging/logTag.hpp @@ -73,6 +73,7 @@ LOG_TAG(exceptions) \ LOG_TAG(exit) \ LOG_TAG(fingerprint) \ + DEBUG_ONLY(LOG_TAG(foreign)) \ LOG_TAG(free) \ LOG_TAG(freelist) \ LOG_TAG(gc) \ diff --git a/src/hotspot/share/opto/callGenerator.cpp b/src/hotspot/share/opto/callGenerator.cpp index 1b809fd5b78..ae05e8b5029 100644 --- a/src/hotspot/share/opto/callGenerator.cpp +++ b/src/hotspot/share/opto/callGenerator.cpp @@ -418,8 +418,9 @@ class LateInlineMHCallGenerator : public LateInlineCallGenerator { bool LateInlineMHCallGenerator::do_late_inline_check(Compile* C, JVMState* jvms) { // Even if inlining is not allowed, a virtual call can be strength-reduced to a direct call. bool allow_inline = C->inlining_incrementally(); - CallGenerator* cg = for_method_handle_inline(jvms, _caller, method(), allow_inline, _input_not_const); - assert(!_input_not_const, "sanity"); // shouldn't have been scheduled for inlining in the first place + bool input_not_const = true; + CallGenerator* cg = for_method_handle_inline(jvms, _caller, method(), allow_inline, input_not_const); + assert(!input_not_const, "sanity"); // shouldn't have been scheduled for inlining in the first place if (cg != NULL) { assert(!cg->is_late_inline() || cg->is_mh_late_inline() || AlwaysIncrementalInline, "we're doing late inlining"); @@ -1054,10 +1055,11 @@ CallGenerator* CallGenerator::for_method_handle_call(JVMState* jvms, ciMethod* c class NativeCallGenerator : public CallGenerator { private: + address _call_addr; ciNativeEntryPoint* _nep; public: - NativeCallGenerator(ciMethod* m, ciNativeEntryPoint* nep) - : CallGenerator(m), _nep(nep) {} + NativeCallGenerator(ciMethod* m, address call_addr, ciNativeEntryPoint* nep) + : CallGenerator(m), _call_addr(call_addr), _nep(nep) {} virtual JVMState* generate(JVMState* jvms); }; @@ -1065,13 +1067,12 @@ public: JVMState* NativeCallGenerator::generate(JVMState* jvms) { GraphKit kit(jvms); - Node* call = kit.make_native_call(tf(), method()->arg_size(), _nep); // -fallback, - nep + Node* call = kit.make_native_call(_call_addr, tf(), method()->arg_size(), _nep); // -fallback, - nep if (call == NULL) return NULL; kit.C->print_inlining_update(this); - address addr = _nep->entry_point(); if (kit.C->log() != NULL) { - kit.C->log()->elem("l2n_intrinsification_success bci='%d' entry_point='" INTPTR_FORMAT "'", jvms->bci(), p2i(addr)); + kit.C->log()->elem("l2n_intrinsification_success bci='%d' entry_point='" INTPTR_FORMAT "'", jvms->bci(), p2i(_call_addr)); } return kit.transfer_exceptions_into_jvms(); @@ -1204,12 +1205,16 @@ CallGenerator* CallGenerator::for_method_handle_inline(JVMState* jvms, ciMethod* case vmIntrinsics::_linkToNative: { - Node* nep = kit.argument(callee->arg_size() - 1); - if (nep->Opcode() == Op_ConP) { + Node* addr_n = kit.argument(1); // target address + Node* nep_n = kit.argument(callee->arg_size() - 1); // NativeEntryPoint + // This check needs to be kept in sync with the one in CallStaticJavaNode::Ideal + if (addr_n->Opcode() == Op_ConL && nep_n->Opcode() == Op_ConP) { input_not_const = false; - const TypeOopPtr* oop_ptr = nep->bottom_type()->is_oopptr(); - ciNativeEntryPoint* nep = oop_ptr->const_oop()->as_native_entry_point(); - return new NativeCallGenerator(callee, nep); + const TypeLong* addr_t = addr_n->bottom_type()->is_long(); + const TypeOopPtr* nep_t = nep_n->bottom_type()->is_oopptr(); + address addr = (address) addr_t->get_con(); + ciNativeEntryPoint* nep = nep_t->const_oop()->as_native_entry_point(); + return new NativeCallGenerator(callee, addr, nep); } else { print_inlining_failure(C, callee, jvms->depth() - 1, jvms->bci(), "NativeEntryPoint not constant"); diff --git a/src/hotspot/share/opto/callnode.cpp b/src/hotspot/share/opto/callnode.cpp index 78716a153f8..d09d643fe5f 100644 --- a/src/hotspot/share/opto/callnode.cpp +++ b/src/hotspot/share/opto/callnode.cpp @@ -1066,6 +1066,12 @@ Node* CallStaticJavaNode::Ideal(PhaseGVN* phase, bool can_reshape) { phase->C->prepend_late_inline(cg); set_generator(NULL); } + } else if (iid == vmIntrinsics::_linkToNative) { + if (in(TypeFunc::Parms + callee->arg_size() - 1)->Opcode() == Op_ConP /* NEP */ + && in(TypeFunc::Parms + 1)->Opcode() == Op_ConL /* address */) { + phase->C->prepend_late_inline(cg); + set_generator(NULL); + } } else { assert(callee->has_member_arg(), "wrong type of call?"); if (in(TypeFunc::Parms + callee->arg_size() - 1)->Opcode() == Op_ConP) { diff --git a/src/hotspot/share/opto/graphKit.cpp b/src/hotspot/share/opto/graphKit.cpp index 4632f5366aa..3b54aca2da0 100644 --- a/src/hotspot/share/opto/graphKit.cpp +++ b/src/hotspot/share/opto/graphKit.cpp @@ -2566,8 +2566,13 @@ Node* GraphKit::sign_extend_short(Node* in) { } //-----------------------------make_native_call------------------------------- -Node* GraphKit::make_native_call(const TypeFunc* call_type, uint nargs, ciNativeEntryPoint* nep) { - uint n_filtered_args = nargs - 2; // -fallback, -nep; +Node* GraphKit::make_native_call(address call_addr, const TypeFunc* call_type, uint nargs, ciNativeEntryPoint* nep) { + // Select just the actual call args to pass on + // [MethodHandle fallback, long addr, HALF addr, ... args , NativeEntryPoint nep] + // | | + // V V + // [ ... args ] + uint n_filtered_args = nargs - 4; // -fallback, -addr (2), -nep; ResourceMark rm; Node** argument_nodes = NEW_RESOURCE_ARRAY(Node*, n_filtered_args); const Type** arg_types = TypeTuple::fields(n_filtered_args); @@ -2577,7 +2582,7 @@ Node* GraphKit::make_native_call(const TypeFunc* call_type, uint nargs, ciNative { for (uint vm_arg_pos = 0, java_arg_read_pos = 0; vm_arg_pos < n_filtered_args; vm_arg_pos++) { - uint vm_unfiltered_arg_pos = vm_arg_pos + 1; // +1 to skip fallback handle argument + uint vm_unfiltered_arg_pos = vm_arg_pos + 3; // +3 to skip fallback handle argument and addr (2 since long) Node* node = argument(vm_unfiltered_arg_pos); const Type* type = call_type->domain()->field_at(TypeFunc::Parms + vm_unfiltered_arg_pos); VMReg reg = type == Type::HALF @@ -2613,7 +2618,6 @@ Node* GraphKit::make_native_call(const TypeFunc* call_type, uint nargs, ciNative TypeTuple::make(TypeFunc::Parms + n_returns, ret_types) ); - address call_addr = nep->entry_point(); if (nep->need_transition()) { RuntimeStub* invoker = SharedRuntime::make_native_invoker(call_addr, nep->shadow_space(), diff --git a/src/hotspot/share/opto/graphKit.hpp b/src/hotspot/share/opto/graphKit.hpp index 3cda4c1921b..965bbfbcdaf 100644 --- a/src/hotspot/share/opto/graphKit.hpp +++ b/src/hotspot/share/opto/graphKit.hpp @@ -801,7 +801,7 @@ class GraphKit : public Phase { Node* sign_extend_byte(Node* in); Node* sign_extend_short(Node* in); - Node* make_native_call(const TypeFunc* call_type, uint nargs, ciNativeEntryPoint* nep); + Node* make_native_call(address call_addr, const TypeFunc* call_type, uint nargs, ciNativeEntryPoint* nep); enum { // flag values for make_runtime_call RC_NO_FP = 1, // CallLeafNoFPNode diff --git a/src/hotspot/share/opto/lcm.cpp b/src/hotspot/share/opto/lcm.cpp index a8a4ab6f5d8..1ae98e27cc6 100644 --- a/src/hotspot/share/opto/lcm.cpp +++ b/src/hotspot/share/opto/lcm.cpp @@ -880,7 +880,7 @@ uint PhaseCFG::sched_call(Block* block, uint node_cnt, Node_List& worklist, Grow save_policy = _matcher._register_save_policy; break; case Op_CallNative: - // We use the c reg save policy here since Panama + // We use the c reg save policy here since Foreign Linker // only supports the C ABI currently. // TODO compute actual save policy based on nep->abi save_policy = _matcher._c_reg_save_policy; diff --git a/src/hotspot/share/prims/foreign_globals.cpp b/src/hotspot/share/prims/foreign_globals.cpp index 239791e51c4..3ec6c93f808 100644 --- a/src/hotspot/share/prims/foreign_globals.cpp +++ b/src/hotspot/share/prims/foreign_globals.cpp @@ -59,23 +59,28 @@ const BufferLayout ForeignGlobals::parse_buffer_layout(jobject jlayout) { return instance().parse_buffer_layout_impl(jlayout); } +const CallRegs ForeignGlobals::parse_call_regs(jobject jconv) { + return instance().parse_call_regs_impl(jconv); +} + ForeignGlobals::ForeignGlobals() { JavaThread* current_thread = JavaThread::current(); ResourceMark rm(current_thread); // ABIDescriptor InstanceKlass* k_ABI = find_InstanceKlass(FOREIGN_ABI "ABIDescriptor", current_thread); - const char* strVMSArray = "[[L" FOREIGN_ABI "VMStorage;"; - Symbol* symVMSArray = SymbolTable::new_symbol(strVMSArray, (int)strlen(strVMSArray)); - ABI.inputStorage_offset = field_offset(k_ABI, "inputStorage", symVMSArray); - ABI.outputStorage_offset = field_offset(k_ABI, "outputStorage", symVMSArray); - ABI.volatileStorage_offset = field_offset(k_ABI, "volatileStorage", symVMSArray); + const char* strVMSArrayArray = "[[L" FOREIGN_ABI "VMStorage;"; + Symbol* symVMSArrayArray = SymbolTable::new_symbol(strVMSArrayArray); + ABI.inputStorage_offset = field_offset(k_ABI, "inputStorage", symVMSArrayArray); + ABI.outputStorage_offset = field_offset(k_ABI, "outputStorage", symVMSArrayArray); + ABI.volatileStorage_offset = field_offset(k_ABI, "volatileStorage", symVMSArrayArray); ABI.stackAlignment_offset = field_offset(k_ABI, "stackAlignment", vmSymbols::int_signature()); ABI.shadowSpace_offset = field_offset(k_ABI, "shadowSpace", vmSymbols::int_signature()); // VMStorage InstanceKlass* k_VMS = find_InstanceKlass(FOREIGN_ABI "VMStorage", current_thread); VMS.index_offset = field_offset(k_VMS, "index", vmSymbols::int_signature()); + VMS.type_offset = field_offset(k_VMS, "type", vmSymbols::int_signature()); // BufferLayout InstanceKlass* k_BL = find_InstanceKlass(FOREIGN_ABI "BufferLayout", current_thread); @@ -85,4 +90,41 @@ ForeignGlobals::ForeignGlobals() { BL.stack_args_offset = field_offset(k_BL, "stack_args", vmSymbols::long_signature()); BL.input_type_offsets_offset = field_offset(k_BL, "input_type_offsets", vmSymbols::long_array_signature()); BL.output_type_offsets_offset = field_offset(k_BL, "output_type_offsets", vmSymbols::long_array_signature()); + + // CallRegs + const char* strVMSArray = "[L" FOREIGN_ABI "VMStorage;"; + Symbol* symVMSArray = SymbolTable::new_symbol(strVMSArray); + InstanceKlass* k_CC = find_InstanceKlass(FOREIGN_ABI "ProgrammableUpcallHandler$CallRegs", current_thread); + CallConvOffsets.arg_regs_offset = field_offset(k_CC, "argRegs", symVMSArray); + CallConvOffsets.ret_regs_offset = field_offset(k_CC, "retRegs", symVMSArray); +} + +void CallRegs::calling_convention(BasicType* sig_bt, VMRegPair *parm_regs, uint argcnt) const { + int src_pos = 0; + for (uint i = 0; i < argcnt; i++) { + switch (sig_bt[i]) { + case T_BOOLEAN: + case T_CHAR: + case T_BYTE: + case T_SHORT: + case T_INT: + case T_FLOAT: + assert(src_pos < _args_length, "oob"); + parm_regs[i].set1(_arg_regs[src_pos++]); + break; + case T_LONG: + case T_DOUBLE: + assert((i + 1) < argcnt && sig_bt[i + 1] == T_VOID, "expecting half"); + assert(src_pos < _args_length, "oob"); + parm_regs[i].set2(_arg_regs[src_pos++]); + break; + case T_VOID: // Halves of longs and doubles + assert(i != 0 && (sig_bt[i - 1] == T_LONG || sig_bt[i - 1] == T_DOUBLE), "expecting half"); + parm_regs[i].set_bad(); + break; + default: + ShouldNotReachHere(); + break; + } + } } diff --git a/src/hotspot/share/prims/foreign_globals.hpp b/src/hotspot/share/prims/foreign_globals.hpp index 604cf0b3202..1cb90252ba2 100644 --- a/src/hotspot/share/prims/foreign_globals.hpp +++ b/src/hotspot/share/prims/foreign_globals.hpp @@ -24,12 +24,23 @@ #ifndef SHARE_PRIMS_FOREIGN_GLOBALS #define SHARE_PRIMS_FOREIGN_GLOBALS +#include "code/vmreg.hpp" #include "oops/oopsHierarchy.hpp" #include "utilities/growableArray.hpp" #include "utilities/macros.hpp" #include CPU_HEADER(foreign_globals) +struct CallRegs { + VMReg* _arg_regs; + int _args_length; + + VMReg* _ret_regs; + int _rets_length; + + void calling_convention(BasicType* sig_bt, VMRegPair *parm_regs, uint argcnt) const; +}; + class ForeignGlobals { private: struct { @@ -42,6 +53,7 @@ private: struct { int index_offset; + int type_offset; } VMS; struct { @@ -53,6 +65,11 @@ private: int output_type_offsets_offset; } BL; + struct { + int arg_regs_offset; + int ret_regs_offset; + } CallConvOffsets; + ForeignGlobals(); static const ForeignGlobals& instance(); @@ -65,9 +82,11 @@ private: const ABIDescriptor parse_abi_descriptor_impl(jobject jabi) const; const BufferLayout parse_buffer_layout_impl(jobject jlayout) const; + const CallRegs parse_call_regs_impl(jobject jconv) const; public: static const ABIDescriptor parse_abi_descriptor(jobject jabi); static const BufferLayout parse_buffer_layout(jobject jlayout); + static const CallRegs parse_call_regs(jobject jconv); }; #endif // SHARE_PRIMS_FOREIGN_GLOBALS diff --git a/src/hotspot/share/prims/nativeEntryPoint.cpp b/src/hotspot/share/prims/nativeEntryPoint.cpp index 2ec76da7290..14187727201 100644 --- a/src/hotspot/share/prims/nativeEntryPoint.cpp +++ b/src/hotspot/share/prims/nativeEntryPoint.cpp @@ -37,7 +37,8 @@ static JNINativeMethod NEP_methods[] = { {CC "vmStorageToVMReg", CC "(II)J", FN_PTR(NEP_vmStorageToVMReg)}, }; -JNI_LEAF(void, JVM_RegisterNativeEntryPointMethods(JNIEnv *env, jclass NEP_class)) +JNI_ENTRY(void, JVM_RegisterNativeEntryPointMethods(JNIEnv *env, jclass NEP_class)) + ThreadToNativeFromVM ttnfv(thread); int status = env->RegisterNatives(NEP_class, NEP_methods, sizeof(NEP_methods)/sizeof(JNINativeMethod)); guarantee(status == JNI_OK && !env->ExceptionOccurred(), "register jdk.internal.invoke.NativeEntryPoint natives"); diff --git a/src/hotspot/share/prims/universalNativeInvoker.cpp b/src/hotspot/share/prims/universalNativeInvoker.cpp index 0cb61bf3a40..8c91b471095 100644 --- a/src/hotspot/share/prims/universalNativeInvoker.cpp +++ b/src/hotspot/share/prims/universalNativeInvoker.cpp @@ -59,7 +59,8 @@ static JNINativeMethod PI_methods[] = { {CC "generateAdapter", CC "(" FOREIGN_ABI "/ABIDescriptor;" FOREIGN_ABI "/BufferLayout;" ")J", FN_PTR(PI_generateAdapter)} }; -JNI_LEAF(void, JVM_RegisterProgrammableInvokerMethods(JNIEnv *env, jclass PI_class)) +JNI_ENTRY(void, JVM_RegisterProgrammableInvokerMethods(JNIEnv *env, jclass PI_class)) + ThreadToNativeFromVM ttnfv(thread); int status = env->RegisterNatives(PI_class, PI_methods, sizeof(PI_methods)/sizeof(JNINativeMethod)); guarantee(status == JNI_OK && !env->ExceptionOccurred(), "register jdk.internal.foreign.abi.programmable.ProgrammableInvoker natives"); diff --git a/src/hotspot/share/prims/universalUpcallHandler.cpp b/src/hotspot/share/prims/universalUpcallHandler.cpp index ca9e674290b..852c4c141d1 100644 --- a/src/hotspot/share/prims/universalUpcallHandler.cpp +++ b/src/hotspot/share/prims/universalUpcallHandler.cpp @@ -22,8 +22,10 @@ */ #include "precompiled.hpp" +#include "classfile/javaClasses.hpp" #include "classfile/symbolTable.hpp" #include "classfile/systemDictionary.hpp" +#include "compiler/compilationPolicy.hpp" #include "memory/resourceArea.hpp" #include "prims/universalUpcallHandler.hpp" #include "runtime/interfaceSupport.inline.hpp" @@ -49,17 +51,29 @@ void ProgrammableUpcallHandler::upcall_helper(JavaThread* thread, jobject rec, a JavaCalls::call_static(&result, upcall_method.klass, upcall_method.name, upcall_method.sig, &args, CATCH); } -void ProgrammableUpcallHandler::attach_thread_and_do_upcall(jobject rec, address buff) { +Thread* ProgrammableUpcallHandler::maybe_attach_and_get_thread(bool* should_detach) { Thread* thread = Thread::current_or_null(); - bool should_detach = false; if (thread == nullptr) { JavaVM_ *vm = (JavaVM *)(&main_vm); JNIEnv* p_env = nullptr; // unused jint result = vm->functions->AttachCurrentThread(vm, (void**) &p_env, nullptr); guarantee(result == JNI_OK, "Could not attach thread for upcall. JNI error code: %d", result); - should_detach = true; + *should_detach = true; thread = Thread::current(); + } else { + *should_detach = false; } + return thread; +} + +void ProgrammableUpcallHandler::detach_thread(Thread* thread) { + JavaVM_ *vm = (JavaVM *)(&main_vm); + vm->functions->DetachCurrentThread(vm); +} + +void ProgrammableUpcallHandler::attach_thread_and_do_upcall(jobject rec, address buff) { + bool should_detach = false; + Thread* thread = maybe_attach_and_get_thread(&should_detach); { MACOS_AARCH64_ONLY(ThreadWXEnable wx(WXWrite, thread)); @@ -67,8 +81,7 @@ void ProgrammableUpcallHandler::attach_thread_and_do_upcall(jobject rec, address } if (should_detach) { - JavaVM_ *vm = (JavaVM *)(&main_vm); - vm->functions->DetachCurrentThread(vm); + detach_thread(thread); } } @@ -86,30 +99,59 @@ ProgrammableUpcallHandler::ProgrammableUpcallHandler() { upcall_method.klass = k; upcall_method.name = SymbolTable::new_symbol("invoke"); - upcall_method.sig = SymbolTable::new_symbol("(L" FOREIGN_ABI "ProgrammableUpcallHandler;J)V"); + upcall_method.sig = SymbolTable::new_symbol("(Ljava/lang/invoke/MethodHandle;J)V"); assert(upcall_method.klass->lookup_method(upcall_method.name, upcall_method.sig) != nullptr, "Could not find upcall method: %s.%s%s", upcall_method.klass->external_name(), upcall_method.name->as_C_string(), upcall_method.sig->as_C_string()); } -JNI_ENTRY(jlong, PUH_AllocateUpcallStub(JNIEnv *env, jobject rec, jobject abi, jobject buffer_layout)) +void ProgrammableUpcallHandler::handle_uncaught_exception(oop exception) { + // Based on CATCH macro + tty->print_cr("Uncaught exception:"); + exception->print(); + ShouldNotReachHere(); +} + +JVM_ENTRY(jlong, PUH_AllocateUpcallStub(JNIEnv *env, jclass unused, jobject rec, jobject abi, jobject buffer_layout)) Handle receiver(THREAD, JNIHandles::resolve(rec)); jobject global_rec = JNIHandles::make_global(receiver); return (jlong) ProgrammableUpcallHandler::generate_upcall_stub(global_rec, abi, buffer_layout); JNI_END +JVM_ENTRY(jlong, PUH_AllocateOptimizedUpcallStub(JNIEnv *env, jclass unused, jobject mh, jobject abi, jobject conv)) + Handle mh_h(THREAD, JNIHandles::resolve(mh)); + jobject mh_j = JNIHandles::make_global(mh_h); + + oop lform = java_lang_invoke_MethodHandle::form(mh_h()); + oop vmentry = java_lang_invoke_LambdaForm::vmentry(lform); + Method* entry = java_lang_invoke_MemberName::vmtarget(vmentry); + const methodHandle mh_entry(THREAD, entry); + + assert(entry->method_holder()->is_initialized(), "no clinit barrier"); + CompilationPolicy::compile_if_required(mh_entry, CHECK_0); + + return (jlong) ProgrammableUpcallHandler::generate_optimized_upcall_stub(mh_j, entry, abi, conv); +JVM_END + +JVM_ENTRY(jboolean, PUH_SupportsOptimizedUpcalls(JNIEnv *env, jclass unused)) + return (jboolean) ProgrammableUpcallHandler::supports_optimized_upcalls(); +JVM_END + #define CC (char*) /*cast a literal from (const char*)*/ #define FN_PTR(f) CAST_FROM_FN_PTR(void*, &f) static JNINativeMethod PUH_methods[] = { - {CC "allocateUpcallStub", CC "(L" FOREIGN_ABI "ABIDescriptor;L" FOREIGN_ABI "BufferLayout;" ")J", FN_PTR(PUH_AllocateUpcallStub)}, + {CC "allocateUpcallStub", CC "(" "Ljava/lang/invoke/MethodHandle;" "L" FOREIGN_ABI "ABIDescriptor;" "L" FOREIGN_ABI "BufferLayout;" ")J", FN_PTR(PUH_AllocateUpcallStub)}, + {CC "allocateOptimizedUpcallStub", CC "(" "Ljava/lang/invoke/MethodHandle;" "L" FOREIGN_ABI "ABIDescriptor;" "L" FOREIGN_ABI "ProgrammableUpcallHandler$CallRegs;" ")J", FN_PTR(PUH_AllocateOptimizedUpcallStub)}, + {CC "supportsOptimizedUpcalls", CC "()Z", FN_PTR(PUH_SupportsOptimizedUpcalls)}, }; /** * This one function is exported, used by NativeLookup. */ -JNI_LEAF(void, JVM_RegisterProgrammableUpcallHandlerMethods(JNIEnv *env, jclass PUH_class)) +JNI_ENTRY(void, JVM_RegisterProgrammableUpcallHandlerMethods(JNIEnv *env, jclass PUH_class)) + ThreadToNativeFromVM ttnfv(thread); int status = env->RegisterNatives(PUH_class, PUH_methods, sizeof(PUH_methods)/sizeof(JNINativeMethod)); guarantee(status == JNI_OK && !env->ExceptionOccurred(), "register jdk.internal.foreign.abi.ProgrammableUpcallHandler natives"); diff --git a/src/hotspot/share/prims/universalUpcallHandler.hpp b/src/hotspot/share/prims/universalUpcallHandler.hpp index 6af2e405d63..feb39fce496 100644 --- a/src/hotspot/share/prims/universalUpcallHandler.hpp +++ b/src/hotspot/share/prims/universalUpcallHandler.hpp @@ -45,8 +45,14 @@ private: static void upcall_helper(JavaThread* thread, jobject rec, address buff); static void attach_thread_and_do_upcall(jobject rec, address buff); + + static void handle_uncaught_exception(oop exception); + static Thread* maybe_attach_and_get_thread(bool* should_detach); + static void detach_thread(Thread* thread); public: + static address generate_optimized_upcall_stub(jobject mh, Method* entry, jobject jabi, jobject jconv); static address generate_upcall_stub(jobject rec, jobject abi, jobject buffer_layout); + static bool supports_optimized_upcalls(); }; #endif // SHARE_VM_PRIMS_UNIVERSALUPCALLHANDLER_HPP diff --git a/src/hotspot/share/prims/upcallStubs.cpp b/src/hotspot/share/prims/upcallStubs.cpp index 8ed046f473d..c4f699449d1 100644 --- a/src/hotspot/share/prims/upcallStubs.cpp +++ b/src/hotspot/share/prims/upcallStubs.cpp @@ -36,8 +36,14 @@ JVM_ENTRY(static jboolean, UH_FreeUpcallStub0(JNIEnv *env, jobject _unused, jlon return false; } //free global JNI handle - jobject* rec_ptr = (jobject*)(void*)cb -> content_begin(); - JNIHandles::destroy_global(*rec_ptr); + jobject handle = NULL; + if (cb->is_optimized_entry_blob()) { + handle = ((OptimizedEntryBlob*)cb)->receiver(); + } else { + jobject* handle_ptr = (jobject*)(void*)cb->content_begin(); + handle = *handle_ptr; + } + JNIHandles::destroy_global(handle); //free code blob CodeCache::free(cb); return true; diff --git a/src/hotspot/share/prims/whitebox.cpp b/src/hotspot/share/prims/whitebox.cpp index 061406466e8..a752006a271 100644 --- a/src/hotspot/share/prims/whitebox.cpp +++ b/src/hotspot/share/prims/whitebox.cpp @@ -2300,13 +2300,13 @@ WB_ENTRY(void, WB_CheckThreadObjOfTerminatingThread(JNIEnv* env, jobject wb, job } WB_END -WB_ENTRY(void, WB_VerifyFrames(JNIEnv* env, jobject wb, jboolean log)) +WB_ENTRY(void, WB_VerifyFrames(JNIEnv* env, jobject wb, jboolean log, jboolean update_map)) intx tty_token = -1; if (log) { tty_token = ttyLocker::hold_tty(); tty->print_cr("[WhiteBox::VerifyFrames] Walking Frames"); } - for (StackFrameStream fst(JavaThread::current(), true, true); !fst.is_done(); fst.next()) { + for (StackFrameStream fst(JavaThread::current(), update_map, true); !fst.is_done(); fst.next()) { frame* current_frame = fst.current(); if (log) { current_frame->print_value(); @@ -2566,7 +2566,7 @@ static JNINativeMethod methods[] = { {CC"handshakeWalkStack", CC"(Ljava/lang/Thread;Z)I", (void*)&WB_HandshakeWalkStack }, {CC"asyncHandshakeWalkStack", CC"(Ljava/lang/Thread;)V", (void*)&WB_AsyncHandshakeWalkStack }, {CC"checkThreadObjOfTerminatingThread", CC"(Ljava/lang/Thread;)V", (void*)&WB_CheckThreadObjOfTerminatingThread }, - {CC"verifyFrames", CC"(Z)V", (void*)&WB_VerifyFrames }, + {CC"verifyFrames", CC"(ZZ)V", (void*)&WB_VerifyFrames }, {CC"addCompilerDirective", CC"(Ljava/lang/String;)I", (void*)&WB_AddCompilerDirective }, {CC"removeCompilerDirective", CC"(I)V", (void*)&WB_RemoveCompilerDirective }, diff --git a/src/hotspot/share/runtime/arguments.cpp b/src/hotspot/share/runtime/arguments.cpp index 24cb6d66c05..b68da70a8ff 100644 --- a/src/hotspot/share/runtime/arguments.cpp +++ b/src/hotspot/share/runtime/arguments.cpp @@ -309,6 +309,8 @@ bool needs_module_property_warning = false; #define PATH_LEN 4 #define UPGRADE_PATH "upgrade.path" #define UPGRADE_PATH_LEN 12 +#define ENABLE_NATIVE_ACCESS "enable.native.access" +#define ENABLE_NATIVE_ACCESS_LEN 20 void Arguments::add_init_library(const char* name, char* options) { _libraryList.add(new AgentLibrary(name, options, false, NULL)); @@ -347,7 +349,8 @@ bool Arguments::is_internal_module_property(const char* property) { matches_property_suffix(property_suffix, ADDMODS, ADDMODS_LEN) || matches_property_suffix(property_suffix, LIMITMODS, LIMITMODS_LEN) || matches_property_suffix(property_suffix, PATH, PATH_LEN) || - matches_property_suffix(property_suffix, UPGRADE_PATH, UPGRADE_PATH_LEN)) { + matches_property_suffix(property_suffix, UPGRADE_PATH, UPGRADE_PATH_LEN) || + matches_property_suffix(property_suffix, ENABLE_NATIVE_ACCESS, ENABLE_NATIVE_ACCESS_LEN)) { return true; } } @@ -1964,6 +1967,7 @@ unsigned int addexports_count = 0; unsigned int addopens_count = 0; unsigned int addmods_count = 0; unsigned int patch_mod_count = 0; +unsigned int enable_native_access_count = 0; // Check the consistency of vm_init_args bool Arguments::check_vm_args_consistency() { @@ -2406,6 +2410,10 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args, bool* patch_m if (!create_numbered_module_property("jdk.module.addmods", tail, addmods_count++)) { return JNI_ENOMEM; } + } else if (match_option(option, "--enable-native-access=", &tail)) { + if (!create_numbered_module_property("jdk.module.enable.native.access", tail, enable_native_access_count++)) { + return JNI_ENOMEM; + } } else if (match_option(option, "--limit-modules=", &tail)) { if (!create_module_property("jdk.module.limitmods", tail, InternalProperty)) { return JNI_ENOMEM; diff --git a/src/hotspot/share/runtime/frame.cpp b/src/hotspot/share/runtime/frame.cpp index 60d751df13a..89dbc98b221 100644 --- a/src/hotspot/share/runtime/frame.cpp +++ b/src/hotspot/share/runtime/frame.cpp @@ -158,7 +158,7 @@ void frame::set_pc(address newpc ) { } #endif // ASSERT - // Unsafe to use the is_deoptimzed tester after changing pc + // Unsafe to use the is_deoptimized tester after changing pc _deopt_state = unknown; _pc = newpc; _cb = CodeCache::find_blob_unsafe(_pc); @@ -1067,6 +1067,10 @@ void frame::oops_do_internal(OopClosure* f, CodeBlobClosure* cf, const RegisterM oops_interpreted_do(f, map, use_interpreter_oop_map_cache); } else if (is_entry_frame()) { oops_entry_do(f, map); + } else if (is_optimized_entry_frame()) { + // Nothing to do + // receiver is a global ref + // handle block is for JNI } else if (CodeCache::contains(pc())) { oops_code_blob_do(f, cf, map, derived_mode); } else { @@ -1105,7 +1109,9 @@ void frame::verify(const RegisterMap* map) const { #if COMPILER2_OR_JVMCI assert(DerivedPointerTable::is_empty(), "must be empty before verify"); #endif - oops_do_internal(&VerifyOopClosure::verify_oop, NULL, map, false, DerivedPointerIterationMode::_ignore); + if (map->update_map()) { // The map has to be up-to-date for the current frame + oops_do_internal(&VerifyOopClosure::verify_oop, NULL, map, false, DerivedPointerIterationMode::_ignore); + } } diff --git a/src/hotspot/share/runtime/frame.hpp b/src/hotspot/share/runtime/frame.hpp index 3d03802d2ea..bd02f01b200 100644 --- a/src/hotspot/share/runtime/frame.hpp +++ b/src/hotspot/share/runtime/frame.hpp @@ -138,6 +138,7 @@ class frame { bool is_compiled_frame() const; bool is_safepoint_blob_frame() const; bool is_deoptimized_frame() const; + bool is_optimized_entry_frame() const; // testers bool is_first_frame() const; // oldest frame? (has no sender) @@ -172,6 +173,7 @@ class frame { frame sender_for_entry_frame(RegisterMap* map) const; frame sender_for_interpreter_frame(RegisterMap* map) const; frame sender_for_native_frame(RegisterMap* map) const; + frame sender_for_optimized_entry_frame(RegisterMap* map) const; bool is_entry_frame_valid(JavaThread* thread) const; diff --git a/src/hotspot/share/runtime/frame.inline.hpp b/src/hotspot/share/runtime/frame.inline.hpp index 8c12db82eb8..3784cdc6ba4 100644 --- a/src/hotspot/share/runtime/frame.inline.hpp +++ b/src/hotspot/share/runtime/frame.inline.hpp @@ -53,6 +53,10 @@ inline bool frame::is_first_frame() const { return is_entry_frame() && entry_frame_is_first(); } +inline bool frame::is_optimized_entry_frame() const { + return _cb != NULL && _cb->is_optimized_entry_blob(); +} + inline address frame::oopmapreg_to_location(VMReg reg, const RegisterMap* reg_map) const { if(reg->is_reg()) { // If it is passed in a register, it got spilled in the stub frame. diff --git a/src/hotspot/share/runtime/globals.hpp b/src/hotspot/share/runtime/globals.hpp index 5ffb1e755cc..62ad33fc65a 100644 --- a/src/hotspot/share/runtime/globals.hpp +++ b/src/hotspot/share/runtime/globals.hpp @@ -2080,6 +2080,9 @@ const intx ObjectAlignmentInBytes = 8; false AARCH64_ONLY(DEBUG_ONLY(||true)), \ "Mark all threads after a safepoint, and clear on a modify " \ "fence. Add cleanliness checks.") \ + \ + develop(bool, TraceOptimizedUpcallStubs, false, \ + "Trace optimized upcall stub generation") \ // end of RUNTIME_FLAGS diff --git a/src/hotspot/share/runtime/reflection.cpp b/src/hotspot/share/runtime/reflection.cpp index d864a78a46f..a7e232124b1 100644 --- a/src/hotspot/share/runtime/reflection.cpp +++ b/src/hotspot/share/runtime/reflection.cpp @@ -764,7 +764,6 @@ static Handle new_type(Symbol* signature, Klass* k, TRAPS) { return Handle(THREAD, nt); } - oop Reflection::new_method(const methodHandle& method, bool for_constant_pool_access, TRAPS) { // Allow sun.reflect.ConstantPool to refer to methods as java.lang.reflect.Methods. assert(!method()->is_initializer() || diff --git a/src/hotspot/share/runtime/sharedRuntime.cpp b/src/hotspot/share/runtime/sharedRuntime.cpp index 130ad243508..dfebed9c146 100644 --- a/src/hotspot/share/runtime/sharedRuntime.cpp +++ b/src/hotspot/share/runtime/sharedRuntime.cpp @@ -512,6 +512,9 @@ address SharedRuntime::raw_exception_handler_for_return_address(JavaThread* curr // JavaCallWrapper::~JavaCallWrapper return StubRoutines::catch_exception_entry(); } + if (blob != NULL && blob->is_optimized_entry_blob()) { + return ((OptimizedEntryBlob*)blob)->exception_handler(); + } // Interpreted code if (Interpreter::contains(return_address)) { // The deferred StackWatermarkSet::after_unwind check will be performed in @@ -1439,7 +1442,7 @@ JRT_BLOCK_ENTRY(address, SharedRuntime::handle_wrong_method_ic_miss(JavaThread* frame stub_frame = current->last_frame(); assert(stub_frame.is_runtime_frame(), "sanity check"); frame caller_frame = stub_frame.sender(®_map); - assert(!caller_frame.is_interpreted_frame() && !caller_frame.is_entry_frame(), "unexpected frame"); + assert(!caller_frame.is_interpreted_frame() && !caller_frame.is_entry_frame() && !caller_frame.is_optimized_entry_frame(), "unexpected frame"); #endif /* ASSERT */ methodHandle callee_method; @@ -1471,7 +1474,8 @@ JRT_BLOCK_ENTRY(address, SharedRuntime::handle_wrong_method(JavaThread* current) frame caller_frame = stub_frame.sender(®_map); if (caller_frame.is_interpreted_frame() || - caller_frame.is_entry_frame()) { + caller_frame.is_entry_frame() || + caller_frame.is_optimized_entry_frame()) { Method* callee = current->callee_target(); guarantee(callee != NULL && callee->is_method(), "bad handshake"); current->set_vm_result_2(callee); diff --git a/src/hotspot/share/runtime/sharedRuntime.hpp b/src/hotspot/share/runtime/sharedRuntime.hpp index abd8bcf7ea1..91f0c14f2e9 100644 --- a/src/hotspot/share/runtime/sharedRuntime.hpp +++ b/src/hotspot/share/runtime/sharedRuntime.hpp @@ -462,6 +462,11 @@ class SharedRuntime: AllStatic { static void save_native_result(MacroAssembler *_masm, BasicType ret_type, int frame_slots); static void restore_native_result(MacroAssembler *_masm, BasicType ret_type, int frame_slots); + static void move32_64(MacroAssembler* masm, VMRegPair src, VMRegPair dst); + static void long_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst); + static void float_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst); + static void double_move(MacroAssembler* masm, VMRegPair src, VMRegPair dst); + // Generate a native wrapper for a given method. The method takes arguments // in the Java compiled code convention, marshals them to the native // convention (handlizes oops, etc), transitions to native, makes the call, @@ -513,6 +518,12 @@ class SharedRuntime: AllStatic { const GrowableArray& output_registers); #endif + static void compute_move_order(const BasicType* in_sig_bt, + int total_in_args, const VMRegPair* in_regs, + int total_out_args, VMRegPair* out_regs, + GrowableArray& arg_order, + VMRegPair tmp_vmreg); + #ifndef PRODUCT // Collect and print inline cache miss statistics diff --git a/src/java.base/share/classes/java/lang/Module.java b/src/java.base/share/classes/java/lang/Module.java index a0c034888f6..62b5e671843 100644 --- a/src/java.base/share/classes/java/lang/Module.java +++ b/src/java.base/share/classes/java/lang/Module.java @@ -109,6 +109,8 @@ public final class Module implements AnnotatedElement { // the module descriptor private final ModuleDescriptor descriptor; + // true, if this module allows restricted native access + private volatile boolean enableNativeAccess; /** * Creates a new named Module. The resulting Module will be defined to the @@ -133,6 +135,10 @@ public final class Module implements AnnotatedElement { String loc = Objects.toString(uri, null); Object[] packages = descriptor.packages().toArray(); defineModule0(this, isOpen, vs, loc, packages); + if (loader == null || loader == ClassLoaders.platformClassLoader()) { + // boot/builtin modules are always native + implAddEnableNativeAccess(); + } } @@ -244,6 +250,30 @@ public final class Module implements AnnotatedElement { return null; } + /** + * Update this module to allow access to restricted methods. + */ + Module implAddEnableNativeAccess() { + enableNativeAccess = true; + return this; + } + + /** + * Update all unnamed modules to allow access to restricted methods. + */ + static void implAddEnableNativeAccessAllUnnamed() { + ALL_UNNAMED_MODULE.enableNativeAccess = true; + } + + /** + * Returns true if module m can access restricted methods. + */ + boolean implIsEnableNativeAccess() { + return isNamed() ? + enableNativeAccess : + ALL_UNNAMED_MODULE.enableNativeAccess; + } + // -- // special Module to mean "all unnamed modules" diff --git a/src/java.base/share/classes/java/lang/System.java b/src/java.base/share/classes/java/lang/System.java index 474188e4077..56e85a896d1 100644 --- a/src/java.base/share/classes/java/lang/System.java +++ b/src/java.base/share/classes/java/lang/System.java @@ -62,7 +62,6 @@ import java.util.Set; import java.util.function.Supplier; import java.util.concurrent.ConcurrentHashMap; import java.util.stream.Stream; - import jdk.internal.misc.Unsafe; import jdk.internal.util.StaticProperty; import jdk.internal.module.ModuleBootstrap; @@ -2274,6 +2273,15 @@ public final class System { public boolean isReflectivelyOpened(Module m, String pn, Module other) { return m.isReflectivelyOpened(pn, other); } + public Module addEnableNativeAccess(Module m) { + return m.implAddEnableNativeAccess(); + } + public void addEnableNativeAccessAllUnnamed() { + Module.implAddEnableNativeAccessAllUnnamed(); + } + public boolean isEnableNativeAccess(Module m) { + return m.implIsEnableNativeAccess(); + } public ServicesCatalog getServicesCatalog(ModuleLayer layer) { return layer.getServicesCatalog(); } diff --git a/src/java.base/share/classes/java/lang/invoke/MethodHandleImpl.java b/src/java.base/share/classes/java/lang/invoke/MethodHandleImpl.java index 846a55423d3..be6340dfba8 100644 --- a/src/java.base/share/classes/java/lang/invoke/MethodHandleImpl.java +++ b/src/java.base/share/classes/java/lang/invoke/MethodHandleImpl.java @@ -1467,6 +1467,11 @@ abstract class MethodHandleImpl { return GenerateJLIClassesHelper.generateHolderClasses(traces); } + @Override + public void ensureCustomized(MethodHandle mh) { + mh.customize(); + } + @Override public VarHandle memoryAccessVarHandle(Class carrier, boolean skipAlignmentMaskCheck, long alignmentMask, ByteOrder order) { diff --git a/src/java.base/share/classes/java/lang/invoke/NativeMethodHandle.java b/src/java.base/share/classes/java/lang/invoke/NativeMethodHandle.java index 46c7cd84c34..4c4cd474e19 100644 --- a/src/java.base/share/classes/java/lang/invoke/NativeMethodHandle.java +++ b/src/java.base/share/classes/java/lang/invoke/NativeMethodHandle.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -58,7 +58,7 @@ import static java.lang.invoke.MethodHandleStatics.newInternalError; throw new IllegalArgumentException("Type must only contain primitives: " + type); if (type != fallback.type()) - throw new IllegalArgumentException("Type of fallback must match"); + throw new IllegalArgumentException("Type of fallback must match: " + type + " != " + fallback.type()); LambdaForm lform = preparedLambdaForm(type); return new NativeMethodHandle(type, lform, fallback, nep); diff --git a/src/java.base/share/classes/java/nio/Buffer.java b/src/java.base/share/classes/java/nio/Buffer.java index 613176f7aea..5aa98ba3ca1 100644 --- a/src/java.base/share/classes/java/nio/Buffer.java +++ b/src/java.base/share/classes/java/nio/Buffer.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2021, 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 @@ -30,6 +30,7 @@ import jdk.internal.access.SharedSecrets; import jdk.internal.access.foreign.MemorySegmentProxy; import jdk.internal.access.foreign.UnmapperProxy; import jdk.internal.misc.ScopedMemoryAccess; +import jdk.internal.misc.ScopedMemoryAccess.Scope; import jdk.internal.misc.Unsafe; import jdk.internal.misc.VM.BufferPool; import jdk.internal.vm.annotation.ForceInline; @@ -821,6 +822,18 @@ public abstract class Buffer { return buffer.segment; } + @Override + public Scope.Handle acquireScope(Buffer buffer, boolean async) { + var scope = buffer.scope(); + if (scope == null) { + return null; + } + if (async && scope.ownerThread() != null) { + throw new IllegalStateException("Confined scope not supported"); + } + return scope.acquire(); + } + @Override public void force(FileDescriptor fd, long address, boolean isSync, long offset, long size) { MappedMemoryUtils.force(fd, address, isSync, offset, size); diff --git a/src/java.base/share/classes/jdk/internal/access/JavaLangAccess.java b/src/java.base/share/classes/jdk/internal/access/JavaLangAccess.java index 54b18e430af..8eaf2b60412 100644 --- a/src/java.base/share/classes/jdk/internal/access/JavaLangAccess.java +++ b/src/java.base/share/classes/jdk/internal/access/JavaLangAccess.java @@ -256,6 +256,21 @@ public interface JavaLangAccess { */ boolean isReflectivelyOpened(Module module, String pn, Module other); + /** + * Updates module m to allow access to restricted methods. + */ + Module addEnableNativeAccess(Module m); + + /** + * Updates all unnamed modules to allow access to restricted methods. + */ + void addEnableNativeAccessAllUnnamed(); + + /** + * Returns true if module m can access restricted methods. + */ + boolean isEnableNativeAccess(Module m); + /** * Returns the ServicesCatalog for the given Layer. */ diff --git a/src/java.base/share/classes/jdk/internal/access/JavaLangInvokeAccess.java b/src/java.base/share/classes/jdk/internal/access/JavaLangInvokeAccess.java index 767108cdb8e..02feb046286 100644 --- a/src/java.base/share/classes/jdk/internal/access/JavaLangInvokeAccess.java +++ b/src/java.base/share/classes/jdk/internal/access/JavaLangInvokeAccess.java @@ -132,4 +132,11 @@ public interface JavaLangInvokeAccess { * @return the native method handle */ MethodHandle nativeMethodHandle(NativeEntryPoint nep, MethodHandle fallback); + + /** + * Ensure given method handle is customized + * + * @param mh the method handle + */ + void ensureCustomized(MethodHandle mh); } diff --git a/src/java.base/share/classes/jdk/internal/access/JavaNioAccess.java b/src/java.base/share/classes/jdk/internal/access/JavaNioAccess.java index aedac5960f8..c639e4ced83 100644 --- a/src/java.base/share/classes/jdk/internal/access/JavaNioAccess.java +++ b/src/java.base/share/classes/jdk/internal/access/JavaNioAccess.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2007, 2018, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2007, 2021, 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 @@ -27,6 +27,7 @@ package jdk.internal.access; import jdk.internal.access.foreign.MemorySegmentProxy; import jdk.internal.access.foreign.UnmapperProxy; +import jdk.internal.misc.ScopedMemoryAccess.Scope; import jdk.internal.misc.VM.BufferPool; import java.io.FileDescriptor; @@ -85,6 +86,14 @@ public interface JavaNioAccess { */ MemorySegmentProxy bufferSegment(Buffer buffer); + /** + * Used by I/O operations to make a buffer's resource scope non-closeable + * (for the duration of the I/O operation) by acquiring a new resource + * scope handle. Null is returned if the buffer has no scope, or + * acquiring is not required to guarantee safety. + */ + Scope.Handle acquireScope(Buffer buffer, boolean async); + /** * Used by {@code jdk.internal.foreign.MappedMemorySegmentImpl} and byte buffer var handle views. */ diff --git a/src/java.base/share/classes/jdk/internal/invoke/NativeEntryPoint.java b/src/java.base/share/classes/jdk/internal/invoke/NativeEntryPoint.java index d3947724de7..12e832a582d 100644 --- a/src/java.base/share/classes/jdk/internal/invoke/NativeEntryPoint.java +++ b/src/java.base/share/classes/jdk/internal/invoke/NativeEntryPoint.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -37,8 +37,6 @@ public class NativeEntryPoint { registerNatives(); } - private final long addr; - private final int shadowSpace; // encoded as VMRegImpl* @@ -49,9 +47,8 @@ public class NativeEntryPoint { private final MethodType methodType; // C2 sees erased version (byte -> int), so need this explicitly private final String name; - private NativeEntryPoint(long addr, int shadowSpace, long[] argMoves, long[] returnMoves, + private NativeEntryPoint(int shadowSpace, long[] argMoves, long[] returnMoves, boolean needTransition, MethodType methodType, String name) { - this.addr = addr; this.shadowSpace = shadowSpace; this.argMoves = Objects.requireNonNull(argMoves); this.returnMoves = Objects.requireNonNull(returnMoves); @@ -60,14 +57,15 @@ public class NativeEntryPoint { this.name = name; } - public static NativeEntryPoint make(long addr, String name, ABIDescriptorProxy abi, VMStorageProxy[] argMoves, VMStorageProxy[] returnMoves, + public static NativeEntryPoint make(String name, ABIDescriptorProxy abi, + VMStorageProxy[] argMoves, VMStorageProxy[] returnMoves, boolean needTransition, MethodType methodType) { if (returnMoves.length > 1) { throw new IllegalArgumentException("Multiple register return not supported"); } - return new NativeEntryPoint( - addr, abi.shadowSpaceBytes(), encodeVMStorages(argMoves), encodeVMStorages(returnMoves), needTransition, methodType, name); + return new NativeEntryPoint(abi.shadowSpaceBytes(), encodeVMStorages(argMoves), encodeVMStorages(returnMoves), + needTransition, methodType, name); } private static long[] encodeVMStorages(VMStorageProxy[] moves) { diff --git a/src/java.base/share/classes/jdk/internal/loader/BootLoader.java b/src/java.base/share/classes/jdk/internal/loader/BootLoader.java index a714320bccd..c862706f91e 100644 --- a/src/java.base/share/classes/jdk/internal/loader/BootLoader.java +++ b/src/java.base/share/classes/jdk/internal/loader/BootLoader.java @@ -62,7 +62,9 @@ public class BootLoader { private static final String JAVA_HOME = StaticProperty.javaHome(); static { - UNNAMED_MODULE = SharedSecrets.getJavaLangAccess().defineUnnamedModule(null); + JavaLangAccess jla = SharedSecrets.getJavaLangAccess(); + UNNAMED_MODULE = jla.defineUnnamedModule(null); + jla.addEnableNativeAccess(UNNAMED_MODULE); setBootLoaderUnnamedModule0(UNNAMED_MODULE); } diff --git a/src/java.base/share/classes/jdk/internal/misc/X-ScopedMemoryAccess.java.template b/src/java.base/share/classes/jdk/internal/misc/X-ScopedMemoryAccess.java.template index 7e6823718ae..9ca80a1296e 100644 --- a/src/java.base/share/classes/jdk/internal/misc/X-ScopedMemoryAccess.java.template +++ b/src/java.base/share/classes/jdk/internal/misc/X-ScopedMemoryAccess.java.template @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -71,7 +71,7 @@ import jdk.internal.vm.annotation.ForceInline; */ public class ScopedMemoryAccess { - private static Unsafe UNSAFE = Unsafe.getUnsafe(); + private static final Unsafe UNSAFE = Unsafe.getUnsafe(); private static native void registerNatives(); static { @@ -97,10 +97,21 @@ public class ScopedMemoryAccess { * which embodies the temporal checks associated with a given memory region. */ public interface Scope { + + interface Handle { + Scope scope(); + } + void checkValidState(); Thread ownerThread(); + boolean isImplicit(); + + Handle acquire(); + + void release(Handle handle); + /** * Error thrown when memory access fails because the memory has already been released. * Note: for performance reasons, this exception is never created by client; instead a shared instance diff --git a/src/java.base/share/classes/jdk/internal/module/ModuleBootstrap.java b/src/java.base/share/classes/jdk/internal/module/ModuleBootstrap.java index b6b3dd7f086..37abc9819a9 100644 --- a/src/java.base/share/classes/jdk/internal/module/ModuleBootstrap.java +++ b/src/java.base/share/classes/jdk/internal/module/ModuleBootstrap.java @@ -38,6 +38,7 @@ import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.HashSet; +import java.util.Iterator; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; @@ -264,7 +265,9 @@ public final class ModuleBootstrap { if (baseUri == null) throw new InternalError(JAVA_BASE + " does not have a location"); BootLoader.loadModule(base); - Modules.defineModule(null, base.descriptor(), baseUri); + + Module baseModule = Modules.defineModule(null, base.descriptor(), baseUri); + JLA.addEnableNativeAccess(baseModule); // Step 2a: Scan all modules when --validate-modules specified @@ -455,6 +458,9 @@ public final class ModuleBootstrap { addExtraReads(bootLayer); boolean extraExportsOrOpens = addExtraExportsAndOpens(bootLayer); + // add enable native access + addEnableNativeAccess(bootLayer); + Counters.add("jdk.module.boot.7.adjustModulesTime"); // save module finders for later use @@ -766,6 +772,47 @@ public final class ModuleBootstrap { } } + /** + * Process the --enable-native-access option to grant access to restricted methods to selected modules. + */ + private static void addEnableNativeAccess(ModuleLayer layer) { + for (String name : decodeEnableNativeAccess()) { + if (name.equals("ALL-UNNAMED")) { + JLA.addEnableNativeAccessAllUnnamed(); + } else { + Optional module = layer.findModule(name); + if (module.isPresent()) { + JLA.addEnableNativeAccess(module.get()); + } else { + warnUnknownModule(ENABLE_NATIVE_ACCESS, name); + } + } + } + } + + /** + * Returns the set of module names specified by --enable-native-access options. + */ + private static Set decodeEnableNativeAccess() { + String prefix = "jdk.module.enable.native.access."; + int index = 0; + // the system property is removed after decoding + String value = getAndRemoveProperty(prefix + index); + Set modules = new HashSet<>(); + if (value == null) { + return modules; + } + while (value != null) { + for (String s : value.split(",")) { + if (!s.isEmpty()) + modules.add(s); + } + index++; + value = getAndRemoveProperty(prefix + index); + } + return modules; + } + /** * Decodes the values of --add-reads, -add-exports, --add-opens or * --patch-modules options that are encoded in system properties. @@ -889,7 +936,7 @@ public final class ModuleBootstrap { private static final String ADD_OPENS = "--add-opens"; private static final String ADD_READS = "--add-reads"; private static final String PATCH_MODULE = "--patch-module"; - + private static final String ENABLE_NATIVE_ACCESS = "--enable-native-access"; /* * Returns the command-line option name corresponds to the specified diff --git a/src/java.base/share/classes/jdk/internal/reflect/Reflection.java b/src/java.base/share/classes/jdk/internal/reflect/Reflection.java index f97793aa531..ce9655f7a78 100644 --- a/src/java.base/share/classes/jdk/internal/reflect/Reflection.java +++ b/src/java.base/share/classes/jdk/internal/reflect/Reflection.java @@ -106,6 +106,13 @@ public class Reflection { } } + public static void ensureNativeAccess(Class currentClass) { + Module module = currentClass.getModule(); + if (!SharedSecrets.getJavaLangAccess().isEnableNativeAccess(module)) { + throw new IllegalCallerException("Illegal native access from: " + module); + } + } + /** * Verify access to a member and return {@code true} if it is granted. * diff --git a/src/java.base/share/classes/module-info.java b/src/java.base/share/classes/module-info.java index e09ff7e8de8..a81b787e77a 100644 --- a/src/java.base/share/classes/module-info.java +++ b/src/java.base/share/classes/module-info.java @@ -212,7 +212,8 @@ module java.base { jdk.jartool, jdk.jfr, jdk.jlink, - jdk.jpackage; + jdk.jpackage, + jdk.incubator.foreign; exports jdk.internal.perf to java.management, jdk.management.agent, @@ -229,7 +230,8 @@ module java.base { java.sql.rowset, jdk.dynalink, jdk.internal.vm.ci, - jdk.unsupported; + jdk.unsupported, + jdk.incubator.foreign; exports jdk.internal.vm to jdk.internal.jvmstat, jdk.management.agent; diff --git a/src/java.base/share/classes/sun/launcher/resources/launcher.properties b/src/java.base/share/classes/sun/launcher/resources/launcher.properties index 2ecb6b254cc..efcc4d69969 100644 --- a/src/java.base/share/classes/sun/launcher/resources/launcher.properties +++ b/src/java.base/share/classes/sun/launcher/resources/launcher.properties @@ -60,6 +60,9 @@ java.launcher.opt.footer = \ \ root modules to resolve in addition to the initial module.\n\ \ can also be ALL-DEFAULT, ALL-SYSTEM,\n\ \ ALL-MODULE-PATH.\n\ +\ --enable-native-access [,...]\n\ +\ modules that are permitted to perform restricted native operations.\n\ +\ can also be ALL-UNNAMED.\n\ \ --list-modules\n\ \ list observable modules and exit\n\ \ -d \n\ diff --git a/src/java.base/share/classes/sun/nio/ch/IOUtil.java b/src/java.base/share/classes/sun/nio/ch/IOUtil.java index 02af6c39af7..b9b320a194f 100644 --- a/src/java.base/share/classes/sun/nio/ch/IOUtil.java +++ b/src/java.base/share/classes/sun/nio/ch/IOUtil.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2021, 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 @@ -28,7 +28,10 @@ package sun.nio.ch; import java.io.FileDescriptor; import java.io.IOException; import java.nio.ByteBuffer; - +import java.util.Objects; +import jdk.internal.access.JavaNioAccess; +import jdk.internal.access.SharedSecrets; +import jdk.internal.misc.ScopedMemoryAccess.Scope; /** * File-descriptor based I/O utilities that are shared by NIO classes. @@ -47,15 +50,30 @@ public class IOUtil { NativeDispatcher nd) throws IOException { - return write(fd, src, position, false, -1, nd); + return write(fd, src, position, false, false, -1, nd); + } + + static int write(FileDescriptor fd, ByteBuffer src, long position, + boolean async, NativeDispatcher nd) + throws IOException + { + return write(fd, src, position, false, async, -1, nd); } static int write(FileDescriptor fd, ByteBuffer src, long position, boolean directIO, int alignment, NativeDispatcher nd) throws IOException + { + return write(fd, src, position, directIO, false, alignment, nd); + } + + static int write(FileDescriptor fd, ByteBuffer src, long position, + boolean directIO, boolean async, int alignment, + NativeDispatcher nd) + throws IOException { if (src instanceof DirectBuffer) { - return writeFromNativeBuffer(fd, src, position, directIO, alignment, nd); + return writeFromNativeBuffer(fd, src, position, directIO, async, alignment, nd); } // Substitute a native buffer @@ -76,7 +94,7 @@ public class IOUtil { // Do not update src until we see how many bytes were written src.position(pos); - int n = writeFromNativeBuffer(fd, bb, position, directIO, alignment, nd); + int n = writeFromNativeBuffer(fd, bb, position, directIO, async, alignment, nd); if (n > 0) { // now update src src.position(pos + n); @@ -89,7 +107,8 @@ public class IOUtil { private static int writeFromNativeBuffer(FileDescriptor fd, ByteBuffer bb, long position, boolean directIO, - int alignment, NativeDispatcher nd) + boolean async, int alignment, + NativeDispatcher nd) throws IOException { int pos = bb.position(); @@ -105,46 +124,62 @@ public class IOUtil { int written = 0; if (rem == 0) return 0; - if (position != -1) { - written = nd.pwrite(fd, - ((DirectBuffer)bb).address() + pos, - rem, position); - } else { - written = nd.write(fd, ((DirectBuffer)bb).address() + pos, rem); + var handle = acquireScope(bb, async); + try { + if (position != -1) { + written = nd.pwrite(fd, bufferAddress(bb) + pos, rem, position); + } else { + written = nd.write(fd, bufferAddress(bb) + pos, rem); + } + } finally { + releaseScope(handle); } if (written > 0) bb.position(pos + written); return written; } - static long write(FileDescriptor fd, ByteBuffer[] bufs, NativeDispatcher nd) + static long write(FileDescriptor fd, ByteBuffer[] bufs, boolean async, + NativeDispatcher nd) throws IOException { - return write(fd, bufs, 0, bufs.length, false, -1, nd); + return write(fd, bufs, 0, bufs.length, false, async, -1, nd); } static long write(FileDescriptor fd, ByteBuffer[] bufs, int offset, int length, NativeDispatcher nd) throws IOException { - return write(fd, bufs, offset, length, false, -1, nd); + return write(fd, bufs, offset, length, false, false, -1, nd); } static long write(FileDescriptor fd, ByteBuffer[] bufs, int offset, int length, - boolean directIO, int alignment, NativeDispatcher nd) + boolean direct, int alignment, NativeDispatcher nd) + throws IOException + { + return write(fd, bufs, offset, length, direct, false, alignment, nd); + } + + static long write(FileDescriptor fd, ByteBuffer[] bufs, int offset, int length, + boolean directIO, boolean async, + int alignment, NativeDispatcher nd) throws IOException { IOVecWrapper vec = IOVecWrapper.get(length); boolean completed = false; int iov_len = 0; + Runnable handleReleasers = null; try { - // Iterate over buffers to populate native iovec array. int count = offset + length; int i = offset; while (i < count && iov_len < IOV_MAX) { ByteBuffer buf = bufs[i]; + var h = acquireScope(buf, async); + if (h != null) { + handleReleasers = LinkedRunnable.of(Releaser.of(h), handleReleasers); + } int pos = buf.position(); int lim = buf.limit(); assert (pos <= lim); @@ -170,7 +205,7 @@ public class IOUtil { pos = shadow.position(); } - vec.putBase(iov_len, ((DirectBuffer)buf).address() + pos); + vec.putBase(iov_len, bufferAddress(buf) + pos); vec.putLen(iov_len, rem); iov_len++; } @@ -203,6 +238,7 @@ public class IOUtil { return bytesWritten; } finally { + releaseScopes(handleReleasers); // if an error occurred then clear refs to buffers and return any shadow // buffers to cache if (!completed) { @@ -220,17 +256,32 @@ public class IOUtil { NativeDispatcher nd) throws IOException { - return read(fd, dst, position, false, -1, nd); + return read(fd, dst, position, false, false, -1, nd); + } + + static int read(FileDescriptor fd, ByteBuffer dst, long position, + boolean async, NativeDispatcher nd) + throws IOException + { + return read(fd, dst, position, false, async, -1, nd); } static int read(FileDescriptor fd, ByteBuffer dst, long position, boolean directIO, int alignment, NativeDispatcher nd) throws IOException + { + return read(fd, dst, position, directIO, false, alignment, nd); + } + + static int read(FileDescriptor fd, ByteBuffer dst, long position, + boolean directIO, boolean async, + int alignment, NativeDispatcher nd) + throws IOException { if (dst.isReadOnly()) throw new IllegalArgumentException("Read-only buffer"); if (dst instanceof DirectBuffer) - return readIntoNativeBuffer(fd, dst, position, directIO, alignment, nd); + return readIntoNativeBuffer(fd, dst, position, directIO, async, alignment, nd); // Substitute a native buffer ByteBuffer bb; @@ -242,7 +293,7 @@ public class IOUtil { bb = Util.getTemporaryDirectBuffer(rem); } try { - int n = readIntoNativeBuffer(fd, bb, position, directIO, alignment,nd); + int n = readIntoNativeBuffer(fd, bb, position, directIO, async, alignment, nd); bb.flip(); if (n > 0) dst.put(bb); @@ -254,7 +305,8 @@ public class IOUtil { private static int readIntoNativeBuffer(FileDescriptor fd, ByteBuffer bb, long position, boolean directIO, - int alignment, NativeDispatcher nd) + boolean async, int alignment, + NativeDispatcher nd) throws IOException { int pos = bb.position(); @@ -270,10 +322,15 @@ public class IOUtil { if (rem == 0) return 0; int n = 0; - if (position != -1) { - n = nd.pread(fd, ((DirectBuffer)bb).address() + pos, rem, position); - } else { - n = nd.read(fd, ((DirectBuffer)bb).address() + pos, rem); + var handle = acquireScope(bb, async); + try { + if (position != -1) { + n = nd.pread(fd, bufferAddress(bb) + pos, rem, position); + } else { + n = nd.read(fd, bufferAddress(bb) + pos, rem); + } + } finally { + releaseScope(handle); } if (n > 0) bb.position(pos + n); @@ -283,26 +340,43 @@ public class IOUtil { static long read(FileDescriptor fd, ByteBuffer[] bufs, NativeDispatcher nd) throws IOException { - return read(fd, bufs, 0, bufs.length, false, -1, nd); + return read(fd, bufs, 0, bufs.length, false, false, -1, nd); + } + + static long read(FileDescriptor fd, ByteBuffer[] bufs, boolean async, + NativeDispatcher nd) + throws IOException + { + return read(fd, bufs, 0, bufs.length, false, async, -1, nd); } static long read(FileDescriptor fd, ByteBuffer[] bufs, int offset, int length, NativeDispatcher nd) throws IOException { - return read(fd, bufs, offset, length, false, -1, nd); + return read(fd, bufs, offset, length, false, false, -1, nd); } static long read(FileDescriptor fd, ByteBuffer[] bufs, int offset, int length, boolean directIO, int alignment, NativeDispatcher nd) + + throws IOException + { + return read(fd, bufs, offset, length, directIO, false, alignment, nd); + } + + static long read(FileDescriptor fd, ByteBuffer[] bufs, int offset, int length, + boolean directIO, boolean async, + int alignment, NativeDispatcher nd) + throws IOException { IOVecWrapper vec = IOVecWrapper.get(length); boolean completed = false; int iov_len = 0; + Runnable handleReleasers = null; try { - // Iterate over buffers to populate native iovec array. int count = offset + length; int i = offset; @@ -310,6 +384,10 @@ public class IOUtil { ByteBuffer buf = bufs[i]; if (buf.isReadOnly()) throw new IllegalArgumentException("Read-only buffer"); + var h = acquireScope(buf, async); + if (h != null) { + handleReleasers = LinkedRunnable.of(Releaser.of(h), handleReleasers); + } int pos = buf.position(); int lim = buf.limit(); assert (pos <= lim); @@ -334,7 +412,7 @@ public class IOUtil { pos = shadow.position(); } - vec.putBase(iov_len, ((DirectBuffer)buf).address() + pos); + vec.putBase(iov_len, bufferAddress(buf) + pos); vec.putLen(iov_len, rem); iov_len++; } @@ -371,6 +449,7 @@ public class IOUtil { return bytesRead; } finally { + releaseScopes(handleReleasers); // if an error occurred then clear refs to buffers and return any shadow // buffers to cache if (!completed) { @@ -384,6 +463,83 @@ public class IOUtil { } } + private static final JavaNioAccess NIO_ACCESS = SharedSecrets.getJavaNioAccess(); + + static Scope.Handle acquireScope(ByteBuffer bb, boolean async) { + return NIO_ACCESS.acquireScope(bb, async); + } + + private static void releaseScope(Scope.Handle handle) { + if (handle == null) + return; + try { + handle.scope().release(handle); + } catch (Exception e) { + throw new IllegalStateException(e); + } + } + + static Runnable acquireScopes(ByteBuffer[] buffers) { + return acquireScopes(null, buffers); + } + + static Runnable acquireScopes(ByteBuffer buf, ByteBuffer[] buffers) { + if (buffers == null) { + assert buf != null; + return IOUtil.Releaser.ofNullable(IOUtil.acquireScope(buf, true)); + } else { + assert buf == null; + Runnable handleReleasers = null; + for (var b : buffers) { + var h = IOUtil.acquireScope(b, true); + if (h != null) { + handleReleasers = IOUtil.LinkedRunnable.of(IOUtil.Releaser.of(h), handleReleasers); + } + } + return handleReleasers; + } + } + + static void releaseScopes(Runnable releasers) { + if (releasers != null) + releasers.run(); + } + + static record LinkedRunnable(Runnable node, Runnable next) + implements Runnable + { + LinkedRunnable { + Objects.requireNonNull(node); + } + @Override + public void run() { + try { + node.run(); + } finally { + if (next != null) + next.run(); + } + } + static LinkedRunnable of(Runnable first, Runnable second) { + return new LinkedRunnable(first, second); + } + } + + static record Releaser(Scope.Handle handle) implements Runnable { + Releaser { Objects.requireNonNull(handle) ; } + @Override public void run() { releaseScope(handle); } + static Runnable of(Scope.Handle handle) { return new Releaser(handle); } + static Runnable ofNullable(Scope.Handle handle) { + if (handle == null) + return () -> { }; + return new Releaser(handle); + } + } + + static long bufferAddress(ByteBuffer buf) { + return NIO_ACCESS.getBufferAddress(buf); + } + public static FileDescriptor newFD(int i) { FileDescriptor fd = new FileDescriptor(); setfdVal(fd, i); diff --git a/src/java.base/share/native/libjli/java.c b/src/java.base/share/native/libjli/java.c index f4b702db7bb..e519686cb78 100644 --- a/src/java.base/share/native/libjli/java.c +++ b/src/java.base/share/native/libjli/java.c @@ -599,6 +599,7 @@ IsModuleOption(const char* name) { JLI_StrCmp(name, "-p") == 0 || JLI_StrCmp(name, "--upgrade-module-path") == 0 || JLI_StrCmp(name, "--add-modules") == 0 || + JLI_StrCmp(name, "--enable-native-access") == 0 || JLI_StrCmp(name, "--limit-modules") == 0 || JLI_StrCmp(name, "--add-exports") == 0 || JLI_StrCmp(name, "--add-opens") == 0 || @@ -611,6 +612,7 @@ IsLongFormModuleOption(const char* name) { return JLI_StrCCmp(name, "--module-path=") == 0 || JLI_StrCCmp(name, "--upgrade-module-path=") == 0 || JLI_StrCCmp(name, "--add-modules=") == 0 || + JLI_StrCCmp(name, "--enable-native-access=") == 0 || JLI_StrCCmp(name, "--limit-modules=") == 0 || JLI_StrCCmp(name, "--add-exports=") == 0 || JLI_StrCCmp(name, "--add-reads=") == 0 || diff --git a/src/java.base/unix/classes/sun/nio/ch/UnixAsynchronousSocketChannelImpl.java b/src/java.base/unix/classes/sun/nio/ch/UnixAsynchronousSocketChannelImpl.java index fce9e1791fe..8cb5b92de07 100644 --- a/src/java.base/unix/classes/sun/nio/ch/UnixAsynchronousSocketChannelImpl.java +++ b/src/java.base/unix/classes/sun/nio/ch/UnixAsynchronousSocketChannelImpl.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2008, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2008, 2021, 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 @@ -76,6 +76,7 @@ class UnixAsynchronousSocketChannelImpl private boolean isScatteringRead; private ByteBuffer readBuffer; private ByteBuffer[] readBuffers; + private Runnable readScopeHandleReleasers; private CompletionHandler readHandler; private Object readAttachment; private PendingFuture readFuture; @@ -86,6 +87,7 @@ class UnixAsynchronousSocketChannelImpl private boolean isGatheringWrite; private ByteBuffer writeBuffer; private ByteBuffer[] writeBuffers; + private Runnable writeScopeHandleReleasers; private CompletionHandler writeHandler; private Object writeAttachment; private PendingFuture writeFuture; @@ -392,9 +394,9 @@ class UnixAsynchronousSocketChannelImpl begin(); if (scattering) { - n = (int)IOUtil.read(fd, readBuffers, nd); + n = (int)IOUtil.read(fd, readBuffers, true, nd); } else { - n = IOUtil.read(fd, readBuffer, -1, nd); + n = IOUtil.read(fd, readBuffer, -1, true, nd); } if (n == IOStatus.UNAVAILABLE) { // spurious wakeup, is this possible? @@ -409,6 +411,7 @@ class UnixAsynchronousSocketChannelImpl this.readBuffers = null; this.readAttachment = null; this.readHandler = null; + IOUtil.releaseScopes(readScopeHandleReleasers); // allow another read to be initiated enableReading(); @@ -516,9 +519,9 @@ class UnixAsynchronousSocketChannelImpl if (attemptRead) { if (isScatteringRead) { - n = (int)IOUtil.read(fd, dsts, nd); + n = (int)IOUtil.read(fd, dsts, true, nd); } else { - n = IOUtil.read(fd, dst, -1, nd); + n = IOUtil.read(fd, dst, -1, true, nd); } } @@ -526,6 +529,7 @@ class UnixAsynchronousSocketChannelImpl PendingFuture result = null; synchronized (updateLock) { this.isScatteringRead = isScatteringRead; + this.readScopeHandleReleasers = IOUtil.acquireScopes(dst, dsts); this.readBuffer = dst; this.readBuffers = dsts; if (handler == null) { @@ -592,9 +596,9 @@ class UnixAsynchronousSocketChannelImpl begin(); if (gathering) { - n = (int)IOUtil.write(fd, writeBuffers, nd); + n = (int)IOUtil.write(fd, writeBuffers, true, nd); } else { - n = IOUtil.write(fd, writeBuffer, -1, nd); + n = IOUtil.write(fd, writeBuffer, -1, true, nd); } if (n == IOStatus.UNAVAILABLE) { // spurious wakeup, is this possible? @@ -609,6 +613,7 @@ class UnixAsynchronousSocketChannelImpl this.writeBuffers = null; this.writeAttachment = null; this.writeHandler = null; + IOUtil.releaseScopes(writeScopeHandleReleasers); // allow another write to be initiated enableWriting(); @@ -702,9 +707,9 @@ class UnixAsynchronousSocketChannelImpl if (attemptWrite) { if (isGatheringWrite) { - n = (int)IOUtil.write(fd, srcs, nd); + n = (int)IOUtil.write(fd, srcs, true, nd); } else { - n = IOUtil.write(fd, src, -1, nd); + n = IOUtil.write(fd, src, -1, true, nd); } } @@ -712,6 +717,7 @@ class UnixAsynchronousSocketChannelImpl PendingFuture result = null; synchronized (updateLock) { this.isGatheringWrite = isGatheringWrite; + this.writeScopeHandleReleasers = IOUtil.acquireScopes(src, srcs); this.writeBuffer = src; this.writeBuffers = srcs; if (handler == null) { diff --git a/src/java.base/windows/classes/sun/nio/ch/WindowsAsynchronousSocketChannelImpl.java b/src/java.base/windows/classes/sun/nio/ch/WindowsAsynchronousSocketChannelImpl.java index 1fc793a4e66..364afd8b869 100644 --- a/src/java.base/windows/classes/sun/nio/ch/WindowsAsynchronousSocketChannelImpl.java +++ b/src/java.base/windows/classes/sun/nio/ch/WindowsAsynchronousSocketChannelImpl.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2008, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2008, 2021, 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 @@ -400,6 +400,7 @@ class WindowsAsynchronousSocketChannelImpl // set by run method private ByteBuffer[] shadow; + private Runnable scopeHandleReleasers; ReadTask(ByteBuffer[] bufs, boolean scatteringRead, @@ -416,6 +417,7 @@ class WindowsAsynchronousSocketChannelImpl * it substitutes non-direct buffers with direct buffers. */ void prepareBuffers() { + scopeHandleReleasers = IOUtil.acquireScopes(bufs); shadow = new ByteBuffer[numBufs]; long address = readBufferArray; for (int i=0; iaddressable. An addressable type is one which can be projected down to * a memory address instance (see {@link #address()}). Examples of addressable types are {@link MemorySegment}, - * {@link MemoryAddress}, {@link LibraryLookup.Symbol} and {@link CLinker.VaList}. - * - * @apiNote In the future, if the Java language permits, {@link Addressable} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types, such as {@link MemorySegment}, {@link MemoryAddress}, {@link LibraryLookup.Symbol} - * and {@link CLinker.VaList}. + * {@link MemoryAddress} and {@link CLinker.VaList}. * * @implSpec * Implementations of this interface are value-based. diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/CLinker.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/CLinker.java index 786c2d8d9c7..2e19005fcbd 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/CLinker.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/CLinker.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -25,10 +25,15 @@ */ package jdk.incubator.foreign; +import jdk.internal.foreign.AbstractCLinker; import jdk.internal.foreign.NativeMemorySegmentImpl; import jdk.internal.foreign.PlatformLayouts; -import jdk.internal.foreign.Utils; import jdk.internal.foreign.abi.SharedUtils; +import jdk.internal.foreign.abi.aarch64.AArch64VaList; +import jdk.internal.foreign.abi.x64.sysv.SysVVaList; +import jdk.internal.foreign.abi.x64.windows.WinVaList; +import jdk.internal.reflect.CallerSensitive; +import jdk.internal.reflect.Reflection; import java.lang.constant.Constable; import java.lang.invoke.MethodHandle; @@ -36,7 +41,6 @@ import java.lang.invoke.MethodType; import java.nio.charset.Charset; import java.util.Objects; import java.util.function.Consumer; -import java.util.stream.Stream; import static jdk.internal.foreign.PlatformLayouts.*; @@ -65,7 +69,8 @@ import static jdk.internal.foreign.PlatformLayouts.*; * carrier type can be used to match the native {@code va_list} type. *

* For the linking process to be successful, some requirements must be satisfied; if {@code M} and {@code F} are - * the method type and the function descriptor, respectively, used during the linking process, then it must be that: + * the method type (obtained after dropping any prefix arguments) and the function descriptor, respectively, + * used during the linking process, then it must be that: *

    *
  • The arity of {@code M} is the same as that of {@code F};
  • *
  • If the return type of {@code M} is {@code void}, then {@code F} should have no return layout @@ -100,33 +105,37 @@ import static jdk.internal.foreign.PlatformLayouts.*; *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} * elements to a method in this class causes a {@link NullPointerException NullPointerException} to be thrown.

    * - * @apiNote In the future, if the Java language permits, {@link CLinker} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types. - * * @implSpec * Implementations of this interface are immutable, thread-safe and value-based. */ -public interface CLinker { +public sealed interface CLinker permits AbstractCLinker { /** * Returns the C linker for the current platform. *

    - * This method is restricted. Restricted method are unsafe, and, if used incorrectly, their use might crash + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. + * * @return a linker for this system. - * @throws IllegalAccessError if the runtime property {@code foreign.restricted} is not set to either - * {@code permit}, {@code warn} or {@code debug} (the default value is set to {@code deny}). + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ + @CallerSensitive static CLinker getInstance() { - Utils.checkRestrictedAccess("CLinker.getInstance"); + Reflection.ensureNativeAccess(Reflection.getCallerClass()); return SharedUtils.getSystemLinker(); } /** - * Obtain a foreign method handle, with given type, which can be used to call a - * target foreign function at a given address and featuring a given function descriptor. + * Obtains a foreign method handle, with the given type and featuring the given function descriptor, + * which can be used to call a target foreign function at the given address. + *

    + * If the provided method type's return type is {@code MemorySegment}, then the resulting method handle features + * an additional prefix parameter, of type {@link SegmentAllocator}, which will be used by the linker runtime + * to allocate structs returned by-value. * * @see LibraryLookup#lookup(String) * @@ -139,20 +148,58 @@ public interface CLinker { MethodHandle downcallHandle(Addressable symbol, MethodType type, FunctionDescriptor function); /** - * Allocates a native segment whose base address (see {@link MemorySegment#address}) can be - * passed to other foreign functions (as a function pointer); calling such a function pointer - * from native code will result in the execution of the provided method handle. + * Obtain a foreign method handle, with the given type and featuring the given function descriptor, + * which can be used to call a target foreign function at the given address. + *

    + * If the provided method type's return type is {@code MemorySegment}, then the provided allocator will be used by + * the linker runtime to allocate structs returned by-value. * - *

    The returned segment is shared, and it only features - * the {@link MemorySegment#CLOSE} access mode. When the returned segment is closed, - * the corresponding native stub will be deallocated.

    + * @see LibraryLookup#lookup(String) + * + * @param symbol downcall symbol. + * @param allocator the segment allocator. + * @param type the method type. + * @param function the function descriptor. + * @return the downcall method handle. + * @throws IllegalArgumentException in the case of a method type and function descriptor mismatch. + */ + MethodHandle downcallHandle(Addressable symbol, SegmentAllocator allocator, MethodType type, FunctionDescriptor function); + + /** + * Obtains a foreign method handle, with the given type and featuring the given function descriptor, which can be + * used to call a target foreign function at an address. + * The resulting method handle features a prefix parameter (as the first parameter) corresponding to the address, of + * type {@link Addressable}. + *

    + * If the provided method type's return type is {@code MemorySegment}, then the resulting method handle features an + * additional prefix parameter (inserted immediately after the address parameter), of type {@link SegmentAllocator}), + * which will be used by the linker runtime to allocate structs returned by-value. + * + * @see LibraryLookup#lookup(String) + * + * @param type the method type. + * @param function the function descriptor. + * @return the downcall method handle. + * @throws IllegalArgumentException in the case of a method type and function descriptor mismatch. + */ + MethodHandle downcallHandle(MethodType type, FunctionDescriptor function); + + /** + * Allocates a native stub with given scope which can be passed to other foreign functions (as a function pointer); + * calling such a function pointer from native code will result in the execution of the provided method handle. + * + *

    The returned memory address is associated with the provided scope. When such scope is closed, + * the corresponding native stub will be deallocated. * * @param target the target method handle. * @param function the function descriptor. + * @param scope the upcall stub scope. * @return the native stub segment. * @throws IllegalArgumentException if the target's method type and the function descriptor mismatch. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. */ - MemorySegment upcallStub(MethodHandle target, FunctionDescriptor function); + MemoryAddress upcallStub(MethodHandle target, FunctionDescriptor function, ResourceScope scope); /** * The layout for the {@code char} C type @@ -205,8 +252,8 @@ public interface CLinker { } /** - * Converts a Java string into a null-terminated C string, using the - * platform's default charset, storing the result into a new native memory segment. + * Converts a Java string into a null-terminated C string, using the platform's default charset, + * storing the result into a native memory segment allocated using the provided allocator. *

    * This method always replaces malformed-input and unmappable-character * sequences with this charset's default replacement byte array. The @@ -214,35 +261,18 @@ public interface CLinker { * control over the encoding process is required. * * @param str the Java string to be converted into a C string. + * @param allocator the allocator to be used for the native segment allocation. * @return a new native memory segment containing the converted C string. */ - static MemorySegment toCString(String str) { + static MemorySegment toCString(String str, SegmentAllocator allocator) { Objects.requireNonNull(str); - return toCString(str.getBytes()); - } - - /** - * Converts a Java string into a null-terminated C string, using the given {@link java.nio.charset.Charset charset}, - * storing the result into a new native memory segment. - *

    - * This method always replaces malformed-input and unmappable-character - * sequences with this charset's default replacement byte array. The - * {@link java.nio.charset.CharsetEncoder} class should be used when more - * control over the encoding process is required. - * - * @param str the Java string to be converted into a C string. - * @param charset The {@link java.nio.charset.Charset} to be used to compute the contents of the C string. - * @return a new native memory segment containing the converted C string. - */ - static MemorySegment toCString(String str, Charset charset) { - Objects.requireNonNull(str); - Objects.requireNonNull(charset); - return toCString(str.getBytes(charset)); + Objects.requireNonNull(allocator); + return toCString(str.getBytes(), allocator); } /** * Converts a Java string into a null-terminated C string, using the platform's default charset, - * storing the result into a native memory segment allocated using the provided scope. + * storing the result into a native memory segment associated with the provided resource scope. *

    * This method always replaces malformed-input and unmappable-character * sequences with this charset's default replacement byte array. The @@ -250,18 +280,18 @@ public interface CLinker { * control over the encoding process is required. * * @param str the Java string to be converted into a C string. - * @param scope the scope to be used for the native segment allocation. + * @param scope the resource scope to be associated with the returned segment. * @return a new native memory segment containing the converted C string. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. */ - static MemorySegment toCString(String str, NativeScope scope) { - Objects.requireNonNull(str); - Objects.requireNonNull(scope); - return toCString(str.getBytes(), scope); + static MemorySegment toCString(String str, ResourceScope scope) { + return toCString(str, SegmentAllocator.ofScope(scope)); } /** - * Converts a Java string into a null-terminated C string, using the given {@link java.nio.charset.Charset charset}, - * storing the result into a new native memory segment native memory segment allocated using the provided scope. + * Converts a Java string into a null-terminated C string, using the given {@linkplain java.nio.charset.Charset charset}, + * storing the result into a new native memory segment native memory segment allocated using the provided allocator. *

    * This method always replaces malformed-input and unmappable-character * sequences with this charset's default replacement byte array. The @@ -270,14 +300,34 @@ public interface CLinker { * * @param str the Java string to be converted into a C string. * @param charset The {@link java.nio.charset.Charset} to be used to compute the contents of the C string. - * @param scope the scope to be used for the native segment allocation. + * @param allocator the allocator to be used for the native segment allocation. * @return a new native memory segment containing the converted C string. */ - static MemorySegment toCString(String str, Charset charset, NativeScope scope) { + static MemorySegment toCString(String str, Charset charset, SegmentAllocator allocator) { Objects.requireNonNull(str); Objects.requireNonNull(charset); - Objects.requireNonNull(scope); - return toCString(str.getBytes(charset), scope); + Objects.requireNonNull(allocator); + return toCString(str.getBytes(charset), allocator); + } + + /** + * Converts a Java string into a null-terminated C string, using the given {@linkplain java.nio.charset.Charset charset}, + * storing the result into a native memory segment associated with the provided resource scope. + *

    + * This method always replaces malformed-input and unmappable-character + * sequences with this charset's default replacement byte array. The + * {@link java.nio.charset.CharsetEncoder} class should be used when more + * control over the encoding process is required. + * + * @param str the Java string to be converted into a C string. + * @param charset The {@link java.nio.charset.Charset} to be used to compute the contents of the C string. + * @param scope the resource scope to be associated with the returned segment. + * @return a new native memory segment containing the converted C string. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. + */ + static MemorySegment toCString(String str, Charset charset, ResourceScope scope) { + return toCString(str, charset, SegmentAllocator.ofScope(scope)); } /** @@ -288,37 +338,49 @@ public interface CLinker { * java.nio.charset.CharsetDecoder} class should be used when more control * over the decoding process is required. *

    - * This method is restricted. Restricted method are unsafe, and, if used incorrectly, their use might crash + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. + * * @param addr the address at which the string is stored. * @return a Java string with the contents of the null-terminated C string at given address. * @throws IllegalArgumentException if the size of the native string is greater than the largest string supported by the platform. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - static String toJavaStringRestricted(MemoryAddress addr) { - Utils.checkRestrictedAccess("CLinker.toJavaStringRestricted"); + @CallerSensitive + static String toJavaString(MemoryAddress addr) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); Objects.requireNonNull(addr); return SharedUtils.toJavaStringInternal(NativeMemorySegmentImpl.EVERYTHING, addr.toRawLongValue(), Charset.defaultCharset()); } /** - * Converts a null-terminated C string stored at given address into a Java string, using the given {@link java.nio.charset.Charset charset}. + * Converts a null-terminated C string stored at given address into a Java string, using the given {@linkplain java.nio.charset.Charset charset}. *

    * This method always replaces malformed-input and unmappable-character * sequences with this charset's default replacement string. The {@link * java.nio.charset.CharsetDecoder} class should be used when more control * over the decoding process is required. *

    - * This method is restricted. Restricted method are unsafe, and, if used incorrectly, their use might crash + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. + * * @param addr the address at which the string is stored. * @param charset The {@link java.nio.charset.Charset} to be used to compute the contents of the Java string. * @return a Java string with the contents of the null-terminated C string at given address. * @throws IllegalArgumentException if the size of the native string is greater than the largest string supported by the platform. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - static String toJavaStringRestricted(MemoryAddress addr, Charset charset) { - Utils.checkRestrictedAccess("CLinker.toJavaStringRestricted"); + @CallerSensitive + static String toJavaString(MemoryAddress addr, Charset charset) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); Objects.requireNonNull(addr); Objects.requireNonNull(charset); return SharedUtils.toJavaStringInternal(NativeMemorySegmentImpl.EVERYTHING, addr.toRawLongValue(), charset); @@ -343,7 +405,7 @@ public interface CLinker { } /** - * Converts a null-terminated C string stored at given address into a Java string, using the given {@link java.nio.charset.Charset charset}. + * Converts a null-terminated C string stored at given address into a Java string, using the given {@linkplain java.nio.charset.Charset charset}. *

    * This method always replaces malformed-input and unmappable-character * sequences with this charset's default replacement string. The {@link @@ -368,14 +430,8 @@ public interface CLinker { MemoryAccess.setByteAtOffset(addr, bytes.length, (byte)0); } - private static MemorySegment toCString(byte[] bytes) { - MemorySegment segment = MemorySegment.allocateNative(bytes.length + 1, 1L); - copy(segment, bytes); - return segment; - } - - private static MemorySegment toCString(byte[] bytes, NativeScope scope) { - MemorySegment addr = scope.allocate(bytes.length + 1, 1L); + private static MemorySegment toCString(byte[] bytes, SegmentAllocator allocator) { + MemorySegment addr = allocator.allocate(bytes.length + 1, 1L); copy(addr, bytes); return addr; } @@ -383,16 +439,21 @@ public interface CLinker { /** * Allocates memory of given size using malloc. *

    - * This method is restricted. Restricted method are unsafe, and, if used incorrectly, their use might crash + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. * * @param size memory size to be allocated * @return addr memory address of the allocated memory * @throws OutOfMemoryError if malloc could not allocate the required amount of native memory. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - static MemoryAddress allocateMemoryRestricted(long size) { - Utils.checkRestrictedAccess("CLinker.allocateMemoryRestricted"); + @CallerSensitive + static MemoryAddress allocateMemory(long size) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); MemoryAddress addr = SharedUtils.allocateMemoryInternal(size); if (addr.equals(MemoryAddress.NULL)) { throw new OutOfMemoryError(); @@ -404,14 +465,19 @@ public interface CLinker { /** * Frees the memory pointed by the given memory address. *

    - * This method is restricted. Restricted method are unsafe, and, if used incorrectly, their use might crash + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. * * @param addr memory address of the native memory to be freed + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - static void freeMemoryRestricted(MemoryAddress addr) { - Utils.checkRestrictedAccess("CLinker.freeMemoryRestricted"); + @CallerSensitive + static void freeMemory(MemoryAddress addr) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); Objects.requireNonNull(addr); SharedUtils.freeMemoryInternal(addr); } @@ -431,21 +497,16 @@ public interface CLinker { * *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} * elements to a method in this class causes a {@link NullPointerException NullPointerException} to be thrown.

    - * - * @apiNote In the future, if the Java language permits, {@link VaList} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types. - * */ - interface VaList extends Addressable, AutoCloseable { + sealed interface VaList extends Addressable permits WinVaList, SysVVaList, AArch64VaList, SharedUtils.EmptyVaList { /** * Reads the next value as an {@code int} and advances this va list's position. * * @param layout the layout of the value * @return the value read as an {@code int} - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). + * @throws IllegalStateException if the resource scope associated with this instance has been closed + * (see {@link #scope()}). * @throws IllegalArgumentException if the given memory layout is not compatible with {@code int} */ int vargAsInt(MemoryLayout layout); @@ -455,8 +516,8 @@ public interface CLinker { * * @param layout the layout of the value * @return the value read as an {@code long} - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). + * @throws IllegalStateException if the resource scope associated with this instance has been closed + * (see {@link #scope()}). * @throws IllegalArgumentException if the given memory layout is not compatible with {@code long} */ long vargAsLong(MemoryLayout layout); @@ -466,8 +527,8 @@ public interface CLinker { * * @param layout the layout of the value * @return the value read as an {@code double} - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). + * @throws IllegalStateException if the resource scope associated with this instance has been closed + * (see {@link #scope()}). * @throws IllegalArgumentException if the given memory layout is not compatible with {@code double} */ double vargAsDouble(MemoryLayout layout); @@ -477,8 +538,8 @@ public interface CLinker { * * @param layout the layout of the value * @return the value read as an {@code MemoryAddress} - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). + * @throws IllegalStateException if the resource scope associated with this instance has been closed + * (see {@link #scope()}). * @throws IllegalArgumentException if the given memory layout is not compatible with {@code MemoryAddress} */ MemoryAddress vargAsAddress(MemoryLayout layout); @@ -486,102 +547,70 @@ public interface CLinker { /** * Reads the next value as a {@code MemorySegment}, and advances this va list's position. *

    - * The memory segment returned by this method will be allocated using - * {@link MemorySegment#allocateNative(long, long)}, and will have to be closed separately. + * The memory segment returned by this method will be allocated using the given {@link SegmentAllocator}. * * @param layout the layout of the value + * @param allocator the allocator to be used for the native segment allocation * @return the value read as an {@code MemorySegment} - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). + * @throws IllegalStateException if the resource scope associated with this instance has been closed + * (see {@link #scope()}). * @throws IllegalArgumentException if the given memory layout is not compatible with {@code MemorySegment} */ - MemorySegment vargAsSegment(MemoryLayout layout); + MemorySegment vargAsSegment(MemoryLayout layout, SegmentAllocator allocator); /** * Reads the next value as a {@code MemorySegment}, and advances this va list's position. *

    - * The memory segment returned by this method will be allocated using the given {@code NativeScope}. + * The memory segment returned by this method will be associated with the given {@link ResourceScope}. * * @param layout the layout of the value - * @param scope the scope to allocate the segment in + * @param scope the resource scope to be associated with the returned segment * @return the value read as an {@code MemorySegment} - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). + * @throws IllegalStateException if the resource scope associated with this instance has been closed + * (see {@link #scope()}). * @throws IllegalArgumentException if the given memory layout is not compatible with {@code MemorySegment} + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. */ - MemorySegment vargAsSegment(MemoryLayout layout, NativeScope scope); + MemorySegment vargAsSegment(MemoryLayout layout, ResourceScope scope); /** * Skips a number of elements with the given memory layouts, and advances this va list's position. * * @param layouts the layout of the value - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). + * @throws IllegalStateException if the resource scope associated with this instance has been closed + * (see {@link #scope()}). */ void skip(MemoryLayout... layouts); /** - * A predicate used to check if the memory associated with the C {@code va_list} modelled - * by this instance is still valid to use. - * - * @return true, if the memory associated with the C {@code va_list} modelled by this instance is still valid - * @see #close() + * Returns the resource scope associated with this instance. + * @return the resource scope associated with this instance. */ - boolean isAlive(); - - /** - * Releases the underlying C {@code va_list} modelled by this instance, and any native memory that is attached - * to this va list that holds its elements (see {@link VaList#make(Consumer)}). - *

    - * After calling this method, {@link #isAlive()} will return {@code false} and further attempts to read values - * from this va list will result in an exception. - * - * @see #isAlive() - */ - void close(); + ResourceScope scope(); /** * Copies this C {@code va_list} at its current position. Copying is useful to traverse the va list's elements * starting from the current position, without affecting the state of the original va list, essentially * allowing the elements to be traversed multiple times. *

    - * If this method needs to allocate native memory for the copy, it will use - * {@link MemorySegment#allocateNative(long, long)} to do so. {@link #close()} will have to be called on the - * returned va list instance to release the allocated memory. + * Any native resource required by the execution of this method will be allocated in the resource scope + * associated with this instance (see {@link #scope()}). *

    * This method only copies the va list cursor itself and not the memory that may be attached to the * va list which holds its elements. That means that if this va list was created with the - * {@link #make(Consumer)} method, closing this va list will also release the native memory that holds its + * {@link #make(Consumer, ResourceScope)} method, closing this va list will also release the native memory that holds its * elements, making the copy unusable. * * @return a copy of this C {@code va_list}. - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). + * @throws IllegalStateException if the resource scope associated with this instance has been closed + * (see {@link #scope()}). */ VaList copy(); - /** - * Copies this C {@code va_list} at its current position. Copying is useful to traverse the va list's elements - * starting from the current position, without affecting the state of the original va list, essentially - * allowing the elements to be traversed multiple times. - *

    - * If this method needs to allocate native memory for the copy, it will use - * the given {@code NativeScope} to do so. - *

    - * This method only copies the va list cursor itself and not the memory that may be attached to the - * va list which holds its elements. That means that if this va list was created with the - * {@link #make(Consumer)} method, closing this va list will also release the native memory that holds its - * elements, making the copy unusable. - * - * @param scope the scope to allocate the copy in - * @return a copy of this C {@code va_list}. - * @throws IllegalStateException if the C {@code va_list} associated with this instance is no longer valid - * (see {@link #close()}). - */ - VaList copy(NativeScope scope); - /** * Returns the memory address of the C {@code va_list} associated with this instance. + * The returned memory address is associated with same resource scope as that associated with this instance. * * @return the memory address of the C {@code va_list} associated with this instance. */ @@ -589,51 +618,58 @@ public interface CLinker { MemoryAddress address(); /** - * Constructs a new {@code VaList} instance out of a memory address pointing to an existing C {@code va_list}. + * Constructs a new {@code VaList} instance out of a memory address pointing to an existing C {@code va_list}, + * backed by the {@linkplain ResourceScope#globalScope() global} resource scope. *

    - * This method is restricted. Restricted method are unsafe, and, if used incorrectly, their use might crash + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. * * @param address a memory address pointing to an existing C {@code va_list}. * @return a new {@code VaList} instance backed by the C {@code va_list} at {@code address}. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - static VaList ofAddressRestricted(MemoryAddress address) { - Utils.checkRestrictedAccess("VaList.ofAddressRestricted"); - Objects.requireNonNull(address); - return SharedUtils.newVaListOfAddress(address); + @CallerSensitive + static VaList ofAddress(MemoryAddress address) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); + return SharedUtils.newVaListOfAddress(address, ResourceScope.globalScope()); } /** - * Constructs a new {@code VaList} using a builder (see {@link Builder}). + * Constructs a new {@code VaList} instance out of a memory address pointing to an existing C {@code va_list}, + * with given resource scope. *

    - * If this method needs to allocate native memory for the va list, it will use - * {@link MemorySegment#allocateNative(long, long)} to do so. - *

    - * This method will allocate native memory to hold the elements in the va list. This memory - * will be 'attached' to the returned va list instance, and will be released when {@link VaList#close()} - * is called. - *

    - * Note that when there are no elements added to the created va list, - * this method will return the same as {@link #empty()}. + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash + * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on + * restricted methods, and use safe and supported functionalities, where possible. * - * @param actions a consumer for a builder (see {@link Builder}) which can be used to specify the elements - * of the underlying C {@code va_list}. - * @return a new {@code VaList} instance backed by a fresh C {@code va_list}. + * @param address a memory address pointing to an existing C {@code va_list}. + * @param scope the resource scope to be associated with the returned {@code VaList} instance. + * @return a new {@code VaList} instance backed by the C {@code va_list} at {@code address}. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - static VaList make(Consumer actions) { - Objects.requireNonNull(actions); - return SharedUtils.newVaList(actions, MemorySegment::allocateNative); + @CallerSensitive + static VaList ofAddress(MemoryAddress address, ResourceScope scope) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); + Objects.requireNonNull(address); + Objects.requireNonNull(scope); + return SharedUtils.newVaListOfAddress(address, scope); } /** - * Constructs a new {@code VaList} using a builder (see {@link Builder}). + * Constructs a new {@code VaList} using a builder (see {@link Builder}), associated with a given + * {@linkplain ResourceScope resource scope}. *

    - * If this method needs to allocate native memory for the va list, it will use - * the given {@code NativeScope} to do so. - *

    - * This method will allocate native memory to hold the elements in the va list. This memory - * will be managed by the given {@code NativeScope}, and will be released when the scope is closed. + * If this method needs to allocate native memory, such memory will be managed by the given + * {@linkplain ResourceScope resource scope}, and will be released when the resource scope is {@linkplain ResourceScope#close closed}. *

    * Note that when there are no elements added to the created va list, * this method will return the same as {@link #empty()}. @@ -642,11 +678,13 @@ public interface CLinker { * of the underlying C {@code va_list}. * @param scope the scope to be used for the valist allocation. * @return a new {@code VaList} instance backed by a fresh C {@code va_list}. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. */ - static VaList make(Consumer actions, NativeScope scope) { + static VaList make(Consumer actions, ResourceScope scope) { Objects.requireNonNull(actions); Objects.requireNonNull(scope); - return SharedUtils.newVaList(actions, SharedUtils.Allocator.ofScope(scope)); + return SharedUtils.newVaList(actions, scope); } /** @@ -665,13 +703,8 @@ public interface CLinker { * *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} * elements to a method in this class causes a {@link NullPointerException NullPointerException} to be thrown.

    - * - * @apiNote In the future, if the Java language permits, {@link Builder} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types. - * */ - interface Builder { + sealed interface Builder permits WinVaList.Builder, SysVVaList.Builder, AArch64VaList.Builder { /** * Adds a native value represented as an {@code int} to the C {@code va_list} being constructed. @@ -729,7 +762,7 @@ public interface CLinker { * A C type kind. Each kind corresponds to a particular C language builtin type, and can be attached to * {@link ValueLayout} instances using the {@link MemoryLayout#withAttribute(String, Constable)} in order * to obtain a layout which can be classified accordingly by {@link CLinker#downcallHandle(Addressable, MethodType, FunctionDescriptor)} - * and {@link CLinker#upcallStub(MethodHandle, FunctionDescriptor)}. + * and {@link CLinker#upcallStub(MethodHandle, FunctionDescriptor, ResourceScope)}. */ enum TypeKind { /** @@ -806,6 +839,6 @@ public interface CLinker { TypeKind = layout.attribute(TypeKind.ATTR_NAME).orElse(null); * } */ - public final static String ATTR_NAME = "abi/kind"; + public static final String ATTR_NAME = "abi/kind"; } } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/GroupLayout.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/GroupLayout.java index 6e09a2fc56f..bcb56052527 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/GroupLayout.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/GroupLayout.java @@ -42,8 +42,8 @@ import java.util.stream.Collectors; /** * A group layout is used to combine together multiple member layouts. There are two ways in which member layouts * can be combined: if member layouts are laid out one after the other, the resulting group layout is said to be a struct - * (see {@link MemoryLayout#ofStruct(MemoryLayout...)}); conversely, if all member layouts are laid out at the same starting offset, - * the resulting group layout is said to be a union (see {@link MemoryLayout#ofUnion(MemoryLayout...)}). + * (see {@link MemoryLayout#structLayout(MemoryLayout...)}); conversely, if all member layouts are laid out at the same starting offset, + * the resulting group layout is said to be a union (see {@link MemoryLayout#unionLayout(MemoryLayout...)}). *

    * This is a value-based * class; programmers should treat instances that are @@ -58,7 +58,7 @@ import java.util.stream.Collectors; * @implSpec * This class is immutable and thread-safe. */ -public final class GroupLayout extends AbstractLayout { +public final class GroupLayout extends AbstractLayout implements MemoryLayout { /** * The group kind. @@ -118,8 +118,8 @@ public final class GroupLayout extends AbstractLayout { * Returns the member layouts associated with this group. * * @apiNote the order in which member layouts are returned is the same order in which member layouts have - * been passed to one of the group layout factory methods (see {@link MemoryLayout#ofStruct(MemoryLayout...)}, - * {@link MemoryLayout#ofUnion(MemoryLayout...)}). + * been passed to one of the group layout factory methods (see {@link MemoryLayout#structLayout(MemoryLayout...)}, + * {@link MemoryLayout#unionLayout(MemoryLayout...)}). * * @return the member layouts associated with this group. */ diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/LibraryLookup.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/LibraryLookup.java index b92a3addc6f..3fa35cbc0e3 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/LibraryLookup.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/LibraryLookup.java @@ -26,6 +26,8 @@ package jdk.incubator.foreign; import jdk.internal.foreign.LibrariesHelper; +import jdk.internal.reflect.CallerSensitive; +import jdk.internal.reflect.Reflection; import java.io.File; import java.lang.invoke.MethodType; @@ -37,23 +39,22 @@ import java.util.Optional; * A native library lookup. Exposes a lookup operation for searching symbols, see {@link LibraryLookup#lookup(String)}. * A given native library remains loaded as long as there is at least one live library lookup instance referring * to it. - * All symbol instances (see {@link LibraryLookup.Symbol}) generated by a given library lookup object contain a strong reference - * to said lookup object, therefore preventing library unloading; in turn method handle instances obtained from - * {@link CLinker#downcallHandle(Addressable, MethodType, FunctionDescriptor)}) also maintain a strong reference - * to the addressable parameter used for their construction. This means that there is always a strong reachability chain - * from a native method handle to a lookup object (the one that was used to lookup the native library symbol the method handle - * refers to); this is useful to prevent situations where a native library is unloaded in the middle of a native call. - *

    - * In cases where a client wants to create a memory segment out of a lookup symbol, the client might want to attach the - * lookup symbol to the newly created segment, so that the symbol will be kept reachable as long as the memory segment - * is reachable; this can be achieved by creating the segment using the {@link MemoryAddress#asSegmentRestricted(long, Runnable, Object)} - * restricted segment factory, as follows: + * All instances generated by a given library lookup object contain a strong reference to said lookup object, + * therefore preventing library unloading. For {@linkplain #lookup(String, MemoryLayout) memory segments} obtained from a library lookup object, + * this means that clients can safely dereference memory associated with lookup symbols, as follows: *
    {@code
    -LibraryLookup defaultLookup = LibraryLookup.defaultLookup();
    -LibraryLookup.Symbol errno = defaultLookup.lookup("errno");
    -MemorySegment errnoSegment = errno.address().asRestrictedSegment(4, errno);
    + * LibraryLookup defaultLookup = LibraryLookup.ofDefault();
    + * MemorySegment errnoSegment = defaultLookup.lookup("errno", MemoryLayouts.JAVA_INT).get();
    + * int errno = MemoryAccess.getInt(errnoSegment);
      * }
    *

    + * For {@linkplain #lookup(String) memory addresses} obtained from a library lookup object, + * since {@linkplain CLinker#downcallHandle(Addressable, MethodType, FunctionDescriptor) native method handles} + * also maintain a strong reference to the addressable parameter used for their construction, there is + * always a strong reachability chain from a native method handle to a lookup object (the one that was used to lookup + * the native library symbol the method handle refers to). This is useful to prevent situations where a native library + * is unloaded in the middle of a native call. + *

    * To allow for a library to be unloaded, a client will have to discard any strong references it * maintains, directly, or indirectly to a lookup object associated with given library. * @@ -63,45 +64,42 @@ MemorySegment errnoSegment = errno.address().asRestrictedSegment(4, errno); public interface LibraryLookup { /** - * A symbol retrieved during a library lookup. A lookup symbol has a name and can be projected - * into a memory address (see {@link #name()} and {@link #address()}, respectively). + * Looks up a symbol with given name in this library. The returned memory address maintains a strong reference to this lookup object. * - * @apiNote In the future, if the Java language permits, {@link Symbol} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types. - * - * @implSpec - * Implementations of this interface are immutable, thread-safe and value-based. + * @param name the symbol name. + * @return the memory address associated with the library symbol (if any). */ - interface Symbol extends Addressable { - /** - * The name of this lookup symbol. - * @return the name of this lookup symbol. - */ - String name(); - - /** - * The memory address of this lookup symbol. If the memory associated with this symbol needs to be dereferenced, - * clients can obtain a segment from this symbol's address using the {@link MemoryAddress#asSegmentRestricted(long, Runnable, Object)}, - * and making sure that the created segment maintains a strong reference to this symbol, to prevent library unloading. - * @return the memory address of this lookup symbol. - */ - @Override - MemoryAddress address(); - } + Optional lookup(String name); /** - * Looks up a symbol with given name in this library. The returned symbol maintains a strong reference to this lookup object. + * Looks up a symbol with given name in this library. The returned memory segment has a size that matches that of + * the specified layout, and maintains a strong reference to this lookup object. This method can be useful + * to lookup global variable symbols in a foreign library. + * * @param name the symbol name. - * @return the library symbol (if any). + * @param layout the layout to be associated with the library symbol. + * @return the memory segment associated with the library symbol (if any). + * @throws IllegalArgumentException if the address associated with the lookup symbol do not match the + * {@linkplain MemoryLayout#byteAlignment() alignment constraints} in {@code layout}. */ - Optional lookup(String name); + Optional lookup(String name, MemoryLayout layout); /** * Obtain a default library lookup object. + *

    + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash + * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on + * restricted methods, and use safe and supported functionalities, where possible. + * * @return the default library lookup object. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ + @CallerSensitive static LibraryLookup ofDefault() { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); SecurityManager security = System.getSecurityManager(); if (security != null) { security.checkPermission(new RuntimePermission("java.foreign.getDefaultLibrary")); @@ -111,12 +109,23 @@ public interface LibraryLookup { /** * Obtain a library lookup object corresponding to a library identified by given path. + *

    + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash + * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on + * restricted methods, and use safe and supported functionalities, where possible. + * * @param path the library absolute path. * @return a library lookup object for given path. * @throws IllegalArgumentException if the specified path does not correspond to an absolute path, * e.g. if {@code !path.isAbsolute()}. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ + @CallerSensitive static LibraryLookup ofPath(Path path) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); Objects.requireNonNull(path); if (!path.isAbsolute()) { throw new IllegalArgumentException("Not an absolute path: " + path.toString()); @@ -134,10 +143,21 @@ public interface LibraryLookup { * is decorated according to the platform conventions (e.g. on Linux, the {@code lib} prefix is added, * as well as the {@code .so} extension); the resulting name is then looked up in the standard native * library path (which can be overriden, by setting the java.library.path property). + *

    + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash + * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on + * restricted methods, and use safe and supported functionalities, where possible. + * * @param libName the library name. * @return a library lookup object for given library name. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ + @CallerSensitive static LibraryLookup ofLibrary(String libName) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); Objects.requireNonNull(libName); SecurityManager security = System.getSecurityManager(); if (security != null) { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MappedMemorySegments.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MappedMemorySegments.java deleted file mode 100644 index 9c92761044a..00000000000 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MappedMemorySegments.java +++ /dev/null @@ -1,167 +0,0 @@ -/* - * Copyright (c) 2020, 2021, 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. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * 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. - */ - -package jdk.incubator.foreign; - -import jdk.internal.foreign.MappedMemorySegmentImpl; - -import java.io.UncheckedIOException; -import java.nio.MappedByteBuffer; -import java.util.Objects; - -/** - * This class provides capabilities to manipulate mapped memory segments, such as {@link #force(MemorySegment)}, - * and {@link #load(MemorySegment)}. The methods in these class are suitable replacements for some of the - * functionality in the {@link java.nio.MappedByteBuffer} class. Note that, while it is possible to map a segment - * into a byte buffer (see {@link MemorySegment#asByteBuffer()}), and call e.g. {@link MappedByteBuffer#force()} that way, - * this can only be done when the source segment is small enough, due to the size limitation inherent to the - * ByteBuffer API. - *

    - * Clients requiring sophisticated, low-level control over mapped memory segments, should consider writing - * custom mapped memory segment factories; using JNI, e.g. on Linux, it is possible to call {@code mmap} - * with the desired parameters; the returned address can be easily wrapped into a memory segment, using - * {@link MemoryAddress#ofLong(long)} and {@link MemoryAddress#asSegmentRestricted(long, Runnable, Object)}. - * - *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} - * elements to a method in this class causes a {@link NullPointerException NullPointerException} to be thrown.

    - * - * @implNote - * The behavior of some the methods in this class (see {@link #load(MemorySegment)}, {@link #unload(MemorySegment)} and - * {@link #isLoaded(MemorySegment)}) is highly platform-dependent; as a result, calling these methods might - * be a no-op on certain platforms. - */ -public final class MappedMemorySegments { - private MappedMemorySegments() { - // no thanks - } - - /** - * Tells whether or not the contents of the given segment is resident in physical - * memory. - * - *

    A return value of {@code true} implies that it is highly likely - * that all of the data in the given segment is resident in physical memory and - * may therefore be accessed without incurring any virtual-memory page - * faults or I/O operations. A return value of {@code false} does not - * necessarily imply that the segment's content is not resident in physical - * memory. - * - *

    The returned value is a hint, rather than a guarantee, because the - * underlying operating system may have paged out some of the segment's data - * by the time that an invocation of this method returns.

    - * - * @param segment the segment whose contents are to be tested. - * @return {@code true} if it is likely that the contents of the given segment - * is resident in physical memory - * - * @throws IllegalStateException if the given segment is not alive, or if the given segment is confined - * and this method is called from a thread other than the segment's owner thread. - * @throws UnsupportedOperationException if the given segment is not a mapped memory segment, e.g. if - * {@code segment.isMapped() == false}. - */ - public static boolean isLoaded(MemorySegment segment) { - return toMappedSegment(segment).isLoaded(); - } - - /** - * Loads the contents of the given segment into physical memory. - * - *

    This method makes a best effort to ensure that, when it returns, - * this contents of the given segment is resident in physical memory. Invoking this - * method may cause some number of page faults and I/O operations to - * occur.

    - * - * @param segment the segment whose contents are to be loaded. - * - * @throws IllegalStateException if the given segment is not alive, or if the given segment is confined - * and this method is called from a thread other than the segment's owner thread. - * @throws UnsupportedOperationException if the given segment is not a mapped memory segment, e.g. if - * {@code segment.isMapped() == false}. - */ - public static void load(MemorySegment segment) { - toMappedSegment(segment).load(); - } - - /** - * Unloads the contents of the given segment from physical memory. - * - *

    This method makes a best effort to ensure that the contents of the given segment are - * are no longer resident in physical memory. Accessing this segment's contents - * after invoking this method may cause some number of page faults and I/O operations to - * occur (as this segment's contents might need to be paged back in).

    - * - * @param segment the segment whose contents are to be unloaded. - * - * @throws IllegalStateException if the given segment is not alive, or if the given segment is confined - * and this method is called from a thread other than the segment's owner thread. - * @throws UnsupportedOperationException if the given segment is not a mapped memory segment, e.g. if - * {@code segment.isMapped() == false}. - */ - public static void unload(MemorySegment segment) { - toMappedSegment(segment).unload(); - } - - /** - * Forces any changes made to the contents of the given segment to be written to the - * storage device described by the mapped segment's file descriptor. - * - *

    If this mapping's file descriptor resides on a local storage - * device then when this method returns it is guaranteed that all changes - * made to the segment since it was created, or since this method was last - * invoked, will have been written to that device. - * - *

    If this mapping's file descriptor does not reside on a local device then no such guarantee - * is made. - * - *

    If the given segment was not mapped in read/write mode ({@link - * java.nio.channels.FileChannel.MapMode#READ_WRITE}) then - * invoking this method may have no effect. In particular, the - * method has no effect for segments mapped in read-only or private - * mapping modes. This method may or may not have an effect for - * implementation-specific mapping modes. - *

    - * - * @param segment the segment whose contents are to be written to the storage device described by the - * segment's file descriptor. - * - * @throws IllegalStateException if the given segment is not alive, or if the given segment is confined - * and this method is called from a thread other than the segment's owner thread. - * @throws UnsupportedOperationException if the given segment is not a mapped memory segment, e.g. if - * {@code segment.isMapped() == false}. - * @throws UncheckedIOException if there is an I/O error writing the contents of the segment to the associated storage device - */ - public static void force(MemorySegment segment) { - toMappedSegment(segment).force(); - } - - static MappedMemorySegmentImpl toMappedSegment(MemorySegment segment) { - Objects.requireNonNull(segment); - if (segment instanceof MappedMemorySegmentImpl) { - return (MappedMemorySegmentImpl)segment; - } else { - throw new UnsupportedOperationException("Not a mapped memory segment"); - } - } -} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryAddress.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryAddress.java index 3348506f8ce..e13fe0156e6 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryAddress.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryAddress.java @@ -26,19 +26,28 @@ package jdk.incubator.foreign; -import jdk.internal.foreign.AbstractMemorySegmentImpl; import jdk.internal.foreign.MemoryAddressImpl; -import jdk.internal.foreign.NativeMemorySegmentImpl; -import jdk.internal.foreign.Utils; +import jdk.internal.ref.CleanerFactory; +import jdk.internal.reflect.CallerSensitive; import java.lang.ref.Cleaner; /** * A memory address models a reference into a memory location. Memory addresses are typically obtained using the - * {@link MemorySegment#address()} method, and can refer to either off-heap or on-heap memory. + * {@link MemorySegment#address()} method, and can refer to either off-heap or on-heap memory. Off-heap memory + * addresses are referred to as native memory addresses (see {@link #isNative()}). Native memory addresses + * allow clients to obtain a raw memory address (expressed as a long value) which can then be used e.g. when interacting + * with native code. + *

    * Given an address, it is possible to compute its offset relative to a given segment, which can be useful * when performing memory dereference operations using a memory access var handle (see {@link MemoryHandles}). *

    + * A memory address is associated with a {@linkplain ResourceScope resource scope}; the resource scope determines the + * lifecycle of the memory address, and whether the address can be used from multiple threads. Memory addresses + * obtained from {@linkplain #ofLong(long) numeric values}, or from native code, are associated with the + * {@linkplain ResourceScope#globalScope() global resource scope}. Memory addresses obtained from segments + * are associated with the same scope as the segment from which they have been obtained. + *

    * All implementations of this interface must be value-based; * programmers should treat instances that are {@linkplain #equals(Object) equal} as interchangeable and should not * use instances for synchronization, or unpredictable behavior may occur. For example, in a future release, @@ -49,14 +58,10 @@ import java.lang.ref.Cleaner; *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} * elements to a method in this class causes a {@link NullPointerException NullPointerException} to be thrown.

    * - * @apiNote In the future, if the Java language permits, {@link MemoryAddress} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types. - * * @implSpec * Implementations of this interface are immutable, thread-safe and value-based. */ -public interface MemoryAddress extends Addressable { +public sealed interface MemoryAddress extends Addressable permits MemoryAddressImpl { @Override default MemoryAddress address() { @@ -70,9 +75,15 @@ public interface MemoryAddress extends Addressable { */ MemoryAddress addOffset(long offset); + /** + * Returns the resource scope associated with this memory address. + * @return the resource scope associated with this memory address. + */ + ResourceScope scope(); + /** * Returns the offset of this memory address into the given segment. More specifically, if both the segment's - * base address and this address are off-heap addresses, the result is computed as + * base address and this address are native addresses, the result is computed as * {@code this.toRawLongValue() - segment.address().toRawLongValue()}. Otherwise, if both addresses in the form * {@code (B, O1)}, {@code (B, O2)}, where {@code B} is the same base heap object and {@code O1}, {@code O2} * are byte offsets (relative to the base object) associated with this address and the segment's base address, @@ -86,82 +97,94 @@ public interface MemoryAddress extends Addressable { * @return the offset of this memory address into the given segment. * @param segment the segment relative to which this address offset should be computed * @throws IllegalArgumentException if {@code segment} is not compatible with this address; this can happen, for instance, - * when {@code segment} models an heap memory region, while this address models an off-heap memory address. + * when {@code segment} models an heap memory region, while this address is a {@linkplain #isNative() native} address. */ long segmentOffset(MemorySegment segment); /** - * Returns a new confined native memory segment with given size, and whose base address is this address; the returned segment has its own temporal - * bounds, and can therefore be closed. This method can be useful when interacting with custom native memory sources (e.g. custom allocators), - * where an address to some underlying memory region is typically obtained from native code (often as a plain {@code long} value). - *

    - * The returned segment will feature all access modes - * (see {@link MemorySegment#ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). + Returns a new native memory segment with given size and resource scope (replacing the scope already associated + * with this address), and whose base address is this address. This method can be useful when interacting with custom + * native memory sources (e.g. custom allocators), where an address to some + * underlying memory region is typically obtained from native code (often as a plain {@code long} value). + * The returned segment is not read-only (see {@link MemorySegment#isReadOnly()}), and is associated with the + * provided resource scope. *

    * Clients should ensure that the address and bounds refers to a valid region of memory that is accessible for reading and, * if appropriate, writing; an attempt to access an invalid memory location from Java code will either return an arbitrary value, * have no visible effect, or cause an unspecified exception to be thrown. *

    - * Calling {@link MemorySegment#close()} on the returned segment will not result in releasing any - * memory resources which might implicitly be associated with the segment. This method is equivalent to the following code: + * This method is equivalent to the following code: *

    {@code
    -    asSegmentRestricted(byteSize, null, null);
    +    asSegment(byteSize, null, scope);
          * }
    - * This method is restricted. Restricted methods are unsafe, and, if used incorrectly, their use might crash + *

    + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. * * @param bytesSize the desired size. - * @return a new confined native memory segment with given base address and size. + * @param scope the native segment scope. + * @return a new native memory segment with given base address, size and scope. * @throws IllegalArgumentException if {@code bytesSize <= 0}. - * @throws UnsupportedOperationException if this address is an heap address. - * @throws IllegalAccessError if the runtime property {@code foreign.restricted} is not set to either - * {@code permit}, {@code warn} or {@code debug} (the default value is set to {@code deny}). + * @throws IllegalStateException if either the scope associated with this address or the provided scope + * have been already closed, or if access occurs from a thread other than the thread owning either + * scopes. + * @throws UnsupportedOperationException if this address is not a {@linkplain #isNative() native} address. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - default MemorySegment asSegmentRestricted(long bytesSize) { - return asSegmentRestricted(bytesSize, null, null); - } + @CallerSensitive + MemorySegment asSegment(long bytesSize, ResourceScope scope); /** - * Returns a new confined native memory segment with given size, and whose base address is this address; the returned segment has its own temporal - * bounds, and can therefore be closed. This method can be useful when interacting with custom native memory sources (e.g. custom allocators), - * where an address to some underlying memory region is typically obtained from native code (often as a plain {@code long} value). - *

    - * The returned segment will feature all access modes - * (see {@link MemorySegment#ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). - * Moreover, the returned segment will keep a strong reference to the supplied attachment object (if any), which can - * be useful in cases where the lifecycle of the segment is dependent on that of some other external resource. + * Returns a new native memory segment with given size and resource scope (replacing the scope already associated + * with this address), and whose base address is this address. This method can be useful when interacting with custom + * native memory sources (e.g. custom allocators), where an address to some + * underlying memory region is typically obtained from native code (often as a plain {@code long} value). + * The returned segment is associated with the provided resource scope. *

    * Clients should ensure that the address and bounds refers to a valid region of memory that is accessible for reading and, * if appropriate, writing; an attempt to access an invalid memory location from Java code will either return an arbitrary value, * have no visible effect, or cause an unspecified exception to be thrown. *

    - * Calling {@link MemorySegment#close()} on the returned segment will not result in releasing any - * memory resources which might implicitly be associated with the segment, but will result in calling the - * provided cleanup action (if any). + * Calling {@link ResourceScope#close()} on the scope associated with the returned segment will result in calling + * the provided cleanup action (if any). *

    - * Both the cleanup action and the attachment object (if any) will be preserved under terminal operations such as - * {@link MemorySegment#handoff(Thread)}, {@link MemorySegment#share()} and {@link MemorySegment#registerCleaner(Cleaner)}. - *

    - * This method is restricted. Restricted methods are unsafe, and, if used incorrectly, their use might crash + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. * * @param bytesSize the desired size. * @param cleanupAction the cleanup action; can be {@code null}. - * @param attachment an attachment object that will be kept strongly reachable by the returned segment; can be {@code null}. - * @return a new confined native memory segment with given base address and size. + * @param scope the native segment scope. + * @return a new native memory segment with given base address, size and scope. * @throws IllegalArgumentException if {@code bytesSize <= 0}. - * @throws UnsupportedOperationException if this address is an heap address. - * @throws IllegalAccessError if the runtime property {@code foreign.restricted} is not set to either - * {@code permit}, {@code warn} or {@code debug} (the default value is set to {@code deny}). + * @throws IllegalStateException if either the scope associated with this address or the provided scope + * have been already closed, or if access occurs from a thread other than the thread owning either + * scopes. + * @throws UnsupportedOperationException if this address is not a {@linkplain #isNative() native} address. + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - MemorySegment asSegmentRestricted(long bytesSize, Runnable cleanupAction, Object attachment); + @CallerSensitive + MemorySegment asSegment(long bytesSize, Runnable cleanupAction, ResourceScope scope); /** - * Returns the raw long value associated with this memory address. - * @return The raw long value associated with this memory address. - * @throws UnsupportedOperationException if this memory address is an heap address. + * Is this an off-heap memory address? + * @return true, if this is an off-heap memory address. + */ + boolean isNative(); + + /** + * Returns the raw long value associated with this native memory address. + * @return The raw long value associated with this native memory address. + * @throws UnsupportedOperationException if this memory address is not a {@linkplain #isNative() native} address. + * @throws IllegalStateException if the scope associated with this segment has been already closed, + * or if access occurs from a thread other than the thread owning either segment. */ long toRawLongValue(); @@ -169,11 +192,10 @@ public interface MemoryAddress extends Addressable { * Compares the specified object with this address for equality. Returns {@code true} if and only if the specified * object is also an address, and it refers to the same memory location as this address. * - * @apiNote two addresses might be considered equal despite their associated segments differ. This - * can happen, for instance, if the segment associated with one address is a slice - * (see {@link MemorySegment#asSlice(long, long)}) of the segment associated with the other address. Moreover, - * two addresses might be considered equals despite differences in the temporal bounds associated with their - * corresponding segments. + * @apiNote two addresses might be considered equal despite their associated resource scopes differ. This + * can happen, for instance, if the same memory address is used to create memory segments with different + * scopes (using {@link #asSegment(long, ResourceScope)}), and the base address of the resulting segments is + * then compared. * * @param that the object to be compared for equality with this address. * @return {@code true} if the specified object is equal to this address. @@ -189,12 +211,14 @@ public interface MemoryAddress extends Addressable { int hashCode(); /** - * The off-heap memory address instance modelling the {@code NULL} address. + * The native memory address instance modelling the {@code NULL} address, associated + * with the {@linkplain ResourceScope#globalScope() global} resource scope. */ - MemoryAddress NULL = new MemoryAddressImpl(null, 0L); + MemoryAddress NULL = new MemoryAddressImpl(null, 0L); /** - * Obtain an off-heap memory address instance from given long address. + * Obtain a native memory address instance from given long address. The returned address is associated + * with the {@linkplain ResourceScope#globalScope() global} resource scope. * @param value the long address. * @return the new memory address instance. */ diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryHandles.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryHandles.java index 6bd7ad0be07..31dbc4e9ce5 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryHandles.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryHandles.java @@ -44,16 +44,16 @@ import java.util.Objects; * (see {@link MemoryHandles#varHandle(Class, ByteOrder)}, * {@link MemoryHandles#varHandle(Class, long, ByteOrder)}). This determines the variable type * (all primitive types but {@code void} and {@code boolean} are supported), as well as the alignment constraint and the - * byte order associated to a memory access var handle. The resulting memory access var handle can then be combined in various ways + * byte order associated with a memory access var handle. The resulting memory access var handle can then be combined in various ways * to emulate different addressing modes. The var handles created by this class feature a mandatory coordinate type * (of type {@link MemorySegment}), and one {@code long} coordinate type, which represents the offset, in bytes, relative * to the segment, at which dereference should occur. *

    * As an example, consider the memory layout expressed by a {@link GroupLayout} instance constructed as follows: *

    {@code
    -GroupLayout seq = MemoryLayout.ofStruct(
    -        MemoryLayout.ofPaddingBits(32),
    -        MemoryLayout.ofValueBits(32, ByteOrder.BIG_ENDIAN).withName("value")
    +GroupLayout seq = MemoryLayout.structLayout(
    +        MemoryLayout.paddingLayout(32),
    +        MemoryLayout.valueLayout(32, ByteOrder.BIG_ENDIAN).withName("value")
     );
      * }
    * To access the member layout named {@code value}, we can construct a memory access var handle as follows: @@ -103,7 +103,7 @@ handle = MemoryHandles.insertCoordinates(handle, 1, 4); //(MemorySegment) -> int */ public final class MemoryHandles { - private final static JavaLangInvokeAccess JLI = SharedSecrets.getJavaLangInvokeAccess(); + private static final JavaLangInvokeAccess JLI = SharedSecrets.getJavaLangInvokeAccess(); private MemoryHandles() { //sorry, just the one! @@ -254,7 +254,7 @@ public final class MemoryHandles { * Java {@code int} to avoid dealing with negative values, which would be * the case if modeled as a Java {@code short}. This is illustrated in the following example: *
    {@code
    -    MemorySegment segment = MemorySegment.allocateNative(2);
    +    MemorySegment segment = MemorySegment.allocateNative(2, ResourceScope.newImplicitScope());
         VarHandle SHORT_VH = MemoryLayouts.JAVA_SHORT.varHandle(short.class);
         VarHandle INT_VH = MemoryHandles.asUnsigned(SHORT_VH, int.class);
         SHORT_VH.set(segment, (short)-1);
    diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayout.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayout.java
    index 9c9e5ffc679..44793be750f 100644
    --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayout.java
    +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayout.java
    @@ -49,7 +49,7 @@ import java.util.stream.Stream;
      * A memory layout can be used to describe the contents of a memory segment in a language neutral fashion.
      * There are two leaves in the layout hierarchy, value layouts, which are used to represent values of given size and kind (see
      * {@link ValueLayout}) and padding layouts which are used, as the name suggests, to represent a portion of a memory
    - * segment whose contents should be ignored, and which are primarily present for alignment reasons (see {@link MemoryLayout#ofPaddingBits(long)}).
    + * segment whose contents should be ignored, and which are primarily present for alignment reasons (see {@link MemoryLayout#paddingLayout(long)}).
      * Some common value layout constants are defined in the {@link MemoryLayouts} class.
      * 

    * More complex layouts can be derived from simpler ones: a sequence layout denotes a repetition of one or more @@ -68,11 +68,11 @@ import java.util.stream.Stream; * The above declaration can be modelled using a layout object, as follows: * *

    {@code
    -SequenceLayout taggedValues = MemoryLayout.ofSequence(5,
    -    MemoryLayout.ofStruct(
    -        MemoryLayout.ofValueBits(8, ByteOrder.nativeOrder()).withName("kind"),
    -        MemoryLayout.ofPaddingBits(24),
    -        MemoryLayout.ofValueBits(32, ByteOrder.nativeOrder()).withName("value")
    +SequenceLayout taggedValues = MemoryLayout.sequenceLayout(5,
    +    MemoryLayout.structLayout(
    +        MemoryLayout.valueLayout(8, ByteOrder.nativeOrder()).withName("kind"),
    +        MemoryLayout.paddingLayout(24),
    +        MemoryLayout.valueLayout(32, ByteOrder.nativeOrder()).withName("value")
         )
     ).withName("TaggedValues");
      * }
    @@ -144,17 +144,17 @@ MemoryLayout value = taggedValues.select(PathElement.sequenceElement(), * * And, we can also replace the layout named {@code value} with another layout, as follows: *
    {@code
    -MemoryLayout taggedValuesWithHole = taggedValues.map(l -> MemoryLayout.ofPadding(32),
    +MemoryLayout taggedValuesWithHole = taggedValues.map(l -> MemoryLayout.paddingLayout(32),
                                                 PathElement.sequenceElement(), PathElement.groupElement("value"));
      * }
    * * That is, the above declaration is identical to the following, more verbose one: *
    {@code
    -MemoryLayout taggedValuesWithHole = MemoryLayout.ofSequence(5,
    -    MemoryLayout.ofStruct(
    -        MemoryLayout.ofValueBits(8, ByteOrder.nativeOrder()).withName("kind").
    -        MemoryLayout.ofPaddingBits(32),
    -        MemoryLayout.ofPaddingBits(32)
    +MemoryLayout taggedValuesWithHole = MemoryLayout.sequenceLayout(5,
    +    MemoryLayout.structLayout(
    +        MemoryLayout.valueLayout(8, ByteOrder.nativeOrder()).withName("kind"),
    +        MemoryLayout.paddingLayout(32),
    +        MemoryLayout.paddingLayout(32)
     ));
      * }
    * @@ -191,17 +191,13 @@ long offset2 = (long) offsetHandle.invokeExact(2L); // 16 * * Layouts can be optionally associated with one or more attributes. A layout attribute forms a name/value * pair, where the name is a {@link String} and the value is a {@link Constable}. The most common form of layout attribute - * is the layout name (see {@link #LAYOUT_NAME}), a custom name that can be associated to memory layouts and that can be referred to when + * is the layout name (see {@link #LAYOUT_NAME}), a custom name that can be associated with memory layouts and that can be referred to when * constructing layout paths. * - * @apiNote In the future, if the Java language permits, {@link MemoryLayout} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types. - * * @implSpec * Implementations of this interface are immutable, thread-safe and value-based. */ -public interface MemoryLayout extends Constable { +public sealed interface MemoryLayout extends Constable permits AbstractLayout, SequenceLayout, GroupLayout, PaddingLayout, ValueLayout { /** * Returns an {@link Optional} containing the nominal descriptor for this @@ -218,7 +214,7 @@ public interface MemoryLayout extends Constable { * Does this layout have a specified size? A layout does not have a specified size if it is (or contains) a sequence layout whose * size is unspecified (see {@link SequenceLayout#elementCount()}). * - * Value layouts (see {@link ValueLayout}) and padding layouts (see {@link MemoryLayout#ofPaddingBits(long)}) + * Value layouts (see {@link ValueLayout}) and padding layouts (see {@link MemoryLayout#paddingLayout(long)}) * always have a specified size, therefore this method always returns {@code true} in these cases. * * @return {@code true}, if this layout has a specified size. @@ -267,7 +263,7 @@ public interface MemoryLayout extends Constable { * }
    * * @param name the layout name. - * @return a new layout which is the same as this layout, except for the name associated to it. + * @return a new layout which is the same as this layout, except for the name associated with it. * @see MemoryLayout#name() */ MemoryLayout withName(String name); @@ -316,7 +312,7 @@ public interface MemoryLayout extends Constable { * Creates a new layout which features the desired alignment constraint. * * @param bitAlignment the layout alignment constraint, expressed in bits. - * @return a new layout which is the same as this layout, except for the alignment constraint associated to it. + * @return a new layout which is the same as this layout, except for the alignment constraint associated with it. * @throws IllegalArgumentException if {@code bitAlignment} is not a power of two, or if it's less than than 8. */ MemoryLayout withBitAlignment(long bitAlignment); @@ -357,6 +353,8 @@ public interface MemoryLayout extends Constable { * layout path contains one or more path elements that select multiple sequence element indices * (see {@link PathElement#sequenceElement()} and {@link PathElement#sequenceElement(long, long)}). * @throws UnsupportedOperationException if one of the layouts traversed by the layout path has unspecified size. + * @throws NullPointerException if either {@code elements == null}, or if any of the elements + * in {@code elements} is {@code null}. */ default long bitOffset(PathElement... elements) { return computePathOp(LayoutPath.rootPath(this, MemoryLayout::bitSize), LayoutPath::offset, @@ -380,8 +378,9 @@ public interface MemoryLayout extends Constable { * } * * where {@code x_1}, {@code x_2}, ... {@code x_n} are dynamic values provided as {@code long} - * arguments, whereas {@code c_1}, {@code c_2}, ... {@code c_m} and {@code s_0}, {@code s_1}, ... {@code s_n} are - * static stride constants which are derived from the layout path. + * arguments, whereas {@code c_1}, {@code c_2}, ... {@code c_m} are static offset constants + * and {@code s_0}, {@code s_1}, ... {@code s_n} are static stride constants which are derived from + * the layout path. * * @param elements the layout path elements. * @return a method handle that can be used to compute the bit offset of the layout element @@ -406,6 +405,8 @@ public interface MemoryLayout extends Constable { * (see {@link PathElement#sequenceElement()} and {@link PathElement#sequenceElement(long, long)}). * @throws UnsupportedOperationException if one of the layouts traversed by the layout path has unspecified size, * or if {@code bitOffset(elements)} is not a multiple of 8. + * @throws NullPointerException if either {@code elements == null}, or if any of the elements + * in {@code elements} is {@code null}. */ default long byteOffset(PathElement... elements) { return Utils.bitsToBytesOrThrow(bitOffset(elements), Utils.bitsToBytesThrowOffset); @@ -429,8 +430,9 @@ public interface MemoryLayout extends Constable { * } * * where {@code x_1}, {@code x_2}, ... {@code x_n} are dynamic values provided as {@code long} - * arguments, whereas {@code c_1}, {@code c_2}, ... {@code c_m} and {@code s_0}, {@code s_1}, ... {@code s_n} are - * static stride constants which are derived from the layout path. + * arguments, whereas {@code c_1}, {@code c_2}, ... {@code c_m} are static offset constants + * and {@code s_0}, {@code s_1}, ... {@code s_n} are static stride constants which are derived from + * the layout path. * *

    The method handle will throw an {@link UnsupportedOperationException} if the computed * offset in bits is not a multiple of 8. @@ -466,9 +468,10 @@ public interface MemoryLayout extends Constable { offset = c_1 + c_2 + ... + c_m + (x_1 * s_1) + (x_2 * s_2) + ... + (x_n * s_n) * } * - * where {@code x_1}, {@code x_2}, ... {@code x_n} are dynamic values provided as optional {@code long} - * access coordinates, whereas {@code c_1}, {@code c_2}, ... {@code c_m} and {@code s_0}, {@code s_1}, ... {@code s_n} are - * static stride constants which are derived from the layout path. + * where {@code x_1}, {@code x_2}, ... {@code x_n} are dynamic values provided as {@code long} + * arguments, whereas {@code c_1}, {@code c_2}, ... {@code c_m} are static offset constants + * and {@code s_0}, {@code s_1}, ... {@code s_n} are static stride constants which are derived from + * the layout path. * * @apiNote the resulting var handle will feature an additional {@code long} access coordinate for every * unspecified sequence access component contained in this layout path. Moreover, the resulting var handle @@ -489,6 +492,50 @@ public interface MemoryLayout extends Constable { Set.of(), elements); } + /** + * Creates a method handle which, given a memory segment, returns a {@linkplain MemorySegment#asSlice(long,long) slice} + * corresponding to the layout selected by a given layout path, where the path is considered rooted in this layout. + * + *

    The returned method handle has a return type of {@code MemorySegment}, features a {@code MemorySegment} + * parameter as leading parameter representing the segment to be sliced, and features as many trailing {@code long} + * parameter types as there are free dimensions in the provided layout path (see {@link PathElement#sequenceElement()}, + * where the order of the parameters corresponds to the order of the path elements. + * The returned method handle can be used to create a slice similar to using {@link MemorySegment#asSlice(long, long)}, + * but where the offset argument is dynamically compute based on indices specified when invoking the method handle. + * + *

    The offset of the returned segment is computed as follows: + * + *

    {@code
    +    bitOffset = c_1 + c_2 + ... + c_m + (x_1 * s_1) + (x_2 * s_2) + ... + (x_n * s_n)
    +    offset = bitOffset / 8
    +     * }
    + * + * where {@code x_1}, {@code x_2}, ... {@code x_n} are dynamic values provided as {@code long} + * arguments, whereas {@code c_1}, {@code c_2}, ... {@code c_m} are static offset constants + * and {@code s_0}, {@code s_1}, ... {@code s_n} are static stride constants which are derived from + * the layout path. + * + *

    After the offset is computed, the returned segment is create as if by calling: + *

    {@code
    +    segment.asSlice(offset, layout.byteSize());
    +     * }
    + * + * where {@code segment} is the segment to be sliced, and where {@code layout} is the layout selected by the given + * layout path, as per {@link MemoryLayout#select(PathElement...)}. + * + *

    The method handle will throw an {@link UnsupportedOperationException} if the computed + * offset in bits is not a multiple of 8. + * + * @param elements the layout path elements. + * @return a method handle which can be used to create a slice of the selected layout element, given a segment. + * @throws UnsupportedOperationException if the size of the selected layout in bits is not a multiple of 8. + */ + default MethodHandle sliceHandle(PathElement... elements) { + return computePathOp(LayoutPath.rootPath(this, MemoryLayout::bitSize), LayoutPath::sliceHandle, + Set.of(), elements); + } + + /** * Selects the layout from a path rooted in this layout. * @@ -535,7 +582,7 @@ public interface MemoryLayout extends Constable { } /** - * Is this a padding layout (e.g. a layout created from {@link #ofPaddingBits(long)}) ? + * Is this a padding layout (e.g. a layout created from {@link #paddingLayout(long)}) ? * @return true, if this layout is a padding layout. */ boolean isPadding(); @@ -554,14 +601,10 @@ public interface MemoryLayout extends Constable { *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} * elements to a method in this class causes a {@link NullPointerException NullPointerException} to be thrown.

    * - * @apiNote In the future, if the Java language permits, {@link PathElement} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types. - * * @implSpec * Implementations of this interface are immutable and thread-safe. */ - interface PathElement { + sealed interface PathElement permits LayoutPath.PathElementImpl { /** * Returns a path element which selects a member layout with given name from a given group layout. @@ -679,7 +722,7 @@ E * (S + I * F) * @return the new selector layout. * @throws IllegalArgumentException if {@code size <= 0}. */ - static MemoryLayout ofPaddingBits(long size) { + static MemoryLayout paddingLayout(long size) { AbstractLayout.checkSize(size); return new PaddingLayout(size); } @@ -692,7 +735,7 @@ E * (S + I * F) * @return a new value layout. * @throws IllegalArgumentException if {@code size <= 0}. */ - static ValueLayout ofValueBits(long size, ByteOrder order) { + static ValueLayout valueLayout(long size, ByteOrder order) { Objects.requireNonNull(order); AbstractLayout.checkSize(size); return new ValueLayout(order, size); @@ -706,7 +749,7 @@ E * (S + I * F) * @return the new sequence layout with given element layout and size. * @throws IllegalArgumentException if {@code elementCount < 0}. */ - static SequenceLayout ofSequence(long elementCount, MemoryLayout elementLayout) { + static SequenceLayout sequenceLayout(long elementCount, MemoryLayout elementLayout) { AbstractLayout.checkSize(elementCount, true); OptionalLong size = OptionalLong.of(elementCount); return new SequenceLayout(size, Objects.requireNonNull(elementLayout)); @@ -718,7 +761,7 @@ E * (S + I * F) * @param elementLayout the element layout of the sequence layout. * @return the new sequence layout with given element layout. */ - static SequenceLayout ofSequence(MemoryLayout elementLayout) { + static SequenceLayout sequenceLayout(MemoryLayout elementLayout) { return new SequenceLayout(OptionalLong.empty(), Objects.requireNonNull(elementLayout)); } @@ -728,7 +771,7 @@ E * (S + I * F) * @param elements The member layouts of the struct group layout. * @return a new struct group layout with given member layouts. */ - static GroupLayout ofStruct(MemoryLayout... elements) { + static GroupLayout structLayout(MemoryLayout... elements) { Objects.requireNonNull(elements); return new GroupLayout(GroupLayout.Kind.STRUCT, Stream.of(elements) @@ -742,7 +785,7 @@ E * (S + I * F) * @param elements The member layouts of the union layout. * @return a new union group layout with given member layouts. */ - static GroupLayout ofUnion(MemoryLayout... elements) { + static GroupLayout unionLayout(MemoryLayout... elements) { Objects.requireNonNull(elements); return new GroupLayout(GroupLayout.Kind.UNION, Stream.of(elements) diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayouts.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayouts.java index 70a8edbcee8..4aee085821f 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayouts.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayouts.java @@ -46,87 +46,87 @@ public final class MemoryLayouts { /** * A value layout constant with size of one byte, and byte order set to {@link ByteOrder#LITTLE_ENDIAN}. */ - public static final ValueLayout BITS_8_LE = MemoryLayout.ofValueBits(8, ByteOrder.LITTLE_ENDIAN); + public static final ValueLayout BITS_8_LE = MemoryLayout.valueLayout(8, ByteOrder.LITTLE_ENDIAN); /** * A value layout constant with size of two bytes, and byte order set to {@link ByteOrder#LITTLE_ENDIAN}. */ - public static final ValueLayout BITS_16_LE = MemoryLayout.ofValueBits(16, ByteOrder.LITTLE_ENDIAN); + public static final ValueLayout BITS_16_LE = MemoryLayout.valueLayout(16, ByteOrder.LITTLE_ENDIAN); /** * A value layout constant with size of four bytes, and byte order set to {@link ByteOrder#LITTLE_ENDIAN}. */ - public static final ValueLayout BITS_32_LE = MemoryLayout.ofValueBits(32, ByteOrder.LITTLE_ENDIAN); + public static final ValueLayout BITS_32_LE = MemoryLayout.valueLayout(32, ByteOrder.LITTLE_ENDIAN); /** * A value layout constant with size of eight bytes, and byte order set to {@link ByteOrder#LITTLE_ENDIAN}. */ - public static final ValueLayout BITS_64_LE = MemoryLayout.ofValueBits(64, ByteOrder.LITTLE_ENDIAN); + public static final ValueLayout BITS_64_LE = MemoryLayout.valueLayout(64, ByteOrder.LITTLE_ENDIAN); /** * A value layout constant with size of one byte, and byte order set to {@link ByteOrder#BIG_ENDIAN}. */ - public static final ValueLayout BITS_8_BE = MemoryLayout.ofValueBits(8, ByteOrder.BIG_ENDIAN); + public static final ValueLayout BITS_8_BE = MemoryLayout.valueLayout(8, ByteOrder.BIG_ENDIAN); /** * A value layout constant with size of two bytes, and byte order set to {@link ByteOrder#BIG_ENDIAN}. */ - public static final ValueLayout BITS_16_BE = MemoryLayout.ofValueBits(16, ByteOrder.BIG_ENDIAN); + public static final ValueLayout BITS_16_BE = MemoryLayout.valueLayout(16, ByteOrder.BIG_ENDIAN); /** * A value layout constant with size of four bytes, and byte order set to {@link ByteOrder#BIG_ENDIAN}. */ - public static final ValueLayout BITS_32_BE = MemoryLayout.ofValueBits(32, ByteOrder.BIG_ENDIAN); + public static final ValueLayout BITS_32_BE = MemoryLayout.valueLayout(32, ByteOrder.BIG_ENDIAN); /** * A value layout constant with size of eight bytes, and byte order set to {@link ByteOrder#BIG_ENDIAN}. */ - public static final ValueLayout BITS_64_BE = MemoryLayout.ofValueBits(64, ByteOrder.BIG_ENDIAN); + public static final ValueLayout BITS_64_BE = MemoryLayout.valueLayout(64, ByteOrder.BIG_ENDIAN); /** * A padding layout constant with size of one byte. */ - public static final MemoryLayout PAD_8 = MemoryLayout.ofPaddingBits(8); + public static final MemoryLayout PAD_8 = MemoryLayout.paddingLayout(8); /** * A padding layout constant with size of two bytes. */ - public static final MemoryLayout PAD_16 = MemoryLayout.ofPaddingBits(16); + public static final MemoryLayout PAD_16 = MemoryLayout.paddingLayout(16); /** * A padding layout constant with size of four bytes. */ - public static final MemoryLayout PAD_32 = MemoryLayout.ofPaddingBits(32); + public static final MemoryLayout PAD_32 = MemoryLayout.paddingLayout(32); /** * A padding layout constant with size of eight bytes. */ - public static final MemoryLayout PAD_64 = MemoryLayout.ofPaddingBits(64); + public static final MemoryLayout PAD_64 = MemoryLayout.paddingLayout(64); /** * A value layout constant whose size is the same as that of a machine address (e.g. {@code size_t}), and byte order set to {@link ByteOrder#nativeOrder()}. */ - public static final ValueLayout ADDRESS = MemoryLayout.ofValueBits(Unsafe.ADDRESS_SIZE * 8, ByteOrder.nativeOrder()); + public static final ValueLayout ADDRESS = MemoryLayout.valueLayout(Unsafe.ADDRESS_SIZE * 8, ByteOrder.nativeOrder()); /** * A value layout constant whose size is the same as that of a Java {@code byte}, and byte order set to {@link ByteOrder#nativeOrder()}. */ - public static final ValueLayout JAVA_BYTE = MemoryLayout.ofValueBits(8, ByteOrder.nativeOrder()); + public static final ValueLayout JAVA_BYTE = MemoryLayout.valueLayout(8, ByteOrder.nativeOrder()); /** * A value layout constant whose size is the same as that of a Java {@code char}, and byte order set to {@link ByteOrder#nativeOrder()}. */ - public static final ValueLayout JAVA_CHAR = MemoryLayout.ofValueBits(16, ByteOrder.nativeOrder()); + public static final ValueLayout JAVA_CHAR = MemoryLayout.valueLayout(16, ByteOrder.nativeOrder()); /** * A value layout constant whose size is the same as that of a Java {@code short}, and byte order set to {@link ByteOrder#nativeOrder()}. */ - public static final ValueLayout JAVA_SHORT = MemoryLayout.ofValueBits(16, ByteOrder.nativeOrder()); + public static final ValueLayout JAVA_SHORT = MemoryLayout.valueLayout(16, ByteOrder.nativeOrder()); /** * A value layout constant whose size is the same as that of a Java {@code int}, and byte order set to {@link ByteOrder#nativeOrder()}. */ - public static final ValueLayout JAVA_INT = MemoryLayout.ofValueBits(32, ByteOrder.nativeOrder()); + public static final ValueLayout JAVA_INT = MemoryLayout.valueLayout(32, ByteOrder.nativeOrder()); /** * A value layout constant whose size is the same as that of a Java {@code long}, and byte order set to {@link ByteOrder#nativeOrder()}. @@ -136,13 +136,13 @@ public final class MemoryLayouts { MemoryLayouts.JAVA_LONG.byteAlignment() == MemoryLayouts.ADDRESS.byteSize(); * } */ - public static final ValueLayout JAVA_LONG = MemoryLayout.ofValueBits(64, ByteOrder.nativeOrder()) + public static final ValueLayout JAVA_LONG = MemoryLayout.valueLayout(64, ByteOrder.nativeOrder()) .withBitAlignment(ADDRESS.bitSize()); /** * A value layout constant whose size is the same as that of a Java {@code float}, and byte order set to {@link ByteOrder#nativeOrder()}. */ - public static final ValueLayout JAVA_FLOAT = MemoryLayout.ofValueBits(32, ByteOrder.nativeOrder()); + public static final ValueLayout JAVA_FLOAT = MemoryLayout.valueLayout(32, ByteOrder.nativeOrder()); /** * A value layout constant whose size is the same as that of a Java {@code double}, and byte order set to {@link ByteOrder#nativeOrder()}. @@ -152,6 +152,6 @@ public final class MemoryLayouts { MemoryLayouts.JAVA_DOUBLE.byteAlignment() == MemoryLayouts.ADDRESS.byteSize(); * } */ - public static final ValueLayout JAVA_DOUBLE = MemoryLayout.ofValueBits(64, ByteOrder.nativeOrder()) + public static final ValueLayout JAVA_DOUBLE = MemoryLayout.valueLayout(64, ByteOrder.nativeOrder()) .withBitAlignment(ADDRESS.bitSize()); } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java index a8bd56e4b61..c8b3fff6727 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java @@ -26,28 +26,29 @@ package jdk.incubator.foreign; -import java.io.FileDescriptor; -import java.lang.ref.Cleaner; +import java.io.UncheckedIOException; import java.nio.ByteBuffer; import jdk.internal.foreign.AbstractMemorySegmentImpl; import jdk.internal.foreign.HeapMemorySegmentImpl; import jdk.internal.foreign.MappedMemorySegmentImpl; +import jdk.internal.foreign.ResourceScopeImpl; import jdk.internal.foreign.NativeMemorySegmentImpl; -import jdk.internal.foreign.Utils; +import jdk.internal.reflect.CallerSensitive; +import jdk.internal.reflect.Reflection; import java.io.IOException; import java.nio.channels.FileChannel; import java.nio.file.Path; import java.util.Objects; -import java.util.Optional; import java.util.Spliterator; +import java.util.stream.Stream; /** * A memory segment models a contiguous region of memory. A memory segment is associated with both spatial - * and temporal bounds. Spatial bounds ensure that memory access operations on a memory segment cannot affect a memory location + * and temporal bounds (e.g. a {@link ResourceScope}). Spatial bounds ensure that memory access operations on a memory segment cannot affect a memory location * which falls outside the boundaries of the memory segment being accessed. Temporal bounds ensure that memory access - * operations on a segment cannot occur after a memory segment has been closed (see {@link MemorySegment#close()}). + * operations on a segment cannot occur after the resource scope associated with a memory segment has been closed (see {@link ResourceScope#close()}). *

    * All implementations of this interface must be value-based; * programmers should treat instances that are {@linkplain Object#equals(Object) equal} as interchangeable and should not @@ -62,8 +63,8 @@ import java.util.Spliterator; *

    Constructing memory segments

    * * There are multiple ways to obtain a memory segment. First, memory segments backed by off-heap memory can - * be allocated using one of the many factory methods provided (see {@link MemorySegment#allocateNative(MemoryLayout)}, - * {@link MemorySegment#allocateNative(long)} and {@link MemorySegment#allocateNative(long, long)}). Memory segments obtained + * be allocated using one of the many factory methods provided (see {@link MemorySegment#allocateNative(MemoryLayout, ResourceScope)}, + * {@link MemorySegment#allocateNative(long, ResourceScope)} and {@link MemorySegment#allocateNative(long, long, ResourceScope)}). Memory segments obtained * in this way are called native memory segments. *

    * It is also possible to obtain a memory segment backed by an existing heap-allocated Java array, @@ -77,161 +78,89 @@ import java.util.Spliterator; * depending on the characteristics of the byte buffer instance the segment is associated with. For instance, a buffer memory * segment obtained from a byte buffer created with the {@link ByteBuffer#allocateDirect(int)} method will be backed * by native memory. - *

    - * Finally, it is also possible to obtain a memory segment backed by a memory-mapped file using the factory method - * {@link MemorySegment#mapFile(Path, long, long, FileChannel.MapMode)}. Such memory segments are called mapped memory segments; - * mapped memory segments are associated with an underlying file descriptor. For more operations on mapped memory segments, please refer to the - * {@link MappedMemorySegments} class. - *

    - * Array and buffer segments are effectively views over existing memory regions which might outlive the - * lifecycle of the segments derived from them, and can even be manipulated directly (e.g. via array access, or direct use - * of the {@link ByteBuffer} API) by other clients. As a result, while sharing array or buffer segments is possible, - * it is strongly advised that clients wishing to do so take extra precautions to make sure that the underlying memory sources - * associated with such segments remain inaccessible, and that said memory sources are never aliased by more than one segment - * at a time - e.g. so as to prevent concurrent modifications of the contents of an array, or buffer segment. * - *

    Explicit deallocation

    + *

    Mapping memory segments from files

    * - * Memory segments are closed explicitly (see {@link MemorySegment#close()}). When a segment is closed, it is no longer - * alive (see {@link #isAlive()}, and subsequent operation on the segment (or on any {@link MemoryAddress} instance - * derived from it) will fail with {@link IllegalStateException}. + * It is also possible to obtain a native memory segment backed by a memory-mapped file using the factory method + * {@link MemorySegment#mapFile(Path, long, long, FileChannel.MapMode, ResourceScope)}. Such native memory segments are + * called mapped memory segments; mapped memory segments are associated with an underlying file descriptor. *

    - * Closing a segment might trigger the releasing of the underlying memory resources associated with said segment, depending on - * the kind of memory segment being considered: - *

      - *
    • closing a native memory segment results in freeing the native memory associated with it
    • - *
    • closing a mapped memory segment results in the backing memory-mapped file to be unmapped
    • - *
    • closing a buffer, or a heap segment does not have any side-effect, other than marking the segment - * as not alive (see {@link MemorySegment#isAlive()}). Also, since the buffer and heap segments might keep - * strong references to the original buffer or array instance, it is the responsibility of clients to ensure that - * these segments are discarded in a timely manner, so as not to prevent garbage collection to reclaim the underlying - * objects.
    • - *
    - * - *

    Access modes

    - * - * Memory segments supports zero or more access modes. Supported access modes are {@link #READ}, - * {@link #WRITE}, {@link #CLOSE}, {@link #SHARE} and {@link #HANDOFF}. The set of access modes supported by a segment alters the - * set of operations that are supported by that segment. For instance, attempting to call {@link #close()} on - * a segment which does not support the {@link #CLOSE} access mode will result in an exception. + * Contents of mapped memory segments can be {@linkplain #force() persisted} and {@linkplain #load() loaded} to and from the underlying file; + * these capabilities are suitable replacements for some of the functionality in the {@link java.nio.MappedByteBuffer} class. + * Note that, while it is possible to map a segment into a byte buffer (see {@link MemorySegment#asByteBuffer()}), + * and then call e.g. {@link java.nio.MappedByteBuffer#force()} that way, this can only be done when the source segment + * is small enough, due to the size limitation inherent to the ByteBuffer API. *

    - * The set of supported access modes can only be made stricter (by supporting fewer access modes). This means - * that restricting the set of access modes supported by a segment before sharing it with other clients - * is generally a good practice if the creator of the segment wants to retain some control over how the segment - * is going to be accessed. + * Clients requiring sophisticated, low-level control over mapped memory segments, should consider writing + * custom mapped memory segment factories; using {@link CLinker}, e.g. on Linux, it is possible to call {@code mmap} + * with the desired parameters; the returned address can be easily wrapped into a memory segment, using + * {@link MemoryAddress#ofLong(long)} and {@link MemoryAddress#asSegment(long, Runnable, ResourceScope)}. + * + *

    Lifecycle and confinement

    + * + * Memory segments are associated with a resource scope (see {@link ResourceScope}), which can be accessed using + * the {@link #scope()} method. As for all resources associated with a resource scope, a segment cannot be + * accessed after its corresponding scope has been closed. For instance, the following code will result in an + * exception: + *
    {@code
    +MemorySegment segment = null;
    +try (ResourceScope scope = ResourceScope.newConfinedScope()) {
    +    segment = MemorySegment.allocateNative(8, 1, scope);
    +}
    +MemoryAccess.getLong(segment); // already closed!
    + * }
    + * Additionally, access to a memory segment is subject to the thread-confinement checks enforced by the owning scope; that is, + * if the segment is associated with a shared scope, it can be accessed by multiple threads; if it is associated with a confined + * scope, it can only be accessed by the thread which owns the scope. + *

    + * Heap and buffer segments are always associated with a global, shared scope. This scope cannot be closed, + * and can be considered as always alive. * *

    Memory segment views

    * - * Memory segments support views. For instance, it is possible to alter the set of supported access modes, - * by creating an immutable view of a memory segment, as follows: + * Memory segments support views. For instance, it is possible to create an immutable view of a memory segment, as follows: *
    {@code
     MemorySegment segment = ...
    -MemorySegment roSegment = segment.withAccessModes(segment.accessModes() & ~WRITE);
    +MemorySegment roSegment = segment.asReadOnly();
      * }
    * It is also possible to create views whose spatial bounds are stricter than the ones of the original segment * (see {@link MemorySegment#asSlice(long, long)}). *

    - * Temporal bounds of the original segment are inherited by the view; that is, closing a segment view, such as a sliced - * view, will cause the original segment to be closed; as such special care must be taken when sharing views - * between multiple clients. If a client want to protect itself against early closure of a segment by - * another actor, it is the responsibility of that client to take protective measures, such as removing {@link #CLOSE} - * from the set of supported access modes, before sharing the view with another client. + * Temporal bounds of the original segment are inherited by the views; that is, when the scope associated with a segment + * is closed, all the views associated with that segment will also be rendered inaccessible. *

    * To allow for interoperability with existing code, a byte buffer view can be obtained from a memory segment * (see {@link #asByteBuffer()}). This can be useful, for instance, for those clients that want to keep using the * {@link ByteBuffer} API, but need to operate on large memory segments. Byte buffers obtained in such a way support - * the same spatial and temporal access restrictions associated to the memory segment from which they originated. + * the same spatial and temporal access restrictions associated with the memory segment from which they originated. * - *

    Thread confinement

    + *

    Stream support

    * - * Memory segments support strong thread-confinement guarantees. Upon creation, they are assigned an owner thread, - * typically the thread which initiated the creation operation. After creation, only the owner thread will be allowed - * to directly manipulate the memory segment (e.g. close the memory segment) or access the underlying memory associated with - * the segment using a memory access var handle. Any attempt to perform such operations from a thread other than the - * owner thread will result in a runtime failure. - *

    - * The {@link #handoff(Thread)} method can be used to change the thread-confinement properties of a memory segment. - * This method is, like {@link #close()}, a terminal operation which marks the original segment as not alive - * (see {@link #isAlive()}) and creates a new segment with the desired thread-confinement properties. Calling - * {@link #handoff(Thread)} is only possible if the segment features the corresponding {@link #HANDOFF} access mode. - *

    - * For instance, if a client wants to transfer ownership of a segment to another (known) thread, it can do so as follows: + * A client might obtain a {@link Stream} from a segment, which can then be used to slice the segment (according to a given + * element layout) and even allow multiple threads to work in parallel on disjoint segment slices + * (to do this, the segment has to be associated with a shared scope). The following code can be used to sum all int + * values in a memory segment in parallel: * *

    {@code
    -MemorySegment segment = ...
    -MemorySegment aSegment = segment.handoff(threadA);
    - * }
    - * - * By doing so, the original segment is marked as not alive, and a new segment is returned whose owner thread - * is {@code threadA}; this allows, for instance, for two threads {@code A} and {@code B} to share - * a segment in a controlled, cooperative and race-free fashion (also known as serial thread confinement). - *

    - * Alternatively, the {@link #share()} method can be used to remove thread ownership altogether; this is only possible - * if the segment features the corresponding {@link #SHARE} access mode. The following code shows how clients can - * obtain a shared segment: - * - *

    {@code
    -MemorySegment segment = ...
    -MemorySegment sharedSegment = segment.share();
    - * }
    - * - * Again here, the original segment is marked as not alive, and a new shared segment is returned which features no owner - * thread (e.g. {@link #ownerThread()} returns {@code null}). This might be useful when multiple threads need to process - * the contents of the same memory segment concurrently (e.g. in the case of parallel processing). For instance, a client - * might obtain a {@link Spliterator} from a shared segment, which can then be used to slice the segment and allow multiple - * threads to work in parallel on disjoint segment slices. The following code can be used to sum all int values in a memory segment in parallel: - * - *
    {@code
    -SequenceLayout SEQUENCE_LAYOUT = MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_INT);
    -try (MemorySegment segment = MemorySegment.allocateNative(SEQUENCE_LAYOUT).share()) {
    +try (ResourceScope scope = ResourceScope.newSharedScope()) {
    +    SequenceLayout SEQUENCE_LAYOUT = MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_INT);
    +    MemorySegment segment = MemorySegment.allocateNative(SEQUENCE_LAYOUT, scope);
         VarHandle VH_int = SEQUENCE_LAYOUT.elementLayout().varHandle(int.class);
    -    int sum = StreamSupport.stream(segment.spliterator(SEQUENCE_LAYOUT), true)
    +    int sum = segment.elements(MemoryLayouts.JAVA_INT).parallel()
                                .mapToInt(s -> (int)VH_int.get(s.address()))
                                .sum();
     }
      * }
    * - * Once shared, a segment can be claimed back by a given thread (again using {@link #handoff(Thread)}); in fact, many threads - * can attempt to gain ownership of the same segment, concurrently, and only one of them is guaranteed to succeed. - *

    - * When using shared segments, clients should make sure that no other thread is accessing the segment while - * the segment is being closed. If one or more threads attempts to access a segment concurrently while the - * segment is being closed, an exception might occur on both the accessing and the closing threads. Clients should - * refrain from attempting to close a segment repeatedly (e.g. keep calling {@link #close()} until no exception is thrown); - * such exceptions should instead be seen as an indication that the client code is lacking appropriate synchronization between the threads - * accessing/closing the segment. - * - *

    Implicit deallocation

    - * - * Clients can register a memory segment against a {@link Cleaner}, to make sure that underlying resources associated with - * that segment will be released when the segment becomes unreachable, which can be useful to prevent native memory - * leaks. This can be achieved using the {@link #registerCleaner(Cleaner)} method, as follows: - * - *
    {@code
    -MemorySegment segment = ...
    -MemorySegment gcSegment = segment.registerCleaner(cleaner);
    - * }
    - * - * Here, the original segment is marked as not alive, and a new segment is returned (the owner thread of the returned - * segment set is set to that of the current thread, see {@link #ownerThread()}); the new segment - * will also be registered with the the {@link Cleaner} instance provided to the {@link #registerCleaner(Cleaner)} method; - * as such, if not closed explicitly (see {@link #close()}), the new segment will be automatically closed by the cleaner. - * - * @apiNote In the future, if the Java language permits, {@link MemorySegment} - * may become a {@code sealed} interface, which would prohibit subclassing except by other explicitly permitted subtypes. - * * @implSpec * Implementations of this interface are immutable, thread-safe and value-based. */ -public interface MemorySegment extends Addressable, AutoCloseable { +public sealed interface MemorySegment extends Addressable permits AbstractMemorySegmentImpl { /** - * The base memory address associated with this memory segment. The returned address is - * a checked memory address and can therefore be used in dereference operations - * (see {@link MemoryAddress}). + * The base memory address associated with this memory segment. + * The returned memory address is associated with same resource scope as that associated with this segment. * @return The base memory address. - * @throws IllegalStateException if this segment is not alive, or if access occurs from a thread other than the - * thread owning this segment */ @Override MemoryAddress address(); @@ -241,27 +170,41 @@ public interface MemorySegment extends Addressable, AutoCloseable { * {@link Spliterator#SUBSIZED}, {@link Spliterator#IMMUTABLE}, {@link Spliterator#NONNULL} and {@link Spliterator#ORDERED} * characteristics. *

    - * The returned spliterator splits this segment according to the specified sequence layout; that is, - * if the supplied layout is a sequence layout whose element count is {@code N}, then calling {@link Spliterator#trySplit()} - * will result in a spliterator serving approximatively {@code N/2} elements (depending on whether N is even or not). - * As such, splitting is possible as long as {@code N >= 2}. The spliterator returns segments that feature the same - * access modes as the given segment less the {@link #CLOSE} access mode. + * The returned spliterator splits this segment according to the specified element layout; that is, + * if the supplied layout has size N, then calling {@link Spliterator#trySplit()} will result in a spliterator serving + * approximately {@code S/N/2} elements (depending on whether N is even or not), where {@code S} is the size of + * this segment. As such, splitting is possible as long as {@code S/N >= 2}. The spliterator returns segments that feature the same + * scope as this given segment. *

    * The returned spliterator effectively allows to slice this segment into disjoint sub-segments, which can then - * be processed in parallel by multiple threads (if the segment is shared). + * be processed in parallel by multiple threads. * - * @param layout the layout to be used for splitting. + * @param elementLayout the layout to be used for splitting. * @return the element spliterator for this segment - * @throws IllegalStateException if the segment is not alive, or if access occurs from a thread other than the - * thread owning this segment + * @throws IllegalArgumentException if the {@code elementLayout} size is zero, or the segment size modulo the + * {@code elementLayout} size is greater than zero. */ - Spliterator spliterator(SequenceLayout layout); + Spliterator spliterator(MemoryLayout elementLayout); /** - * The thread owning this segment. - * @return the thread owning this segment. + * Returns a sequential {@code Stream} over disjoint slices (whose size matches that of the specified layout) + * in this segment. Calling this method is equivalent to the following code: + *

    {@code
    +    StreamSupport.stream(segment.spliterator(elementLayout), false);
    +     * }
    + * + * @param elementLayout the layout to be used for splitting. + * @return a sequential {@code Stream} over disjoint slices in this segment. + * @throws IllegalArgumentException if the {@code elementLayout} size is zero, or the segment size modulo the + * {@code elementLayout} size is greater than zero. */ - Thread ownerThread(); + Stream elements(MemoryLayout elementLayout); + + /** + * Returns the resource scope associated with this memory segment. + * @return the resource scope associated with this memory segment. + */ + ResourceScope scope(); /** * The size (in bytes) of this memory segment. @@ -269,33 +212,6 @@ public interface MemorySegment extends Addressable, AutoCloseable { */ long byteSize(); - /** - * Obtains a segment view with specific access modes. Supported access modes are {@link #READ}, {@link #WRITE}, - * {@link #CLOSE}, {@link #SHARE} and {@link #HANDOFF}. It is generally not possible to go from a segment with stricter access modes - * to one with less strict access modes. For instance, attempting to add {@link #WRITE} access mode to a read-only segment - * will be met with an exception. - * @param accessModes an ORed mask of zero or more access modes. - * @return a segment view with specific access modes. - * @throws IllegalArgumentException when {@code mask} is an access mask which is less strict than the one supported by this - * segment, or when {@code mask} contains bits not associated with any of the supported access modes. - */ - MemorySegment withAccessModes(int accessModes); - - /** - * Does this segment support a given set of access modes? - * @param accessModes an ORed mask of zero or more access modes. - * @return true, if the access modes in {@code accessModes} are stricter than the ones supported by this segment. - * @throws IllegalArgumentException when {@code mask} contains bits not associated with any of the supported access modes. - */ - boolean hasAccessModes(int accessModes); - - /** - * Returns the access modes associated with this segment; the result is represented as ORed values from - * {@link #READ}, {@link #WRITE}, {@link #CLOSE}, {@link #SHARE} and {@link #HANDOFF}. - * @return the access modes associated with this segment. - */ - int accessModes(); - /** * Obtains a new memory segment view whose base address is the same as the base address of this segment plus a given offset, * and whose new size is specified by the given argument. @@ -376,135 +292,38 @@ public interface MemorySegment extends Addressable, AutoCloseable { return asSlice(newBase.segmentOffset(this)); } + /** + * Is this segment read-only? + * @return {@code true}, if this segment is read-only. + * @see #asReadOnly() + */ + boolean isReadOnly(); + + /** + * Obtains a read-only view of this segment. The resulting segment will be identical to this one, but + * attempts to overwrite the contents of the returned segment will cause runtime exceptions. + * @return a read-only view of this segment + * @see #isReadOnly() + */ + MemorySegment asReadOnly(); + + /** + * Is this a native segment? Returns true if this segment is a native memory segment, + * created using the {@link #allocateNative(long, ResourceScope)} (and related) factory, or a buffer segment + * derived from a direct {@link java.nio.ByteBuffer} using the {@link #ofByteBuffer(ByteBuffer)} factory, + * or if this is a {@linkplain #isMapped() mapped} segment. + * @return {@code true} if this segment is native segment. + */ + boolean isNative(); + /** * Is this a mapped segment? Returns true if this segment is a mapped memory segment, - * created using the {@link #mapFile(Path, long, long, FileChannel.MapMode)} factory, or a buffer segment + * created using the {@link #mapFile(Path, long, long, FileChannel.MapMode, ResourceScope)} factory, or a buffer segment * derived from a {@link java.nio.MappedByteBuffer} using the {@link #ofByteBuffer(ByteBuffer)} factory. * @return {@code true} if this segment is a mapped segment. */ boolean isMapped(); - /** - * Is this segment alive? - * @return true, if the segment is alive. - * @see MemorySegment#close() - */ - boolean isAlive(); - - /** - * Closes this memory segment. This is a terminal operation; as a side-effect, if this operation completes - * without exceptions, this segment will be marked as not alive, and subsequent operations on this segment - * will fail with {@link IllegalStateException}. - *

    - * Depending on the kind of memory segment being closed, calling this method further triggers deallocation of all the resources - * associated with the memory segment. - * - * @apiNote This operation is not idempotent; that is, closing an already closed segment always results in an - * exception being thrown. This reflects a deliberate design choice: segment state transitions should be - * manifest in the client code; a failure in any of these transitions reveals a bug in the underlying application - * logic. This is especially useful when reasoning about the lifecycle of dependent segment views (see {@link #asSlice(MemoryAddress)}, - * where closing one segment might side-effect multiple segments. In such cases it might in fact not be obvious, looking - * at the code, as to whether a given segment is alive or not. - * - * @throws IllegalStateException if this segment is not alive, or if access occurs from a thread other than the - * thread owning this segment, or if this segment is shared and the segment is concurrently accessed while this method is - * called. - * @throws UnsupportedOperationException if this segment does not support the {@link #CLOSE} access mode. - */ - void close(); - - /** - * Obtains a new confined memory segment backed by the same underlying memory region as this segment. The returned segment will - * be confined on the specified thread, and will feature the same spatial bounds and access modes (see {@link #accessModes()}) - * as this segment. - *

    - * This is a terminal operation; as a side-effect, if this operation completes - * without exceptions, this segment will be marked as not alive, and subsequent operations on this segment - * will fail with {@link IllegalStateException}. - *

    - * In case where the owner thread of the returned segment differs from that of this segment, write accesses to this - * segment's content happens-before - * hand-over from the current owner thread to the new owner thread, which in turn happens before read accesses - * to the returned segment's contents on the new owner thread. - * - * @param thread the new owner thread - * @return a new confined memory segment whose owner thread is set to {@code thread}. - * @throws IllegalStateException if this segment is not alive, or if access occurs from a thread other than the - * thread owning this segment. - * @throws UnsupportedOperationException if this segment does not support the {@link #HANDOFF} access mode. - */ - MemorySegment handoff(Thread thread); - - /** - * Obtains a new confined memory segment backed by the same underlying memory region as this segment, but whose - * temporal bounds are controlled by the provided {@link NativeScope} instance. - *

    - * This is a terminal operation; - * as a side-effect, this segment will be marked as not alive, and subsequent operations on this segment - * will fail with {@link IllegalStateException}. - *

    - * The returned segment will feature only {@link MemorySegment#READ} and {@link MemorySegment#WRITE} access modes - * (assuming these were available in the original segment). As such the returned segment cannot be closed directly - * using {@link MemorySegment#close()} - but it will be closed indirectly when this native scope is closed. The - * returned segment will also be confined by the same thread as the provided native scope (see {@link NativeScope#ownerThread()}). - *

    - * In case where the owner thread of the returned segment differs from that of this segment, write accesses to this - * segment's content happens-before - * hand-over from the current owner thread to the new owner thread, which in turn happens before read accesses - * to the returned segment's contents on the new owner thread. - * - * @param nativeScope the native scope. - * @return a new confined memory segment backed by the same underlying memory region as this segment, but whose life-cycle - * is tied to that of {@code nativeScope}. - * @throws IllegalStateException if this segment is not alive, or if access occurs from a thread other than the - * thread owning this segment. - * @throws UnsupportedOperationException if this segment does not support the {@link #HANDOFF} access mode. - */ - MemorySegment handoff(NativeScope nativeScope); - - /** - * Obtains a new shared memory segment backed by the same underlying memory region as this segment. The returned segment will - * not be confined on any thread and can therefore be accessed concurrently from multiple threads; moreover, the - * returned segment will feature the same spatial bounds and access modes (see {@link #accessModes()}) - * as this segment. - *

    - * This is a terminal operation; as a side-effect, if this operation completes - * without exceptions, this segment will be marked as not alive, and subsequent operations on this segment - * will fail with {@link IllegalStateException}. - *

    - * Write accesses to this segment's content happens-before - * hand-over from the current owner thread to the new owner thread, which in turn happens before read accesses - * to the returned segment's contents on a new thread. - * - * @return a new memory shared segment backed by the same underlying memory region as this segment. - * @throws IllegalStateException if this segment is not alive, or if access occurs from a thread other than the - * thread owning this segment. - */ - MemorySegment share(); - - /** - * Register this memory segment instance against a {@link Cleaner} object, by returning a new memory segment backed - * by the same underlying memory region as this segment. The returned segment will feature the same confinement, - * spatial bounds and access modes (see {@link #accessModes()}) as this segment. Moreover, the returned segment - * will be associated with the specified {@link Cleaner} object; this allows for the segment to be closed - * as soon as it becomes unreachable, which might be helpful in preventing native memory leaks. - *

    - * This is a terminal operation; as a side-effect, if this operation completes - * without exceptions, this segment will be marked as not alive, and subsequent operations on this segment - * will fail with {@link IllegalStateException}. - *

    - * The implicit deallocation behavior associated with the returned segment will be preserved under terminal - * operations such as {@link #handoff(Thread)} and {@link #share()}. - * - * @param cleaner the cleaner object, responsible for implicit deallocation of the returned segment. - * @return a new memory segment backed by the same underlying memory region as this segment, which features - * implicit deallocation. - * @throws IllegalStateException if this segment is not alive, or if access occurs from a thread other than the - * thread owning this segment, or if this segment is already associated with a cleaner. - * @throws UnsupportedOperationException if this segment does not support the {@link #CLOSE} access mode. - */ - MemorySegment registerCleaner(Cleaner cleaner); - /** * Fills a value into this memory segment. *

    @@ -526,9 +345,9 @@ for (long l = 0; l < segment.byteSize(); l++) { * * @param value the value to fill into this segment * @return this memory segment - * @throws IllegalStateException if this segment is not alive, or if access occurs from a thread other than the - * thread owning this segment - * @throws UnsupportedOperationException if this segment does not support the {@link #WRITE} access mode + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope, + * @throws UnsupportedOperationException if this segment is read-only (see {@link #isReadOnly()}). */ MemorySegment fill(byte value); @@ -543,23 +362,22 @@ for (long l = 0; l < segment.byteSize(); l++) { *

    * The result of a bulk copy is unspecified if, in the uncommon case, the source segment and this segment * do not overlap, but refer to overlapping regions of the same backing storage using different addresses. - * For example, this may occur if the same file is {@link MemorySegment#mapFile mapped} to two segments. + * For example, this may occur if the same file is {@linkplain MemorySegment#mapFile mapped} to two segments. * * @param src the source segment. * @throws IndexOutOfBoundsException if {@code src.byteSize() > this.byteSize()}. - * @throws IllegalStateException if either the source segment or this segment have been already closed, - * or if access occurs from a thread other than the thread owning either segment. - * @throws UnsupportedOperationException if either the source segment or this segment do not feature required access modes; - * more specifically, {@code src} should feature at least the {@link MemorySegment#READ} access mode, - * while this segment should feature at least the {@link MemorySegment#WRITE} access mode. + * @throws IllegalStateException if either the scope associated with the source segment or the scope associated + * with this segment have been already closed, or if access occurs from a thread other than the thread owning either + * scopes. + * @throws UnsupportedOperationException if this segment is read-only (see {@link #isReadOnly()}). */ void copyFrom(MemorySegment src); /** * Finds and returns the offset, in bytes, of the first mismatch between * this segment and a given other segment. The offset is relative to the - * {@link #address() base address} of each segment and will be in the - * range of 0 (inclusive) up to the {@link #byteSize() size} (in bytes) of + * {@linkplain #address() base address} of each segment and will be in the + * range of 0 (inclusive) up to the {@linkplain #byteSize() size} (in bytes) of * the smaller memory segment (exclusive). *

    * If the two segments share a common prefix then the returned offset is @@ -573,18 +391,99 @@ for (long l = 0; l < segment.byteSize(); l++) { * @param other the segment to be tested for a mismatch with this segment * @return the relative offset, in bytes, of the first mismatch between this * and the given other segment, otherwise -1 if no mismatch - * @throws IllegalStateException if either this segment of the other segment - * have been already closed, or if access occurs from a thread other than the - * thread owning either segment - * @throws UnsupportedOperationException if either this segment or the other - * segment does not feature at least the {@link MemorySegment#READ} access mode + * @throws IllegalStateException if either the scope associated with this segment or the scope associated + * with the {@code other} segment have been already closed, or if access occurs from a thread other than the thread + * owning either scopes. */ long mismatch(MemorySegment other); + /** + * Tells whether or not the contents of this mapped segment is resident in physical + * memory. + * + *

    A return value of {@code true} implies that it is highly likely + * that all of the data in this segment is resident in physical memory and + * may therefore be accessed without incurring any virtual-memory page + * faults or I/O operations. A return value of {@code false} does not + * necessarily imply that this segment's content is not resident in physical + * memory. + * + *

    The returned value is a hint, rather than a guarantee, because the + * underlying operating system may have paged out some of this segment's data + * by the time that an invocation of this method returns.

    + * + * @return {@code true} if it is likely that the contents of this segment + * is resident in physical memory + * + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope. + * @throws UnsupportedOperationException if this segment is not a mapped memory segment, e.g. if + * {@code isMapped() == false}. + */ + boolean isLoaded(); + + /** + * Loads the contents of this mapped segment into physical memory. + * + *

    This method makes a best effort to ensure that, when it returns, + * this contents of this segment is resident in physical memory. Invoking this + * method may cause some number of page faults and I/O operations to + * occur.

    + * + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope. + * @throws UnsupportedOperationException if this segment is not a mapped memory segment, e.g. if + * {@code isMapped() == false}. + */ + void load(); + + /** + * Unloads the contents of this mapped segment from physical memory. + * + *

    This method makes a best effort to ensure that the contents of this segment are + * are no longer resident in physical memory. Accessing this segment's contents + * after invoking this method may cause some number of page faults and I/O operations to + * occur (as this segment's contents might need to be paged back in).

    + * + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope. + * @throws UnsupportedOperationException if this segment is not a mapped memory segment, e.g. if + * {@code isMapped() == false}. + */ + void unload(); + + /** + * Forces any changes made to the contents of this mapped segment to be written to the + * storage device described by the mapped segment's file descriptor. + * + *

    If the file descriptor associated with this mapped segment resides on a local storage + * device then when this method returns it is guaranteed that all changes + * made to this segment since it was created, or since this method was last + * invoked, will have been written to that device. + * + *

    If the file descriptor associated with this mapped segment does not reside on a local device then + * no such guarantee is made. + * + *

    If this segment was not mapped in read/write mode ({@link + * java.nio.channels.FileChannel.MapMode#READ_WRITE}) then + * invoking this method may have no effect. In particular, the + * method has no effect for segments mapped in read-only or private + * mapping modes. This method may or may not have an effect for + * implementation-specific mapping modes. + *

    + * + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope. + * @throws UnsupportedOperationException if this segment is not a mapped memory segment, e.g. if + * {@code isMapped() == false}. + * @throws UncheckedIOException if there is an I/O error writing the contents of this segment to the associated storage device + */ + void force(); + /** * Wraps this segment in a {@link ByteBuffer}. Some of the properties of the returned buffer are linked to * the properties of this segment. For instance, if this segment is immutable - * (e.g. the segment has access mode {@link #READ} but not {@link #WRITE}), then the resulting buffer is read-only + * (e.g. the segment is a read-only segment, see {@link #isReadOnly()}), then the resulting buffer is read-only * (see {@link ByteBuffer#isReadOnly()}. Additionally, if this is a native memory segment, the resulting buffer is * direct (see {@link ByteBuffer#isDirect()}). *

    @@ -593,14 +492,13 @@ for (long l = 0; l < segment.byteSize(); l++) { * are set to this segment' size (see {@link MemorySegment#byteSize()}). For this reason, a byte buffer cannot be * returned if this segment' size is greater than {@link Integer#MAX_VALUE}. *

    - * The life-cycle of the returned buffer will be tied to that of this segment. That means that if the this segment - * is closed (see {@link MemorySegment#close()}, accessing the returned - * buffer will throw an {@link IllegalStateException}. + * The life-cycle of the returned buffer will be tied to that of this segment. That is, accessing the returned buffer + * after the scope associated with this segment has been closed (see {@link ResourceScope#close()}, will throw an {@link IllegalStateException}. *

    - * If this segment is shared, calling certain I/O operations on the resulting buffer might result in - * an unspecified exception being thrown. Examples of such problematic operations are {@link FileChannel#read(ByteBuffer)}, - * {@link FileChannel#write(ByteBuffer)}, {@link java.nio.channels.SocketChannel#read(ByteBuffer)} and - * {@link java.nio.channels.SocketChannel#write(ByteBuffer)}. + * If this segment is associated with a confined scope, calling read/write I/O operations on the resulting buffer + * might result in an unspecified exception being thrown. Examples of such problematic operations are + * {@link java.nio.channels.AsynchronousSocketChannel#read(ByteBuffer)} and + * {@link java.nio.channels.AsynchronousSocketChannel#write(ByteBuffer)}. *

    * Finally, the resulting buffer's byte order is {@link java.nio.ByteOrder#BIG_ENDIAN}; this can be changed using * {@link ByteBuffer#order(java.nio.ByteOrder)}. @@ -608,83 +506,70 @@ for (long l = 0; l < segment.byteSize(); l++) { * @return a {@link ByteBuffer} view of this memory segment. * @throws UnsupportedOperationException if this segment cannot be mapped onto a {@link ByteBuffer} instance, * e.g. because it models an heap-based segment that is not based on a {@code byte[]}), or if its size is greater - * than {@link Integer#MAX_VALUE}, or if the segment does not support the {@link #READ} access mode. + * than {@link Integer#MAX_VALUE}. */ ByteBuffer asByteBuffer(); /** * Copy the contents of this memory segment into a fresh byte array. * @return a fresh byte array copy of this memory segment. - * @throws UnsupportedOperationException if this segment does not feature the {@link #READ} access mode, or if this - * segment's contents cannot be copied into a {@link byte[]} instance, e.g. its size is greater than {@link Integer#MAX_VALUE}, - * @throws IllegalStateException if this segment has been closed, or if access occurs from a thread other than the - * thread owning this segment. + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope, or if this segment's contents cannot be copied into a {@link byte[]} instance, + * e.g. its size is greater than {@link Integer#MAX_VALUE}. */ byte[] toByteArray(); /** * Copy the contents of this memory segment into a fresh short array. * @return a fresh short array copy of this memory segment. - * @throws UnsupportedOperationException if this segment does not feature the {@link #READ} access mode, or if this - * segment's contents cannot be copied into a {@link short[]} instance, e.g. because {@code byteSize() % 2 != 0}, - * or {@code byteSize() / 2 > Integer#MAX_VALUE}. - * @throws IllegalStateException if this segment has been closed, or if access occurs from a thread other than the - * thread owning this segment. + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope, or if this segment's contents cannot be copied into a {@link short[]} instance, + * e.g. because {@code byteSize() % 2 != 0}, or {@code byteSize() / 2 > Integer#MAX_VALUE} */ short[] toShortArray(); /** * Copy the contents of this memory segment into a fresh char array. * @return a fresh char array copy of this memory segment. - * @throws UnsupportedOperationException if this segment does not feature the {@link #READ} access mode, or if this - * segment's contents cannot be copied into a {@link char[]} instance, e.g. because {@code byteSize() % 2 != 0}, - * or {@code byteSize() / 2 > Integer#MAX_VALUE}. - * @throws IllegalStateException if this segment has been closed, or if access occurs from a thread other than the - * thread owning this segment. + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope, or if this segment's contents cannot be copied into a {@link char[]} instance, + * e.g. because {@code byteSize() % 2 != 0}, or {@code byteSize() / 2 > Integer#MAX_VALUE}. */ char[] toCharArray(); /** * Copy the contents of this memory segment into a fresh int array. * @return a fresh int array copy of this memory segment. - * @throws UnsupportedOperationException if this segment does not feature the {@link #READ} access mode, or if this - * segment's contents cannot be copied into a {@link int[]} instance, e.g. because {@code byteSize() % 4 != 0}, - * or {@code byteSize() / 4 > Integer#MAX_VALUE}. - * @throws IllegalStateException if this segment has been closed, or if access occurs from a thread other than the - * thread owning this segment. + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope, or if this segment's contents cannot be copied into a {@link int[]} instance, + * e.g. because {@code byteSize() % 4 != 0}, or {@code byteSize() / 4 > Integer#MAX_VALUE}. */ int[] toIntArray(); /** * Copy the contents of this memory segment into a fresh float array. * @return a fresh float array copy of this memory segment. - * @throws UnsupportedOperationException if this segment does not feature the {@link #READ} access mode, or if this - * segment's contents cannot be copied into a {@link float[]} instance, e.g. because {@code byteSize() % 4 != 0}, - * or {@code byteSize() / 4 > Integer#MAX_VALUE}. - * @throws IllegalStateException if this segment has been closed, or if access occurs from a thread other than the - * thread owning this segment. + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope, or if this segment's contents cannot be copied into a {@link float[]} instance, + * e.g. because {@code byteSize() % 4 != 0}, or {@code byteSize() / 4 > Integer#MAX_VALUE}. */ float[] toFloatArray(); /** * Copy the contents of this memory segment into a fresh long array. * @return a fresh long array copy of this memory segment. - * @throws UnsupportedOperationException if this segment does not feature the {@link #READ} access mode, or if this - * segment's contents cannot be copied into a {@link long[]} instance, e.g. because {@code byteSize() % 8 != 0}, - * or {@code byteSize() / 8 > Integer#MAX_VALUE}. - * @throws IllegalStateException if this segment has been closed, or if access occurs from a thread other than the - * thread owning this segment. + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope, or if this segment's contents cannot be copied into a {@link long[]} instance, + * e.g. because {@code byteSize() % 8 != 0}, or {@code byteSize() / 8 > Integer#MAX_VALUE}. */ long[] toLongArray(); /** * Copy the contents of this memory segment into a fresh double array. * @return a fresh double array copy of this memory segment. - * @throws UnsupportedOperationException if this segment does not feature the {@link #READ} access mode, or if this - * segment's contents cannot be copied into a {@link double[]} instance, e.g. because {@code byteSize() % 8 != 0}, - * or {@code byteSize() / 8 > Integer#MAX_VALUE}. - * @throws IllegalStateException if this segment has been closed, or if access occurs from a thread other than the - * thread owning this segment. + * @throws IllegalStateException if the scope associated with this segment has been closed, or if access occurs from + * a thread other than the thread owning that scope, or if this segment's contents cannot be copied into a {@link double[]} instance, + * e.g. because {@code byteSize() % 8 != 0}, or {@code byteSize() / 8 > Integer#MAX_VALUE}. */ double[] toDoubleArray(); @@ -693,15 +578,15 @@ for (long l = 0; l < segment.byteSize(); l++) { * buffer. The segment starts relative to the buffer's position (inclusive) * and ends relative to the buffer's limit (exclusive). *

    - * The segment will feature all access modes (see {@link #ALL_ACCESS}), - * unless the given buffer is {@linkplain ByteBuffer#isReadOnly() read-only} in which case the segment will - * not feature the {@link #WRITE} access mode, and its confinement thread is the current thread (see {@link Thread#currentThread()}). + * If the buffer is {@link ByteBuffer#isReadOnly() read-only}, the resulting segment will also be + * {@link ByteBuffer#isReadOnly() read-only}. The scope associated with this segment can either be the + * {@linkplain ResourceScope#globalScope() global} resource scope, in case the buffer has been created independently, + * or to some other (possibly closeable) resource scope, in case the buffer has been obtained using {@link #asByteBuffer()}. *

    - * The resulting memory segment keeps a reference to the backing buffer, to ensure it remains reachable - * for the life-time of the segment. + * The resulting memory segment keeps a reference to the backing buffer, keeping it reachable. * * @param bb the byte buffer backing the buffer memory segment. - * @return a new confined buffer memory segment. + * @return a new buffer memory segment. */ static MemorySegment ofByteBuffer(ByteBuffer bb) { return AbstractMemorySegmentImpl.ofBuffer(bb); @@ -709,13 +594,10 @@ for (long l = 0; l < segment.byteSize(); l++) { /** * Creates a new confined array memory segment that models the memory associated with a given heap-allocated byte array. - *

    - * The resulting memory segment keeps a reference to the backing array, to ensure it remains reachable - * for the life-time of the segment. The segment will feature all access modes - * (see {@link #ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). + * The returned segment's resource scope is set to the {@linkplain ResourceScope#globalScope() global} resource scope. * * @param arr the primitive array backing the array memory segment. - * @return a new confined array memory segment. + * @return a new array memory segment. */ static MemorySegment ofArray(byte[] arr) { return HeapMemorySegmentImpl.OfByte.fromArray(arr); @@ -723,13 +605,10 @@ for (long l = 0; l < segment.byteSize(); l++) { /** * Creates a new confined array memory segment that models the memory associated with a given heap-allocated char array. - *

    - * The resulting memory segment keeps a reference to the backing array, to ensure it remains reachable - * for the life-time of the segment. The segment will feature all access modes - * (see {@link #ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). + * The returned segment's resource scope is set to the {@linkplain ResourceScope#globalScope() global} resource scope. * * @param arr the primitive array backing the array memory segment. - * @return a new confined array memory segment. + * @return a new array memory segment. */ static MemorySegment ofArray(char[] arr) { return HeapMemorySegmentImpl.OfChar.fromArray(arr); @@ -737,13 +616,10 @@ for (long l = 0; l < segment.byteSize(); l++) { /** * Creates a new confined array memory segment that models the memory associated with a given heap-allocated short array. - *

    - * The resulting memory segment keeps a reference to the backing array, to ensure it remains reachable - * for the life-time of the segment. The segment will feature all access modes - * (see {@link #ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). + * The returned segment's resource scope is set to the {@linkplain ResourceScope#globalScope() global} resource scope. * * @param arr the primitive array backing the array memory segment. - * @return a new confined array memory segment. + * @return a new array memory segment. */ static MemorySegment ofArray(short[] arr) { return HeapMemorySegmentImpl.OfShort.fromArray(arr); @@ -751,13 +627,10 @@ for (long l = 0; l < segment.byteSize(); l++) { /** * Creates a new confined array memory segment that models the memory associated with a given heap-allocated int array. - *

    - * The resulting memory segment keeps a reference to the backing array, to ensure it remains reachable - * for the life-time of the segment. The segment will feature all access modes - * (see {@link #ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). + * The returned segment's resource scope is set to the {@linkplain ResourceScope#globalScope() global} resource scope. * * @param arr the primitive array backing the array memory segment. - * @return a new confined array memory segment. + * @return a new array memory segment. */ static MemorySegment ofArray(int[] arr) { return HeapMemorySegmentImpl.OfInt.fromArray(arr); @@ -765,13 +638,10 @@ for (long l = 0; l < segment.byteSize(); l++) { /** * Creates a new confined array memory segment that models the memory associated with a given heap-allocated float array. - *

    - * The resulting memory segment keeps a reference to the backing array, to ensure it remains reachable - * for the life-time of the segment. The segment will feature all access modes - * (see {@link #ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). + * The returned segment's resource scope is set to the {@linkplain ResourceScope#globalScope() global} resource scope. * * @param arr the primitive array backing the array memory segment. - * @return a new confined array memory segment. + * @return a new array memory segment. */ static MemorySegment ofArray(float[] arr) { return HeapMemorySegmentImpl.OfFloat.fromArray(arr); @@ -779,13 +649,10 @@ for (long l = 0; l < segment.byteSize(); l++) { /** * Creates a new confined array memory segment that models the memory associated with a given heap-allocated long array. - *

    - * The resulting memory segment keeps a reference to the backing array, to ensure it remains reachable - * for the life-time of the segment. The segment will feature all access modes - * (see {@link #ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). + * The returned segment's resource scope is set to the {@linkplain ResourceScope#globalScope() global} resource scope. * * @param arr the primitive array backing the array memory segment. - * @return a new confined array memory segment. + * @return a new array memory segment. */ static MemorySegment ofArray(long[] arr) { return HeapMemorySegmentImpl.OfLong.fromArray(arr); @@ -793,65 +660,99 @@ for (long l = 0; l < segment.byteSize(); l++) { /** * Creates a new confined array memory segment that models the memory associated with a given heap-allocated double array. - *

    - * The resulting memory segment keeps a reference to the backing array, to ensure it remains reachable - * for the life-time of the segment. The segment will feature all access modes - * (see {@link #ALL_ACCESS}). + * The returned segment's resource scope is set to the {@linkplain ResourceScope#globalScope() global} resource scope. * * @param arr the primitive array backing the array memory segment. - * @return a new confined array memory segment. + * @return a new array memory segment. */ static MemorySegment ofArray(double[] arr) { return HeapMemorySegmentImpl.OfDouble.fromArray(arr); } /** - * Creates a new confined native memory segment that models a newly allocated block of off-heap memory with given layout. + * Creates a new confined native memory segment that models a newly allocated block of off-heap memory with given layout + * and resource scope. A client is responsible make sure that the resource scope associated with the returned segment is closed + * when the segment is no longer in use. Failure to do so will result in off-heap memory leaks. *

    * This is equivalent to the following code: *

    {@code
    -    allocateNative(layout.bytesSize(), layout.bytesAlignment());
    +    allocateNative(layout.bytesSize(), layout.bytesAlignment(), scope);
          * }
    - * - * @implNote The block of off-heap memory associated with the returned native memory segment is initialized to zero. - * Moreover, a client is responsible to call the {@link MemorySegment#close()} on a native memory segment, - * to make sure the backing off-heap memory block is deallocated accordingly. Failure to do so will result in off-heap memory leaks. + *

    + * The block of off-heap memory associated with the returned native memory segment is initialized to zero. * * @param layout the layout of the off-heap memory block backing the native memory segment. + * @param scope the segment scope. * @return a new native memory segment. * @throws IllegalArgumentException if the specified layout has illegal size or alignment constraint. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. */ - static MemorySegment allocateNative(MemoryLayout layout) { + static MemorySegment allocateNative(MemoryLayout layout, ResourceScope scope) { + Objects.requireNonNull(scope); Objects.requireNonNull(layout); - return allocateNative(layout.byteSize(), layout.byteAlignment()); + return allocateNative(layout.byteSize(), layout.byteAlignment(), scope); } /** - * Creates a new confined native memory segment that models a newly allocated block of off-heap memory with given size (in bytes). + * Creates a new confined native memory segment that models a newly allocated block of off-heap memory with given size (in bytes) + * and resource scope. A client is responsible make sure that the resource scope associated with the returned segment is closed + * when the segment is no longer in use. Failure to do so will result in off-heap memory leaks. *

    * This is equivalent to the following code: *

    {@code
    -allocateNative(bytesSize, 1);
    +    allocateNative(bytesSize, 1, scope);
          * }
    - * - * @implNote The block of off-heap memory associated with the returned native memory segment is initialized to zero. - * Moreover, a client is responsible to call the {@link MemorySegment#close()} on a native memory segment, - * to make sure the backing off-heap memory block is deallocated accordingly. Failure to do so will result in off-heap memory leaks. + *

    + * The block of off-heap memory associated with the returned native memory segment is initialized to zero. * * @param bytesSize the size (in bytes) of the off-heap memory block backing the native memory segment. - * @return a new confined native memory segment. - * @throws IllegalArgumentException if {@code bytesSize < 0}. + * @param scope the segment scope. + * @return a new native memory segment. + * @throws IllegalArgumentException if {@code bytesSize <= 0}. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. */ - static MemorySegment allocateNative(long bytesSize) { - return allocateNative(bytesSize, 1); + static MemorySegment allocateNative(long bytesSize, ResourceScope scope) { + return allocateNative(bytesSize, 1, scope); } /** - * Creates a new confined mapped memory segment that models a memory-mapped region of a file from a given path. + * Creates a new confined native memory segment that models a newly allocated block of off-heap memory with given size + * (in bytes), alignment constraint (in bytes) and resource scope. A client is responsible make sure that the resource + * scope associated with the returned segment is closed when the segment is no longer in use. + * Failure to do so will result in off-heap memory leaks. *

    - * The segment will feature all access modes (see {@link #ALL_ACCESS}), - * unless the given mapping mode is {@linkplain FileChannel.MapMode#READ_ONLY READ_ONLY}, in which case - * the segment will not feature the {@link #WRITE} access mode, and its confinement thread is the current thread (see {@link Thread#currentThread()}). + * The block of off-heap memory associated with the returned native memory segment is initialized to zero. + * + * @param bytesSize the size (in bytes) of the off-heap memory block backing the native memory segment. + * @param alignmentBytes the alignment constraint (in bytes) of the off-heap memory block backing the native memory segment. + * @param scope the segment scope. + * @return a new native memory segment. + * @throws IllegalArgumentException if {@code bytesSize <= 0}, {@code alignmentBytes <= 0}, or if {@code alignmentBytes} + * is not a power of 2. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. + */ + static MemorySegment allocateNative(long bytesSize, long alignmentBytes, ResourceScope scope) { + Objects.requireNonNull(scope); + if (bytesSize <= 0) { + throw new IllegalArgumentException("Invalid allocation size : " + bytesSize); + } + + if (alignmentBytes <= 0 || + ((alignmentBytes & (alignmentBytes - 1)) != 0L)) { + throw new IllegalArgumentException("Invalid alignment constraint : " + alignmentBytes); + } + + return NativeMemorySegmentImpl.makeNativeSegment(bytesSize, alignmentBytes, (ResourceScopeImpl) scope); + } + + /** + * Creates a new mapped memory segment that models a memory-mapped region of a file from a given path. + *

    + * If the specified mapping mode is {@linkplain FileChannel.MapMode#READ_ONLY READ_ONLY}, the resulting segment + * will be read-only (see {@link #isReadOnly()}). *

    * The content of a mapped memory segment can change at any time, for example * if the content of the corresponding region of the mapped file is changed by @@ -874,10 +775,13 @@ allocateNative(bytesSize, 1); * @param bytesOffset the offset (expressed in bytes) within the file at which the mapped segment is to start. * @param bytesSize the size (in bytes) of the mapped memory backing the memory segment. * @param mapMode a file mapping mode, see {@link FileChannel#map(FileChannel.MapMode, long, long)}; the chosen mapping mode - * might affect the behavior of the returned memory mapped segment (see {@link MappedMemorySegments#force(MemorySegment)}). + * might affect the behavior of the returned memory mapped segment (see {@link #force()}). + * @param scope the segment scope. * @return a new confined mapped memory segment. * @throws IllegalArgumentException if {@code bytesOffset < 0}, {@code bytesSize < 0}, or if {@code path} is not associated * with the default file system. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. * @throws UnsupportedOperationException if an unsupported map mode is specified. * @throws IOException if the specified path does not point to an existing file, or if some other I/O error occurs. * @throws SecurityException If a security manager is installed and it denies an unspecified permission required by the implementation. @@ -885,104 +789,33 @@ allocateNative(bytesSize, 1); * read access if the file is opened for reading. The {@link SecurityManager#checkWrite(String)} method is invoked to check * write access if the file is opened for writing. */ - static MemorySegment mapFile(Path path, long bytesOffset, long bytesSize, FileChannel.MapMode mapMode) throws IOException { - return MappedMemorySegmentImpl.makeMappedSegment(path, bytesOffset, bytesSize, mapMode); + static MemorySegment mapFile(Path path, long bytesOffset, long bytesSize, FileChannel.MapMode mapMode, ResourceScope scope) throws IOException { + Objects.requireNonNull(scope); + return MappedMemorySegmentImpl.makeMappedSegment(path, bytesOffset, bytesSize, mapMode, (ResourceScopeImpl) scope); } /** - * Creates a new confined native memory segment that models a newly allocated block of off-heap memory with given size and - * alignment constraint (in bytes). The segment will feature all access modes - * (see {@link #ALL_ACCESS}), and its confinement thread is the current thread (see {@link Thread#currentThread()}). - * - * @implNote The block of off-heap memory associated with the returned native memory segment is initialized to zero. - * Moreover, a client is responsible to call the {@link MemorySegment#close()} on a native memory segment, - * to make sure the backing off-heap memory block is deallocated accordingly. Failure to do so will result in off-heap memory leaks. - * - * @param bytesSize the size (in bytes) of the off-heap memory block backing the native memory segment. - * @param alignmentBytes the alignment constraint (in bytes) of the off-heap memory block backing the native memory segment. - * @return a new confined native memory segment. - * @throws IllegalArgumentException if {@code bytesSize < 0}, {@code alignmentBytes < 0}, or if {@code alignmentBytes} - * is not a power of 2. - */ - static MemorySegment allocateNative(long bytesSize, long alignmentBytes) { - if (bytesSize <= 0) { - throw new IllegalArgumentException("Invalid allocation size : " + bytesSize); - } - - if (alignmentBytes < 0 || - ((alignmentBytes & (alignmentBytes - 1)) != 0L)) { - throw new IllegalArgumentException("Invalid alignment constraint : " + alignmentBytes); - } - - return NativeMemorySegmentImpl.makeNativeSegment(bytesSize, alignmentBytes); - } - - /** - * Returns a shared native memory segment whose base address is {@link MemoryAddress#NULL} and whose size is {@link Long#MAX_VALUE}. + * Returns a native memory segment whose base address is {@link MemoryAddress#NULL} and whose size is {@link Long#MAX_VALUE}. * This method can be very useful when dereferencing memory addresses obtained when interacting with native libraries. - * The segment will feature the {@link #READ} and {@link #WRITE} access modes. + * The returned segment is associated with the global resource scope (see {@link ResourceScope#globalScope()}). * Equivalent to (but likely more efficient than) the following code: *

    {@code
    -    MemoryAddress.NULL.asSegmentRestricted(Long.MAX_VALUE)
    -                 .withOwnerThread(null)
    -                 .withAccessModes(READ | WRITE);
    +    MemoryAddress.NULL.asSegment(Long.MAX_VALUE)
          * }
    *

    - * This method is restricted. Restricted methods are unsafe, and, if used incorrectly, their use might crash + * This method is restricted. + * Restricted method are unsafe, and, if used incorrectly, their use might crash * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on * restricted methods, and use safe and supported functionalities, where possible. * * @return a memory segment whose base address is {@link MemoryAddress#NULL} and whose size is {@link Long#MAX_VALUE}. - * @throws IllegalAccessError if the runtime property {@code foreign.restricted} is not set to either - * {@code permit}, {@code warn} or {@code debug} (the default value is set to {@code deny}). + * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option + * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or + * {@code ALL-UNNAMED} in case {@code M} is an unnamed module. */ - static MemorySegment ofNativeRestricted() { - Utils.checkRestrictedAccess("MemorySegment.ofNativeRestricted"); + @CallerSensitive + static MemorySegment globalNativeSegment() { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); return NativeMemorySegmentImpl.EVERYTHING; } - - // access mode masks - - /** - * Read access mode; read operations are supported by a segment which supports this access mode. - * @see MemorySegment#accessModes() - * @see MemorySegment#withAccessModes(int) - */ - int READ = 1; - - /** - * Write access mode; write operations are supported by a segment which supports this access mode. - * @see MemorySegment#accessModes() - * @see MemorySegment#withAccessModes(int) - */ - int WRITE = READ << 1; - - /** - * Close access mode; calling {@link #close()} is supported by a segment which supports this access mode. - * @see MemorySegment#accessModes() - * @see MemorySegment#withAccessModes(int) - */ - int CLOSE = WRITE << 1; - - /** - * Share access mode; this segment support sharing with threads other than the owner thread (see {@link #share()}). - * @see MemorySegment#accessModes() - * @see MemorySegment#withAccessModes(int) - */ - int SHARE = CLOSE << 1; - - /** - * Handoff access mode; this segment support serial thread-confinement via thread ownership changes - * (see {@link #handoff(NativeScope)} and {@link #handoff(Thread)}). - * @see MemorySegment#accessModes() - * @see MemorySegment#withAccessModes(int) - */ - int HANDOFF = SHARE << 1; - - /** - * Default access mode; this is a union of all the access modes supported by memory segments. - * @see MemorySegment#accessModes() - * @see MemorySegment#withAccessModes(int) - */ - int ALL_ACCESS = READ | WRITE | CLOSE | SHARE | HANDOFF; } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/NativeScope.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/NativeScope.java deleted file mode 100644 index fb521345a15..00000000000 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/NativeScope.java +++ /dev/null @@ -1,472 +0,0 @@ -/* - * 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. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * 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. - * - */ - -package jdk.incubator.foreign; - -import jdk.internal.foreign.AbstractMemorySegmentImpl; -import jdk.internal.foreign.AbstractNativeScope; -import jdk.internal.foreign.Utils; - -import java.lang.invoke.VarHandle; -import java.lang.reflect.Array; -import java.nio.ByteOrder; -import java.util.Objects; -import java.util.OptionalLong; -import java.util.function.Function; -import java.util.stream.Stream; - -/** - * A native scope is an abstraction which provides shared temporal bounds for one or more allocations, backed - * by off-heap memory. Native scopes can be either bounded or unbounded, depending on whether the size - * of the native scope is known statically. If an application knows before-hand how much memory it needs to allocate, - * then using a bounded native scope will typically provide better performance than independently allocating the memory - * for each value (e.g. using {@link MemorySegment#allocateNative(long)}), or using an unbounded native scope. - * For this reason, using a bounded native scope is recommended in cases where programs might need to emulate native stack allocation. - *

    - * Allocation scopes are thread-confined (see {@link #ownerThread()}; as such, the resulting {@link MemorySegment} instances - * returned by the native scope will be backed by memory segments confined by the same owner thread as the native scope's - * owner thread. - *

    - * To allow for more usability, it is possible for a native scope to reclaim ownership of an existing memory segment - * (see {@link MemorySegment#handoff(NativeScope)}). This might be useful to allow one or more segments which were independently - * created to share the same life-cycle as a given native scope - which in turns enables a client to group all memory - * allocation and usage under a single try-with-resources block. - * - *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} - * elements to a method in this class causes a {@link NullPointerException NullPointerException} to be thrown.

    - * - * @apiNote In the future, if the Java language permits, {@link NativeScope} - * may become a {@code sealed} interface, which would prohibit subclassing except by - * explicitly permitted types. - */ -public interface NativeScope extends AutoCloseable { - - /** - * If this native scope is bounded, returns the size, in bytes, of this native scope. - * @return the size, in bytes, of this native scope (if available). - */ - OptionalLong byteSize(); - - /** - * The thread owning this native scope. - * @return the thread owning this native scope. - */ - Thread ownerThread(); - - /** - * Returns the number of allocated bytes in this native scope. - * @return the number of allocated bytes in this native scope. - */ - long allocatedBytes(); - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given byte value. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @param value the value to be set on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a byte value. - */ - default MemorySegment allocate(ValueLayout layout, byte value) { - Objects.requireNonNull(layout); - VarHandle handle = layout.varHandle(byte.class); - MemorySegment addr = allocate(layout); - handle.set(addr, value); - return addr; - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given char value. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @param value the value to be set on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a char value. - */ - default MemorySegment allocate(ValueLayout layout, char value) { - Objects.requireNonNull(layout); - VarHandle handle = layout.varHandle(char.class); - MemorySegment addr = allocate(layout); - handle.set(addr, value); - return addr; - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given short value. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @param value the value to be set on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a short value. - */ - default MemorySegment allocate(ValueLayout layout, short value) { - Objects.requireNonNull(layout); - VarHandle handle = layout.varHandle(short.class); - MemorySegment addr = allocate(layout); - handle.set(addr, value); - return addr; - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given int value. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @param value the value to be set on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a int value. - */ - default MemorySegment allocate(ValueLayout layout, int value) { - Objects.requireNonNull(layout); - VarHandle handle = layout.varHandle(int.class); - MemorySegment addr = allocate(layout); - handle.set(addr, value); - return addr; - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given float value. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @param value the value to be set on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a float value. - */ - default MemorySegment allocate(ValueLayout layout, float value) { - Objects.requireNonNull(layout); - VarHandle handle = layout.varHandle(float.class); - MemorySegment addr = allocate(layout); - handle.set(addr, value); - return addr; - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given long value. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @param value the value to be set on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a long value. - */ - default MemorySegment allocate(ValueLayout layout, long value) { - Objects.requireNonNull(layout); - VarHandle handle = layout.varHandle(long.class); - MemorySegment addr = allocate(layout); - handle.set(addr, value); - return addr; - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given double value. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @param value the value to be set on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a double value. - */ - default MemorySegment allocate(ValueLayout layout, double value) { - Objects.requireNonNull(layout); - VarHandle handle = layout.varHandle(double.class); - MemorySegment addr = allocate(layout); - handle.set(addr, value); - return addr; - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given address value - * (expressed as an {@link Addressable} instance). - * The address value might be narrowed according to the platform address size (see {@link MemoryLayouts#ADDRESS}). - * The segment returned by this method cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @param value the value to be set on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - * @throws IllegalArgumentException if {@code layout.byteSize() != MemoryLayouts.ADDRESS.byteSize()}. - */ - default MemorySegment allocate(ValueLayout layout, Addressable value) { - Objects.requireNonNull(value); - Objects.requireNonNull(layout); - if (MemoryLayouts.ADDRESS.byteSize() != layout.byteSize()) { - throw new IllegalArgumentException("Layout size mismatch - " + layout.byteSize() + " != " + MemoryLayouts.ADDRESS.byteSize()); - } - switch ((int)layout.byteSize()) { - case 4: return allocate(layout, (int)value.address().toRawLongValue()); - case 8: return allocate(layout, value.address().toRawLongValue()); - default: throw new UnsupportedOperationException("Unsupported pointer size"); // should not get here - } - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given byte array. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param elementLayout the element layout of the array to be allocated. - * @param array the array to be copied on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * array.length)}. - * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a byte value. - */ - default MemorySegment allocateArray(ValueLayout elementLayout, byte[] array) { - return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given short array. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param elementLayout the element layout of the array to be allocated. - * @param array the array to be copied on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * array.length)}. - * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a short value. - */ - default MemorySegment allocateArray(ValueLayout elementLayout, short[] array) { - return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given char array. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param elementLayout the element layout of the array to be allocated. - * @param array the array to be copied on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * array.length)}. - * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a char value. - */ - default MemorySegment allocateArray(ValueLayout elementLayout, char[] array) { - return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given int array. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param elementLayout the element layout of the array to be allocated. - * @param array the array to be copied on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * array.length)}. - * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a int value. - */ - default MemorySegment allocateArray(ValueLayout elementLayout, int[] array) { - return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given float array. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param elementLayout the element layout of the array to be allocated. - * @param array the array to be copied on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * array.length)}. - * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a float value. - */ - default MemorySegment allocateArray(ValueLayout elementLayout, float[] array) { - return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given long array. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param elementLayout the element layout of the array to be allocated. - * @param array the array to be copied on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * array.length)}. - * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a long value. - */ - default MemorySegment allocateArray(ValueLayout elementLayout, long[] array) { - return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given double array. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param elementLayout the element layout of the array to be allocated. - * @param array the array to be copied on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * array.length)}. - * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a double value. - */ - default MemorySegment allocateArray(ValueLayout elementLayout, double[] array) { - return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); - } - - /** - * Allocate a block of memory in this native scope with given layout and initialize it with given address array. - * The address value of each array element might be narrowed according to the platform address size (see {@link MemoryLayouts#ADDRESS}). - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, the returned - * segment must conform to the layout alignment constraints. - * @param elementLayout the element layout of the array to be allocated. - * @param array the array to be copied on the newly allocated memory block. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * array.length)}. - * @throws IllegalArgumentException if {@code layout.byteSize() != MemoryLayouts.ADDRESS.byteSize()}. - */ - default MemorySegment allocateArray(ValueLayout elementLayout, Addressable[] array) { - Objects.requireNonNull(elementLayout); - Objects.requireNonNull(array); - Stream.of(array).forEach(Objects::requireNonNull); - if (MemoryLayouts.ADDRESS.byteSize() != elementLayout.byteSize()) { - throw new IllegalArgumentException("Layout size mismatch - " + elementLayout.byteSize() + " != " + MemoryLayouts.ADDRESS.byteSize()); - } - switch ((int)elementLayout.byteSize()) { - case 4: return copyArrayWithSwapIfNeeded(Stream.of(array) - .mapToInt(a -> (int)a.address().toRawLongValue()).toArray(), - elementLayout, MemorySegment::ofArray); - case 8: return copyArrayWithSwapIfNeeded(Stream.of(array) - .mapToLong(a -> a.address().toRawLongValue()).toArray(), - elementLayout, MemorySegment::ofArray); - default: throw new UnsupportedOperationException("Unsupported pointer size"); // should not get here - } - } - - private MemorySegment copyArrayWithSwapIfNeeded(Z array, ValueLayout elementLayout, - Function heapSegmentFactory) { - Objects.requireNonNull(array); - Objects.requireNonNull(elementLayout); - Utils.checkPrimitiveCarrierCompat(array.getClass().componentType(), elementLayout); - MemorySegment addr = allocate(MemoryLayout.ofSequence(Array.getLength(array), elementLayout)); - if (elementLayout.byteSize() == 1 || (elementLayout.order() == ByteOrder.nativeOrder())) { - addr.copyFrom(heapSegmentFactory.apply(array)); - } else { - ((AbstractMemorySegmentImpl)addr).copyFromSwap(heapSegmentFactory.apply(array), elementLayout.byteSize()); - } - return addr; - } - - /** - * Allocate a block of memory in this native scope with given layout. The segment returned by this method is - * associated with a segment which cannot be closed. Moreover, the returned segment must conform to the layout alignment constraints. - * @param layout the layout of the block of memory to be allocated. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < layout.byteSize()}. - */ - default MemorySegment allocate(MemoryLayout layout) { - Objects.requireNonNull(layout); - return allocate(layout.byteSize(), layout.byteAlignment()); - } - - /** - * Allocate a block of memory corresponding to an array with given element layout and size. - * The segment returned by this method is associated with a segment which cannot be closed. - * Moreover, the returned segment must conform to the layout alignment constraints. This is equivalent to the - * following code: - *
    {@code
    -    allocate(MemoryLayout.ofSequence(size, elementLayout));
    -     * }
    - * @param elementLayout the array element layout. - * @param count the array element count. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if this is a - * bounded allocation scope, and {@code byteSize().getAsLong() - allocatedBytes() < (elementLayout.byteSize() * count)}. - */ - default MemorySegment allocateArray(MemoryLayout elementLayout, long count) { - Objects.requireNonNull(elementLayout); - return allocate(MemoryLayout.ofSequence(count, elementLayout)); - } - - /** - * Allocate a block of memory in this native scope with given size. The segment returned by this method is - * associated with a segment which cannot be closed. Moreover, the returned segment must be aligned to {@code size}. - * @param bytesSize the size (in bytes) of the block of memory to be allocated. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if - * {@code limit() - size() < bytesSize}. - */ - default MemorySegment allocate(long bytesSize) { - return allocate(bytesSize, bytesSize); - } - - /** - * Allocate a block of memory in this native scope with given size and alignment constraint. - * The segment returned by this method is associated with a segment which cannot be closed. Moreover, - * the returned segment must be aligned to {@code alignment}. - * @param bytesSize the size (in bytes) of the block of memory to be allocated. - * @param bytesAlignment the alignment (in bytes) of the block of memory to be allocated. - * @return a segment for the newly allocated memory block. - * @throws OutOfMemoryError if there is not enough space left in this native scope, that is, if - * {@code limit() - size() < bytesSize}. - */ - MemorySegment allocate(long bytesSize, long bytesAlignment); - - /** - * Close this native scope; calling this method will render any segment obtained through this native scope - * unusable and might release any backing memory resources associated with this native scope. - */ - @Override - void close(); - - /** - * Creates a new bounded native scope, backed by off-heap memory. - * @param size the size of the native scope. - * @return a new bounded native scope, with given size (in bytes). - */ - static NativeScope boundedScope(long size) { - return new AbstractNativeScope.BoundedNativeScope(size); - } - - /** - * Creates a new unbounded native scope, backed by off-heap memory. - * @return a new unbounded native scope. - */ - static NativeScope unboundedScope() { - return new AbstractNativeScope.UnboundedNativeScope(); - } -} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/ResourceScope.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/ResourceScope.java new file mode 100644 index 00000000000..0a46ab705b3 --- /dev/null +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/ResourceScope.java @@ -0,0 +1,285 @@ +/* + * Copyright (c) 2021, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ + +package jdk.incubator.foreign; + +import jdk.internal.foreign.ResourceScopeImpl; + +import java.lang.invoke.MethodHandle; +import java.lang.ref.Cleaner; +import java.nio.channels.FileChannel; +import java.nio.file.Path; +import java.util.Objects; +import java.util.Spliterator; + +/** + * A resource scope manages the lifecycle of one or more resources. Resources (e.g. {@link MemorySegment}) associated + * with a resource scope can only be accessed while the resource scope is alive (see {@link #isAlive()}), + * and by the thread associated with the resource scope (if any). + * + *

    Explicit resource scopes

    + * + * Resource scopes obtained from {@link #newConfinedScope()}, {@link #newSharedScope()} support deterministic deallocation; + * We call these resource scopes explicit scopes. Explicit resource scopes can be closed explicitly (see {@link ResourceScope#close()}). + * When a resource scope is closed, it is no longer alive (see {@link #isAlive()}, and subsequent operations on + * resources associated with that scope (e.g. attempting to access a {@link MemorySegment} instance) will fail with {@link IllegalStateException}. + *

    + * Closing a resource scope will cause all the cleanup actions associated with that scope (see {@link #addCloseAction(Runnable)}) to be called. + * Moreover, closing a resource scope might trigger the releasing of the underlying memory resources associated with said scope; for instance: + *

      + *
    • closing the scope associated with a native memory segment results in freeing the native memory associated with it + * (see {@link MemorySegment#allocateNative(long, ResourceScope)}, or {@link SegmentAllocator#arenaAllocator(ResourceScope)})
    • + *
    • closing the scope associated with a mapped memory segment results in the backing memory-mapped file to be unmapped + * (see {@link MemorySegment#mapFile(Path, long, long, FileChannel.MapMode, ResourceScope)})
    • + *
    • closing the scope associated with an upcall stub results in releasing the stub + * (see {@link CLinker#upcallStub(MethodHandle, FunctionDescriptor, ResourceScope)}
    • + *
    + *

    + * Sometimes, explicit scopes can be associated with a {@link Cleaner} instance (see {@link #newConfinedScope(Cleaner)} and + * {@link #newSharedScope(Cleaner)}). We call these resource scopes managed resource scopes. A managed resource scope + * is closed automatically once the scope instance becomes unreachable. + *

    + * Managed scopes can be useful to allow for predictable, deterministic resource deallocation, while still prevent accidental native memory leaks. + * In case a managed resource scope is closed explicitly, no further action will be taken when the scope becomes unreachable; + * that is, cleanup actions (see {@link #addCloseAction(Runnable)}) associated with a resource scope, whether managed or not, + * are called exactly once. + * + *

    Implicit resource scopes

    + * + * Resource scopes obtained from {@link #newImplicitScope()} cannot be closed explicitly. We call these resource scopes + * implicit scopes. Calling {@link #close()} on an implicit resource scope always results in an exception. + * Resources associated with implicit scopes are released once the scope instance becomes + * unreachable. + *

    + * An important implicit resource scope is the so called {@linkplain #globalScope() global scope}; the global scope is + * an implicit scope that is guaranteed to never become unreachable. + * As a results, the global scope will never attempt to release resources associated with it. Such resources must, where + * needed, be managed independently by clients. + * + *

    Thread confinement

    + * + * Resource scopes can be further divided into two categories: thread-confined resource scopes, and shared + * resource scopes. + *

    + * Confined resource scopes (see {@link #newConfinedScope()}), support strong thread-confinement guarantees. Upon creation, + * they are assigned an owner thread, typically the thread which initiated the creation operation (see {@link #ownerThread()}). + * After creating a confined resource scope, only the owner thread will be allowed to directly manipulate the resources + * associated with this resource scope. Any attempt to perform resource access from a thread other than the + * owner thread will result in a runtime failure. + *

    + * Shared resource scopes (see {@link #newSharedScope()} and {@link #newImplicitScope()}), on the other hand, have no owner thread; + * as such resources associated with this shared resource scopes can be accessed by multiple threads. + * This might be useful when multiple threads need to access the same resource concurrently (e.g. in the case of parallel processing). + * For instance, a client might obtain a {@link Spliterator} from a shared segment, which can then be used to slice the + * segment and allow multiple threads to work in parallel on disjoint segment slices. The following code can be used to sum + * all int values in a memory segment in parallel: + * + *

    {@code
    +SequenceLayout SEQUENCE_LAYOUT = MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_INT);
    +try (ResourceScope scope = ResourceScope.newSharedScope()) {
    +    MemorySegment segment = MemorySegment.allocateNative(SEQUENCE_LAYOUT, scope);
    +    VarHandle VH_int = SEQUENCE_LAYOUT.elementLayout().varHandle(int.class);
    +    int sum = StreamSupport.stream(segment.spliterator(SEQUENCE_LAYOUT), true)
    +        .mapToInt(s -> (int)VH_int.get(s.address()))
    +        .sum();
    +}
    + * }
    + * + *

    + * Explicit shared resource scopes, while powerful, must be used with caution: if one or more threads accesses + * a resource associated with a shared scope while the scope is being closed from another thread, an exception might occur on both + * the accessing and the closing threads. Clients should refrain from attempting to close a shared resource scope repeatedly + * (e.g. keep calling {@link #close()} until no exception is thrown). Instead, clients of shared resource scopes + * should always ensure that proper synchronization mechanisms (e.g. using resource scope handles, see below) are put in place + * so that threads closing shared resource scopes can never race against threads accessing resources managed by same scopes. + * + *

    Resource scope handles

    + * + * Resource scopes can be made non-closeable by acquiring one or more resource scope handles (see + * {@link #acquire()}. A resource scope handle can be used to make sure that resources associated with a given resource scope + * (either explicit or implicit) cannot be released for a certain period of time - e.g. during a critical region of code + * involving one or more resources associated with the scope. For instance, an explicit resource scope can only be closed + * after all the handles acquired against that scope have been closed (see {@link Handle#close()}). + * This can be useful when clients need to perform a critical operation on a memory segment, during which they have + * to ensure that the segment will not be released; this can be done as follows: + * + *
    {@code
    +MemorySegment segment = ...
    +ResourceScope.Handle segmentHandle = segment.scope().acquire()
    +try {
    +   
    +} finally {
    +   segment.scope().release(segmentHandle);
    +}
    + * }
    + * + * Acquiring implicit resource scopes is also possible, but it is often unnecessary: since resources associated with + * an implicit scope will only be released when the scope becomes unreachable, + * clients can use e.g. {@link java.lang.ref.Reference#reachabilityFence(Object)} to make sure that resources associated + * with implicit scopes are not released prematurely. That said, the above code snippet works (trivially) for implicit scopes too. + * + * @implSpec + * Implementations of this interface are immutable, thread-safe and value-based. + */ +public sealed interface ResourceScope extends AutoCloseable permits ResourceScopeImpl { + /** + * Is this resource scope alive? + * @return true, if this resource scope is alive. + * @see ResourceScope#close() + */ + boolean isAlive(); + + /** + * The thread owning this resource scope. + * @return the thread owning this resource scope, or {@code null} if this resource scope is shared. + */ + Thread ownerThread(); + + /** + * Is this resource scope an implicit scope? + * @return true if this scope is an implicit scope. + * @see #newImplicitScope() + * @see #globalScope() + */ + boolean isImplicit(); + + /** + * Closes this resource scope. As a side-effect, if this operation completes without exceptions, this scope will be marked + * as not alive, and subsequent operations on resources associated with this scope will fail with {@link IllegalStateException}. + * Additionally, upon successful closure, all native resources associated with this resource scope will be released. + * + * @apiNote This operation is not idempotent; that is, closing an already closed resource scope always results in an + * exception being thrown. This reflects a deliberate design choice: resource scope state transitions should be + * manifest in the client code; a failure in any of these transitions reveals a bug in the underlying application + * logic. + * + * @throws IllegalStateException if one of the following condition is met: + *
      + *
    • this resource scope is not alive + *
    • this resource scope is confined, and this method is called from a thread other than the thread owning this resource scope
    • + *
    • this resource scope is shared and a resource associated with this scope is accessed while this method is called
    • + *
    • one or more handles (see {@link #acquire()}) associated with this resource scope have not been {@linkplain #release(Handle) released}
    • + *
    + * @throws UnsupportedOperationException if this resource scope is {@linkplain #isImplicit() implicit}. + */ + void close(); + + /** + * Add a custom cleanup action which will be executed when the resource scope is closed. + * The order in which custom cleanup actions are invoked once the scope is closed is unspecified. + * @param runnable the custom cleanup action to be associated with this scope. + * @throws IllegalStateException if this scope has already been closed. + */ + void addCloseAction(Runnable runnable); + + /** + * Acquires a resource scope handle associated with this resource scope. An explicit resource scope cannot be + * {@linkplain #close() closed} until all the resource scope handles acquired from it have been {@linkplain #release(Handle)} released}. + * @return a resource scope handle. + */ + Handle acquire(); + + /** + * Release the provided resource scope handle. This method is idempotent, that is, releasing the same handle + * multiple times has no effect. + * @param handle the resource scope handle to be released. + * @throws IllegalArgumentException if the provided handle is not associated with this scope. + */ + void release(Handle handle); + + /** + * An abstraction modelling a resource scope handle. A resource scope handle is typically {@linkplain #acquire() acquired} by clients + * in order to prevent an explicit resource scope from being closed while executing a certain operation. + * Once obtained, resource scope handles can be {@linkplain #release(Handle)} released}; an explicit resource scope can + * be closed only after all the resource scope handles acquired from it have been released. + */ + sealed interface Handle permits ResourceScopeImpl.HandleImpl { + + /** + * Returns the resource scope associated with this handle. + * @return the resource scope associated with this handle. + */ + ResourceScope scope(); + } + + /** + * Create a new confined scope. The resulting scope is closeable, and is not managed by a {@link Cleaner}. + * @return a new confined scope. + */ + static ResourceScope newConfinedScope() { + return ResourceScopeImpl.createConfined( null); + } + + /** + * Create a new confined scope managed by a {@link Cleaner}. + * @param cleaner the cleaner to be associated with the returned scope. + * @return a new confined scope, managed by {@code cleaner}. + * @throws NullPointerException if {@code cleaner == null}. + */ + static ResourceScope newConfinedScope(Cleaner cleaner) { + Objects.requireNonNull(cleaner); + return ResourceScopeImpl.createConfined( cleaner); + } + + /** + * Create a new shared scope. The resulting scope is closeable, and is not managed by a {@link Cleaner}. + * @return a new shared scope. + */ + static ResourceScope newSharedScope() { + return ResourceScopeImpl.createShared(null); + } + + /** + * Create a new shared scope managed by a {@link Cleaner}. + * @param cleaner the cleaner to be associated with the returned scope. + * @return a new shared scope, managed by {@code cleaner}. + * @throws NullPointerException if {@code cleaner == null}. + */ + static ResourceScope newSharedScope(Cleaner cleaner) { + Objects.requireNonNull(cleaner); + return ResourceScopeImpl.createShared(cleaner); + } + + /** + * Create a new implicit scope. The implicit scope is a managed, shared, and non-closeable scope which only features + * implicit closure. + * Since implicit scopes can only be closed implicitly by the garbage collector, it is recommended that implicit + * scopes are only used in cases where deallocation performance is not a critical concern, to avoid unnecessary + * memory pressure. + * + * @return a new implicit scope. + */ + static ResourceScope newImplicitScope() { + return ResourceScopeImpl.createImplicitScope(); + } + + /** + * Returns an implicit scope which is assumed to be always alive. + * @return the global scope. + */ + static ResourceScope globalScope() { + return ResourceScopeImpl.GLOBAL; + } +} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/SegmentAllocator.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/SegmentAllocator.java new file mode 100644 index 00000000000..f127dc70330 --- /dev/null +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/SegmentAllocator.java @@ -0,0 +1,466 @@ +/* + * Copyright (c) 2021, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ + +package jdk.incubator.foreign; + +import jdk.internal.foreign.ArenaAllocator; +import jdk.internal.foreign.AbstractMemorySegmentImpl; +import jdk.internal.foreign.ResourceScopeImpl; +import jdk.internal.foreign.Utils; + +import java.lang.invoke.VarHandle; +import java.lang.reflect.Array; +import java.nio.ByteOrder; +import java.util.Objects; +import java.util.function.Function; +import java.util.stream.Stream; + +/** + * This interface models a memory allocator. Clients implementing this interface + * must implement the {@link #allocate(long, long)} method. This interface defines several default methods + * which can be useful to create segments from several kinds of Java values such as primitives and arrays. + * This interface can be seen as a thin wrapper around the basic capabilities for creating native segments + * (e.g. {@link MemorySegment#allocateNative(long, long, ResourceScope)}); since {@link SegmentAllocator} is a functional interface, + * clients can easily obtain a native allocator by using either a lambda expression or a method reference. + *

    + * This interface provides a factory, namely {@link SegmentAllocator#ofScope(ResourceScope)} which can be used to obtain + * a scoped allocator, that is, an allocator which creates segment bound by a given scope. This can be useful + * when working inside a try-with-resources construct: + * + *

    {@code
    +try (ResourceScope scope = ResourceScope.newConfinedScope()) {
    +   SegmentAllocator allocator = SegmentAllocator.ofScope(scope);
    +   ...
    +}
    + * }
    + * + * In addition, this interface also defines factories for commonly used allocators; for instance {@link #arenaAllocator(ResourceScope)} + * and {@link #arenaAllocator(long, ResourceScope)} are arena-style native allocators. Finally {@link #ofSegment(MemorySegment)} + * returns an allocator which wraps a segment (either on-heap or off-heap) and recycles its content upon each new allocation request. + */ +@FunctionalInterface +public interface SegmentAllocator { + + /** + * Allocate a block of memory with given layout and initialize it with given byte value. + * @implSpec the default implementation for this method calls {@code this.allocate(layout)}. + * @param layout the layout of the block of memory to be allocated. + * @param value the value to be set on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a byte value. + */ + default MemorySegment allocate(ValueLayout layout, byte value) { + Objects.requireNonNull(layout); + VarHandle handle = layout.varHandle(byte.class); + MemorySegment addr = allocate(layout); + handle.set(addr, value); + return addr; + } + + /** + * Allocate a block of memory with given layout and initialize it with given char value. + * @implSpec the default implementation for this method calls {@code this.allocate(layout)}. + * @param layout the layout of the block of memory to be allocated. + * @param value the value to be set on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a char value. + */ + default MemorySegment allocate(ValueLayout layout, char value) { + Objects.requireNonNull(layout); + VarHandle handle = layout.varHandle(char.class); + MemorySegment addr = allocate(layout); + handle.set(addr, value); + return addr; + } + + /** + * Allocate a block of memory with given layout and initialize it with given short value. + * @implSpec the default implementation for this method calls {@code this.allocate(layout)}. + * @param layout the layout of the block of memory to be allocated. + * @param value the value to be set on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a short value. + */ + default MemorySegment allocate(ValueLayout layout, short value) { + Objects.requireNonNull(layout); + VarHandle handle = layout.varHandle(short.class); + MemorySegment addr = allocate(layout); + handle.set(addr, value); + return addr; + } + + /** + * Allocate a block of memory with given layout and initialize it with given int value. + * @implSpec the default implementation for this method calls {@code this.allocate(layout)}. + * @param layout the layout of the block of memory to be allocated. + * @param value the value to be set on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a int value. + */ + default MemorySegment allocate(ValueLayout layout, int value) { + Objects.requireNonNull(layout); + VarHandle handle = layout.varHandle(int.class); + MemorySegment addr = allocate(layout); + handle.set(addr, value); + return addr; + } + + /** + * Allocate a block of memory with given layout and initialize it with given float value. + * @implSpec the default implementation for this method calls {@code this.allocate(layout)}. + * @param layout the layout of the block of memory to be allocated. + * @param value the value to be set on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a float value. + */ + default MemorySegment allocate(ValueLayout layout, float value) { + Objects.requireNonNull(layout); + VarHandle handle = layout.varHandle(float.class); + MemorySegment addr = allocate(layout); + handle.set(addr, value); + return addr; + } + + /** + * Allocate a block of memory with given layout and initialize it with given long value. + * @implSpec the default implementation for this method calls {@code this.allocate(layout)}. + * @param layout the layout of the block of memory to be allocated. + * @param value the value to be set on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a long value. + */ + default MemorySegment allocate(ValueLayout layout, long value) { + Objects.requireNonNull(layout); + VarHandle handle = layout.varHandle(long.class); + MemorySegment addr = allocate(layout); + handle.set(addr, value); + return addr; + } + + /** + * Allocate a block of memory with given layout and initialize it with given double value. + * @implSpec the default implementation for this method calls {@code this.allocate(layout)}. + * @param layout the layout of the block of memory to be allocated. + * @param value the value to be set on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize()} does not conform to the size of a double value. + */ + default MemorySegment allocate(ValueLayout layout, double value) { + Objects.requireNonNull(layout); + VarHandle handle = layout.varHandle(double.class); + MemorySegment addr = allocate(layout); + handle.set(addr, value); + return addr; + } + + /** + * Allocate a block of memory with given layout and initialize it with given address value + * (expressed as an {@link Addressable} instance). + * The address value might be narrowed according to the platform address size (see {@link MemoryLayouts#ADDRESS}). + * @implSpec the default implementation for this method calls {@code this.allocate(layout)}. + * @param layout the layout of the block of memory to be allocated. + * @param value the value to be set on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize() != MemoryLayouts.ADDRESS.byteSize()}. + */ + default MemorySegment allocate(ValueLayout layout, Addressable value) { + Objects.requireNonNull(value); + Objects.requireNonNull(layout); + if (MemoryLayouts.ADDRESS.byteSize() != layout.byteSize()) { + throw new IllegalArgumentException("Layout size mismatch - " + layout.byteSize() + " != " + MemoryLayouts.ADDRESS.byteSize()); + } + return switch ((int)layout.byteSize()) { + case 4 -> allocate(layout, (int)value.address().toRawLongValue()); + case 8 -> allocate(layout, value.address().toRawLongValue()); + default -> throw new UnsupportedOperationException("Unsupported pointer size"); // should not get here + }; + } + + /** + * Allocate a block of memory with given layout and initialize it with given byte array. + * @implSpec the default implementation for this method calls {@code this.allocateArray(layout, array.length)}. + * @param elementLayout the element layout of the array to be allocated. + * @param array the array to be copied on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a byte value. + */ + default MemorySegment allocateArray(ValueLayout elementLayout, byte[] array) { + return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); + } + + /** + * Allocate a block of memory with given layout and initialize it with given short array. + * @implSpec the default implementation for this method calls {@code this.allocateArray(layout, array.length)}. + * @param elementLayout the element layout of the array to be allocated. + * @param array the array to be copied on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a short value. + */ + default MemorySegment allocateArray(ValueLayout elementLayout, short[] array) { + return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); + } + + /** + * Allocate a block of memory with given layout and initialize it with given char array. + * @implSpec the default implementation for this method calls {@code this.allocateArray(layout, array.length)}. + * @param elementLayout the element layout of the array to be allocated. + * @param array the array to be copied on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a char value. + */ + default MemorySegment allocateArray(ValueLayout elementLayout, char[] array) { + return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); + } + + /** + * Allocate a block of memory with given layout and initialize it with given int array. + * @implSpec the default implementation for this method calls {@code this.allocateArray(layout, array.length)}. + * @param elementLayout the element layout of the array to be allocated. + * @param array the array to be copied on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a int value. + */ + default MemorySegment allocateArray(ValueLayout elementLayout, int[] array) { + return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); + } + + /** + * Allocate a block of memory with given layout and initialize it with given float array. + * @implSpec the default implementation for this method calls {@code this.allocateArray(layout, array.length)}. + * @param elementLayout the element layout of the array to be allocated. + * @param array the array to be copied on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a float value. + */ + default MemorySegment allocateArray(ValueLayout elementLayout, float[] array) { + return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); + } + + /** + * Allocate a block of memory with given layout and initialize it with given long array. + * @implSpec the default implementation for this method calls {@code this.allocateArray(layout, array.length)}. + * @param elementLayout the element layout of the array to be allocated. + * @param array the array to be copied on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a long value. + */ + default MemorySegment allocateArray(ValueLayout elementLayout, long[] array) { + return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); + } + + /** + * Allocate a block of memory with given layout and initialize it with given double array. + * @implSpec the default implementation for this method calls {@code this.allocateArray(layout, array.length)}. + * @param elementLayout the element layout of the array to be allocated. + * @param array the array to be copied on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code elementLayout.byteSize()} does not conform to the size of a double value. + */ + default MemorySegment allocateArray(ValueLayout elementLayout, double[] array) { + return copyArrayWithSwapIfNeeded(array, elementLayout, MemorySegment::ofArray); + } + + /** + * Allocate a block of memory with given layout and initialize it with given address array. + * The address value of each array element might be narrowed according to the platform address size (see {@link MemoryLayouts#ADDRESS}). + * @implSpec the default implementation for this method calls {@code this.allocateArray(layout, array.length)}. + * @param elementLayout the element layout of the array to be allocated. + * @param array the array to be copied on the newly allocated memory block. + * @return a segment for the newly allocated memory block. + * @throws IllegalArgumentException if {@code layout.byteSize() != MemoryLayouts.ADDRESS.byteSize()}. + */ + default MemorySegment allocateArray(ValueLayout elementLayout, Addressable[] array) { + Objects.requireNonNull(elementLayout); + Objects.requireNonNull(array); + Stream.of(array).forEach(Objects::requireNonNull); + if (MemoryLayouts.ADDRESS.byteSize() != elementLayout.byteSize()) { + throw new IllegalArgumentException("Layout size mismatch - " + elementLayout.byteSize() + " != " + MemoryLayouts.ADDRESS.byteSize()); + } + return switch ((int)elementLayout.byteSize()) { + case 4 -> copyArrayWithSwapIfNeeded(Stream.of(array) + .mapToInt(a -> (int)a.address().toRawLongValue()).toArray(), + elementLayout, MemorySegment::ofArray); + case 8 -> copyArrayWithSwapIfNeeded(Stream.of(array) + .mapToLong(a -> a.address().toRawLongValue()).toArray(), + elementLayout, MemorySegment::ofArray); + default -> throw new UnsupportedOperationException("Unsupported pointer size"); // should not get here + }; + } + + private MemorySegment copyArrayWithSwapIfNeeded(Z array, ValueLayout elementLayout, + Function heapSegmentFactory) { + Objects.requireNonNull(array); + Objects.requireNonNull(elementLayout); + Utils.checkPrimitiveCarrierCompat(array.getClass().componentType(), elementLayout); + MemorySegment addr = allocate(MemoryLayout.sequenceLayout(Array.getLength(array), elementLayout)); + if (elementLayout.byteSize() == 1 || (elementLayout.order() == ByteOrder.nativeOrder())) { + addr.copyFrom(heapSegmentFactory.apply(array)); + } else { + ((AbstractMemorySegmentImpl)addr).copyFromSwap(heapSegmentFactory.apply(array), elementLayout.byteSize()); + } + return addr; + } + + /** + * Allocate a block of memory with given layout. + * @implSpec the default implementation for this method calls {@code this.allocate(layout.byteSize(), layout.byteAlignment())}. + * @param layout the layout of the block of memory to be allocated. + * @return a segment for the newly allocated memory block. + */ + default MemorySegment allocate(MemoryLayout layout) { + Objects.requireNonNull(layout); + return allocate(layout.byteSize(), layout.byteAlignment()); + } + + /** + * Allocate a block of memory corresponding to an array with given element layout and size. + * @implSpec the default implementation for this method calls {@code this.allocate(MemoryLayout.sequenceLayout(count, elementLayout))}. + * @param elementLayout the array element layout. + * @param count the array element count. + * @return a segment for the newly allocated memory block. + */ + default MemorySegment allocateArray(MemoryLayout elementLayout, long count) { + Objects.requireNonNull(elementLayout); + return allocate(MemoryLayout.sequenceLayout(count, elementLayout)); + } + + /** + * Allocate a block of memory with given size, with default alignment (1-byte aligned). + * @implSpec the default implementation for this method calls {@code this.allocate(bytesSize, 1)}. + * @param bytesSize the size (in bytes) of the block of memory to be allocated. + * @return a segment for the newly allocated memory block. + */ + default MemorySegment allocate(long bytesSize) { + return allocate(bytesSize, 1); + } + + /** + * Allocate a block of memory with given size and alignment constraint. + * @param bytesSize the size (in bytes) of the block of memory to be allocated. + * @param bytesAlignment the alignment (in bytes) of the block of memory to be allocated. + * @return a segment for the newly allocated memory block. + */ + MemorySegment allocate(long bytesSize, long bytesAlignment); + + /** + * Returns a native arena-based allocator which allocates a single memory segment, of given size (using malloc), + * and then responds to allocation request by returning different slices of that same segment + * (until no further allocation is possible). + * This can be useful when clients want to perform multiple allocation requests while avoiding the cost associated + * with allocating a new off-heap memory region upon each allocation request. + *

    + * An allocator associated with a shared resource scope is thread-safe and allocation requests may be + * performed concurrently; conversely, if the arena allocator is associated with a confined resource scope, + * allocation requests can only occur from the thread owning the allocator's resource scope. + *

    + * The returned allocator might throw an {@link OutOfMemoryError} if an incoming allocation request exceeds + * the allocator capacity. + * + * @param size the size (in bytes) of the allocation arena. + * @param scope the scope associated with the segments returned by this allocator. + * @return a new bounded arena-based allocator + * @throws IllegalArgumentException if {@code size <= 0}. + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. + */ + static SegmentAllocator arenaAllocator(long size, ResourceScope scope) { + Objects.requireNonNull(scope); + return scope.ownerThread() == null ? + new ArenaAllocator.BoundedSharedArenaAllocator(scope, size) : + new ArenaAllocator.BoundedArenaAllocator(scope, size); + } + + /** + * Returns a native unbounded arena-based allocator. + *

    + * The returned allocator allocates a memory segment {@code S} of a certain fixed size (using malloc) and then + * responds to allocation requests in one of the following ways: + *

      + *
    • if the size of the allocation requests is smaller than the size of {@code S}, and {@code S} has a free + * slice {@code S'} which fits that allocation request, return that {@code S'}. + *
    • if the size of the allocation requests is smaller than the size of {@code S}, and {@code S} has no free + * slices which fits that allocation request, allocate a new segment {@code S'} (using malloc), which has same size as {@code S} + * and set {@code S = S'}; the allocator then tries to respond to the same allocation request again. + *
    • if the size of the allocation requests is bigger than the size of {@code S}, allocate a new segment {@code S'} + * (using malloc), which has a sufficient size to satisfy the allocation request, and return {@code S'}. + *
    + *

    + * This segment allocator can be useful when clients want to perform multiple allocation requests while avoiding the + * cost associated with allocating a new off-heap memory region upon each allocation request. + *

    + * An allocator associated with a shared resource scope is thread-safe and allocation requests may be + * performed concurrently; conversely, if the arena allocator is associated with a confined resource scope, + * allocation requests can only occur from the thread owning the allocator's resource scope. + *

    + * The returned allocator might throw an {@link OutOfMemoryError} if an incoming allocation request exceeds + * the system capacity. + * + * @param scope the scope associated with the segments returned by this allocator. + * @return a new unbounded arena-based allocator + * @throws IllegalStateException if {@code scope} has been already closed, or if access occurs from a thread other + * than the thread owning {@code scope}. + */ + static SegmentAllocator arenaAllocator(ResourceScope scope) { + Objects.requireNonNull(scope); + return scope.ownerThread() == null ? + new ArenaAllocator.UnboundedSharedArenaAllocator(scope) : + new ArenaAllocator.UnboundedArenaAllocator(scope); + } + + /** + * Returns a segment allocator which responds to allocation requests by recycling a single segment; that is, + * each new allocation request will return a new slice starting at the segment offset {@code 0} (alignment + * constraints are ignored by this allocator). This can be useful to limit allocation requests in case a client + * knows that they have fully processed the contents of the allocated segment before the subsequent allocation request + * takes place. + *

    + * While the allocator returned by this method is thread-safe, concurrent access on the same recycling + * allocator might cause a thread to overwrite contents written to the underlying segment by a different thread. + * + * @param segment the memory segment to be recycled by the returned allocator. + * @return an allocator which recycles an existing segment upon each new allocation request. + */ + static SegmentAllocator ofSegment(MemorySegment segment) { + Objects.requireNonNull(segment); + return (size, align) -> segment.asSlice(0, size); + } + + /** + * Returns a native allocator which responds to allocation requests by allocating new segments + * bound by the given resource scope, using the {@link MemorySegment#allocateNative(long, long, ResourceScope)} + * factory. This code is equivalent (but likely more efficient) to the following: + *

    {@code
    +    Resource scope = ...
    +    SegmentAllocator scoped = (size, align) -> MemorySegment.allocateNative(size, align, scope);
    +     * }
    + * + * @param scope the resource scope associated with the segments created by the returned allocator. + * @return an allocator which allocates new memory segment bound by the provided resource scope. + */ + static SegmentAllocator ofScope(ResourceScope scope) { + Objects.requireNonNull(scope); + return (ResourceScopeImpl)scope; + } +} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/SequenceLayout.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/SequenceLayout.java index 584c34e9335..8b79cb2fb9f 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/SequenceLayout.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/SequenceLayout.java @@ -32,7 +32,6 @@ import java.util.Map; import java.util.Objects; import java.util.Optional; import java.util.OptionalLong; -import java.util.stream.LongStream; /** * A sequence layout. A sequence layout is used to denote a repetition of a given layout, also called the sequence layout's element layout. @@ -41,16 +40,16 @@ import java.util.stream.LongStream; * that is equal to the sequence layout's element count. In other words this layout: * *
    {@code
    -MemoryLayout.ofSequence(3, MemoryLayout.ofValueBits(32, ByteOrder.BIG_ENDIAN));
    +MemoryLayout.sequenceLayout(3, MemoryLayout.valueLayout(32, ByteOrder.BIG_ENDIAN));
      * }
    * * is equivalent to the following layout: * *
    {@code
    -MemoryLayout.ofStruct(
    -    MemoryLayout.ofValueBits(32, ByteOrder.BIG_ENDIAN),
    -    MemoryLayout.ofValueBits(32, ByteOrder.BIG_ENDIAN),
    -    MemoryLayout.ofValueBits(32, ByteOrder.BIG_ENDIAN));
    +MemoryLayout.structLayout(
    +    MemoryLayout.valueLayout(32, ByteOrder.BIG_ENDIAN),
    +    MemoryLayout.valueLayout(32, ByteOrder.BIG_ENDIAN),
    +    MemoryLayout.valueLayout(32, ByteOrder.BIG_ENDIAN));
      * }
    * *

    @@ -67,7 +66,7 @@ MemoryLayout.ofStruct( * @implSpec * This class is immutable and thread-safe. */ -public final class SequenceLayout extends AbstractLayout { +public final class SequenceLayout extends AbstractLayout implements MemoryLayout { private final OptionalLong elemCount; private final MemoryLayout elementLayout; @@ -123,11 +122,11 @@ public final class SequenceLayout extends AbstractLayout { *

    * For instance, given a sequence layout of the kind: *

    {@code
    -    var seq = MemoryLayout.ofSequence(4, MemoryLayout.ofSequence(3, MemoryLayouts.JAVA_INT));
    +    var seq = MemoryLayout.sequenceLayout(4, MemoryLayout.sequenceLayout(3, MemoryLayouts.JAVA_INT));
          * }
    * calling {@code seq.reshape(2, 6)} will yield the following sequence layout: *
    {@code
    -    var reshapeSeq = MemoryLayout.ofSequence(2, MemoryLayout.ofSequence(6, MemoryLayouts.JAVA_INT));
    +    var reshapeSeq = MemoryLayout.sequenceLayout(2, MemoryLayout.sequenceLayout(6, MemoryLayouts.JAVA_INT));
          * }
    *

    * If one of the provided element count is the special value {@code -1}, then the element @@ -187,7 +186,7 @@ public final class SequenceLayout extends AbstractLayout { MemoryLayout res = flat.elementLayout(); for (int i = elementCounts.length - 1 ; i >= 0 ; i--) { - res = MemoryLayout.ofSequence(elementCounts[i], res); + res = MemoryLayout.sequenceLayout(elementCounts[i], res); } return (SequenceLayout)res; } @@ -199,11 +198,11 @@ public final class SequenceLayout extends AbstractLayout { * be dropped and their element counts will be incorporated into that of the returned sequence layout. * For instance, given a sequence layout of the kind: *

    {@code
    -    var seq = MemoryLayout.ofSequence(4, MemoryLayout.ofSequence(3, MemoryLayouts.JAVA_INT));
    +    var seq = MemoryLayout.sequenceLayout(4, MemoryLayout.sequenceLayout(3, MemoryLayouts.JAVA_INT));
          * }
    * calling {@code seq.flatten()} will yield the following sequence layout: *
    {@code
    -    var flattenedSeq = MemoryLayout.ofSequence(12, MemoryLayouts.JAVA_INT);
    +    var flattenedSeq = MemoryLayout.sequenceLayout(12, MemoryLayouts.JAVA_INT);
          * }
    * @return a new sequence layout with the same size as this layout (but, possibly, with different * element count), whose element layout is not a sequence layout. @@ -221,7 +220,7 @@ public final class SequenceLayout extends AbstractLayout { count = count * elemSeq.elementCount().orElseThrow(this::badUnboundSequenceLayout); elemLayout = elemSeq.elementLayout(); } - return MemoryLayout.ofSequence(count, elemLayout); + return MemoryLayout.sequenceLayout(count, elemLayout); } private UnsupportedOperationException badUnboundSequenceLayout() { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/package-info.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/package-info.java index 322da7fcc0a..e7c35db0301 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/package-info.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/package-info.java @@ -45,19 +45,15 @@ * ranging from {@code 0} to {@code 9}, we can use the following code: * *
    {@code
    -try (MemorySegment segment = MemorySegment.allocateNative(10 * 4)) {
    -    for (int i = 0 ; i < 10 ; i++) {
    -       MemoryAccess.setIntAtIndex(segment, i);
    -    }
    +MemorySegment segment = MemorySegment.allocateNative(10 * 4, ResourceScope.newImplicitScope());
    +for (int i = 0 ; i < 10 ; i++) {
    +   MemoryAccess.setIntAtIndex(segment, i, 42);
     }
      * }
    * * Here create a native memory segment, that is, a memory segment backed by * off-heap memory; the size of the segment is 40 bytes, enough to store 10 values of the primitive type {@code int}. - * The segment is created inside a try-with-resources construct: this idiom ensures that all the memory resources - * associated with the segment will be released at the end of the block, according to the semantics described in - * Section {@jls 14.20.3} of The Java Language Specification. Inside the try-with-resources block, we initialize - * the contents of the memory segment using the + * Inside a loop, we then initialize the contents of the memory segment using the * {@link jdk.incubator.foreign.MemoryAccess#setIntAtIndex(jdk.incubator.foreign.MemorySegment, long, int)} helper method; * more specifically, if we view the memory segment as a set of 10 adjacent slots, * {@code s[i]}, where {@code 0 <= i < 10}, where the size of each slot is exactly 4 bytes, the initialization logic above will set each slot @@ -66,16 +62,25 @@ try (MemorySegment segment = MemorySegment.allocateNative(10 * 4)) { *

    Deterministic deallocation

    * * When writing code that manipulates memory segments, especially if backed by memory which resides outside the Java heap, it is - * crucial that the resources associated with a memory segment are released when the segment is no longer in use, by calling the {@link jdk.incubator.foreign.MemorySegment#close()} - * method either explicitly, or implicitly, by relying on try-with-resources construct (as demonstrated in the example above). - * Closing a given memory segment is an atomic operation which can either succeed - and result in the underlying - * memory associated with the segment to be released, or fail with an exception. - *

    - * The deterministic deallocation model differs significantly from the implicit strategies adopted within other APIs, most - * notably the {@link java.nio.ByteBuffer} API: in that case, when a native byte buffer is created (see {@link java.nio.ByteBuffer#allocateDirect(int)}), - * the underlying memory is not released until the byte buffer reference becomes unreachable. While implicit deallocation - * models such as this can be very convenient - clients do not have to remember to close a direct buffer - such models can also make it - * hard for clients to ensure that the memory associated with a direct buffer has indeed been released. + * often crucial that the resources associated with a memory segment are released when the segment is no longer in use, + * and in a timely fashion. For this reason, there might be cases where waiting for the garbage collector to determine that a segment + * is unreachable is not optimal. + * Clients that operate under these assumptions might want to programmatically release the memory associated + * with a memory segment. This can be done, using the {@link jdk.incubator.foreign.ResourceScope} abstraction, as shown below: + * + *

    {@code
    +try (ResourceScope scope = ResourceScope.newConfinedScope()) {
    +    MemorySegment segment = MemorySegment.allocateNative(10 * 4, scope);
    +    for (int i = 0 ; i < 10 ; i++) {
    +        MemoryAccess.setIntAtIndex(segment, i, 42);
    +    }
    +}
    + * }
    + * + * This example is almost identical to the prior one; this time we first create a so called resource scope, + * which is used to bind the life-cycle of the segment created immediately afterwards. Note the use of the + * try-with-resources construct: this idiom ensures that all the memory resources associated with the segment will be released + * at the end of the block, according to the semantics described in Section {@jls 14.20.3} of The Java Language Specification. * *

    Safety

    * @@ -86,14 +91,9 @@ try (MemorySegment segment = MemorySegment.allocateNative(10 * 4)) { * Section {@jls 15.10.4} of The Java Language Specification. *

    * Since memory segments can be closed (see above), segments are also validated (upon access) to make sure that - * the segment being accessed has not been closed prematurely. We call this guarantee temporal safety. Note that, - * in the general case, guaranteeing temporal safety can be hard, as multiple threads could attempt to access and/or close - * the same memory segment concurrently. The memory access API addresses this problem by imposing strong - * thread-confinement guarantees on memory segments: upon creation, a memory segment is associated with an owner thread, - * which is the only thread that can either access or close the segment. - *

    - * Together, spatial and temporal safety ensure that each memory access operation either succeeds - and accesses a valid - * memory location - or fails. + * the resource scope associated with the segment being accessed has not been closed prematurely. + * We call this guarantee temporal safety. Together, spatial and temporal safety ensure that each memory access + * operation either succeeds - and accesses a valid memory location - or fails. * *

    Foreign function access

    * The key abstractions introduced to support foreign function access are {@link jdk.incubator.foreign.LibraryLookup} and {@link jdk.incubator.foreign.CLinker}. @@ -106,15 +106,16 @@ try (MemorySegment segment = MemorySegment.allocateNative(10 * 4)) { * we can use the following code: * *
    {@code
    -MethodHandle strlen = CLinker.getInstance().downcallHandle(
    +      MethodHandle strlen = CLinker.getInstance().downcallHandle(
             LibraryLookup.ofDefault().lookup("strlen").get(),
             MethodType.methodType(long.class, MemoryAddress.class),
             FunctionDescriptor.of(CLinker.C_LONG, CLinker.C_POINTER)
    -);
    +      );
     
    -try (var cString = CLinker.toCString("Hello")) {
    -    long len = strlen.invokeExact(cString.address()) // 5
    -}
    +      try (var scope = ResourceScope.newConfinedScope()) {
    +         var cString = CLinker.toCString("Hello", scope);
    +         long len = (long)strlen.invokeExact(cString.address()); // 5
    +      }
      * }
    * * Here, we lookup the {@code strlen} symbol in the default library lookup (see {@link jdk.incubator.foreign.LibraryLookup#ofDefault()}). @@ -126,7 +127,7 @@ try (var cString = CLinker.toCString("Hello")) { * the method handle invocation (here performed using {@link java.lang.invoke.MethodHandle#invokeExact(java.lang.Object...)}) * into a foreign function call, according to the rules specified by the platform C ABI. The {@link jdk.incubator.foreign.CLinker} * class also provides many useful methods for interacting with native code, such as converting Java strings into - * native strings and viceversa (see {@link jdk.incubator.foreign.CLinker#toCString(java.lang.String)} and + * native strings and viceversa (see {@link jdk.incubator.foreign.CLinker#toCString(java.lang.String, ResourceScope)} and * {@link jdk.incubator.foreign.CLinker#toJavaString(jdk.incubator.foreign.MemorySegment)}, respectively), as * demonstrated in the above example. * @@ -146,43 +147,44 @@ try (var cString = CLinker.toCString("Hello")) { * the original segment accordingly, as follows: * *
    {@code
    -MemorySegment segment = MemorySegment.allocateNative(100);
    +MemorySegment segment = MemorySegment.allocateNative(100, scope);
     ...
     MemoryAddress addr = ... //obtain address from native code
     int x = MemoryAccess.getIntAtOffset(segment, addr.segmentOffset(segment));
      * }
    * * Secondly, if the client does not have a segment which contains a given memory address, it can create one unsafely, - * using the {@link jdk.incubator.foreign.MemoryAddress#asSegmentRestricted(long)} factory. This allows the client to + * using the {@link jdk.incubator.foreign.MemoryAddress#asSegment(long, ResourceScope)} factory. This allows the client to * inject extra knowledge about spatial bounds which might, for instance, be available in the documentation of the foreign function * which produced the native address. Here is how an unsafe segment can be created from a native address: * *
    {@code
    +ResourceScope scope = ... // initialize a resource scope object
     MemoryAddress addr = ... //obtain address from native code
    -MemorySegment segment = addr.asSegmentRestricted(4); // segment is 4 bytes long
    +MemorySegment segment = addr.asSegment(4, scope); // segment is 4 bytes long
     int x = MemoryAccess.getInt(segment);
      * }
    * * Alternatively, the client can fall back to use the so called everything segment - that is, a primordial segment - * which covers the entire native heap. This segment can be obtained by calling the {@link jdk.incubator.foreign.MemorySegment#ofNativeRestricted()} + * which covers the entire native heap. This segment can be obtained by calling the {@link jdk.incubator.foreign.MemorySegment#globalNativeSegment()} * method, so that dereference can happen without the need of creating any additional segment instances: * *
    {@code
     MemoryAddress addr = ... //obtain address from native code
    -int x = MemoryAccess.getIntAtOffset(MemorySegment.ofNativeRestricted(), addr.toRawLongValue());
    +int x = MemoryAccess.getIntAtOffset(MemorySegment.globalNativeSegment(), addr.toRawLongValue());
      * }
    * *

    Upcalls

    * The {@link jdk.incubator.foreign.CLinker} interface also allows to turn an existing method handle (which might point - * to a Java method) into a native memory segment (see {@link jdk.incubator.foreign.MemorySegment}), so that Java code + * to a Java method) into a native memory address (see {@link jdk.incubator.foreign.MemoryAddress}), so that Java code * can effectively be passed to other foreign functions. For instance, we can write a method that compares two * integer values, as follows: * *
    {@code
     class IntComparator {
         static int intCompare(MemoryAddress addr1, MemoryAddress addr2) {
    -        return MemoryAccess.getIntAtOffset(MemorySegment.ofNativeRestricted(), addr1.toRawLongValue()) -
    -               MemoryAccess.getIntAtOffset(MemorySegment.ofNativeRestricted(), addr2.toRawLongValue());
    +        return MemoryAccess.getIntAtOffset(MemorySegment.globalNativeSegment(), addr1.toRawLongValue()) -
    +               MemoryAccess.getIntAtOffset(MemorySegment.globalNativeSegment(), addr2.toRawLongValue());
         }
     }
      * }
    @@ -197,39 +199,38 @@ MethodHandle intCompareHandle = MethodHandles.lookup().findStatic(IntComparator. MethodType.methodType(int.class, MemoryAddress.class, MemoryAddress.class)); * } * - * Now that we have a method handle instance, we can link it into a fresh native memory segment, using the {@link jdk.incubator.foreign.CLinker} interface, as follows: + * Now that we have a method handle instance, we can link it into a fresh native memory address, using the {@link jdk.incubator.foreign.CLinker} interface, as follows: * *
    {@code
    -MemorySegment comparFunc = CLinker.getInstance().upcallStub(
    +ResourceScope scope = ...
    +MemoryAddress comparFunc = CLinker.getInstance().upcallStub(
          intCompareHandle,
    -     FunctionDescriptor.of(C_INT, C_POINTER, C_POINTER)
    +     FunctionDescriptor.of(C_INT, C_POINTER, C_POINTER),
    +     scope
     );
      * }
    * * As before, we need to provide a {@link jdk.incubator.foreign.FunctionDescriptor} instance describing the signature * of the function pointer we want to create; as before, this, coupled with the method handle type, uniquely determines the * sequence of steps which will allow foreign code to call {@code intCompareHandle} according to the rules specified - * by the platform C ABI. + * by the platform C ABI. The lifecycle of the memory address returned by + * {@link jdk.incubator.foreign.CLinker#upcallStub(java.lang.invoke.MethodHandle, jdk.incubator.foreign.FunctionDescriptor, jdk.incubator.foreign.ResourceScope)} + * is tied to the {@linkplain jdk.incubator.foreign.ResourceScope resource scope} parameter passed to that method. * + * *

    Restricted methods

    * Some methods in this package are considered restricted. Restricted methods are typically used to bind native - * foreign data and/or functions to first-class Java API elements which can then be used directly by client. For instance - * the restricted method {@link jdk.incubator.foreign.MemoryAddress#asSegmentRestricted(long)} can be used to create + * foreign data and/or functions to first-class Java API elements which can then be used directly by clients. For instance + * the restricted method {@link jdk.incubator.foreign.MemoryAddress#asSegment(long, ResourceScope)} can be used to create * a fresh segment with given spatial bounds out of a native address. *

    * Binding foreign data and/or functions is generally unsafe and, if done incorrectly, can result in VM crashes, or memory corruption when the bound Java API element is accessed. - * For instance, in the case of {@link jdk.incubator.foreign.MemoryAddress#asSegmentRestricted(long)}, if the provided + * For instance, in the case of {@link jdk.incubator.foreign.MemoryAddress#asSegment(long, ResourceScope)}, if the provided * spatial bounds are incorrect, a client of the segment returned by that method might crash the VM, or corrupt * memory when attempting to dereference said segment. For these reasons, it is crucial for code that calls a restricted method * to never pass arguments that might cause incorrect binding of foreign data and/or functions to a Java API. *

    - * Access to restricted methods is disabled by default; to enable restricted methods, the JDK property - * {@code foreign.restricted} must be set to a value other than {@code deny}. The possible values for this property are: - *

      - *
    • {@code deny}: issues a runtime exception on each restricted call. This is the default value;
    • - *
    • {@code permit}: allows restricted calls;
    • - *
    • {@code warn}: like permit, but also prints a one-line warning on each restricted call;
    • - *
    • {@code debug}: like permit, but also dumps the stack corresponding to any given restricted call.
    • - *
    + * Access to restricted methods is disabled by default; to enable restricted methods, the command line option + * {@code --enable-native-access} must mention the name of the caller's module. */ package jdk.incubator.foreign; diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractCLinker.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractCLinker.java new file mode 100644 index 00000000000..b13ccd2fa94 --- /dev/null +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractCLinker.java @@ -0,0 +1,55 @@ +/* + * Copyright (c) 2021, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + * + */ +package jdk.internal.foreign; + +import jdk.incubator.foreign.Addressable; +import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.FunctionDescriptor; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.SegmentAllocator; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; +import java.util.Objects; + +public abstract non-sealed class AbstractCLinker implements CLinker { + + public final MethodHandle downcallHandle(Addressable symbol, MethodType type, FunctionDescriptor function) { + Objects.requireNonNull(symbol); + return MethodHandles.insertArguments(downcallHandle(type, function), 0, symbol); + } + + public final MethodHandle downcallHandle(Addressable symbol, SegmentAllocator allocator, MethodType type, FunctionDescriptor function) { + Objects.requireNonNull(symbol); + Objects.requireNonNull(allocator); + MethodHandle downcall = MethodHandles.insertArguments(downcallHandle(type, function), 0, symbol); + if (type.returnType().equals(MemorySegment.class)) { + downcall = MethodHandles.insertArguments(downcall, 0, allocator); + } + return downcall; + } +} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java index 971533569cf..b54ba68f29f 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java @@ -35,43 +35,42 @@ import jdk.internal.util.ArraysSupport; import jdk.internal.vm.annotation.ForceInline; import sun.security.action.GetPropertyAction; -import java.io.FileDescriptor; -import java.lang.invoke.VarHandle; -import java.lang.ref.Cleaner; import java.nio.ByteBuffer; import java.util.*; import java.util.function.Consumer; import java.util.function.Function; import java.util.function.IntFunction; +import java.util.stream.Stream; +import java.util.stream.StreamSupport; /** * This abstract class provides an immutable implementation for the {@code MemorySegment} interface. This class contains information * about the segment's spatial and temporal bounds; each memory segment implementation is associated with an owner thread which is set at creation time. * Access to certain sensitive operations on the memory segment will fail with {@code IllegalStateException} if the * segment is either in an invalid state (e.g. it has already been closed) or if access occurs from a thread other - * than the owner thread. See {@link MemoryScope} for more details on management of temporal bounds. Subclasses + * than the owner thread. See {@link ResourceScopeImpl} for more details on management of temporal bounds. Subclasses * are defined for each memory segment kind, see {@link NativeMemorySegmentImpl}, {@link HeapMemorySegmentImpl} and * {@link MappedMemorySegmentImpl}. */ -public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy implements MemorySegment { +public abstract non-sealed class AbstractMemorySegmentImpl extends MemorySegmentProxy implements MemorySegment { private static final ScopedMemoryAccess SCOPED_MEMORY_ACCESS = ScopedMemoryAccess.getScopedMemoryAccess(); private static final boolean enableSmallSegments = Boolean.parseBoolean(GetPropertyAction.privilegedGetProperty("jdk.incubator.foreign.SmallSegments", "true")); - final static int FIRST_RESERVED_FLAG = 1 << 16; // upper 16 bits are reserved - final static int SMALL = FIRST_RESERVED_FLAG; - final static long NONCE = new Random().nextLong(); + static final int READ_ONLY = 1; + static final int SMALL = READ_ONLY << 1; + static final long NONCE = new Random().nextLong(); - final static JavaNioAccess nioAccess = SharedSecrets.getJavaNioAccess(); + static final JavaNioAccess nioAccess = SharedSecrets.getJavaNioAccess(); final long length; final int mask; - final MemoryScope scope; + final ResourceScopeImpl scope; @ForceInline - AbstractMemorySegmentImpl(long length, int mask, MemoryScope scope) { + AbstractMemorySegmentImpl(long length, int mask, ResourceScopeImpl scope) { this.length = length; this.mask = mask; this.scope = scope; @@ -81,14 +80,23 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple abstract Object base(); - abstract AbstractMemorySegmentImpl dup(long offset, long size, int mask, MemoryScope scope); + abstract AbstractMemorySegmentImpl dup(long offset, long size, int mask, ResourceScopeImpl scope); abstract ByteBuffer makeByteBuffer(); static int defaultAccessModes(long size) { return (enableSmallSegments && size < Integer.MAX_VALUE) ? - ALL_ACCESS | SMALL : - ALL_ACCESS; + SMALL : 0; + } + + @Override + public AbstractMemorySegmentImpl asReadOnly() { + return dup(0, length, mask | READ_ONLY, scope); + } + + @Override + public boolean isReadOnly() { + return isSet(READ_ONLY); } @Override @@ -108,14 +116,21 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple } @Override - public Spliterator spliterator(SequenceLayout sequenceLayout) { - Objects.requireNonNull(sequenceLayout); - checkValidState(); - if (sequenceLayout.byteSize() != byteSize()) { - throw new IllegalArgumentException(); + public Spliterator spliterator(MemoryLayout elementLayout) { + Objects.requireNonNull(elementLayout); + if (elementLayout.byteSize() == 0) { + throw new IllegalArgumentException("Element layout size cannot be zero"); } - return new SegmentSplitter(sequenceLayout.elementLayout().byteSize(), sequenceLayout.elementCount().getAsLong(), - withAccessModes(accessModes() & ~CLOSE)); + if (byteSize() % elementLayout.byteSize() != 0) { + throw new IllegalArgumentException("Segment size is no a multiple of layout size"); + } + return new SegmentSplitter(elementLayout.byteSize(), byteSize() / elementLayout.byteSize(), + this); + } + + @Override + public Stream elements(MemoryLayout elementLayout) { + return StreamSupport.stream(spliterator(elementLayout), false); } @Override @@ -185,10 +200,10 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple /** * Mismatch over long lengths. */ - private static long vectorizedMismatchLargeForBytes(MemoryScope aScope, MemoryScope bScope, - Object a, long aOffset, - Object b, long bOffset, - long length) { + private static long vectorizedMismatchLargeForBytes(ResourceScopeImpl aScope, ResourceScopeImpl bScope, + Object a, long aOffset, + Object b, long bOffset, + long length) { long off = 0; long remaining = length; int i, size; @@ -217,132 +232,63 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple @Override @ForceInline public final MemoryAddress address() { - checkValidState(); - return new MemoryAddressImpl(base(), min()); + return new MemoryAddressImpl(this, 0L); } @Override public final ByteBuffer asByteBuffer() { - if (!isSet(READ)) { - throw unsupportedAccessMode(READ); - } checkArraySize("ByteBuffer", 1); ByteBuffer _bb = makeByteBuffer(); - if (!isSet(WRITE)) { + if (isSet(READ_ONLY)) { //scope is IMMUTABLE - obtain a RO byte buffer _bb = _bb.asReadOnlyBuffer(); } return _bb; } - @Override - public final int accessModes() { - return mask & ALL_ACCESS; - } - @Override public final long byteSize() { return length; } - @Override public final boolean isAlive() { return scope.isAlive(); } - @Override public Thread ownerThread() { return scope.ownerThread(); } - @Override - public AbstractMemorySegmentImpl withAccessModes(int accessModes) { - checkAccessModes(accessModes); - if ((~accessModes() & accessModes) != 0) { - throw new IllegalArgumentException("Cannot acquire more access modes"); - } - return dup(0, length, (mask & ~ALL_ACCESS) | accessModes, scope); - } - - @Override - public boolean hasAccessModes(int accessModes) { - checkAccessModes(accessModes); - return (accessModes() & accessModes) == accessModes; - } - - private void checkAccessModes(int accessModes) { - if ((accessModes & ~ALL_ACCESS) != 0) { - throw new IllegalArgumentException("Invalid access modes"); - } - } - - public MemorySegment handoff(Thread thread) { - Objects.requireNonNull(thread); - checkValidState(); - if (!isSet(HANDOFF)) { - throw unsupportedAccessMode(HANDOFF); - } - try { - return dup(0L, length, mask, scope.confineTo(thread)); - } finally { - //flush read/writes to segment memory before returning the new segment - VarHandle.fullFence(); - } - } - - @Override - public MemorySegment share() { - checkValidState(); - if (!isSet(SHARE)) { - throw unsupportedAccessMode(SHARE); - } - try { - return dup(0L, length, mask, scope.share()); - } finally { - //flush read/writes to segment memory before returning the new segment - VarHandle.fullFence(); - } - } - - @Override - public MemorySegment handoff(NativeScope scope) { - Objects.requireNonNull(scope); - checkValidState(); - if (!isSet(HANDOFF)) { - throw unsupportedAccessMode(HANDOFF); - } - if (!isSet(CLOSE)) { - throw unsupportedAccessMode(CLOSE); - } - MemorySegment dup = handoff(scope.ownerThread()); - ((AbstractNativeScope)scope).register(dup); - return dup.withAccessModes(accessModes() & (READ | WRITE)); - } - - @Override - public MemorySegment registerCleaner(Cleaner cleaner) { - Objects.requireNonNull(cleaner); - checkValidState(); - if (!isSet(CLOSE)) { - throw unsupportedAccessMode(CLOSE); - } - return dup(0L, length, mask, scope.cleanable(cleaner)); - } - - @Override - public final void close() { - checkValidState(); - if (!isSet(CLOSE)) { - throw unsupportedAccessMode(CLOSE); - } - scope.close(); - } - @Override public boolean isMapped() { return false; } + @Override + public boolean isNative() { + return false; + } + + @Override + public void load() { + throw new UnsupportedOperationException("Not a mapped segment"); + } + + @Override + public void unload() { + throw new UnsupportedOperationException("Not a mapped segment"); + } + + @Override + public boolean isLoaded() { + throw new UnsupportedOperationException("Not a mapped segment"); + } + + @Override + public void force() { + throw new UnsupportedOperationException("Not a mapped segment"); + } + @Override public final byte[] toByteArray() { return toArray(byte[].class, 1, byte[]::new, MemorySegment::ofArray); @@ -393,20 +339,13 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple @Override public void checkAccess(long offset, long length, boolean readOnly) { - if (!readOnly && !isSet(WRITE)) { - throw unsupportedAccessMode(WRITE); - } else if (readOnly && !isSet(READ)) { - throw unsupportedAccessMode(READ); + if (!readOnly && isSet(READ_ONLY)) { + throw new UnsupportedOperationException("Attempt to write a read-only segment"); } checkBounds(offset, length); } - private void checkAccessAndScope(long offset, long length, boolean readOnly) { - checkValidState(); - checkAccess(offset, length, readOnly); - } - - private void checkValidState() { + void checkValidState() { try { scope.checkValidState(); } catch (ScopedMemoryAccess.Scope.ScopedAccessError ex) { @@ -432,17 +371,19 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple private int checkArraySize(String typeName, int elemSize) { if (length % elemSize != 0) { - throw new UnsupportedOperationException(String.format("Segment size is not a multiple of %d. Size: %d", elemSize, length)); + throw new IllegalStateException(String.format("Segment size is not a multiple of %d. Size: %d", elemSize, length)); } long arraySize = length / elemSize; if (arraySize > (Integer.MAX_VALUE - 8)) { //conservative check - throw new UnsupportedOperationException(String.format("Segment is too large to wrap as %s. Size: %d", typeName, length)); + throw new IllegalStateException(String.format("Segment is too large to wrap as %s. Size: %d", typeName, length)); } return (int)arraySize; } private void checkBounds(long offset, long length) { - if (isSmall()) { + if (isSmall() && + offset < Integer.MAX_VALUE && length < Integer.MAX_VALUE && + offset > Integer.MIN_VALUE && length > Integer.MIN_VALUE) { checkBoundsSmall((int)offset, (int)length); } else { if (length < 0 || @@ -454,7 +395,7 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple } @Override - public MemoryScope scope() { + public ResourceScopeImpl scope() { return scope; } @@ -466,31 +407,6 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple } } - UnsupportedOperationException unsupportedAccessMode(int expected) { - return new UnsupportedOperationException((String.format("Required access mode %s ; current access modes: %s", - modeStrings(expected).get(0), modeStrings(mask)))); - } - - private List modeStrings(int mode) { - List modes = new ArrayList<>(); - if ((mode & READ) != 0) { - modes.add("READ"); - } - if ((mode & WRITE) != 0) { - modes.add("WRITE"); - } - if ((mode & CLOSE) != 0) { - modes.add("CLOSE"); - } - if ((mode & SHARE) != 0) { - modes.add("SHARE"); - } - if ((mode & HANDOFF) != 0) { - modes.add("HANDOFF"); - } - return modes; - } - private IndexOutOfBoundsException outOfBoundException(long offset, long length) { return new IndexOutOfBoundsException(String.format("Out of bound access on segment %s; new offset = %d; new length = %d", this, offset, length)); @@ -602,20 +518,20 @@ public abstract class AbstractMemorySegmentImpl extends MemorySegmentProxy imple int size = limit - pos; AbstractMemorySegmentImpl bufferSegment = (AbstractMemorySegmentImpl)nioAccess.bufferSegment(bb); - final MemoryScope bufferScope; + final ResourceScopeImpl bufferScope; int modes; if (bufferSegment != null) { bufferScope = bufferSegment.scope; modes = bufferSegment.mask; } else { - bufferScope = MemoryScope.createConfined(bb, MemoryScope.DUMMY_CLEANUP_ACTION, null); + bufferScope = ResourceScopeImpl.GLOBAL; modes = defaultAccessModes(size); } if (bb.isReadOnly()) { - modes &= ~WRITE; + modes |= READ_ONLY; } if (base != null) { - return new HeapMemorySegmentImpl.OfByte(bbAddress + pos, (byte[])base, size, modes, bufferScope); + return new HeapMemorySegmentImpl.OfByte(bbAddress + pos, (byte[])base, size, modes); } else if (unmapper == null) { return new NativeMemorySegmentImpl(bbAddress + pos, size, modes, bufferScope); } else { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractNativeScope.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractNativeScope.java deleted file mode 100644 index 6d7efc83439..00000000000 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/AbstractNativeScope.java +++ /dev/null @@ -1,172 +0,0 @@ -/* - * 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. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * 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. - */ - -package jdk.internal.foreign; - -import jdk.incubator.foreign.MemorySegment; -import jdk.incubator.foreign.NativeScope; - -import java.util.ArrayList; -import java.util.List; -import java.util.OptionalLong; - -public abstract class AbstractNativeScope implements NativeScope { - - private final List segments = new ArrayList<>(); - private final Thread ownerThread; - - private static final int SCOPE_MASK = MemorySegment.READ | MemorySegment.WRITE; // no terminal operations allowed - - AbstractNativeScope() { - this.ownerThread = Thread.currentThread(); - } - - @Override - public Thread ownerThread() { - return ownerThread; - } - - @Override - public void close() { - segments.forEach(MemorySegment::close); - } - - void checkOwnerThread() { - if (Thread.currentThread() != ownerThread()) { - throw new IllegalStateException("Attempt to access scope from different thread"); - } - } - - MemorySegment newSegment(long size, long align) { - MemorySegment segment = MemorySegment.allocateNative(size, align); - segments.add(segment); - return segment; - } - - MemorySegment newSegment(long size) { - return newSegment(size, size); - } - - public void register(MemorySegment segment) { - segments.add(segment); - } - - public static class UnboundedNativeScope extends AbstractNativeScope { - - private static final long BLOCK_SIZE = 4 * 1024; - private static final long MAX_ALLOC_SIZE = BLOCK_SIZE / 2; - - private MemorySegment segment; - private long sp = 0L; - private long size = 0L; - - @Override - public OptionalLong byteSize() { - return OptionalLong.empty(); - } - - @Override - public long allocatedBytes() { - return size; - } - - public UnboundedNativeScope() { - super(); - this.segment = newSegment(BLOCK_SIZE); - } - - @Override - public MemorySegment allocate(long bytesSize, long bytesAlignment) { - checkOwnerThread(); - if (Utils.alignUp(bytesSize, bytesAlignment) > MAX_ALLOC_SIZE) { - MemorySegment segment = newSegment(bytesSize, bytesAlignment); - return segment.withAccessModes(SCOPE_MASK); - } - // try to slice from current segment first... - MemorySegment slice = trySlice(bytesSize, bytesAlignment); - if (slice == null) { - // ... if that fails, allocate a new segment and slice from there - sp = 0L; - segment = newSegment(BLOCK_SIZE, 1L); - slice = trySlice(bytesSize, bytesAlignment); - if (slice == null) { - // this should not be possible - allocations that do not fit in BLOCK_SIZE should get their own - // standalone segment (see above). - throw new AssertionError("Cannot get here!"); - } - } - return slice; - } - - private MemorySegment trySlice(long bytesSize, long bytesAlignment) { - long min = segment.address().toRawLongValue(); - long start = Utils.alignUp(min + sp, bytesAlignment) - min; - if (segment.byteSize() - start < bytesSize) { - return null; - } else { - MemorySegment slice = segment.asSlice(start, bytesSize) - .withAccessModes(SCOPE_MASK); - sp = start + bytesSize; - size += Utils.alignUp(bytesSize, bytesAlignment); - return slice; - } - } - } - - public static class BoundedNativeScope extends AbstractNativeScope { - private final MemorySegment segment; - private long sp = 0L; - - @Override - public OptionalLong byteSize() { - return OptionalLong.of(segment.byteSize()); - } - - @Override - public long allocatedBytes() { - return sp; - } - - public BoundedNativeScope(long size) { - super(); - this.segment = newSegment(size, 1); - } - - @Override - public MemorySegment allocate(long bytesSize, long bytesAlignment) { - checkOwnerThread(); - long min = segment.address().toRawLongValue(); - long start = Utils.alignUp(min + sp, bytesAlignment) - min; - try { - MemorySegment slice = segment.asSlice(start, bytesSize) - .withAccessModes(SCOPE_MASK); - sp = start + bytesSize; - return slice; - } catch (IndexOutOfBoundsException ex) { - throw new OutOfMemoryError("Not enough space left to allocate"); - } - } - } -} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ArenaAllocator.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ArenaAllocator.java new file mode 100644 index 00000000000..c4d56a22a53 --- /dev/null +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ArenaAllocator.java @@ -0,0 +1,124 @@ +package jdk.internal.foreign; + +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.SegmentAllocator; +import jdk.incubator.foreign.ResourceScope; + +public abstract class ArenaAllocator implements SegmentAllocator { + + protected MemorySegment segment; + + protected long sp = 0L; + + ArenaAllocator(MemorySegment segment) { + this.segment = segment; + } + + MemorySegment trySlice(long bytesSize, long bytesAlignment) { + long min = segment.address().toRawLongValue(); + long start = Utils.alignUp(min + sp, bytesAlignment) - min; + if (segment.byteSize() - start < bytesSize) { + return null; + } else { + MemorySegment slice = segment.asSlice(start, bytesSize); + sp = start + bytesSize; + return slice; + } + } + + void checkConfinementIfNeeded() { + Thread ownerThread = scope().ownerThread(); + if (ownerThread != null && ownerThread != Thread.currentThread()) { + throw new IllegalStateException("Attempt to allocate outside confinement thread"); + } + } + + ResourceScope scope() { + return segment.scope(); + } + + public static class UnboundedArenaAllocator extends ArenaAllocator { + + private static final long DEFAULT_BLOCK_SIZE = 4 * 1024; + + public UnboundedArenaAllocator(ResourceScope scope) { + super(MemorySegment.allocateNative(DEFAULT_BLOCK_SIZE, 1, scope)); + } + + private MemorySegment newSegment(long size, long align) { + return MemorySegment.allocateNative(size, align, segment.scope()); + } + + @Override + public MemorySegment allocate(long bytesSize, long bytesAlignment) { + checkConfinementIfNeeded(); + // try to slice from current segment first... + MemorySegment slice = trySlice(bytesSize, bytesAlignment); + if (slice != null) { + return slice; + } else { + long maxPossibleAllocationSize = bytesSize + bytesAlignment - 1; + if (maxPossibleAllocationSize > DEFAULT_BLOCK_SIZE) { + // too big + return newSegment(bytesSize, bytesAlignment); + } else { + // allocate a new segment and slice from there + sp = 0L; + segment = newSegment(DEFAULT_BLOCK_SIZE, 1L); + return trySlice(bytesSize, bytesAlignment); + } + } + } + } + + public static class BoundedArenaAllocator extends ArenaAllocator { + + public BoundedArenaAllocator(ResourceScope scope, long size) { + super(MemorySegment.allocateNative(size, 1, scope)); + } + + @Override + public MemorySegment allocate(long bytesSize, long bytesAlignment) { + checkConfinementIfNeeded(); + // try to slice from current segment first... + MemorySegment slice = trySlice(bytesSize, bytesAlignment); + if (slice != null) { + return slice; + } else { + throw new OutOfMemoryError("Not enough space left to allocate"); + } + } + } + + public static class BoundedSharedArenaAllocator extends BoundedArenaAllocator { + public BoundedSharedArenaAllocator(ResourceScope scope, long size) { + super(scope, size); + } + + @Override + public synchronized MemorySegment allocate(long bytesSize, long bytesAlignment) { + return super.allocate(bytesSize, bytesAlignment); + } + } + + public static class UnboundedSharedArenaAllocator implements SegmentAllocator { + + final ResourceScope scope; + + final ThreadLocal allocators = new ThreadLocal<>() { + @Override + protected ArenaAllocator initialValue() { + return new UnboundedArenaAllocator(scope); + } + }; + + public UnboundedSharedArenaAllocator(ResourceScope scope) { + this.scope = scope; + } + + @Override + public MemorySegment allocate(long bytesSize, long bytesAlignment) { + return allocators.get().allocate(bytesSize, bytesAlignment); + } + } +} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ConfinedScope.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ConfinedScope.java new file mode 100644 index 00000000000..16749105bd1 --- /dev/null +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ConfinedScope.java @@ -0,0 +1,133 @@ +/* + * Copyright (c) 2021, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ + +package jdk.internal.foreign; + +import jdk.incubator.foreign.ResourceScope; +import jdk.internal.vm.annotation.ForceInline; + +import java.lang.ref.Cleaner; +import java.lang.ref.Reference; + +/** + * A confined scope, which features an owner thread. The liveness check features an additional + * confinement check - that is, calling any operation on this scope from a thread other than the + * owner thread will result in an exception. Because of this restriction, checking the liveness bit + * can be performed in plain mode. + */ +final class ConfinedScope extends ResourceScopeImpl { + + private boolean closed; // = false + private int lockCount = 0; + private final Thread owner; + + public ConfinedScope(Thread owner, Cleaner cleaner) { + super(cleaner, new ConfinedResourceList()); + this.owner = owner; + } + + @ForceInline + public final void checkValidState() { + if (owner != Thread.currentThread()) { + throw new IllegalStateException("Attempted access outside owning thread"); + } + if (closed) { + throw new IllegalStateException("Already closed"); + } + } + + @Override + public boolean isAlive() { + return !closed; + } + + @Override + public HandleImpl acquire() { + checkValidState(); + lockCount++; + return new ConfinedHandle(); + } + + void justClose() { + this.checkValidState(); + if (lockCount == 0) { + closed = true; + } else { + throw new IllegalStateException("Scope is acquired by " + lockCount + " locks"); + } + } + + @Override + public Thread ownerThread() { + return owner; + } + + /** + * A confined resource list; no races are possible here. + */ + static final class ConfinedResourceList extends ResourceList { + @Override + void add(ResourceCleanup cleanup) { + if (fst != ResourceCleanup.CLOSED_LIST) { + cleanup.next = fst; + fst = cleanup; + } else { + throw new IllegalStateException("Already closed!"); + } + } + + @Override + void cleanup() { + if (fst != ResourceCleanup.CLOSED_LIST) { + ResourceCleanup prev = fst; + fst = ResourceCleanup.CLOSED_LIST; + cleanup(prev); + } else { + throw new IllegalStateException("Attempt to cleanup an already closed resource list"); + } + } + } + + /** + * A confined resource scope handle; no races are possible here. + */ + final class ConfinedHandle implements HandleImpl { + boolean released = false; + + @Override + public ResourceScopeImpl scope() { + return ConfinedScope.this; + } + + @Override + public void release() { + checkValidState(); // thread check + if (!released) { + released = true; + lockCount--; + } + } + } +} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java index 30e2fc0608b..755d6cb14dc 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java @@ -34,7 +34,6 @@ import jdk.internal.vm.annotation.ForceInline; import java.nio.ByteBuffer; import java.util.Objects; -import java.util.function.Supplier; /** * Implementation for heap memory segments. An heap memory segment is composed by an offset and @@ -52,8 +51,8 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl final H base; @ForceInline - HeapMemorySegmentImpl(long offset, H base, long length, int mask, MemoryScope scope) { - super(length, mask, scope); + HeapMemorySegmentImpl(long offset, H base, long length, int mask) { + super(length, mask, ResourceScopeImpl.GLOBAL); this.offset = offset; this.base = base; } @@ -67,7 +66,7 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl } @Override - abstract HeapMemorySegmentImpl dup(long offset, long size, int mask, MemoryScope scope); + abstract HeapMemorySegmentImpl dup(long offset, long size, int mask, ResourceScopeImpl scope); @Override ByteBuffer makeByteBuffer() { @@ -75,20 +74,20 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl throw new UnsupportedOperationException("Not an address to an heap-allocated byte array"); } JavaNioAccess nioAccess = SharedSecrets.getJavaNioAccess(); - return nioAccess.newHeapByteBuffer((byte[]) base(), (int)min() - BYTE_ARR_BASE, (int) byteSize(), this); + return nioAccess.newHeapByteBuffer((byte[]) base(), (int)min() - BYTE_ARR_BASE, (int) byteSize(), null); } // factories public static class OfByte extends HeapMemorySegmentImpl { - OfByte(long offset, byte[] base, long length, int mask, MemoryScope scope) { - super(offset, base, length, mask, scope); + OfByte(long offset, byte[] base, long length, int mask) { + super(offset, base, length, mask); } @Override - OfByte dup(long offset, long size, int mask, MemoryScope scope) { - return new OfByte(this.offset + offset, base, size, mask, scope); + OfByte dup(long offset, long size, int mask, ResourceScopeImpl scope) { + return new OfByte(this.offset + offset, base, size, mask); } @Override @@ -99,20 +98,19 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl public static MemorySegment fromArray(byte[] arr) { Objects.requireNonNull(arr); long byteSize = (long)arr.length * Unsafe.ARRAY_BYTE_INDEX_SCALE; - MemoryScope scope = MemoryScope.createConfined(null, MemoryScope.DUMMY_CLEANUP_ACTION, null); - return new OfByte(Unsafe.ARRAY_BYTE_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize), scope); + return new OfByte(Unsafe.ARRAY_BYTE_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize)); } } public static class OfChar extends HeapMemorySegmentImpl { - OfChar(long offset, char[] base, long length, int mask, MemoryScope scope) { - super(offset, base, length, mask, scope); + OfChar(long offset, char[] base, long length, int mask) { + super(offset, base, length, mask); } @Override - OfChar dup(long offset, long size, int mask, MemoryScope scope) { - return new OfChar(this.offset + offset, base, size, mask, scope); + OfChar dup(long offset, long size, int mask, ResourceScopeImpl scope) { + return new OfChar(this.offset + offset, base, size, mask); } @Override @@ -123,20 +121,19 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl public static MemorySegment fromArray(char[] arr) { Objects.requireNonNull(arr); long byteSize = (long)arr.length * Unsafe.ARRAY_CHAR_INDEX_SCALE; - MemoryScope scope = MemoryScope.createConfined(null, MemoryScope.DUMMY_CLEANUP_ACTION, null); - return new OfChar(Unsafe.ARRAY_CHAR_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize), scope); + return new OfChar(Unsafe.ARRAY_CHAR_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize)); } } public static class OfShort extends HeapMemorySegmentImpl { - OfShort(long offset, short[] base, long length, int mask, MemoryScope scope) { - super(offset, base, length, mask, scope); + OfShort(long offset, short[] base, long length, int mask) { + super(offset, base, length, mask); } @Override - OfShort dup(long offset, long size, int mask, MemoryScope scope) { - return new OfShort(this.offset + offset, base, size, mask, scope); + OfShort dup(long offset, long size, int mask, ResourceScopeImpl scope) { + return new OfShort(this.offset + offset, base, size, mask); } @Override @@ -147,20 +144,19 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl public static MemorySegment fromArray(short[] arr) { Objects.requireNonNull(arr); long byteSize = (long)arr.length * Unsafe.ARRAY_SHORT_INDEX_SCALE; - MemoryScope scope = MemoryScope.createConfined(null, MemoryScope.DUMMY_CLEANUP_ACTION, null); - return new OfShort(Unsafe.ARRAY_SHORT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize), scope); + return new OfShort(Unsafe.ARRAY_SHORT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize)); } } public static class OfInt extends HeapMemorySegmentImpl { - OfInt(long offset, int[] base, long length, int mask, MemoryScope scope) { - super(offset, base, length, mask, scope); + OfInt(long offset, int[] base, long length, int mask) { + super(offset, base, length, mask); } @Override - OfInt dup(long offset, long size, int mask, MemoryScope scope) { - return new OfInt(this.offset + offset, base, size, mask, scope); + OfInt dup(long offset, long size, int mask, ResourceScopeImpl scope) { + return new OfInt(this.offset + offset, base, size, mask); } @Override @@ -171,20 +167,19 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl public static MemorySegment fromArray(int[] arr) { Objects.requireNonNull(arr); long byteSize = (long)arr.length * Unsafe.ARRAY_INT_INDEX_SCALE; - MemoryScope scope = MemoryScope.createConfined(null, MemoryScope.DUMMY_CLEANUP_ACTION, null); - return new OfInt(Unsafe.ARRAY_INT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize), scope); + return new OfInt(Unsafe.ARRAY_INT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize)); } } public static class OfLong extends HeapMemorySegmentImpl { - OfLong(long offset, long[] base, long length, int mask, MemoryScope scope) { - super(offset, base, length, mask, scope); + OfLong(long offset, long[] base, long length, int mask) { + super(offset, base, length, mask); } @Override - OfLong dup(long offset, long size, int mask, MemoryScope scope) { - return new OfLong(this.offset + offset, base, size, mask, scope); + OfLong dup(long offset, long size, int mask, ResourceScopeImpl scope) { + return new OfLong(this.offset + offset, base, size, mask); } @Override @@ -195,20 +190,19 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl public static MemorySegment fromArray(long[] arr) { Objects.requireNonNull(arr); long byteSize = (long)arr.length * Unsafe.ARRAY_LONG_INDEX_SCALE; - MemoryScope scope = MemoryScope.createConfined(null, MemoryScope.DUMMY_CLEANUP_ACTION, null); - return new OfLong(Unsafe.ARRAY_LONG_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize), scope); + return new OfLong(Unsafe.ARRAY_LONG_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize)); } } public static class OfFloat extends HeapMemorySegmentImpl { - OfFloat(long offset, float[] base, long length, int mask, MemoryScope scope) { - super(offset, base, length, mask, scope); + OfFloat(long offset, float[] base, long length, int mask) { + super(offset, base, length, mask); } @Override - OfFloat dup(long offset, long size, int mask, MemoryScope scope) { - return new OfFloat(this.offset + offset, base, size, mask, scope); + OfFloat dup(long offset, long size, int mask, ResourceScopeImpl scope) { + return new OfFloat(this.offset + offset, base, size, mask); } @Override @@ -219,20 +213,19 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl public static MemorySegment fromArray(float[] arr) { Objects.requireNonNull(arr); long byteSize = (long)arr.length * Unsafe.ARRAY_FLOAT_INDEX_SCALE; - MemoryScope scope = MemoryScope.createConfined(null, MemoryScope.DUMMY_CLEANUP_ACTION, null); - return new OfFloat(Unsafe.ARRAY_FLOAT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize), scope); + return new OfFloat(Unsafe.ARRAY_FLOAT_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize)); } } public static class OfDouble extends HeapMemorySegmentImpl { - OfDouble(long offset, double[] base, long length, int mask, MemoryScope scope) { - super(offset, base, length, mask, scope); + OfDouble(long offset, double[] base, long length, int mask) { + super(offset, base, length, mask); } @Override - OfDouble dup(long offset, long size, int mask, MemoryScope scope) { - return new OfDouble(this.offset + offset, base, size, mask, scope); + OfDouble dup(long offset, long size, int mask, ResourceScopeImpl scope) { + return new OfDouble(this.offset + offset, base, size, mask); } @Override @@ -243,8 +236,8 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl public static MemorySegment fromArray(double[] arr) { Objects.requireNonNull(arr); long byteSize = (long)arr.length * Unsafe.ARRAY_DOUBLE_INDEX_SCALE; - MemoryScope scope = MemoryScope.createConfined(null, MemoryScope.DUMMY_CLEANUP_ACTION, null); - return new OfDouble(Unsafe.ARRAY_DOUBLE_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize), scope); + return new OfDouble(Unsafe.ARRAY_DOUBLE_BASE_OFFSET, arr, byteSize, defaultAccessModes(byteSize)); } } + } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java index 6a8d4e9236d..346e6882a45 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java @@ -35,7 +35,6 @@ import jdk.internal.access.foreign.MemorySegmentProxy; import jdk.incubator.foreign.GroupLayout; import jdk.incubator.foreign.SequenceLayout; import jdk.incubator.foreign.ValueLayout; -import sun.invoke.util.Wrapper; import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; @@ -62,6 +61,7 @@ public class LayoutPath { private static final MethodHandle ADD_STRIDE; private static final MethodHandle MH_ADD_SCALED_OFFSET; + private static final MethodHandle MH_SLICE; private static final int UNSPECIFIED_ELEM_INDEX = -1; @@ -72,6 +72,8 @@ public class LayoutPath { MethodType.methodType(long.class, MemorySegment.class, long.class, long.class, long.class)); MH_ADD_SCALED_OFFSET = lookup.findStatic(LayoutPath.class, "addScaledOffset", MethodType.methodType(long.class, long.class, long.class, long.class)); + MH_SLICE = lookup.findVirtual(MemorySegment.class, "asSlice", + MethodType.methodType(MemorySegment.class, long.class, long.class)); } catch (Throwable ex) { throw new ExceptionInInitializerError(ex); } @@ -200,6 +202,22 @@ public class LayoutPath { return mh; } + public MethodHandle sliceHandle() { + if (strides.length == 0) { + // trigger checks eagerly + Utils.bitsToBytesOrThrow(offset, Utils.bitsToBytesThrowOffset); + } + + MethodHandle offsetHandle = offsetHandle(); // bit offset + offsetHandle = MethodHandles.filterReturnValue(offsetHandle, Utils.MH_bitsToBytesOrThrowForOffset); // byte offset + + MethodHandle sliceHandle = MH_SLICE; // (MS, long, long) -> MS + sliceHandle = MethodHandles.insertArguments(sliceHandle, 2, layout.byteSize()); // (MS, long) -> MS + sliceHandle = MethodHandles.collectArguments(sliceHandle, 1, offsetHandle); // (MS, ...) -> MS + + return sliceHandle; + } + public MemoryLayout layout() { return layout; } @@ -211,9 +229,9 @@ public class LayoutPath { } else if (enclosing.layout instanceof SequenceLayout) { SequenceLayout seq = (SequenceLayout)enclosing.layout; if (seq.elementCount().isPresent()) { - return enclosing.map(l -> dup(l, MemoryLayout.ofSequence(seq.elementCount().getAsLong(), newLayout))); + return enclosing.map(l -> dup(l, MemoryLayout.sequenceLayout(seq.elementCount().getAsLong(), newLayout))); } else { - return enclosing.map(l -> dup(l, MemoryLayout.ofSequence(newLayout))); + return enclosing.map(l -> dup(l, MemoryLayout.sequenceLayout(newLayout))); } } else if (enclosing.layout instanceof GroupLayout) { GroupLayout g = (GroupLayout)enclosing.layout; @@ -221,9 +239,9 @@ public class LayoutPath { //if we selected a layout in a group we must have a valid index newElements.set((int)elementIndex, newLayout); if (g.isUnion()) { - return enclosing.map(l -> dup(l, MemoryLayout.ofUnion(newElements.toArray(new MemoryLayout[0])))); + return enclosing.map(l -> dup(l, MemoryLayout.unionLayout(newElements.toArray(new MemoryLayout[0])))); } else { - return enclosing.map(l -> dup(l, MemoryLayout.ofStruct(newElements.toArray(new MemoryLayout[0])))); + return enclosing.map(l -> dup(l, MemoryLayout.structLayout(newElements.toArray(new MemoryLayout[0])))); } } else { return newLayout; @@ -299,7 +317,7 @@ public class LayoutPath { * This class provides an immutable implementation for the {@code PathElement} interface. A path element implementation * is simply a pointer to one of the selector methods provided by the {@code LayoutPath} class. */ - public static class PathElementImpl implements MemoryLayout.PathElement, UnaryOperator { + public static final class PathElementImpl implements MemoryLayout.PathElement, UnaryOperator { public enum PathKind { SEQUENCE_ELEMENT("unbound sequence element"), diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LibrariesHelper.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LibrariesHelper.java index ea2c42576c2..7705c9a9e6d 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LibrariesHelper.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LibrariesHelper.java @@ -29,27 +29,27 @@ import jdk.incubator.foreign.MemoryAddress; import java.io.File; import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.internal.loader.NativeLibraries; import jdk.internal.loader.NativeLibrary; -import jdk.internal.ref.CleanerFactory; -import java.nio.file.Files; -import java.nio.file.Path; -import java.util.Arrays; -import java.util.IdentityHashMap; +import java.lang.ref.Reference; +import java.lang.ref.WeakReference; import java.util.Map; import java.util.Objects; import java.util.Optional; -import java.util.concurrent.atomic.AtomicInteger; +import java.util.concurrent.ConcurrentHashMap; import java.util.function.Supplier; public final class LibrariesHelper { private LibrariesHelper() {} - private final static NativeLibraries nativeLibraries = + private static final NativeLibraries nativeLibraries = NativeLibraries.rawNativeLibraries(LibrariesHelper.class, true); - private final static Map loadedLibraries = new IdentityHashMap<>(); + private static final Map> loadedLibraries = new ConcurrentHashMap<>(); /** * Load the specified shared library. @@ -76,67 +76,69 @@ public final class LibrariesHelper { "Library not found: " + path); } - // return the absolute path of the library of given name by searching - // in the given array of paths. - private static Optional findLibraryPath(Path[] paths, String libName) { - return Arrays.stream(paths). - map(p -> p.resolve(System.mapLibraryName(libName))). - filter(Files::isRegularFile).map(Path::toAbsolutePath).findFirst(); - } - public static LibraryLookup getDefaultLibrary() { return LibraryLookupImpl.DEFAULT_LOOKUP; } - synchronized static LibraryLookupImpl lookup(Supplier librarySupplier, String notFoundMsg) { + static LibraryLookupImpl lookup(Supplier librarySupplier, String notFoundMsg) { NativeLibrary library = librarySupplier.get(); if (library == null) { throw new IllegalArgumentException(notFoundMsg); } - AtomicInteger refCount = loadedLibraries.computeIfAbsent(library, lib -> new AtomicInteger()); - refCount.incrementAndGet(); - LibraryLookupImpl lookup = new LibraryLookupImpl(library); - CleanerFactory.cleaner().register(lookup, () -> tryUnload(library)); - return lookup; - } - - synchronized static void tryUnload(NativeLibrary library) { - AtomicInteger refCount = loadedLibraries.get(library); - if (refCount.decrementAndGet() == 0) { - loadedLibraries.remove(library); - nativeLibraries.unload(library); + ResourceScope[] holder = new ResourceScope[1]; + try { + WeakReference scopeRef = loadedLibraries.computeIfAbsent(library, lib -> { + ResourceScopeImpl s = ResourceScopeImpl.createImplicitScope(); + holder[0] = s; // keep the scope alive at least until the outer method returns + s.addOrCleanupIfFail(ResourceScopeImpl.ResourceList.ResourceCleanup.ofRunnable(() -> { + nativeLibraries.unload(library); + loadedLibraries.remove(library); + })); + return new WeakReference<>(s); + }); + return new LibraryLookupImpl(library, scopeRef.get()); + } finally { + Reference.reachabilityFence(holder); } } - static class LibraryLookupImpl implements LibraryLookup { + //Todo: in principle we could expose a scope accessor, so that users could unload libraries at will + static final class LibraryLookupImpl implements LibraryLookup { final NativeLibrary library; + final MemorySegment librarySegment; - LibraryLookupImpl(NativeLibrary library) { + LibraryLookupImpl(NativeLibrary library, ResourceScope scope) { this.library = library; + this.librarySegment = MemoryAddress.NULL.asSegment(Long.MAX_VALUE, scope); } @Override - public Optional lookup(String name) { + public final Optional lookup(String name) { try { Objects.requireNonNull(name); MemoryAddress addr = MemoryAddress.ofLong(library.lookup(name)); - return Optional.of(new Symbol() { // inner class - retains a link to enclosing lookup - @Override - public String name() { - return name; - } - - @Override - public MemoryAddress address() { - return addr; - } - }); + return Optional.of(librarySegment.asSlice(addr).address()); } catch (NoSuchMethodException ex) { return Optional.empty(); } } - static LibraryLookup DEFAULT_LOOKUP = new LibraryLookupImpl(NativeLibraries.defaultLibrary); + @Override + public final Optional lookup(String name, MemoryLayout layout) { + try { + Objects.requireNonNull(name); + Objects.requireNonNull(layout); + MemoryAddress addr = MemoryAddress.ofLong(library.lookup(name)); + if (addr.toRawLongValue() % layout.byteAlignment() != 0) { + throw new IllegalArgumentException("Bad layout alignment constraints: " + layout.byteAlignment()); + } + return Optional.of(librarySegment.asSlice(addr, layout.byteSize())); + } catch (NoSuchMethodException ex) { + return Optional.empty(); + } + } + + static LibraryLookup DEFAULT_LOOKUP = new LibraryLookupImpl(NativeLibraries.defaultLibrary, ResourceScopeImpl.GLOBAL); } /* used for testing */ diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MappedMemorySegmentImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MappedMemorySegmentImpl.java index 36a0e83323e..85332b66a5b 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MappedMemorySegmentImpl.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MappedMemorySegmentImpl.java @@ -27,10 +27,10 @@ package jdk.internal.foreign; import jdk.incubator.foreign.MemorySegment; import jdk.internal.access.foreign.UnmapperProxy; +import jdk.internal.misc.ExtendedMapMode; import jdk.internal.misc.ScopedMemoryAccess; import sun.nio.ch.FileChannelImpl; -import java.io.FileDescriptor; import java.io.IOException; import java.nio.ByteBuffer; import java.nio.channels.FileChannel; @@ -40,7 +40,6 @@ import java.nio.file.OpenOption; import java.nio.file.Path; import java.nio.file.StandardOpenOption; import java.util.Objects; -import java.util.Optional; /** * Implementation for a mapped memory segments. A mapped memory segment is a native memory segment, which @@ -54,18 +53,19 @@ public class MappedMemorySegmentImpl extends NativeMemorySegmentImpl { static ScopedMemoryAccess SCOPED_MEMORY_ACCESS = ScopedMemoryAccess.getScopedMemoryAccess(); - MappedMemorySegmentImpl(long min, UnmapperProxy unmapper, long length, int mask, MemoryScope scope) { + MappedMemorySegmentImpl(long min, UnmapperProxy unmapper, long length, int mask, ResourceScopeImpl scope) { super(min, length, mask, scope); this.unmapper = unmapper; } @Override ByteBuffer makeByteBuffer() { - return nioAccess.newMappedByteBuffer(unmapper, min, (int)length, null, this); + return nioAccess.newMappedByteBuffer(unmapper, min, (int)length, null, + scope == ResourceScopeImpl.GLOBAL ? null : this); } @Override - MappedMemorySegmentImpl dup(long offset, long size, int mask, MemoryScope scope) { + MappedMemorySegmentImpl dup(long offset, long size, int mask, ResourceScopeImpl scope) { return new MappedMemorySegmentImpl(min + offset, unmapper, size, mask, scope); } @@ -77,11 +77,6 @@ public class MappedMemorySegmentImpl extends NativeMemorySegmentImpl { return (MappedMemorySegmentImpl)super.asSlice(offset, newSize); } - @Override - public MappedMemorySegmentImpl withAccessModes(int accessModes) { - return (MappedMemorySegmentImpl)super.withAccessModes(accessModes); - } - @Override public boolean isMapped() { return true; @@ -111,9 +106,10 @@ public class MappedMemorySegmentImpl extends NativeMemorySegmentImpl { // factories - public static MemorySegment makeMappedSegment(Path path, long bytesOffset, long bytesSize, FileChannel.MapMode mapMode) throws IOException { + public static MemorySegment makeMappedSegment(Path path, long bytesOffset, long bytesSize, FileChannel.MapMode mapMode, ResourceScopeImpl scope) throws IOException { Objects.requireNonNull(path); Objects.requireNonNull(mapMode); + scope.checkValidStateSlow(); if (bytesSize < 0) throw new IllegalArgumentException("Requested bytes size must be >= 0."); if (bytesOffset < 0) throw new IllegalArgumentException("Requested bytes offset must be >= 0."); FileSystem fs = path.getFileSystem(); @@ -125,22 +121,31 @@ public class MappedMemorySegmentImpl extends NativeMemorySegmentImpl { UnmapperProxy unmapperProxy = ((FileChannelImpl)channelImpl).mapInternal(mapMode, bytesOffset, bytesSize); int modes = defaultAccessModes(bytesSize); if (mapMode == FileChannel.MapMode.READ_ONLY) { - modes &= ~WRITE; + modes |= READ_ONLY; } if (unmapperProxy != null) { - MemoryScope scope = MemoryScope.createConfined(null, unmapperProxy::unmap, null); - return new MappedMemorySegmentImpl(unmapperProxy.address(), unmapperProxy, bytesSize, + AbstractMemorySegmentImpl segment = new MappedMemorySegmentImpl(unmapperProxy.address(), unmapperProxy, bytesSize, modes, scope); + scope.addOrCleanupIfFail(new ResourceScopeImpl.ResourceList.ResourceCleanup() { + @Override + public void cleanup() { + unmapperProxy.unmap(); + } + }); + return segment; } else { - return new EmptyMappedMemorySegmentImpl(modes); + return new EmptyMappedMemorySegmentImpl(modes, scope); } } } private static OpenOption[] openOptions(FileChannel.MapMode mapMode) { - if (mapMode == FileChannel.MapMode.READ_ONLY) { + if (mapMode == FileChannel.MapMode.READ_ONLY || + mapMode == ExtendedMapMode.READ_ONLY_SYNC) { return new OpenOption[] { StandardOpenOption.READ }; - } else if (mapMode == FileChannel.MapMode.READ_WRITE || mapMode == FileChannel.MapMode.PRIVATE) { + } else if (mapMode == FileChannel.MapMode.READ_WRITE || + mapMode == FileChannel.MapMode.PRIVATE || + mapMode == ExtendedMapMode.READ_WRITE_SYNC) { return new OpenOption[] { StandardOpenOption.READ, StandardOpenOption.WRITE }; } else { throw new UnsupportedOperationException("Unsupported map mode: " + mapMode); @@ -149,9 +154,8 @@ public class MappedMemorySegmentImpl extends NativeMemorySegmentImpl { static class EmptyMappedMemorySegmentImpl extends MappedMemorySegmentImpl { - public EmptyMappedMemorySegmentImpl(int modes) { - super(0, null, 0, modes, - MemoryScope.createConfined(null, MemoryScope.DUMMY_CLEANUP_ACTION, null)); + public EmptyMappedMemorySegmentImpl(int modes, ResourceScopeImpl scope) { + super(0, null, 0, modes, scope); } @Override diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryAddressImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryAddressImpl.java index f7d39bd7944..2330feefd4e 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryAddressImpl.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryAddressImpl.java @@ -27,7 +27,10 @@ package jdk.internal.foreign; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemorySegment; +import jdk.internal.reflect.CallerSensitive; +import jdk.internal.reflect.Reflection; +import jdk.incubator.foreign.ResourceScope; import java.util.Objects; /** @@ -36,52 +39,75 @@ import java.util.Objects; */ public final class MemoryAddressImpl implements MemoryAddress { - private final Object base; + private final AbstractMemorySegmentImpl segment; private final long offset; - public MemoryAddressImpl(Object base, long offset) { - this.base = base; + public MemoryAddressImpl(AbstractMemorySegmentImpl segment, long offset) { + this.segment = segment; this.offset = offset; } + Object base() { + return segment != null ? segment.base() : null; + } + + long offset() { + return segment != null ? + segment.min() + offset : offset; + } + // MemoryAddress methods + @Override + public ResourceScope scope() { + return segment != null ? + segment.scope() : ResourceScope.globalScope(); + } + + @Override + public MemoryAddress addOffset(long offset) { + return new MemoryAddressImpl(segment, this.offset + offset); + } + @Override public long segmentOffset(MemorySegment segment) { Objects.requireNonNull(segment); AbstractMemorySegmentImpl segmentImpl = (AbstractMemorySegmentImpl)segment; - if (segmentImpl.base() != base) { - throw new IllegalArgumentException("Invalid segment: " + segment); + if (segmentImpl.base() != base()) { + throw new IllegalArgumentException("Incompatible segment: " + segment); } - return offset - segmentImpl.min(); + return offset() - segmentImpl.min(); + } + + @Override + public boolean isNative() { + return base() == null; } @Override public long toRawLongValue() { - if (base != null) { - throw new UnsupportedOperationException("Not a native address"); + if (segment != null) { + if (segment.base() != null) { + throw new UnsupportedOperationException("Not a native address"); + } + segment.checkValidState(); } - return offset; - } - - @Override - public MemoryAddress addOffset(long bytes) { - return new MemoryAddressImpl(base, offset + bytes); + return offset(); } // Object methods @Override public int hashCode() { - return Objects.hash(base, offset); + return Objects.hash(base(), offset()); } @Override public boolean equals(Object that) { if (that instanceof MemoryAddressImpl) { MemoryAddressImpl addr = (MemoryAddressImpl)that; - return Objects.equals(base, addr.base) && - offset == addr.offset; + return Objects.equals(base(), addr.base()) && + offset() == addr.offset(); } else { return false; } @@ -89,23 +115,38 @@ public final class MemoryAddressImpl implements MemoryAddress { @Override public String toString() { - return "MemoryAddress{ base: " + base + " offset=0x" + Long.toHexString(offset) + " }"; + return "MemoryAddress{ base: " + base() + " offset=0x" + Long.toHexString(offset()) + " }"; } @Override - public MemorySegment asSegmentRestricted(long bytesSize, Runnable cleanupAction, Object attachment) { - Utils.checkRestrictedAccess("MemoryAddress.asSegmentRestricted"); + @CallerSensitive + public final MemorySegment asSegment(long bytesSize, ResourceScope scope) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); + return asSegment(bytesSize, null, scope); + } + + @Override + @CallerSensitive + public final MemorySegment asSegment(long bytesSize, Runnable cleanupAction, ResourceScope scope) { + Reflection.ensureNativeAccess(Reflection.getCallerClass()); + Objects.requireNonNull(scope); if (bytesSize <= 0) { throw new IllegalArgumentException("Invalid size : " + bytesSize); } - return NativeMemorySegmentImpl.makeNativeSegmentUnchecked(this, bytesSize, cleanupAction, attachment); + return NativeMemorySegmentImpl.makeNativeSegmentUnchecked(this, bytesSize, + cleanupAction, + (ResourceScopeImpl) scope); } public static MemorySegment ofLongUnchecked(long value) { return ofLongUnchecked(value, Long.MAX_VALUE); } + public static MemorySegment ofLongUnchecked(long value, long byteSize, ResourceScopeImpl resourceScope) { + return NativeMemorySegmentImpl.makeNativeSegmentUnchecked(MemoryAddress.ofLong(value), byteSize, null, resourceScope); + } + public static MemorySegment ofLongUnchecked(long value, long byteSize) { - return MemoryAddress.ofLong(value).asSegmentRestricted(byteSize).share(); + return NativeMemorySegmentImpl.makeNativeSegmentUnchecked(MemoryAddress.ofLong(value), byteSize, null, ResourceScopeImpl.GLOBAL); } } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryScope.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryScope.java deleted file mode 100644 index 624e5e5ba7c..00000000000 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/MemoryScope.java +++ /dev/null @@ -1,307 +0,0 @@ -/* - * Copyright (c) 2019, 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. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * 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. - * - */ - -package jdk.internal.foreign; - -import jdk.internal.misc.ScopedMemoryAccess; -import jdk.internal.ref.PhantomCleanable; -import jdk.internal.vm.annotation.ForceInline; - -import java.lang.invoke.MethodHandles; -import java.lang.invoke.VarHandle; -import java.lang.ref.Cleaner; -import java.lang.ref.Reference; -import java.util.Objects; - -/** - * This class manages the temporal bounds associated with a memory segment as well - * as thread confinement. A scope has a liveness bit, which is updated when the scope is closed - * (this operation is triggered by {@link AbstractMemorySegmentImpl#close()}). This bit is consulted prior - * to memory access (see {@link #checkValidState()}). - * There are two kinds of memory scope: confined memory scope and shared memory scope. - * A confined memory scope has an associated owner thread that confines some operations to - * associated owner thread such as {@link #close()} or {@link #checkValidState()}. - * Shared scopes do not feature an owner thread - meaning their operations can be called, in a racy - * manner, by multiple threads. To guarantee temporal safety in the presence of concurrent thread, - * shared scopes use a more sophisticated synchronization mechanism, which guarantees that no concurrent - * access is possible when a scope is being closed (see {@link jdk.internal.misc.ScopedMemoryAccess}). - */ -abstract class MemoryScope implements ScopedMemoryAccess.Scope { - - static final Runnable DUMMY_CLEANUP_ACTION = () -> { }; - - private MemoryScope(Object ref, Runnable cleanupAction, Cleaner cleaner) { - Objects.requireNonNull(cleanupAction); - this.ref = ref; - this.cleanupAction = cleanupAction; - this.scopeCleanable = cleaner != null ? - new ScopeCleanable(this, cleaner, cleanupAction) : - null; - } - - /** - * Creates a confined memory scope with given attachment and cleanup action. The returned scope - * is assumed to be confined on the current thread. - * @param ref an optional reference to an instance that needs to be kept reachable - * @param cleanupAction a cleanup action to be executed when returned scope is closed - * @return a confined memory scope - */ - static MemoryScope createConfined(Object ref, Runnable cleanupAction, Cleaner cleaner) { - return new ConfinedScope(Thread.currentThread(), ref, cleanupAction, cleaner); - } - - /** - * Creates a shared memory scope with given attachment and cleanup action. - * @param ref an optional reference to an instance that needs to be kept reachable - * @param cleanupAction a cleanup action to be executed when returned scope is closed - * @return a shared memory scope - */ - static MemoryScope createShared(Object ref, Runnable cleanupAction, Cleaner cleaner) { - return new SharedScope(ref, cleanupAction, cleaner); - } - - protected final Object ref; - protected final ScopeCleanable scopeCleanable; - protected final Runnable cleanupAction; - - /** - * Closes this scope, executing any cleanup action (where provided). - * @throws IllegalStateException if this scope is already closed or if this is - * a confined scope and this method is called outside of the owner thread. - */ - final void close() { - try { - justClose(); - cleanupAction.run(); - if (scopeCleanable != null) { - scopeCleanable.clear(); - } - } finally { - Reference.reachabilityFence(this); - } - } - - abstract void justClose(); - - /** - * Duplicates this scope with given new "owner" thread and {@link #close() closes} it. - * @param newOwner new owner thread of the returned memory scope - * @return a new confined scope, which is a duplicate of this scope, but with a new owner thread. - * @throws IllegalStateException if this scope is already closed or if this is - * a confined scope and this method is called outside of the owner thread. - */ - final MemoryScope confineTo(Thread newOwner) { - try { - justClose(); - if (scopeCleanable != null) { - scopeCleanable.clear(); - } - return new ConfinedScope(newOwner, ref, cleanupAction, scopeCleanable != null ? - scopeCleanable.cleaner : null); - } finally { - Reference.reachabilityFence(this); - } - } - - /** - * Duplicates this scope with given new "owner" thread and {@link #close() closes} it. - * @return a new shared scope, which is a duplicate of this scope. - * @throws IllegalStateException if this scope is already closed or if this is - * a confined scope and this method is called outside of the owner thread, - * or if this is already a shared scope. - */ - final MemoryScope share() { - try { - justClose(); - if (scopeCleanable != null) { - scopeCleanable.clear(); - } - return new SharedScope(ref, cleanupAction, scopeCleanable != null ? - scopeCleanable.cleaner : null); - } finally { - Reference.reachabilityFence(this); - } - } - - final MemoryScope cleanable(Cleaner cleaner) { - if (scopeCleanable != null) { - throw new IllegalStateException("Already registered with a cleaner"); - } - try { - justClose(); - return ownerThread() == null ? - new SharedScope(ref, cleanupAction, cleaner) : - new ConfinedScope(ownerThread(), ref, cleanupAction, cleaner); - } finally { - Reference.reachabilityFence(this); - } - } - - /** - * Returns "owner" thread of this scope. - * @return owner thread (or null for a shared scope) - */ - public abstract Thread ownerThread(); - - /** - * Returns true, if this scope is still alive. This method may be called in any thread. - * @return {@code true} if this scope is not closed yet. - */ - public abstract boolean isAlive(); - - /** - * Checks that this scope is still alive (see {@link #isAlive()}). - * @throws IllegalStateException if this scope is already closed or if this is - * a confined scope and this method is called outside of the owner thread. - */ - public abstract void checkValidState(); - - @Override - protected Object clone() throws CloneNotSupportedException { - throw new CloneNotSupportedException(); - } - - /** - * A confined scope, which features an owner thread. The liveness check features an additional - * confinement check - that is, calling any operation on this scope from a thread other than the - * owner thread will result in an exception. Because of this restriction, checking the liveness bit - * can be performed in plain mode (see {@link #checkAliveRaw(MemoryScope)}). - */ - static class ConfinedScope extends MemoryScope { - - private boolean closed; // = false - final Thread owner; - - public ConfinedScope(Thread owner, Object ref, Runnable cleanupAction, Cleaner cleaner) { - super(ref, cleanupAction, cleaner); - this.owner = owner; - } - - @ForceInline - public final void checkValidState() { - if (owner != Thread.currentThread()) { - throw new IllegalStateException("Attempted access outside owning thread"); - } - if (closed) { - throw ScopedAccessError.INSTANCE; - } - } - - @Override - public boolean isAlive() { - return !closed; - } - - void justClose() { - checkValidState(); - closed = true; - } - - @Override - public Thread ownerThread() { - return owner; - } - } - - /** - * A shared scope, which can be shared across multiple threads. Closing a shared scope has to ensure that - * (i) only one thread can successfully close a scope (e.g. in a close vs. close race) and that - * (ii) no other thread is accessing the memory associated with this scope while the segment is being - * closed. To ensure the former condition, a CAS is performed on the liveness bit. Ensuring the latter - * is trickier, and require a complex synchronization protocol (see {@link jdk.internal.misc.ScopedMemoryAccess}). - * Since it is the responsibility of the closing thread to make sure that no concurrent access is possible, - * checking the liveness bit upon access can be performed in plain mode (see {@link #checkAliveRaw(MemoryScope)}), - * as in the confined case. - */ - static class SharedScope extends MemoryScope { - - static ScopedMemoryAccess SCOPED_MEMORY_ACCESS = ScopedMemoryAccess.getScopedMemoryAccess(); - - final static int ALIVE = 0; - final static int CLOSING = 1; - final static int CLOSED = 2; - - int state = ALIVE; - - private static final VarHandle STATE; - - static { - try { - STATE = MethodHandles.lookup().findVarHandle(SharedScope.class, "state", int.class); - } catch (Throwable ex) { - throw new ExceptionInInitializerError(ex); - } - } - - SharedScope(Object ref, Runnable cleanupAction, Cleaner cleaner) { - super(ref, cleanupAction, cleaner); - } - - @Override - public Thread ownerThread() { - return null; - } - - @Override - public void checkValidState() { - if (state != ALIVE) { - throw ScopedAccessError.INSTANCE; - } - } - - void justClose() { - if (!STATE.compareAndSet(this, ALIVE, CLOSING)) { - throw new IllegalStateException("Already closed"); - } - boolean success = SCOPED_MEMORY_ACCESS.closeScope(this); - STATE.setVolatile(this, success ? CLOSED : ALIVE); - if (!success) { - throw new IllegalStateException("Cannot close while another thread is accessing the segment"); - } - } - - @Override - public boolean isAlive() { - return (int)STATE.getVolatile(this) != CLOSED; - } - } - - static class ScopeCleanable extends PhantomCleanable { - final Cleaner cleaner; - final Runnable cleanupAction; - - public ScopeCleanable(MemoryScope referent, Cleaner cleaner, Runnable cleanupAction) { - super(referent, cleaner); - this.cleaner = cleaner; - this.cleanupAction = cleanupAction; - } - - @Override - protected void performCleanup() { - cleanupAction.run(); - } - } -} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java index 395f21f4d51..8d61c9b79b5 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java @@ -28,6 +28,8 @@ package jdk.internal.foreign; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; import jdk.internal.misc.Unsafe; import jdk.internal.misc.VM; import jdk.internal.vm.annotation.ForceInline; @@ -41,34 +43,40 @@ import java.nio.ByteBuffer; */ public class NativeMemorySegmentImpl extends AbstractMemorySegmentImpl { - public static final MemorySegment EVERYTHING = makeNativeSegmentUnchecked(MemoryAddress.NULL, Long.MAX_VALUE, MemoryScope.DUMMY_CLEANUP_ACTION, null) - .share() - .withAccessModes(READ | WRITE); + public static final MemorySegment EVERYTHING = makeNativeSegmentUnchecked(MemoryAddress.NULL, Long.MAX_VALUE, null, ResourceScopeImpl.GLOBAL); private static final Unsafe unsafe = Unsafe.getUnsafe(); + public static final SegmentAllocator IMPLICIT_ALLOCATOR = (size, align) -> MemorySegment.allocateNative(size, align, ResourceScope.newImplicitScope()); + // The maximum alignment supported by malloc - typically 16 on // 64-bit platforms and 8 on 32-bit platforms. - private final static long MAX_MALLOC_ALIGN = Unsafe.ADDRESS_SIZE == 4 ? 8 : 16; + private static final long MAX_MALLOC_ALIGN = Unsafe.ADDRESS_SIZE == 4 ? 8 : 16; private static final boolean skipZeroMemory = GetBooleanAction.privilegedGetProperty("jdk.internal.foreign.skipZeroMemory"); final long min; @ForceInline - NativeMemorySegmentImpl(long min, long length, int mask, MemoryScope scope) { + NativeMemorySegmentImpl(long min, long length, int mask, ResourceScopeImpl scope) { super(length, mask, scope); this.min = min; } @Override - NativeMemorySegmentImpl dup(long offset, long size, int mask, MemoryScope scope) { + NativeMemorySegmentImpl dup(long offset, long size, int mask, ResourceScopeImpl scope) { return new NativeMemorySegmentImpl(min + offset, size, mask, scope); } @Override ByteBuffer makeByteBuffer() { - return nioAccess.newDirectByteBuffer(min(), (int) this.length, null, this); + return nioAccess.newDirectByteBuffer(min(), (int) this.length, null, + scope == ResourceScopeImpl.GLOBAL ? null : this); + } + + @Override + public boolean isNative() { + return true; } @Override @@ -83,7 +91,8 @@ public class NativeMemorySegmentImpl extends AbstractMemorySegmentImpl { // factories - public static MemorySegment makeNativeSegment(long bytesSize, long alignmentBytes) { + public static MemorySegment makeNativeSegment(long bytesSize, long alignmentBytes, ResourceScopeImpl scope) { + scope.checkValidStateSlow(); if (VM.isDirectMemoryPageAligned()) { alignmentBytes = Math.max(alignmentBytes, nioAccess.pageSize()); } @@ -98,12 +107,15 @@ public class NativeMemorySegmentImpl extends AbstractMemorySegmentImpl { unsafe.setMemory(buf, alignedSize, (byte)0); } long alignedBuf = Utils.alignUp(buf, alignmentBytes); - MemoryScope scope = MemoryScope.createConfined(null, () -> { + AbstractMemorySegmentImpl segment = new NativeMemorySegmentImpl(buf, alignedSize, + defaultAccessModes(alignedSize), scope); + scope.addOrCleanupIfFail(new ResourceScopeImpl.ResourceList.ResourceCleanup() { + @Override + public void cleanup() { unsafe.freeMemory(buf); nioAccess.unreserveMemory(alignedSize, bytesSize); - }, null); - MemorySegment segment = new NativeMemorySegmentImpl(buf, alignedSize, - defaultAccessModes(alignedSize), scope); + } + }); if (alignedSize != bytesSize) { long delta = alignedBuf - buf; segment = segment.asSlice(delta, bytesSize); @@ -111,8 +123,12 @@ public class NativeMemorySegmentImpl extends AbstractMemorySegmentImpl { return segment; } - public static MemorySegment makeNativeSegmentUnchecked(MemoryAddress min, long bytesSize, Runnable cleanupAction, Object ref) { - return new NativeMemorySegmentImpl(min.toRawLongValue(), bytesSize, defaultAccessModes(bytesSize), - MemoryScope.createConfined(ref, cleanupAction == null ? MemoryScope.DUMMY_CLEANUP_ACTION : cleanupAction, null)); + public static MemorySegment makeNativeSegmentUnchecked(MemoryAddress min, long bytesSize, Runnable cleanupAction, ResourceScopeImpl scope) { + scope.checkValidStateSlow(); + AbstractMemorySegmentImpl segment = new NativeMemorySegmentImpl(min.toRawLongValue(), bytesSize, defaultAccessModes(bytesSize), scope); + if (cleanupAction != null) { + scope.addCloseAction(cleanupAction); + } + return segment; } } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/PlatformLayouts.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/PlatformLayouts.java index 16e268acc68..1b378c8a80e 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/PlatformLayouts.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/PlatformLayouts.java @@ -32,7 +32,6 @@ import jdk.incubator.foreign.ValueLayout; import java.nio.ByteOrder; import static java.nio.ByteOrder.LITTLE_ENDIAN; -import static jdk.incubator.foreign.MemoryLayouts.ADDRESS; public class PlatformLayouts { public static Z pick(Z sysv, Z win64, Z aarch64) { @@ -51,42 +50,42 @@ public class PlatformLayouts { } private static ValueLayout ofChar(ByteOrder order, long bitSize) { - return MemoryLayout.ofValueBits(bitSize, order) + return MemoryLayout.valueLayout(bitSize, order) .withAttribute(CLinker.TypeKind.ATTR_NAME, CLinker.TypeKind.CHAR); } private static ValueLayout ofShort(ByteOrder order, long bitSize) { - return MemoryLayout.ofValueBits(bitSize, order) + return MemoryLayout.valueLayout(bitSize, order) .withAttribute(CLinker.TypeKind.ATTR_NAME, CLinker.TypeKind.SHORT); } private static ValueLayout ofInt(ByteOrder order, long bitSize) { - return MemoryLayout.ofValueBits(bitSize, order) + return MemoryLayout.valueLayout(bitSize, order) .withAttribute(CLinker.TypeKind.ATTR_NAME, CLinker.TypeKind.INT); } private static ValueLayout ofLong(ByteOrder order, long bitSize) { - return MemoryLayout.ofValueBits(bitSize, order) + return MemoryLayout.valueLayout(bitSize, order) .withAttribute(CLinker.TypeKind.ATTR_NAME, CLinker.TypeKind.LONG); } private static ValueLayout ofLongLong(ByteOrder order, long bitSize) { - return MemoryLayout.ofValueBits(bitSize, order) + return MemoryLayout.valueLayout(bitSize, order) .withAttribute(CLinker.TypeKind.ATTR_NAME, CLinker.TypeKind.LONG_LONG); } private static ValueLayout ofFloat(ByteOrder order, long bitSize) { - return MemoryLayout.ofValueBits(bitSize, order) + return MemoryLayout.valueLayout(bitSize, order) .withAttribute(CLinker.TypeKind.ATTR_NAME, CLinker.TypeKind.FLOAT); } private static ValueLayout ofDouble(ByteOrder order, long bitSize) { - return MemoryLayout.ofValueBits(bitSize, order) + return MemoryLayout.valueLayout(bitSize, order) .withAttribute(CLinker.TypeKind.ATTR_NAME, CLinker.TypeKind.DOUBLE); } private static ValueLayout ofPointer(ByteOrder order, long bitSize) { - return MemoryLayout.ofValueBits(bitSize, order) + return MemoryLayout.valueLayout(bitSize, order) .withAttribute(CLinker.TypeKind.ATTR_NAME, CLinker.TypeKind.POINTER); } @@ -162,7 +161,7 @@ public class PlatformLayouts { * The name of the layout attribute (see {@link MemoryLayout#attributes()} used to mark variadic parameters. The * attribute value must be a boolean. */ - public final static String VARARGS_ATTRIBUTE_NAME = "abi/windows/varargs"; + public static final String VARARGS_ATTRIBUTE_NAME = "abi/windows/varargs"; /** * The {@code char} native type. diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ResourceScopeImpl.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ResourceScopeImpl.java new file mode 100644 index 00000000000..ffd225a6b26 --- /dev/null +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/ResourceScopeImpl.java @@ -0,0 +1,322 @@ +/* + * Copyright (c) 2019, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + * + */ + +package jdk.internal.foreign; + +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; +import jdk.internal.misc.ScopedMemoryAccess; +import jdk.internal.ref.CleanerFactory; + +import java.lang.ref.Cleaner; +import java.lang.ref.Reference; +import java.util.Objects; + +/** + * This class manages the temporal bounds associated with a memory segment as well + * as thread confinement. A scope has a liveness bit, which is updated when the scope is closed + * (this operation is triggered by {@link ResourceScope#close()}). This bit is consulted prior + * to memory access (see {@link #checkValidState()}). + * There are two kinds of memory scope: confined memory scope and shared memory scope. + * A confined memory scope has an associated owner thread that confines some operations to + * associated owner thread such as {@link #close()} or {@link #checkValidState()}. + * Shared scopes do not feature an owner thread - meaning their operations can be called, in a racy + * manner, by multiple threads. To guarantee temporal safety in the presence of concurrent thread, + * shared scopes use a more sophisticated synchronization mechanism, which guarantees that no concurrent + * access is possible when a scope is being closed (see {@link jdk.internal.misc.ScopedMemoryAccess}). + */ +public abstract non-sealed class ResourceScopeImpl implements ResourceScope, ScopedMemoryAccess.Scope, SegmentAllocator { + + final ResourceList resourceList; + + @Override + public void addCloseAction(Runnable runnable) { + Objects.requireNonNull(runnable); + addInternal(ResourceList.ResourceCleanup.ofRunnable(runnable)); + } + + @Override + public boolean isImplicit() { + return false; + } + + /** + * Add a cleanup action. If a failure occurred (because of a add vs. close race), call the cleanup action. + * This semantics is useful when allocating new memory segments, since we first do a malloc/mmap and _then_ + * we register the cleanup (free/munmap) against the scope; so, if registration fails, we still have to + * cleanup memory. From the perspective of the client, such a failure would manifest as a factory + * returning a segment that is already "closed" - which is always possible anyway (e.g. if the scope + * is closed _after_ the cleanup for the segment is registered but _before_ the factory returns the + * new segment to the client). For this reason, it's not worth adding extra complexity to the segment + * initialization logic here - and using an optimistic logic works well in practice. + */ + public void addOrCleanupIfFail(ResourceList.ResourceCleanup resource) { + try { + addInternal(resource); + } catch (Throwable ex) { + resource.cleanup(); + } + } + + void addInternal(ResourceList.ResourceCleanup resource) { + try { + checkValidStateSlow(); + resourceList.add(resource); + } catch (ScopedMemoryAccess.Scope.ScopedAccessError err) { + throw new IllegalStateException("Already closed"); + } + } + + protected ResourceScopeImpl(Cleaner cleaner, ResourceList resourceList) { + this.resourceList = resourceList; + if (cleaner != null) { + cleaner.register(this, resourceList); + } + } + + public static ResourceScopeImpl createImplicitScope() { + return new ImplicitScopeImpl(CleanerFactory.cleaner()); + } + + public static ResourceScopeImpl createConfined(Thread thread, Cleaner cleaner) { + return new ConfinedScope(thread, cleaner); + } + + /** + * Creates a confined memory scope with given attachment and cleanup action. The returned scope + * is assumed to be confined on the current thread. + * @return a confined memory scope + */ + public static ResourceScopeImpl createConfined(Cleaner cleaner) { + return new ConfinedScope(Thread.currentThread(), cleaner); + } + + /** + * Creates a shared memory scope with given attachment and cleanup action. + * @return a shared memory scope + */ + public static ResourceScopeImpl createShared(Cleaner cleaner) { + return new SharedScope(cleaner); + } + + private final void release0(HandleImpl handle) { + try { + Objects.requireNonNull(handle); + if (handle.scope() != this) { + throw new IllegalArgumentException("Cannot release an handle acquired from another scope"); + } + handle.release(); + } finally { + Reference.reachabilityFence(this); + } + } + + @Override + public final void release(ResourceScope.Handle handle) { + release0((HandleImpl)handle); + } + + @Override + public final void release(ScopedMemoryAccess.Scope.Handle handle) { + release0((HandleImpl)handle); + } + + @Override + public abstract HandleImpl acquire(); + + /** + * Internal interface used to implement resource scope handles. + */ + public non-sealed interface HandleImpl extends ResourceScope.Handle, ScopedMemoryAccess.Scope.Handle { + + @Override + ResourceScopeImpl scope(); + + void release(); + } + + /** + * Closes this scope, executing any cleanup action (where provided). + * @throws IllegalStateException if this scope is already closed or if this is + * a confined scope and this method is called outside of the owner thread. + */ + public void close() { + try { + justClose(); + resourceList.cleanup(); + } finally { + Reference.reachabilityFence(this); + } + } + + abstract void justClose(); + + /** + * Returns "owner" thread of this scope. + * @return owner thread (or null for a shared scope) + */ + public abstract Thread ownerThread(); + + /** + * Returns true, if this scope is still alive. This method may be called in any thread. + * @return {@code true} if this scope is not closed yet. + */ + public abstract boolean isAlive(); + + + /** + * This is a faster version of {@link #checkValidStateSlow()}, which is called upon memory access, and which + * relies on invariants associated with the memory scope implementations (typically, volatile access + * to the closed state bit is replaced with plain access, and ownership check is removed where not needed. + * Should be used with care. + */ + public abstract void checkValidState(); + + /** + * Checks that this scope is still alive (see {@link #isAlive()}). + * @throws IllegalStateException if this scope is already closed or if this is + * a confined scope and this method is called outside of the owner thread. + */ + public final void checkValidStateSlow() { + if (ownerThread() != null && Thread.currentThread() != ownerThread()) { + throw new IllegalStateException("Attempted access outside owning thread"); + } else if (!isAlive()) { + throw new IllegalStateException("Already closed"); + } + } + + @Override + protected Object clone() throws CloneNotSupportedException { + throw new CloneNotSupportedException(); + } + + /** + * Allocates a segment using this scope. Used by {@link SegmentAllocator#ofScope(ResourceScope)}. + */ + @Override + public MemorySegment allocate(long bytesSize, long bytesAlignment) { + return MemorySegment.allocateNative(bytesSize, bytesAlignment, this); + } + + /** + * A non-closeable, shared scope. Similar to a shared scope, but its {@link #close()} method throws unconditionally. + * In addition, non-closeable scopes feature a much simpler scheme for generating resource scope handles, where + * the scope itself also acts as a resource scope handle and is returned by {@link #acquire()}. + */ + static class ImplicitScopeImpl extends SharedScope implements HandleImpl { + + public ImplicitScopeImpl(Cleaner cleaner) { + super(cleaner); + } + + @Override + public HandleImpl acquire() { + return this; + } + + @Override + public boolean isImplicit() { + return true; + } + + @Override + public void close() { + throw new UnsupportedOperationException("Scope cannot be closed"); + } + + @Override + public void release() { + // do nothing + } + + @Override + public ResourceScopeImpl scope() { + return this; + } + } + + /** + * The global, always alive, non-closeable, shared scope. This is like a {@link ImplicitScopeImpl non-closeable scope}, + * except that the operation which adds new resources to the global scope does nothing: as the scope can never + * become not-alive, there is nothing to track. + */ + public static final ResourceScopeImpl GLOBAL = new ImplicitScopeImpl( null) { + @Override + void addInternal(ResourceList.ResourceCleanup resource) { + // do nothing + } + }; + + /** + * A list of all cleanup actions associated with a resource scope. Cleanup actions are modelled as instances + * of the {@link ResourceCleanup} class, and, together, form a linked list. Depending on whether a scope + * is shared or confined, different implementations of this class will be used, see {@link ConfinedScope.ConfinedResourceList} + * and {@link SharedScope.SharedResourceList}. + */ + public abstract static class ResourceList implements Runnable { + ResourceCleanup fst; + + abstract void add(ResourceCleanup cleanup); + + abstract void cleanup(); + + public final void run() { + cleanup(); // cleaner interop + } + + static void cleanup(ResourceCleanup first) { + ResourceCleanup current = first; + while (current != null) { + current.cleanup(); + current = current.next; + } + } + + public static abstract class ResourceCleanup { + ResourceCleanup next; + + public abstract void cleanup(); + + static final ResourceCleanup CLOSED_LIST = new ResourceCleanup() { + @Override + public void cleanup() { + throw new IllegalStateException("This resource list has already been closed!"); + } + }; + + static ResourceCleanup ofRunnable(Runnable cleanupAction) { + return new ResourceCleanup() { + @Override + public void cleanup() { + cleanupAction.run(); + } + }; + } + } + + } +} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/SharedScope.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/SharedScope.java new file mode 100644 index 00000000000..8bbeaaab829 --- /dev/null +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/SharedScope.java @@ -0,0 +1,196 @@ +/* + * Copyright (c) 2021, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ + +package jdk.internal.foreign; + +import jdk.incubator.foreign.ResourceScope; +import jdk.internal.misc.ScopedMemoryAccess; + +import java.lang.invoke.MethodHandles; +import java.lang.invoke.VarHandle; +import java.lang.ref.Cleaner; +import java.lang.ref.Reference; +import java.util.concurrent.atomic.AtomicBoolean; + +/** + * A shared scope, which can be shared across multiple threads. Closing a shared scope has to ensure that + * (i) only one thread can successfully close a scope (e.g. in a close vs. close race) and that + * (ii) no other thread is accessing the memory associated with this scope while the segment is being + * closed. To ensure the former condition, a CAS is performed on the liveness bit. Ensuring the latter + * is trickier, and require a complex synchronization protocol (see {@link jdk.internal.misc.ScopedMemoryAccess}). + * Since it is the responsibility of the closing thread to make sure that no concurrent access is possible, + * checking the liveness bit upon access can be performed in plain mode, as in the confined case. + */ +class SharedScope extends ResourceScopeImpl { + + private static final ScopedMemoryAccess SCOPED_MEMORY_ACCESS = ScopedMemoryAccess.getScopedMemoryAccess(); + + private static final int ALIVE = 0; + private static final int CLOSING = -1; + private static final int CLOSED = -2; + private static final int MAX_FORKS = Integer.MAX_VALUE; + + private int state = ALIVE; + + private static final VarHandle STATE; + + static { + try { + STATE = MethodHandles.lookup().findVarHandle(jdk.internal.foreign.SharedScope.class, "state", int.class); + } catch (Throwable ex) { + throw new ExceptionInInitializerError(ex); + } + } + + SharedScope(Cleaner cleaner) { + super(cleaner, new SharedResourceList()); + } + + @Override + public Thread ownerThread() { + return null; + } + + @Override + public void checkValidState() { + if (state < ALIVE) { + throw ScopedAccessError.INSTANCE; + } + } + + @Override + public HandleImpl acquire() { + int value; + do { + value = (int) STATE.getVolatile(this); + if (value < ALIVE) { + //segment is not alive! + throw new IllegalStateException("Already closed"); + } else if (value == MAX_FORKS) { + //overflow + throw new IllegalStateException("Segment acquire limit exceeded"); + } + } while (!STATE.compareAndSet(this, value, value + 1)); + return new SharedHandle(); + } + + void justClose() { + int prevState = (int) STATE.compareAndExchange(this, ALIVE, CLOSING); + if (prevState < 0) { + throw new IllegalStateException("Already closed"); + } else if (prevState != ALIVE) { + throw new IllegalStateException("Scope is acquired by " + prevState + " locks"); + } + boolean success = SCOPED_MEMORY_ACCESS.closeScope(this); + STATE.setVolatile(this, success ? CLOSED : ALIVE); + if (!success) { + throw new IllegalStateException("Cannot close while another thread is accessing the segment"); + } + } + + @Override + public boolean isAlive() { + return (int) STATE.getVolatile(this) != CLOSED; + } + + /** + * A shared resource list; this implementation has to handle add vs. add races, as well as add vs. cleanup races. + */ + static class SharedResourceList extends ResourceList { + + static final VarHandle FST; + + static { + try { + FST = MethodHandles.lookup().findVarHandle(ResourceList.class, "fst", ResourceCleanup.class); + } catch (Throwable ex) { + throw new ExceptionInInitializerError(); + } + } + + @Override + void add(ResourceCleanup cleanup) { + while (true) { + ResourceCleanup prev = (ResourceCleanup) FST.getAcquire(this); + cleanup.next = prev; + ResourceCleanup newSegment = (ResourceCleanup) FST.compareAndExchangeRelease(this, prev, cleanup); + if (newSegment == ResourceCleanup.CLOSED_LIST) { + // too late + throw new IllegalStateException("Already closed"); + } else if (newSegment == prev) { + return; //victory + } + // keep trying + } + } + + void cleanup() { + // At this point we are only interested about add vs. close races - not close vs. close + // (because MemoryScope::justClose ensured that this thread won the race to close the scope). + // So, the only "bad" thing that could happen is that some other thread adds to this list + // while we're closing it. + if (FST.getAcquire(this) != ResourceCleanup.CLOSED_LIST) { + //ok now we're really closing down + ResourceCleanup prev = null; + while (true) { + prev = (ResourceCleanup) FST.getAcquire(this); + // no need to check for DUMMY, since only one thread can get here! + if (FST.weakCompareAndSetRelease(this, prev, ResourceCleanup.CLOSED_LIST)) { + break; + } + } + cleanup(prev); + } else { + throw new IllegalStateException("Attempt to cleanup an already closed resource list"); + } + } + } + + /** + * A shared resource scope handle; this implementation has to handle close vs. close races. + */ + class SharedHandle implements HandleImpl { + final AtomicBoolean released = new AtomicBoolean(false); + + @Override + public ResourceScopeImpl scope() { + return SharedScope.this; + } + + @Override + public void release() { + if (released.compareAndSet(false, true)) { + int value; + do { + value = (int) STATE.getVolatile(jdk.internal.foreign.SharedScope.this); + if (value <= ALIVE) { + //cannot get here - we can't close segment twice + throw new IllegalStateException("Already closed"); + } + } while (!STATE.compareAndSet(jdk.internal.foreign.SharedScope.this, value, value - 1)); + } + } + } +} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/Utils.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/Utils.java index 4aea34250e7..f3b0240d2d0 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/Utils.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/Utils.java @@ -44,14 +44,10 @@ import static sun.security.action.GetPropertyAction.*; * This class contains misc helper functions to support creation of memory segments. */ public final class Utils { - // used when testing invoke exact behavior of memory access handles private static final boolean SHOULD_ADAPT_HANDLES = Boolean.parseBoolean(privilegedGetProperty("jdk.internal.foreign.SHOULD_ADAPT_HANDLES", "true")); - private static final String foreignRestrictedAccess = Optional.ofNullable(VM.getSavedProperty("foreign.restricted")) - .orElse("deny"); - private static final MethodHandle SEGMENT_FILTER; public static final MethodHandle MH_bitsToBytesOrThrowForOffset; @@ -107,27 +103,6 @@ public final class Utils { return (AbstractMemorySegmentImpl)segment; } - public static void checkRestrictedAccess(String method) { - switch (foreignRestrictedAccess) { - case "deny" -> throwIllegalAccessError(foreignRestrictedAccess, method); - case "warn" -> System.err.println("WARNING: Accessing restricted foreign method: " + method); - case "debug" -> { - StringBuilder sb = new StringBuilder("DEBUG: restricted foreign method: \" + method"); - StackWalker.getInstance().forEach(f -> sb.append(System.lineSeparator()) - .append("\tat ") - .append(f)); - System.err.println(sb.toString()); - } - case "permit" -> {} - default -> throwIllegalAccessError(foreignRestrictedAccess, method); - } - } - - private static void throwIllegalAccessError(String value, String method) { - throw new IllegalAccessError("Illegal access to restricted foreign method: " + method + - " ; system property 'foreign.restricted' is set to '" + value + "'"); - } - public static void checkPrimitiveCarrierCompat(Class carrier, MemoryLayout layout) { checkLayoutType(layout, ValueLayout.class); if (!isValidPrimitiveCarrier(carrier)) @@ -150,5 +125,4 @@ public final class Utils { if (!layoutType.isInstance(layout)) throw new IllegalArgumentException("Expected a " + layoutType.getSimpleName() + ": " + layout); } - } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/Binding.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/Binding.java index e0771ddb7ae..5b8468eafd3 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/Binding.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/Binding.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -28,11 +28,13 @@ import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; import jdk.internal.foreign.MemoryAddressImpl; +import jdk.internal.foreign.ResourceScopeImpl; import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; -import java.lang.invoke.MethodType; import java.util.ArrayList; import java.util.Deque; import java.util.List; @@ -44,7 +46,6 @@ import java.nio.ByteOrder; import static java.lang.invoke.MethodHandles.collectArguments; import static java.lang.invoke.MethodHandles.filterArguments; import static java.lang.invoke.MethodHandles.insertArguments; -import static java.lang.invoke.MethodHandles.permuteArguments; import static java.lang.invoke.MethodType.methodType; /** @@ -218,16 +219,99 @@ public abstract class Binding { MH_BASE_ADDRESS = lookup.findVirtual(MemorySegment.class, "address", methodType(MemoryAddress.class)); MH_COPY_BUFFER = lookup.findStatic(Binding.Copy.class, "copyBuffer", - methodType(MemorySegment.class, MemorySegment.class, long.class, long.class, SharedUtils.Allocator.class)); + methodType(MemorySegment.class, MemorySegment.class, long.class, long.class, Context.class)); MH_ALLOCATE_BUFFER = lookup.findStatic(Binding.Allocate.class, "allocateBuffer", - methodType(MemorySegment.class, long.class, long.class, SharedUtils.Allocator.class)); + methodType(MemorySegment.class, long.class, long.class, Context.class)); MH_TO_SEGMENT = lookup.findStatic(Binding.ToSegment.class, "toSegment", - methodType(MemorySegment.class, MemoryAddress.class, long.class)); + methodType(MemorySegment.class, MemoryAddress.class, long.class, Context.class)); } catch (ReflectiveOperationException e) { throw new RuntimeException(e); } } + /** + * A binding context is used as an helper to carry out evaluation of certain bindings; for instance, + * it helps {@link Allocate} bindings, by providing the {@link SegmentAllocator} that should be used for + * the allocation operation, or {@link ToSegment} bindings, by providing the {@link ResourceScope} that + * should be used to create an unsafe struct from a memory address. + */ + public static class Context implements AutoCloseable { + private final SegmentAllocator allocator; + private final ResourceScope scope; + + private Context(SegmentAllocator allocator, ResourceScope scope) { + this.allocator = allocator; + this.scope = scope; + } + + public SegmentAllocator allocator() { + return allocator; + } + + public ResourceScope scope() { + return scope; + } + + @Override + public void close() { + scope().close(); + } + + /** + * Create a binding context from given native scope. + */ + public static Context ofBoundedAllocator(long size) { + ResourceScope scope = ResourceScope.newConfinedScope(); + return new Context(SegmentAllocator.arenaAllocator(size, scope), scope); + } + + /** + * Create a binding context from given segment allocator. The resulting context will throw when + * the context's scope is accessed. + */ + public static Context ofAllocator(SegmentAllocator allocator) { + return new Context(allocator, null) { + @Override + public ResourceScope scope() { + throw new UnsupportedOperationException(); + } + }; + } + + /** + * Create a binding context from given scope. The resulting context will throw when + * the context's allocator is accessed. + */ + public static Context ofScope() { + ResourceScope scope = ResourceScope.newConfinedScope(); + return new Context(null, scope) { + @Override + public SegmentAllocator allocator() { throw new UnsupportedOperationException(); } + }; + } + + /** + * Dummy binding context. Throws exceptions when attempting to access scope, return a throwing allocator, and has + * an idempotent {@link #close()}. + */ + public static final Context DUMMY = new Context(null, null) { + @Override + public SegmentAllocator allocator() { + return SharedUtils.THROWING_ALLOCATOR; + } + + @Override + public ResourceScope scope() { + throw new UnsupportedOperationException(); + } + + @Override + public void close() { + // do nothing + } + }; + } + enum Tag { VM_STORE, VM_LOAD, @@ -255,31 +339,10 @@ public abstract class Binding { public abstract void verify(Deque> stack); public abstract void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator); + BindingInterpreter.LoadFunc loadFunc, Context context); public abstract MethodHandle specialize(MethodHandle specializedHandle, int insertPos, int allocatorPos); - private static MethodHandle mergeArguments(MethodHandle mh, int sourceIndex, int destIndex) { - MethodType oldType = mh.type(); - Class sourceType = oldType.parameterType(sourceIndex); - Class destType = oldType.parameterType(destIndex); - if (sourceType != destType) { - // TODO meet? - throw new IllegalArgumentException("Parameter types differ: " + sourceType + " != " + destType); - } - MethodType newType = oldType.dropParameterTypes(destIndex, destIndex + 1); - int[] reorder = new int[oldType.parameterCount()]; - assert destIndex > sourceIndex; - for (int i = 0, index = 0; i < reorder.length; i++) { - if (i != destIndex) { - reorder[i] = index++; - } else { - reorder[i] = sourceIndex; - } - } - return permuteArguments(mh, newType, reorder); - } - private static void checkType(Class type) { if (!type.isPrimitive() || type == void.class || type == boolean.class) throw new IllegalArgumentException("Illegal type: " + type); @@ -477,7 +540,7 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { storeFunc.store(storage(), type(), stack.pop()); } @@ -512,7 +575,7 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { stack.push(loadFunc.load(storage(), type())); } @@ -564,7 +627,10 @@ public abstract class Binding { } public VarHandle varHandle() { - return MemoryHandles.insertCoordinates(MemoryHandles.varHandle(type, ByteOrder.nativeOrder()), 1, offset); + // alignment is set to 1 byte here to avoid exceptions for cases where we do super word + // copies of e.g. 2 int fields of a struct as a single long, while the struct is only + // 4-byte-aligned (since it only contains ints) + return MemoryHandles.insertCoordinates(MemoryHandles.varHandle(type, 1, ByteOrder.nativeOrder()), 1, offset); } } @@ -589,7 +655,7 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { Object value = stack.pop(); MemorySegment operand = (MemorySegment) stack.pop(); MemorySegment writeAddress = operand.asSlice(offset()); @@ -633,7 +699,7 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { MemorySegment operand = (MemorySegment) stack.pop(); MemorySegment readAddress = operand.asSlice(offset()); stack.push(SharedUtils.read(readAddress, type())); @@ -673,8 +739,8 @@ public abstract class Binding { } private static MemorySegment copyBuffer(MemorySegment operand, long size, long alignment, - SharedUtils.Allocator allocator) { - MemorySegment copy = allocator.allocate(size, alignment); + Context context) { + MemorySegment copy = context.allocator().allocate(size, alignment); copy.copyFrom(operand.asSlice(0, size)); return copy; } @@ -705,9 +771,9 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { MemorySegment operand = (MemorySegment) stack.pop(); - MemorySegment copy = copyBuffer(operand, size, alignment, allocator); + MemorySegment copy = copyBuffer(operand, size, alignment, context); stack.push(copy); } @@ -715,7 +781,7 @@ public abstract class Binding { public MethodHandle specialize(MethodHandle specializedHandle, int insertPos, int allocatorPos) { MethodHandle filter = insertArguments(MH_COPY_BUFFER, 1, size, alignment); specializedHandle = collectArguments(specializedHandle, insertPos, filter); - return mergeArguments(specializedHandle, allocatorPos, insertPos + 1); + return SharedUtils.mergeArguments(specializedHandle, allocatorPos, insertPos + 1); } @Override @@ -748,8 +814,8 @@ public abstract class Binding { this.alignment = alignment; } - private static MemorySegment allocateBuffer(long size, long allignment, SharedUtils.Allocator allocator) { - return allocator.allocate(size, allignment); + private static MemorySegment allocateBuffer(long size, long allignment, Context context) { + return context.allocator().allocate(size, allignment); } public long size() { @@ -776,15 +842,15 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { - stack.push(allocateBuffer(size, alignment, allocator)); + BindingInterpreter.LoadFunc loadFunc, Context context) { + stack.push(allocateBuffer(size, alignment, context)); } @Override public MethodHandle specialize(MethodHandle specializedHandle, int insertPos, int allocatorPos) { MethodHandle allocateBuffer = insertArguments(MH_ALLOCATE_BUFFER, 0, size, alignment); specializedHandle = collectArguments(specializedHandle, insertPos, allocateBuffer); - return mergeArguments(specializedHandle, allocatorPos, insertPos); + return SharedUtils.mergeArguments(specializedHandle, allocatorPos, insertPos); } @Override @@ -823,7 +889,7 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { stack.push(((MemoryAddress)stack.pop()).toRawLongValue()); } @@ -839,7 +905,7 @@ public abstract class Binding { } /** - * Box_ADDRESS() + * BOX_ADDRESS() * Pops a 'long' from the operand stack, converts it to a 'MemoryAddress', * and pushes that onto the operand stack. */ @@ -858,7 +924,7 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { stack.push(MemoryAddress.ofLong((long) stack.pop())); } @@ -893,7 +959,7 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { stack.push(((MemorySegment) stack.pop()).address()); } @@ -909,8 +975,8 @@ public abstract class Binding { } /** - * BASE_ADDRESS([size]) - * Pops a MemoryAddress from the operand stack, and takes the converts it to a MemorySegment + * TO_SEGMENT([size]) + * Pops a MemoryAddress from the operand stack, and converts it to a MemorySegment * with the given size, and pushes that onto the operand stack */ public static class ToSegment extends Binding { @@ -922,9 +988,8 @@ public abstract class Binding { this.size = size; } - // FIXME should register with scope - private static MemorySegment toSegment(MemoryAddress operand, long size) { - return MemoryAddressImpl.ofLongUnchecked(operand.toRawLongValue(), size); + private static MemorySegment toSegment(MemoryAddress operand, long size, Context context) { + return MemoryAddressImpl.ofLongUnchecked(operand.toRawLongValue(), size, (ResourceScopeImpl) context.scope); } @Override @@ -936,16 +1001,17 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { MemoryAddress operand = (MemoryAddress) stack.pop(); - MemorySegment segment = toSegment(operand, size); + MemorySegment segment = toSegment(operand, size, context); stack.push(segment); } @Override public MethodHandle specialize(MethodHandle specializedHandle, int insertPos, int allocatorPos) { MethodHandle toSegmentHandle = insertArguments(MH_TO_SEGMENT, 1, size); - return filterArguments(specializedHandle, insertPos, toSegmentHandle); + specializedHandle = collectArguments(specializedHandle, insertPos, toSegmentHandle); + return SharedUtils.mergeArguments(specializedHandle, allocatorPos, insertPos + 1); } @Override @@ -988,7 +1054,7 @@ public abstract class Binding { @Override public void interpret(Deque stack, BindingInterpreter.StoreFunc storeFunc, - BindingInterpreter.LoadFunc loadFunc, SharedUtils.Allocator allocator) { + BindingInterpreter.LoadFunc loadFunc, Context context) { stack.push(stack.peekLast()); } @@ -1012,7 +1078,7 @@ public abstract class Binding { */ @Override public MethodHandle specialize(MethodHandle specializedHandle, int insertPos, int allocatorPos) { - return mergeArguments(specializedHandle, insertPos, insertPos + 1); + return SharedUtils.mergeArguments(specializedHandle, insertPos, insertPos + 1); } @Override diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/BindingInterpreter.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/BindingInterpreter.java index 019ccbbe040..3a29c5fbc01 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/BindingInterpreter.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/BindingInterpreter.java @@ -30,19 +30,19 @@ import java.util.List; public class BindingInterpreter { - static void unbox(Object arg, List bindings, StoreFunc storeFunc, SharedUtils.Allocator allocator) { + static void unbox(Object arg, List bindings, StoreFunc storeFunc, Binding.Context context) { Deque stack = new ArrayDeque<>(); stack.push(arg); for (Binding b : bindings) { - b.interpret(stack, storeFunc, null, allocator); + b.interpret(stack, storeFunc, null, context); } } - static Object box(List bindings, LoadFunc loadFunc, SharedUtils.Allocator allocator) { + static Object box(List bindings, LoadFunc loadFunc, Binding.Context context) { Deque stack = new ArrayDeque<>(); for (Binding b : bindings) { - b.interpret(stack, null, loadFunc, allocator); + b.interpret(stack, null, loadFunc, context); } return stack.pop(); } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/BufferLayout.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/BufferLayout.java index ac5ca419ca7..d290a36cc73 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/BufferLayout.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/BufferLayout.java @@ -26,6 +26,7 @@ package jdk.internal.foreign.abi; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.internal.foreign.MemoryAddressImpl; import java.io.PrintStream; import java.lang.invoke.VarHandle; @@ -115,7 +116,7 @@ class BufferLayout { return Long.toHexString((long) VH_LONG.get(buffer.asSlice(offset))); } - private static void dumpValues(jdk.internal.foreign.abi.Architecture arch, MemorySegment buff, PrintStream stream, + private void dumpValues(jdk.internal.foreign.abi.Architecture arch, MemorySegment buff, PrintStream stream, Map offsets) { for (var entry : offsets.entrySet()) { VMStorage storage = entry.getKey(); @@ -128,6 +129,14 @@ class BufferLayout { } stream.println("}"); } + long stack_ptr = (long) VH_LONG.get(buff.asSlice(stack_args)); + long stack_bytes = (long) VH_LONG.get(buff.asSlice(stack_args_bytes)); + MemorySegment stackArgs = MemoryAddressImpl.ofLongUnchecked(stack_ptr, stack_bytes); + stream.println("Stack {"); + for (int i = 0; i < stack_bytes / 8; i += 8) { + stream.printf(" @%d: %s%n", i, getLongString(stackArgs, i)); + } + stream.println("}"); } void dump(Architecture arch, MemorySegment buff, PrintStream stream) { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/ProgrammableInvoker.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/ProgrammableInvoker.java index d2aa871040f..b8db59f7727 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/ProgrammableInvoker.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/ProgrammableInvoker.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -25,12 +25,13 @@ package jdk.internal.foreign.abi; import jdk.incubator.foreign.Addressable; +import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; -import jdk.incubator.foreign.NativeScope; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; import jdk.internal.access.JavaLangInvokeAccess; import jdk.internal.access.SharedSecrets; -import jdk.internal.foreign.Utils; import jdk.internal.invoke.NativeEntryPoint; import jdk.internal.invoke.VMStorageProxy; import sun.security.action.GetPropertyAction; @@ -39,24 +40,19 @@ import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; import java.lang.invoke.VarHandle; +import java.lang.ref.Reference; import java.util.Arrays; import java.util.List; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; -import java.util.stream.Collectors; -import java.util.stream.IntStream; import java.util.stream.Stream; import static java.lang.invoke.MethodHandles.collectArguments; import static java.lang.invoke.MethodHandles.dropArguments; -import static java.lang.invoke.MethodHandles.empty; import static java.lang.invoke.MethodHandles.filterArguments; import static java.lang.invoke.MethodHandles.identity; import static java.lang.invoke.MethodHandles.insertArguments; -import static java.lang.invoke.MethodHandles.tryFinally; import static java.lang.invoke.MethodType.methodType; -import static jdk.internal.foreign.abi.SharedUtils.Allocator.THROWING_ALLOCATOR; -import static jdk.internal.foreign.abi.SharedUtils.DEFAULT_ALLOCATOR; import static sun.security.action.GetBooleanAction.privilegedGetProperty; /** @@ -78,26 +74,27 @@ public class ProgrammableInvoker { private static final MethodHandle MH_INVOKE_MOVES; private static final MethodHandle MH_INVOKE_INTERP_BINDINGS; - - private static final MethodHandle MH_MAKE_SCOPE; - private static final MethodHandle MH_CLOSE_SCOPE; - private static final MethodHandle MH_WRAP_SCOPE; + private static final MethodHandle MH_ADDR_TO_LONG; + private static final MethodHandle MH_WRAP_ALLOCATOR; private static final Map adapterStubs = new ConcurrentHashMap<>(); + private static final MethodHandle EMPTY_OBJECT_ARRAY_HANDLE = MethodHandles.constant(Object[].class, new Object[0]); + static { try { MethodHandles.Lookup lookup = MethodHandles.lookup(); MH_INVOKE_MOVES = lookup.findVirtual(ProgrammableInvoker.class, "invokeMoves", - methodType(Object.class, Object[].class, Binding.VMStore[].class, Binding.VMLoad[].class)); + methodType(Object.class, long.class, Object[].class, Binding.VMStore[].class, Binding.VMLoad[].class)); MH_INVOKE_INTERP_BINDINGS = lookup.findVirtual(ProgrammableInvoker.class, "invokeInterpBindings", - methodType(Object.class, Object[].class, MethodHandle.class, Map.class, Map.class)); - MH_MAKE_SCOPE = lookup.findStatic(NativeScope.class, "boundedScope", - methodType(NativeScope.class, long.class)); - MH_CLOSE_SCOPE = lookup.findVirtual(NativeScope.class, "close", - methodType(void.class)); - MH_WRAP_SCOPE = lookup.findStatic(SharedUtils.Allocator.class, "ofScope", - methodType(SharedUtils.Allocator.class, NativeScope.class)); + methodType(Object.class, Addressable.class, SegmentAllocator.class, Object[].class, MethodHandle.class, Map.class, Map.class)); + MH_WRAP_ALLOCATOR = lookup.findStatic(Binding.Context.class, "ofAllocator", + methodType(Binding.Context.class, SegmentAllocator.class)); + MethodHandle MH_Addressable_address = lookup.findVirtual(Addressable.class, "address", + methodType(MemoryAddress.class)); + MethodHandle MH_MemoryAddress_toRawLongValue = lookup.findVirtual(MemoryAddress.class, "toRawLongValue", + methodType(long.class)); + MH_ADDR_TO_LONG = filterArguments(MH_MemoryAddress_toRawLongValue, 0, MH_Addressable_address); } catch (ReflectiveOperationException e) { throw new RuntimeException(e); } @@ -109,17 +106,15 @@ public class ProgrammableInvoker { private final CallingSequence callingSequence; - private final Addressable addr; private final long stubAddress; private final long bufferCopySize; - public ProgrammableInvoker(ABIDescriptor abi, Addressable addr, CallingSequence callingSequence) { + public ProgrammableInvoker(ABIDescriptor abi, CallingSequence callingSequence) { this.abi = abi; this.layout = BufferLayout.of(abi); this.stubAddress = adapterStubs.computeIfAbsent(abi, key -> generateAdapter(key, layout)); - this.addr = addr; this.callingSequence = callingSequence; this.stackArgsBytes = argMoveBindingsStream(callingSequence) @@ -128,24 +123,7 @@ public class ProgrammableInvoker { .count() * abi.arch.typeSize(abi.arch.stackType()); - this.bufferCopySize = bufferCopySize(callingSequence); - } - - private static long bufferCopySize(CallingSequence callingSequence) { - // FIXME: > 16 bytes alignment might need extra space since the - // starting address of the allocator might be un-aligned. - long size = 0; - for (int i = 0; i < callingSequence.argumentCount(); i++) { - List bindings = callingSequence.argumentBindings(i); - for (Binding b : bindings) { - if (b instanceof Binding.Copy) { - Binding.Copy c = (Binding.Copy) b; - size = Utils.alignUp(size, c.alignment()); - size += c.size(); - } - } - } - return size; + this.bufferCopySize = SharedUtils.bufferCopySize(callingSequence); } public MethodHandle getBoundMethodHandle() { @@ -160,41 +138,53 @@ public class ProgrammableInvoker { : Object[].class; MethodType leafType = methodType(returnType, argMoveTypes); + MethodType leafTypeWithAddress = leafType.insertParameterTypes(0, long.class); - MethodHandle handle = insertArguments(MH_INVOKE_MOVES.bindTo(this), 1, argMoves, retMoves) - .asCollector(Object[].class, leafType.parameterCount()) - .asType(leafType); + MethodHandle handle = insertArguments(MH_INVOKE_MOVES.bindTo(this), 2, argMoves, retMoves); + MethodHandle collector = makeCollectorHandle(leafType); + handle = collectArguments(handle, 1, collector); + handle = handle.asType(leafTypeWithAddress); boolean isSimple = !(retMoves.length > 1); boolean usesStackArgs = stackArgsBytes != 0; if (USE_INTRINSICS && isSimple && !usesStackArgs) { NativeEntryPoint nep = NativeEntryPoint.make( - addr.address().toRawLongValue(), "native_call", abi, toStorageArray(argMoves), toStorageArray(retMoves), !callingSequence.isTrivial(), - leafType + leafTypeWithAddress ); handle = JLIA.nativeMethodHandle(nep, handle); } + handle = filterArguments(handle, 0, MH_ADDR_TO_LONG); if (USE_SPEC && isSimple) { handle = specialize(handle); } else { - Map argIndexMap = indexMap(argMoves); - Map retIndexMap = indexMap(retMoves); + Map argIndexMap = SharedUtils.indexMap(argMoves); + Map retIndexMap = SharedUtils.indexMap(retMoves); - handle = insertArguments(MH_INVOKE_INTERP_BINDINGS.bindTo(this), 1, handle, argIndexMap, retIndexMap); - handle = handle.asCollector(Object[].class, callingSequence.methodType().parameterCount()) - .asType(callingSequence.methodType()); + handle = insertArguments(MH_INVOKE_INTERP_BINDINGS.bindTo(this), 3, handle, argIndexMap, retIndexMap); + MethodHandle collectorInterp = makeCollectorHandle(callingSequence.methodType()); + handle = collectArguments(handle, 2, collectorInterp); + handle = handle.asType(handle.type().changeReturnType(callingSequence.methodType().returnType())); } return handle; } + // Funnel from type to Object[] + private static MethodHandle makeCollectorHandle(MethodType type) { + return type.parameterCount() == 0 + ? EMPTY_OBJECT_ARRAY_HANDLE + : identity(Object[].class) + .asCollector(Object[].class, type.parameterCount()) + .asType(type.changeReturnType(Object[].class)); + } + private Stream argMoveBindingsStream(CallingSequence callingSequence) { return callingSequence.argumentBindings() .filter(Binding.VMStore.class::isInstance) @@ -215,17 +205,12 @@ public class ProgrammableInvoker { private MethodHandle specialize(MethodHandle leafHandle) { MethodType highLevelType = callingSequence.methodType(); - MethodType leafType = leafHandle.type(); - MethodHandle specializedHandle = leafHandle; // initial + int argInsertPos = 1; + int argContextPos = 1; + + MethodHandle specializedHandle = dropArguments(leafHandle, argContextPos, Binding.Context.class); - int argInsertPos = -1; - int argAllocatorPos = -1; - if (bufferCopySize > 0) { - argAllocatorPos = 0; - specializedHandle = dropArguments(specializedHandle, argAllocatorPos, SharedUtils.Allocator.class); - argInsertPos++; - } for (int i = 0; i < highLevelType.parameterCount(); i++) { List bindings = callingSequence.argumentBindings(i); argInsertPos += bindings.stream().filter(Binding.VMStore.class::isInstance).count() + 1; @@ -235,48 +220,37 @@ public class ProgrammableInvoker { if (binding.tag() == Binding.Tag.VM_STORE) { argInsertPos--; } else { - specializedHandle = binding.specialize(specializedHandle, argInsertPos, argAllocatorPos); + specializedHandle = binding.specialize(specializedHandle, argInsertPos, argContextPos); } } } if (highLevelType.returnType() != void.class) { MethodHandle returnFilter = identity(highLevelType.returnType()); - int retAllocatorPos = 0; + int retContextPos = 0; int retInsertPos = 1; - returnFilter = dropArguments(returnFilter, retAllocatorPos, SharedUtils.Allocator.class); + returnFilter = dropArguments(returnFilter, retContextPos, Binding.Context.class); List bindings = callingSequence.returnBindings(); for (int j = bindings.size() - 1; j >= 0; j--) { Binding binding = bindings.get(j); - returnFilter = binding.specialize(returnFilter, retInsertPos, retAllocatorPos); + returnFilter = binding.specialize(returnFilter, retInsertPos, retContextPos); } - returnFilter = insertArguments(returnFilter, retAllocatorPos, DEFAULT_ALLOCATOR); - specializedHandle = MethodHandles.filterReturnValue(specializedHandle, returnFilter); + returnFilter = MethodHandles.filterArguments(returnFilter, retContextPos, MH_WRAP_ALLOCATOR); + // (SegmentAllocator, Addressable, Context, ...) -> ... + specializedHandle = MethodHandles.collectArguments(returnFilter, retInsertPos, specializedHandle); + // (Addressable, SegmentAllocator, Context, ...) -> ... + specializedHandle = SharedUtils.swapArguments(specializedHandle, 0, 1); // normalize parameter order + } else { + specializedHandle = MethodHandles.dropArguments(specializedHandle, 1, SegmentAllocator.class); } - if (bufferCopySize > 0) { - // insert try-finally to close the NativeScope used for Binding.Copy - MethodHandle closer = leafType.returnType() == void.class - // (Throwable, NativeScope) -> void - ? collectArguments(empty(methodType(void.class, Throwable.class)), 1, MH_CLOSE_SCOPE) - // (Throwable, V, NativeScope) -> V - : collectArguments(dropArguments(identity(specializedHandle.type().returnType()), 0, Throwable.class), - 2, MH_CLOSE_SCOPE); - // Handle takes a SharedUtils.Allocator, so need to wrap our NativeScope - specializedHandle = filterArguments(specializedHandle, argAllocatorPos, MH_WRAP_SCOPE); - specializedHandle = tryFinally(specializedHandle, closer); - MethodHandle makeScopeHandle = insertArguments(MH_MAKE_SCOPE, 0, bufferCopySize); - specializedHandle = collectArguments(specializedHandle, argAllocatorPos, makeScopeHandle); - } + // now bind the internal context parameter + + argContextPos++; // skip over the return SegmentAllocator (inserted by the above code) + specializedHandle = SharedUtils.wrapWithAllocator(specializedHandle, argContextPos, bufferCopySize, false); return specializedHandle; } - private static Map indexMap(Binding.Move[] moves) { - return IntStream.range(0, moves.length) - .boxed() - .collect(Collectors.toMap(i -> moves[i].storage(), i -> i)); - } - /** * Does a native invocation by moving primitive values from the arg array into an intermediate buffer * and calling the assembly stub that forwards arguments from the buffer to the target function @@ -286,14 +260,15 @@ public class ProgrammableInvoker { * @param returnBindings Binding.Move values describing how return values should be copied * @return null, a single primitive value, or an Object[] of primitive values */ - Object invokeMoves(Object[] args, Binding.VMStore[] argBindings, Binding.VMLoad[] returnBindings) { + Object invokeMoves(long addr, Object[] args, Binding.VMStore[] argBindings, Binding.VMLoad[] returnBindings) { MemorySegment stackArgsSeg = null; - try (MemorySegment argBuffer = MemorySegment.allocateNative(layout.size, 64)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment argBuffer = MemorySegment.allocateNative(layout.size, 64, scope); if (stackArgsBytes > 0) { - stackArgsSeg = MemorySegment.allocateNative(stackArgsBytes, 8); + stackArgsSeg = MemorySegment.allocateNative(stackArgsBytes, 8, scope); } - VH_LONG.set(argBuffer.asSlice(layout.arguments_next_pc), addr.address().toRawLongValue()); + VH_LONG.set(argBuffer.asSlice(layout.arguments_next_pc), addr); VH_LONG.set(argBuffer.asSlice(layout.stack_args_bytes), stackArgsBytes); VH_LONG.set(argBuffer.asSlice(layout.stack_args), stackArgsSeg == null ? 0L : stackArgsSeg.address().toRawLongValue()); @@ -333,32 +308,33 @@ public class ProgrammableInvoker { } return returns; } - } finally { - if (stackArgsSeg != null) { - stackArgsSeg.close(); - } } } - Object invokeInterpBindings(Object[] args, MethodHandle leaf, + Object invokeInterpBindings(Addressable address, SegmentAllocator allocator, Object[] args, MethodHandle leaf, Map argIndexMap, Map retIndexMap) throws Throwable { - SharedUtils.Allocator unboxAllocator = bufferCopySize != 0 - ? SharedUtils.Allocator.ofScope(NativeScope.boundedScope(bufferCopySize)) - : THROWING_ALLOCATOR; - try (unboxAllocator) { + Binding.Context unboxContext = bufferCopySize != 0 + ? Binding.Context.ofBoundedAllocator(bufferCopySize) + : Binding.Context.DUMMY; + try (unboxContext) { // do argument processing, get Object[] as result - Object[] moves = new Object[leaf.type().parameterCount()]; + Object[] leafArgs = new Object[leaf.type().parameterCount()]; + leafArgs[0] = address; // addr for (int i = 0; i < args.length; i++) { Object arg = args[i]; BindingInterpreter.unbox(arg, callingSequence.argumentBindings(i), (storage, type, value) -> { - moves[argIndexMap.get(storage)] = value; - }, unboxAllocator); + leafArgs[argIndexMap.get(storage) + 1] = value; // +1 to skip addr + }, unboxContext); } // call leaf - Object o = leaf.invokeWithArguments(moves); + Object o = leaf.invokeWithArguments(leafArgs); + // make sure arguments are reachable during the call + // technically we only need to do all Addressable parameters here + Reference.reachabilityFence(address); + Reference.reachabilityFence(args); // return value processing if (o == null) { @@ -366,10 +342,10 @@ public class ProgrammableInvoker { } else if (o instanceof Object[]) { Object[] oArr = (Object[]) o; return BindingInterpreter.box(callingSequence.returnBindings(), - (storage, type) -> oArr[retIndexMap.get(storage)], DEFAULT_ALLOCATOR); + (storage, type) -> oArr[retIndexMap.get(storage)], Binding.Context.ofAllocator(allocator)); } else { return BindingInterpreter.box(callingSequence.returnBindings(), (storage, type) -> o, - DEFAULT_ALLOCATOR); + Binding.Context.ofAllocator(allocator)); } } } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/ProgrammableUpcallHandler.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/ProgrammableUpcallHandler.java index 810e69a65b0..b52a9d8dcdc 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/ProgrammableUpcallHandler.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/ProgrammableUpcallHandler.java @@ -26,22 +26,32 @@ package jdk.internal.foreign.abi; import jdk.incubator.foreign.MemoryAddress; -import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; +import jdk.internal.access.JavaLangInvokeAccess; +import jdk.internal.access.SharedSecrets; import jdk.internal.foreign.MemoryAddressImpl; -import jdk.internal.foreign.Utils; -import jdk.internal.vm.annotation.Stable; +import sun.security.action.GetPropertyAction; import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; import java.lang.invoke.VarHandle; -import java.nio.ByteOrder; -import java.util.ArrayList; import java.util.Arrays; +import java.util.List; +import java.util.Map; import java.util.Objects; +import java.util.stream.Stream; -import static jdk.internal.foreign.abi.SharedUtils.DEFAULT_ALLOCATOR; +import static java.lang.invoke.MethodHandles.dropArguments; +import static java.lang.invoke.MethodHandles.filterReturnValue; +import static java.lang.invoke.MethodHandles.identity; +import static java.lang.invoke.MethodHandles.insertArguments; +import static java.lang.invoke.MethodHandles.lookup; +import static java.lang.invoke.MethodType.methodType; +import static jdk.internal.foreign.abi.SharedUtils.mergeArguments; import static sun.security.action.GetBooleanAction.privilegedGetProperty; /** @@ -49,59 +59,214 @@ import static sun.security.action.GetBooleanAction.privilegedGetProperty; * takes an array of storage pointers, which describes the state of the CPU at the time of the upcall. This can be used * by the Java code to fetch the upcall arguments and to store the results to the desired location, as per system ABI. */ -public class ProgrammableUpcallHandler implements UpcallHandler { - +public class ProgrammableUpcallHandler { private static final boolean DEBUG = privilegedGetProperty("jdk.internal.foreign.ProgrammableUpcallHandler.DEBUG"); + private static final boolean USE_SPEC = Boolean.parseBoolean( + GetPropertyAction.privilegedGetProperty("jdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC", "true")); + private static final boolean USE_INTRINSICS = Boolean.parseBoolean( + GetPropertyAction.privilegedGetProperty("jdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS", "true")); + + private static final JavaLangInvokeAccess JLI = SharedSecrets.getJavaLangInvokeAccess(); private static final VarHandle VH_LONG = MemoryLayouts.JAVA_LONG.varHandle(long.class); - @Stable - private final MethodHandle mh; - private final MethodType type; - private final CallingSequence callingSequence; - private final long entryPoint; + private static final MethodHandle MH_invokeMoves; + private static final MethodHandle MH_invokeInterpBindings; - private final ABIDescriptor abi; - private final BufferLayout layout; - - public ProgrammableUpcallHandler(ABIDescriptor abi, MethodHandle target, CallingSequence callingSequence) { - this.abi = abi; - this.layout = BufferLayout.of(abi); - this.type = callingSequence.methodType(); - this.callingSequence = callingSequence; - this.mh = target.asSpreader(Object[].class, callingSequence.methodType().parameterCount()); - this.entryPoint = allocateUpcallStub(abi, layout); - } - - @Override - public long entryPoint() { - return entryPoint; - } - - public static void invoke(ProgrammableUpcallHandler handler, long address) { - handler.invoke(MemoryAddress.ofLong(address)); - } - - private void invoke(MemoryAddress buffer) { + static { try { - MemorySegment bufferBase = MemoryAddressImpl.ofLongUnchecked(buffer.toRawLongValue(), layout.size); + MethodHandles.Lookup lookup = lookup(); + MH_invokeMoves = lookup.findStatic(ProgrammableUpcallHandler.class, "invokeMoves", + methodType(void.class, MemoryAddress.class, MethodHandle.class, + Binding.VMLoad[].class, Binding.VMStore[].class, ABIDescriptor.class, BufferLayout.class)); + MH_invokeInterpBindings = lookup.findStatic(ProgrammableUpcallHandler.class, "invokeInterpBindings", + methodType(Object.class, Object[].class, MethodHandle.class, Map.class, Map.class, + CallingSequence.class, long.class)); + } catch (ReflectiveOperationException e) { + throw new InternalError(e); + } + } - if (DEBUG) { - System.err.println("Buffer state before:"); - layout.dump(abi.arch, bufferBase, System.err); + public static UpcallHandler make(ABIDescriptor abi, MethodHandle target, CallingSequence callingSequence) { + Binding.VMLoad[] argMoves = argMoveBindings(callingSequence); + Binding.VMStore[] retMoves = retMoveBindings(callingSequence); + + boolean isSimple = !(retMoves.length > 1); + + Class llReturn = !isSimple + ? Object[].class + : retMoves.length == 1 + ? retMoves[0].type() + : void.class; + Class[] llParams = Arrays.stream(argMoves).map(Binding.Move::type).toArray(Class[]::new); + MethodType llType = MethodType.methodType(llReturn, llParams); + + MethodHandle doBindings; + long bufferCopySize = SharedUtils.bufferCopySize(callingSequence); + if (USE_SPEC && isSimple) { + doBindings = specializedBindingHandle(target, callingSequence, llReturn, bufferCopySize); + assert doBindings.type() == llType; + } else { + Map argIndices = SharedUtils.indexMap(argMoves); + Map retIndices = SharedUtils.indexMap(retMoves); + target = target.asSpreader(Object[].class, callingSequence.methodType().parameterCount()); + doBindings = insertArguments(MH_invokeInterpBindings, 1, target, argIndices, retIndices, callingSequence, + bufferCopySize); + doBindings = doBindings.asCollector(Object[].class, llType.parameterCount()); + doBindings = doBindings.asType(llType); + } + + long entryPoint; + boolean usesStackArgs = argMoveBindingsStream(callingSequence) + .map(Binding.VMLoad::storage) + .anyMatch(s -> abi.arch.isStackType(s.type())); + if (USE_INTRINSICS && isSimple && !usesStackArgs && supportsOptimizedUpcalls()) { + checkPrimitive(doBindings.type()); + JLI.ensureCustomized(doBindings); + VMStorage[] args = Arrays.stream(argMoves).map(Binding.Move::storage).toArray(VMStorage[]::new); + VMStorage[] rets = Arrays.stream(retMoves).map(Binding.Move::storage).toArray(VMStorage[]::new); + CallRegs conv = new CallRegs(args, rets); + entryPoint = allocateOptimizedUpcallStub(doBindings, abi, conv); + } else { + BufferLayout layout = BufferLayout.of(abi); + MethodHandle doBindingsErased = doBindings.asSpreader(Object[].class, doBindings.type().parameterCount()); + MethodHandle invokeMoves = insertArguments(MH_invokeMoves, 1, doBindingsErased, argMoves, retMoves, abi, layout); + entryPoint = allocateUpcallStub(invokeMoves, abi, layout); + } + return () -> entryPoint; + } + + private static void checkPrimitive(MethodType type) { + if (!type.returnType().isPrimitive() + || type.parameterList().stream().anyMatch(p -> !p.isPrimitive())) + throw new IllegalArgumentException("MethodHandle type must be primitive: " + type); + } + + private static Stream argMoveBindingsStream(CallingSequence callingSequence) { + return callingSequence.argumentBindings() + .filter(Binding.VMLoad.class::isInstance) + .map(Binding.VMLoad.class::cast); + } + + private static Binding.VMLoad[] argMoveBindings(CallingSequence callingSequence) { + return argMoveBindingsStream(callingSequence) + .toArray(Binding.VMLoad[]::new); + } + + private static Binding.VMStore[] retMoveBindings(CallingSequence callingSequence) { + return callingSequence.returnBindings().stream() + .filter(Binding.VMStore.class::isInstance) + .map(Binding.VMStore.class::cast) + .toArray(Binding.VMStore[]::new); + } + + private static MethodHandle specializedBindingHandle(MethodHandle target, CallingSequence callingSequence, + Class llReturn, long bufferCopySize) { + MethodType highLevelType = callingSequence.methodType(); + + MethodHandle specializedHandle = target; // initial + + int argAllocatorPos = 0; + int argInsertPos = 1; + specializedHandle = dropArguments(specializedHandle, argAllocatorPos, Binding.Context.class); + for (int i = 0; i < highLevelType.parameterCount(); i++) { + MethodHandle filter = identity(highLevelType.parameterType(i)); + int filterAllocatorPos = 0; + int filterInsertPos = 1; // +1 for allocator + filter = dropArguments(filter, filterAllocatorPos, Binding.Context.class); + + List bindings = callingSequence.argumentBindings(i); + for (int j = bindings.size() - 1; j >= 0; j--) { + Binding binding = bindings.get(j); + filter = binding.specialize(filter, filterInsertPos, filterAllocatorPos); } + specializedHandle = MethodHandles.collectArguments(specializedHandle, argInsertPos, filter); + specializedHandle = mergeArguments(specializedHandle, argAllocatorPos, argInsertPos + filterAllocatorPos); + argInsertPos += filter.type().parameterCount() - 1; // -1 for allocator + } - MemorySegment stackArgsBase = MemoryAddressImpl.ofLongUnchecked((long)VH_LONG.get(bufferBase.asSlice(layout.stack_args))); - Object[] args = new Object[type.parameterCount()]; - for (int i = 0 ; i < type.parameterCount() ; i++) { + if (llReturn != void.class) { + int retAllocatorPos = -1; // assumed not needed + int retInsertPos = 0; + MethodHandle filter = identity(llReturn); + List bindings = callingSequence.returnBindings(); + for (int j = bindings.size() - 1; j >= 0; j--) { + Binding binding = bindings.get(j); + filter = binding.specialize(filter, retInsertPos, retAllocatorPos); + } + specializedHandle = filterReturnValue(specializedHandle, filter); + } + + specializedHandle = SharedUtils.wrapWithAllocator(specializedHandle, argAllocatorPos, bufferCopySize, true); + + return specializedHandle; + } + + public static void invoke(MethodHandle mh, long address) throws Throwable { + mh.invokeExact(MemoryAddress.ofLong(address)); + } + + private static void invokeMoves(MemoryAddress buffer, MethodHandle leaf, + Binding.VMLoad[] argBindings, Binding.VMStore[] returnBindings, + ABIDescriptor abi, BufferLayout layout) throws Throwable { + MemorySegment bufferBase = MemoryAddressImpl.ofLongUnchecked(buffer.toRawLongValue(), layout.size); + + if (DEBUG) { + System.err.println("Buffer state before:"); + layout.dump(abi.arch, bufferBase, System.err); + } + + MemorySegment stackArgsBase = MemoryAddressImpl.ofLongUnchecked((long)VH_LONG.get(bufferBase.asSlice(layout.stack_args))); + Object[] moves = new Object[argBindings.length]; + for (int i = 0; i < moves.length; i++) { + Binding.VMLoad binding = argBindings[i]; + VMStorage storage = binding.storage(); + MemorySegment ptr = abi.arch.isStackType(storage.type()) + ? stackArgsBase.asSlice(storage.index() * abi.arch.typeSize(abi.arch.stackType())) + : bufferBase.asSlice(layout.argOffset(storage)); + moves[i] = SharedUtils.read(ptr, binding.type()); + } + + // invokeInterpBindings, and then actual target + Object o = leaf.invoke(moves); + + if (o == null) { + // nop + } else if (o instanceof Object[] returns) { + for (int i = 0; i < returnBindings.length; i++) { + Binding.VMStore binding = returnBindings[i]; + VMStorage storage = binding.storage(); + MemorySegment ptr = bufferBase.asSlice(layout.retOffset(storage)); + SharedUtils.writeOverSized(ptr, binding.type(), returns[i]); + } + } else { // single Object + Binding.VMStore binding = returnBindings[0]; + VMStorage storage = binding.storage(); + MemorySegment ptr = bufferBase.asSlice(layout.retOffset(storage)); + SharedUtils.writeOverSized(ptr, binding.type(), o); + } + + if (DEBUG) { + System.err.println("Buffer state after:"); + layout.dump(abi.arch, bufferBase, System.err); + } + } + + private static Object invokeInterpBindings(Object[] moves, MethodHandle leaf, + Map argIndexMap, + Map retIndexMap, + CallingSequence callingSequence, + long bufferCopySize) throws Throwable { + Binding.Context allocator = bufferCopySize != 0 + ? Binding.Context.ofBoundedAllocator(bufferCopySize) + : Binding.Context.ofScope(); + try (allocator) { + /// Invoke interpreter, got array of high-level arguments back + Object[] args = new Object[callingSequence.methodType().parameterCount()]; + for (int i = 0; i < args.length; i++) { args[i] = BindingInterpreter.box(callingSequence.argumentBindings(i), - (storage, type) -> { - MemorySegment ptr = abi.arch.isStackType(storage.type()) - ? stackArgsBase.asSlice(storage.index() * abi.arch.typeSize(abi.arch.stackType())) - : bufferBase.asSlice(layout.argOffset(storage)); - return SharedUtils.read(ptr, type); - }, DEFAULT_ALLOCATOR); + (storage, type) -> moves[argIndexMap.get(storage)], allocator); } if (DEBUG) { @@ -109,31 +274,36 @@ public class ProgrammableUpcallHandler implements UpcallHandler { System.err.println(Arrays.toString(args).indent(2)); } - Object o = mh.invoke(args); + // invoke our target + Object o = leaf.invoke(args); if (DEBUG) { System.err.println("Java return:"); System.err.println(Objects.toString(o).indent(2)); } - if (mh.type().returnType() != void.class) { + Object[] returnMoves = new Object[retIndexMap.size()]; + if (leaf.type().returnType() != void.class) { BindingInterpreter.unbox(o, callingSequence.returnBindings(), - (storage, type, value) -> { - MemorySegment ptr = bufferBase.asSlice(layout.retOffset(storage)); - SharedUtils.writeOverSized(ptr, type, value); - }, null); + (storage, type, value) -> returnMoves[retIndexMap.get(storage)] = value, null); } - if (DEBUG) { - System.err.println("Buffer state after:"); - layout.dump(abi.arch, bufferBase, System.err); + if (returnMoves.length == 0) { + return null; + } else if (returnMoves.length == 1) { + return returnMoves[0]; + } else { + return returnMoves; } - } catch (Throwable t) { - throw new IllegalStateException(t); } } - public native long allocateUpcallStub(ABIDescriptor abi, BufferLayout layout); + // used for transporting data into native code + private static record CallRegs(VMStorage[] argRegs, VMStorage[] retRegs) {} + + static native long allocateOptimizedUpcallStub(MethodHandle mh, ABIDescriptor abi, CallRegs conv); + static native long allocateUpcallStub(MethodHandle mh, ABIDescriptor abi, BufferLayout layout); + static native boolean supportsOptimizedUpcalls(); private static native void registerNatives(); static { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/SharedUtils.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/SharedUtils.java index 32530d57341..0e6ec717237 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/SharedUtils.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/SharedUtils.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -24,6 +24,7 @@ */ package jdk.internal.foreign.abi; +import jdk.incubator.foreign.Addressable; import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.GroupLayout; import jdk.incubator.foreign.MemoryAccess; @@ -32,7 +33,8 @@ import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; import jdk.incubator.foreign.LibraryLookup; -import jdk.incubator.foreign.NativeScope; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; import jdk.incubator.foreign.SequenceLayout; import jdk.incubator.foreign.CLinker; import jdk.incubator.foreign.ValueLayout; @@ -47,15 +49,24 @@ import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; import java.lang.invoke.VarHandle; +import java.lang.ref.Reference; import java.nio.charset.Charset; import java.util.List; +import java.util.Map; import java.util.function.Consumer; +import java.util.stream.Collectors; import java.util.stream.IntStream; import static java.lang.invoke.MethodHandles.collectArguments; +import static java.lang.invoke.MethodHandles.constant; +import static java.lang.invoke.MethodHandles.dropArguments; +import static java.lang.invoke.MethodHandles.dropReturn; +import static java.lang.invoke.MethodHandles.empty; +import static java.lang.invoke.MethodHandles.filterArguments; import static java.lang.invoke.MethodHandles.identity; import static java.lang.invoke.MethodHandles.insertArguments; import static java.lang.invoke.MethodHandles.permuteArguments; +import static java.lang.invoke.MethodHandles.tryFinally; import static java.lang.invoke.MethodType.methodType; import static jdk.incubator.foreign.CLinker.*; @@ -64,27 +75,35 @@ public class SharedUtils { private static final MethodHandle MH_ALLOC_BUFFER; private static final MethodHandle MH_BASEADDRESS; private static final MethodHandle MH_BUFFER_COPY; - - static final Allocator DEFAULT_ALLOCATOR = MemorySegment::allocateNative; + private static final MethodHandle MH_MAKE_CONTEXT_NO_ALLOCATOR; + private static final MethodHandle MH_MAKE_CONTEXT_BOUNDED_ALLOCATOR; + private static final MethodHandle MH_CLOSE_CONTEXT; + private static final MethodHandle MH_REACHBILITY_FENCE; static { try { - var lookup = MethodHandles.lookup(); - MH_ALLOC_BUFFER = lookup.findStatic(SharedUtils.class, "allocateNative", + MethodHandles.Lookup lookup = MethodHandles.lookup(); + MH_ALLOC_BUFFER = lookup.findVirtual(SegmentAllocator.class, "allocate", methodType(MemorySegment.class, MemoryLayout.class)); MH_BASEADDRESS = lookup.findVirtual(MemorySegment.class, "address", methodType(MemoryAddress.class)); MH_BUFFER_COPY = lookup.findStatic(SharedUtils.class, "bufferCopy", methodType(MemoryAddress.class, MemoryAddress.class, MemorySegment.class)); + MH_MAKE_CONTEXT_NO_ALLOCATOR = lookup.findStatic(Binding.Context.class, "ofScope", + methodType(Binding.Context.class)); + MH_MAKE_CONTEXT_BOUNDED_ALLOCATOR = lookup.findStatic(Binding.Context.class, "ofBoundedAllocator", + methodType(Binding.Context.class, long.class)); + MH_CLOSE_CONTEXT = lookup.findVirtual(Binding.Context.class, "close", + methodType(void.class)); + MH_REACHBILITY_FENCE = lookup.findStatic(Reference.class, "reachabilityFence", + methodType(void.class, Object.class)); } catch (ReflectiveOperationException e) { throw new BootstrapMethodError(e); } } - // workaround for https://bugs.openjdk.java.net/browse/JDK-8239083 - private static MemorySegment allocateNative(MemoryLayout layout) { - return MemorySegment.allocateNative(layout); - } + // this allocator should be used when no allocation is expected + public static final SegmentAllocator THROWING_ALLOCATOR = (size, align) -> { throw new IllegalStateException("Cannot get here"); }; /** * Align the specified type from a given address @@ -154,22 +173,19 @@ public class SharedUtils { */ public static MethodHandle adaptDowncallForIMR(MethodHandle handle, FunctionDescriptor cDesc) { if (handle.type().returnType() != void.class) - throw new IllegalArgumentException("return expected to be void for in memory returns"); - if (handle.type().parameterType(0) != MemoryAddress.class) - throw new IllegalArgumentException("MemoryAddress expected as first param"); + throw new IllegalArgumentException("return expected to be void for in memory returns: " + handle.type()); + if (handle.type().parameterType(2) != MemoryAddress.class) + throw new IllegalArgumentException("MemoryAddress expected as third param: " + handle.type()); if (cDesc.returnLayout().isEmpty()) throw new IllegalArgumentException("Return layout needed: " + cDesc); MethodHandle ret = identity(MemorySegment.class); // (MemorySegment) MemorySegment - handle = collectArguments(ret, 1, handle); // (MemorySegment, MemoryAddress ...) MemorySegment - handle = collectArguments(handle, 1, MH_BASEADDRESS); // (MemorySegment, MemorySegment ...) MemorySegment - MethodType oldType = handle.type(); // (MemorySegment, MemorySegment, ...) MemorySegment - MethodType newType = oldType.dropParameterTypes(0, 1); // (MemorySegment, ...) MemorySegment - int[] reorder = IntStream.range(-1, newType.parameterCount()).toArray(); - reorder[0] = 0; // [0, 0, 1, 2, 3, ...] - handle = permuteArguments(handle, newType, reorder); // (MemorySegment, ...) MemoryAddress - handle = collectArguments(handle, 0, insertArguments(MH_ALLOC_BUFFER, 0, cDesc.returnLayout().get())); // (...) MemoryAddress - + handle = collectArguments(ret, 1, handle); // (MemorySegment, Addressable, SegmentAllocator, MemoryAddress, ...) MemorySegment + handle = collectArguments(handle, 3, MH_BASEADDRESS); // (MemorySegment, Addressable, SegmentAllocator, MemorySegment, ...) MemorySegment + handle = mergeArguments(handle, 0, 3); // (MemorySegment, Addressable, SegmentAllocator, ...) MemorySegment + handle = collectArguments(handle, 0, insertArguments(MH_ALLOC_BUFFER, 1, cDesc.returnLayout().get())); // (SegmentAllocator, Addressable, SegmentAllocator, ...) MemoryAddress + handle = mergeArguments(handle, 0, 2); // (SegmentAllocator, Addressable, ...) MemoryAddress + handle = swapArguments(handle, 0, 1); // (Addressable, SegmentAllocator, ...) MemoryAddress return handle; } @@ -181,12 +197,16 @@ public class SharedUtils { * @param target the target handle to adapt * @return the adapted handle */ - public static MethodHandle adaptUpcallForIMR(MethodHandle target) { + public static MethodHandle adaptUpcallForIMR(MethodHandle target, boolean dropReturn) { if (target.type().returnType() != MemorySegment.class) throw new IllegalArgumentException("Must return MemorySegment for IMR"); target = collectArguments(MH_BUFFER_COPY, 1, target); // (MemoryAddress, ...) MemoryAddress + if (dropReturn) { // no handling for return value, need to drop it + target = dropReturn(target); + } + return target; } @@ -223,18 +243,26 @@ public class SharedUtils { cDesc.returnLayout().ifPresent(rl -> checkCompatibleType(mt.returnType(), rl, addressSize)); } - public static Class primitiveCarrierForSize(long size) { - if (size == 1) { - return byte.class; - } else if(size == 2) { - return short.class; - } else if (size <= 4) { - return int.class; - } else if (size <= 8) { - return long.class; + public static Class primitiveCarrierForSize(long size, boolean useFloat) { + if (useFloat) { + if (size == 4) { + return float.class; + } else if (size == 8) { + return double.class; + } + } else { + if (size == 1) { + return byte.class; + } else if (size == 2) { + return short.class; + } else if (size <= 4) { + return int.class; + } else if (size <= 8) { + return long.class; + } } - throw new IllegalArgumentException("Size too large: " + size); + throw new IllegalArgumentException("No type for size: " + size + " isFloat=" + useFloat); } public static CLinker getSystemLinker() { @@ -264,16 +292,136 @@ public class SharedUtils { throw new IllegalArgumentException("String too large"); } + static long bufferCopySize(CallingSequence callingSequence) { + // FIXME: > 16 bytes alignment might need extra space since the + // starting address of the allocator might be un-aligned. + long size = 0; + for (int i = 0; i < callingSequence.argumentCount(); i++) { + List bindings = callingSequence.argumentBindings(i); + for (Binding b : bindings) { + if (b instanceof Binding.Copy) { + Binding.Copy c = (Binding.Copy) b; + size = Utils.alignUp(size, c.alignment()); + size += c.size(); + } else if (b instanceof Binding.Allocate) { + Binding.Allocate c = (Binding.Allocate) b; + size = Utils.alignUp(size, c.alignment()); + size += c.size(); + } + } + } + return size; + } + + static Map indexMap(Binding.Move[] moves) { + return IntStream.range(0, moves.length) + .boxed() + .collect(Collectors.toMap(i -> moves[i].storage(), i -> i)); + } + + static MethodHandle mergeArguments(MethodHandle mh, int sourceIndex, int destIndex) { + MethodType oldType = mh.type(); + Class sourceType = oldType.parameterType(sourceIndex); + Class destType = oldType.parameterType(destIndex); + if (sourceType != destType) { + // TODO meet? + throw new IllegalArgumentException("Parameter types differ: " + sourceType + " != " + destType); + } + MethodType newType = oldType.dropParameterTypes(destIndex, destIndex + 1); + int[] reorder = new int[oldType.parameterCount()]; + assert destIndex > sourceIndex; + for (int i = 0, index = 0; i < reorder.length; i++) { + if (i != destIndex) { + reorder[i] = index++; + } else { + reorder[i] = sourceIndex; + } + } + return permuteArguments(mh, newType, reorder); + } + + + static MethodHandle swapArguments(MethodHandle mh, int firstArg, int secondArg) { + MethodType mtype = mh.type(); + int[] perms = new int[mtype.parameterCount()]; + MethodType swappedType = MethodType.methodType(mtype.returnType()); + for (int i = 0 ; i < perms.length ; i++) { + int dst = i; + if (i == firstArg) dst = secondArg; + if (i == secondArg) dst = firstArg; + perms[i] = dst; + swappedType = swappedType.appendParameterTypes(mtype.parameterType(dst)); + } + return permuteArguments(mh, swappedType, perms); + } + + private static MethodHandle reachabilityFenceHandle(Class type) { + return MH_REACHBILITY_FENCE.asType(MethodType.methodType(void.class, type)); + } + + static MethodHandle wrapWithAllocator(MethodHandle specializedHandle, + int allocatorPos, long bufferCopySize, + boolean upcall) { + // insert try-finally to close the NativeScope used for Binding.Copy + MethodHandle closer; + int insertPos; + if (specializedHandle.type().returnType() == void.class) { + closer = empty(methodType(void.class, Throwable.class)); // (Throwable) -> void + insertPos = 1; + } else { + closer = identity(specializedHandle.type().returnType()); // (V) -> V + closer = dropArguments(closer, 0, Throwable.class); // (Throwable, V) -> V + insertPos = 2; + } + + // downcalls get the leading Addressable/SegmentAllocator param as well + if (!upcall) { + closer = collectArguments(closer, insertPos++, reachabilityFenceHandle(Addressable.class)); + closer = dropArguments(closer, insertPos++, SegmentAllocator.class); // (Throwable, V?, Addressable, SegmentAllocator) -> V/void + } + + closer = collectArguments(closer, insertPos++, MH_CLOSE_CONTEXT); // (Throwable, V?, Addressable?, BindingContext) -> V/void + + if (!upcall) { + // now for each Addressable parameter, add a reachability fence + MethodType specType = specializedHandle.type(); + // skip 3 for address, segment allocator, and binding context + for (int i = 3; i < specType.parameterCount(); i++) { + Class param = specType.parameterType(i); + if (Addressable.class.isAssignableFrom(param)) { + closer = collectArguments(closer, insertPos++, reachabilityFenceHandle(param)); + } else { + closer = dropArguments(closer, insertPos++, param); + } + } + } + + MethodHandle contextFactory; + + if (bufferCopySize > 0) { + contextFactory = MethodHandles.insertArguments(MH_MAKE_CONTEXT_BOUNDED_ALLOCATOR, 0, bufferCopySize); + } else if (upcall) { + contextFactory = MH_MAKE_CONTEXT_NO_ALLOCATOR; + } else { + // this path is probably never used now, since ProgrammableInvoker never calls this routine with bufferCopySize == 0 + contextFactory = constant(Binding.Context.class, Binding.Context.DUMMY); + } + + specializedHandle = tryFinally(specializedHandle, closer); + specializedHandle = collectArguments(specializedHandle, allocatorPos, contextFactory); + return specializedHandle; + } + // lazy init MH_ALLOC and MH_FREE handles private static class AllocHolder { - final static LibraryLookup LOOKUP = LibraryLookup.ofDefault(); + static final LibraryLookup LOOKUP = LibraryLookup.ofDefault(); - final static MethodHandle MH_MALLOC = getSystemLinker().downcallHandle(LOOKUP.lookup("malloc").get(), + static final MethodHandle MH_MALLOC = getSystemLinker().downcallHandle(LOOKUP.lookup("malloc").get(), MethodType.methodType(MemoryAddress.class, long.class), FunctionDescriptor.of(C_POINTER, C_LONG_LONG)); - final static MethodHandle MH_FREE = getSystemLinker().downcallHandle(LOOKUP.lookup("free").get(), + static final MethodHandle MH_FREE = getSystemLinker().downcallHandle(LOOKUP.lookup("free").get(), MethodType.methodType(void.class, MemoryAddress.class), FunctionDescriptor.ofVoid(C_POINTER)); } @@ -294,25 +442,25 @@ public class SharedUtils { } } - public static VaList newVaList(Consumer actions, Allocator allocator) { + public static VaList newVaList(Consumer actions, ResourceScope scope) { return switch (CABI.current()) { - case Win64 -> Windowsx64Linker.newVaList(actions, allocator); - case SysV -> SysVx64Linker.newVaList(actions, allocator); - case AArch64 -> AArch64Linker.newVaList(actions, allocator); + case Win64 -> Windowsx64Linker.newVaList(actions, scope); + case SysV -> SysVx64Linker.newVaList(actions, scope); + case AArch64 -> AArch64Linker.newVaList(actions, scope); }; } public static VarHandle vhPrimitiveOrAddress(Class carrier, MemoryLayout layout) { return carrier == MemoryAddress.class - ? MemoryHandles.asAddressVarHandle(layout.varHandle(primitiveCarrierForSize(layout.byteSize()))) + ? MemoryHandles.asAddressVarHandle(layout.varHandle(primitiveCarrierForSize(layout.byteSize(), false))) : layout.varHandle(carrier); } - public static VaList newVaListOfAddress(MemoryAddress ma) { + public static VaList newVaListOfAddress(MemoryAddress ma, ResourceScope scope) { return switch (CABI.current()) { - case Win64 -> Windowsx64Linker.newVaListOfAddress(ma); - case SysV -> SysVx64Linker.newVaListOfAddress(ma); - case AArch64 -> AArch64Linker.newVaListOfAddress(ma); + case Win64 -> Windowsx64Linker.newVaListOfAddress(ma, scope); + case SysV -> SysVx64Linker.newVaListOfAddress(ma, scope); + case AArch64 -> AArch64Linker.newVaListOfAddress(ma, scope); }; } @@ -336,7 +484,7 @@ public class SharedUtils { public static MethodHandle unboxVaLists(MethodType type, MethodHandle handle, MethodHandle unboxer) { for (int i = 0; i < type.parameterCount(); i++) { if (type.parameterType(i) == VaList.class) { - handle = MethodHandles.filterArguments(handle, i, unboxer); + handle = filterArguments(handle, i + 1, unboxer); // +1 for leading address } } return handle; @@ -346,7 +494,7 @@ public class SharedUtils { MethodType type = handle.type(); for (int i = 0; i < type.parameterCount(); i++) { if (type.parameterType(i) == VaList.class) { - handle = MethodHandles.filterArguments(handle, i, boxer); + handle = filterArguments(handle, i, boxer); } } return handle; @@ -365,37 +513,6 @@ public class SharedUtils { .orElse(false); } - public interface Allocator extends AutoCloseable { - Allocator THROWING_ALLOCATOR = (size, align) -> { throw new UnsupportedOperationException("Null allocator"); }; - - default MemorySegment allocate(MemoryLayout layout) { - return allocate(layout.byteSize(), layout.byteAlignment()); - } - - default MemorySegment allocate(long size) { - return allocate(size, 1); - } - - @Override - default void close() {} - - MemorySegment allocate(long size, long align); - - static Allocator ofScope(NativeScope scope) { - return new Allocator() { - @Override - public MemorySegment allocate(long size, long align) { - return scope.allocate(size, align); - } - - @Override - public void close() { - scope.close(); - } - }; - } - } - public static class SimpleVaArg { public final Class carrier; public final MemoryLayout layout; @@ -409,12 +526,12 @@ public class SharedUtils { public VarHandle varHandle() { return carrier == MemoryAddress.class - ? MemoryHandles.asAddressVarHandle(layout.varHandle(primitiveCarrierForSize(layout.byteSize()))) + ? MemoryHandles.asAddressVarHandle(layout.varHandle(primitiveCarrierForSize(layout.byteSize(), false))) : layout.varHandle(carrier); } } - public static class EmptyVaList implements VaList { + public static non-sealed class EmptyVaList implements VaList { private final MemoryAddress address; @@ -447,12 +564,12 @@ public class SharedUtils { } @Override - public MemorySegment vargAsSegment(MemoryLayout layout) { + public MemorySegment vargAsSegment(MemoryLayout layout, SegmentAllocator allocator) { throw uoe(); } @Override - public MemorySegment vargAsSegment(MemoryLayout layout, NativeScope scope) { + public MemorySegment vargAsSegment(MemoryLayout layout, ResourceScope scope) { throw uoe(); } @@ -462,13 +579,8 @@ public class SharedUtils { } @Override - public boolean isAlive() { - return true; - } - - @Override - public void close() { - throw uoe(); + public ResourceScope scope() { + return ResourceScope.globalScope(); } @Override @@ -476,11 +588,6 @@ public class SharedUtils { return this; } - @Override - public VaList copy(NativeScope scope) { - throw uoe(); - } - @Override public MemoryAddress address() { return address; diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/UpcallStubs.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/UpcallStubs.java index d64300e533d..d417b8ad9ab 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/UpcallStubs.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/UpcallStubs.java @@ -26,18 +26,16 @@ package jdk.internal.foreign.abi; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemorySegment; -import jdk.internal.foreign.MemoryAddressImpl; +import jdk.internal.foreign.ResourceScopeImpl; import jdk.internal.foreign.NativeMemorySegmentImpl; public class UpcallStubs { - public static MemorySegment upcallAddress(UpcallHandler handler) { + public static MemoryAddress upcallAddress(UpcallHandler handler, ResourceScopeImpl scope) { long stubAddress = handler.entryPoint(); - return NativeMemorySegmentImpl.makeNativeSegmentUnchecked( - MemoryAddress.ofLong(stubAddress), 0, () -> freeUpcallStub(stubAddress), null) - .share() - .withAccessModes(MemorySegment.CLOSE | MemorySegment.HANDOFF | MemorySegment.SHARE); - }; + return NativeMemorySegmentImpl.makeNativeSegmentUnchecked(MemoryAddress.ofLong(stubAddress), 0, + () -> freeUpcallStub(stubAddress), scope).address(); + } private static void freeUpcallStub(long stubAddress) { if (!freeUpcallStub0(stubAddress)) { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/AArch64Linker.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/AArch64Linker.java index ad0661180f3..4bacc721238 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/AArch64Linker.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/AArch64Linker.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2019, 2020, Arm Limited. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * @@ -25,12 +25,12 @@ */ package jdk.internal.foreign.abi.aarch64; -import jdk.incubator.foreign.Addressable; import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.MemoryAddress; -import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; -import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.ResourceScope; +import jdk.internal.foreign.AbstractCLinker; +import jdk.internal.foreign.ResourceScopeImpl; import jdk.internal.foreign.abi.SharedUtils; import jdk.internal.foreign.abi.UpcallStubs; @@ -40,13 +40,11 @@ import java.lang.invoke.MethodType; import java.util.Objects; import java.util.function.Consumer; -import static jdk.internal.foreign.PlatformLayouts.*; - /** * ABI implementation based on ARM document "Procedure Call Standard for * the ARM 64-bit Architecture". */ -public class AArch64Linker implements CLinker { +public final class AArch64Linker extends AbstractCLinker { private static AArch64Linker instance; static final long ADDRESS_SIZE = 64; // bits @@ -59,8 +57,8 @@ public class AArch64Linker implements CLinker { MethodHandles.Lookup lookup = MethodHandles.lookup(); MH_unboxVaList = lookup.findVirtual(VaList.class, "address", MethodType.methodType(MemoryAddress.class)); - MH_boxVaList = lookup.findStatic(AArch64Linker.class, "newVaListOfAddress", - MethodType.methodType(VaList.class, MemoryAddress.class)); + MH_boxVaList = MethodHandles.insertArguments(lookup.findStatic(AArch64Linker.class, "newVaListOfAddress", + MethodType.methodType(VaList.class, MemoryAddress.class, ResourceScope.class)), 1, ResourceScope.globalScope()); } catch (ReflectiveOperationException e) { throw new ExceptionInInitializerError(e); } @@ -74,32 +72,36 @@ public class AArch64Linker implements CLinker { } @Override - public MethodHandle downcallHandle(Addressable symbol, MethodType type, FunctionDescriptor function) { - Objects.requireNonNull(symbol); + public final MethodHandle downcallHandle(MethodType type, FunctionDescriptor function) { Objects.requireNonNull(type); Objects.requireNonNull(function); MethodType llMt = SharedUtils.convertVaListCarriers(type, AArch64VaList.CARRIER); - MethodHandle handle = CallArranger.arrangeDowncall(symbol, llMt, function); + MethodHandle handle = CallArranger.arrangeDowncall(llMt, function); + if (!type.returnType().equals(MemorySegment.class)) { + // not returning segment, just insert a throwing allocator + handle = MethodHandles.insertArguments(handle, 1, SharedUtils.THROWING_ALLOCATOR); + } handle = SharedUtils.unboxVaLists(type, handle, MH_unboxVaList); return handle; } @Override - public MemorySegment upcallStub(MethodHandle target, FunctionDescriptor function) { + public final MemoryAddress upcallStub(MethodHandle target, FunctionDescriptor function, ResourceScope scope) { + Objects.requireNonNull(scope); Objects.requireNonNull(target); Objects.requireNonNull(function); target = SharedUtils.boxVaLists(target, MH_boxVaList); - return UpcallStubs.upcallAddress(CallArranger.arrangeUpcall(target, target.type(), function)); + return UpcallStubs.upcallAddress(CallArranger.arrangeUpcall(target, target.type(), function), (ResourceScopeImpl) scope); } - public static VaList newVaList(Consumer actions, SharedUtils.Allocator allocator) { - AArch64VaList.Builder builder = AArch64VaList.builder(allocator); + public static VaList newVaList(Consumer actions, ResourceScope scope) { + AArch64VaList.Builder builder = AArch64VaList.builder(scope); actions.accept(builder); return builder.build(); } - public static VaList newVaListOfAddress(MemoryAddress ma) { - return AArch64VaList.ofAddress(ma); + public static VaList newVaListOfAddress(MemoryAddress ma, ResourceScope scope) { + return AArch64VaList.ofAddress(ma, scope); } public static VaList emptyVaList() { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/AArch64VaList.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/AArch64VaList.java index b822ed123a6..ace707632c7 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/AArch64VaList.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/AArch64VaList.java @@ -26,7 +26,6 @@ package jdk.internal.foreign.abi.aarch64; import jdk.incubator.foreign.*; -import jdk.internal.foreign.NativeMemorySegmentImpl; import jdk.internal.foreign.Utils; import jdk.internal.foreign.abi.SharedUtils; import jdk.internal.misc.Unsafe; @@ -42,11 +41,12 @@ import static jdk.internal.foreign.PlatformLayouts.AArch64; import static jdk.incubator.foreign.CLinker.VaList; import static jdk.incubator.foreign.MemoryLayout.PathElement.groupElement; import static jdk.internal.foreign.abi.SharedUtils.SimpleVaArg; +import static jdk.internal.foreign.abi.SharedUtils.THROWING_ALLOCATOR; import static jdk.internal.foreign.abi.SharedUtils.checkCompatibleType; import static jdk.internal.foreign.abi.SharedUtils.vhPrimitiveOrAddress; import static jdk.internal.foreign.abi.aarch64.CallArranger.MAX_REGISTER_ARGUMENTS; -public class AArch64VaList implements VaList { +public non-sealed class AArch64VaList implements VaList { private static final Unsafe U = Unsafe.getUnsafe(); static final Class CARRIER = MemoryAddress.class; @@ -62,7 +62,7 @@ public class AArch64VaList implements VaList { // int __vr_offs; // offset from __vr_top to next FP/SIMD register arg // } va_list; - static final GroupLayout LAYOUT = MemoryLayout.ofStruct( + static final GroupLayout LAYOUT = MemoryLayout.structLayout( AArch64.C_POINTER.withName("__stack"), AArch64.C_POINTER.withName("__gr_top"), AArch64.C_POINTER.withName("__vr_top"), @@ -71,14 +71,14 @@ public class AArch64VaList implements VaList { ).withName("__va_list"); private static final MemoryLayout GP_REG - = MemoryLayout.ofValueBits(64, ByteOrder.nativeOrder()); + = MemoryLayout.valueLayout(64, ByteOrder.nativeOrder()); private static final MemoryLayout FP_REG - = MemoryLayout.ofValueBits(128, ByteOrder.nativeOrder()); + = MemoryLayout.valueLayout(128, ByteOrder.nativeOrder()); private static final MemoryLayout LAYOUT_GP_REGS - = MemoryLayout.ofSequence(MAX_REGISTER_ARGUMENTS, GP_REG); + = MemoryLayout.sequenceLayout(MAX_REGISTER_ARGUMENTS, GP_REG); private static final MemoryLayout LAYOUT_FP_REGS - = MemoryLayout.ofSequence(MAX_REGISTER_ARGUMENTS, FP_REG); + = MemoryLayout.sequenceLayout(MAX_REGISTER_ARGUMENTS, FP_REG); private static final int GP_SLOT_SIZE = (int) GP_REG.byteSize(); private static final int FP_SLOT_SIZE = (int) FP_REG.byteSize(); @@ -104,31 +104,27 @@ public class AArch64VaList implements VaList { private final MemorySegment segment; private final MemorySegment gpRegsArea; private final MemorySegment fpRegsArea; - private final List attachedSegments; - private AArch64VaList(MemorySegment segment, MemorySegment gpRegsArea, MemorySegment fpRegsArea, - List attachedSegments) { + private AArch64VaList(MemorySegment segment, MemorySegment gpRegsArea, MemorySegment fpRegsArea) { this.segment = segment; this.gpRegsArea = gpRegsArea; this.fpRegsArea = fpRegsArea; - this.attachedSegments = attachedSegments; } private static AArch64VaList readFromSegment(MemorySegment segment) { - MemorySegment gpRegsArea = handoffIfNeeded(grTop(segment).addOffset(-MAX_GP_OFFSET) - .asSegmentRestricted(MAX_GP_OFFSET), segment.ownerThread()); + MemorySegment gpRegsArea = grTop(segment).addOffset(-MAX_GP_OFFSET).asSegment( + MAX_GP_OFFSET, segment.scope()); - MemorySegment fpRegsArea = handoffIfNeeded(vrTop(segment).addOffset(-MAX_FP_OFFSET) - .asSegmentRestricted(MAX_FP_OFFSET), segment.ownerThread()); - return new AArch64VaList(segment, gpRegsArea, fpRegsArea, List.of(gpRegsArea, fpRegsArea)); + MemorySegment fpRegsArea = vrTop(segment).addOffset(-MAX_FP_OFFSET).asSegment( + MAX_FP_OFFSET, segment.scope()); + return new AArch64VaList(segment, gpRegsArea, fpRegsArea); } private static MemoryAddress emptyListAddress() { long ptr = U.allocateMemory(LAYOUT.byteSize()); - MemorySegment ms = MemoryAddress.ofLong(ptr) - .asSegmentRestricted(LAYOUT.byteSize(), () -> U.freeMemory(ptr), null) - .share(); - cleaner.register(AArch64VaList.class, ms::close); + MemorySegment ms = MemoryAddress.ofLong(ptr).asSegment( + LAYOUT.byteSize(), () -> U.freeMemory(ptr), ResourceScope.newSharedScope()); + cleaner.register(AArch64VaList.class, () -> ms.scope().close()); VH_stack.set(ms, MemoryAddress.NULL); VH_gr_top.set(ms, MemoryAddress.NULL); VH_vr_top.set(ms, MemoryAddress.NULL); @@ -234,21 +230,21 @@ public class AArch64VaList implements VaList { } @Override - public MemorySegment vargAsSegment(MemoryLayout layout) { - return (MemorySegment) read(MemorySegment.class, layout); + public MemorySegment vargAsSegment(MemoryLayout layout, SegmentAllocator allocator) { + Objects.requireNonNull(allocator); + return (MemorySegment) read(MemorySegment.class, layout, allocator); } @Override - public MemorySegment vargAsSegment(MemoryLayout layout, NativeScope scope) { - Objects.requireNonNull(scope); - return (MemorySegment) read(MemorySegment.class, layout, SharedUtils.Allocator.ofScope(scope)); + public MemorySegment vargAsSegment(MemoryLayout layout, ResourceScope scope) { + return vargAsSegment(layout, SegmentAllocator.ofScope(scope)); } private Object read(Class carrier, MemoryLayout layout) { - return read(carrier, layout, MemorySegment::allocateNative); + return read(carrier, layout, THROWING_ALLOCATOR); } - private Object read(Class carrier, MemoryLayout layout, SharedUtils.Allocator allocator) { + private Object read(Class carrier, MemoryLayout layout, SegmentAllocator allocator) { Objects.requireNonNull(layout); checkCompatibleType(carrier, layout, AArch64Linker.ADDRESS_SIZE); @@ -257,22 +253,18 @@ public class AArch64VaList implements VaList { preAlignStack(layout); return switch (typeClass) { case STRUCT_REGISTER, STRUCT_HFA, STRUCT_REFERENCE -> { - try (MemorySegment slice = handoffIfNeeded(stackPtr() - .asSegmentRestricted(layout.byteSize()), segment.ownerThread())) { - MemorySegment seg = allocator.allocate(layout); - seg.copyFrom(slice); - postAlignStack(layout); - yield seg; - } + MemorySegment slice = stackPtr().asSegment(layout.byteSize(), scope()); + MemorySegment seg = allocator.allocate(layout); + seg.copyFrom(slice); + postAlignStack(layout); + yield seg; } case POINTER, INTEGER, FLOAT -> { VarHandle reader = vhPrimitiveOrAddress(carrier, layout); - try (MemorySegment slice = handoffIfNeeded(stackPtr() - .asSegmentRestricted(layout.byteSize()), segment.ownerThread())) { - Object res = reader.get(slice); - postAlignStack(layout); - yield res; - } + MemorySegment slice = stackPtr().asSegment(layout.byteSize(), scope()); + Object res = reader.get(slice); + postAlignStack(layout); + yield res; } }; } else { @@ -314,12 +306,10 @@ public class AArch64VaList implements VaList { gpRegsArea.asSlice(currentGPOffset())); consumeGPSlots(1); - try (MemorySegment slice = handoffIfNeeded(ptr - .asSegmentRestricted(layout.byteSize()), segment.ownerThread())) { - MemorySegment seg = allocator.allocate(layout); - seg.copyFrom(slice); - yield seg; - } + MemorySegment slice = ptr.asSegment(layout.byteSize(), scope()); + MemorySegment seg = allocator.allocate(layout); + seg.copyFrom(slice); + yield seg; } case POINTER, INTEGER -> { VarHandle reader = SharedUtils.vhPrimitiveOrAddress(carrier, layout); @@ -356,40 +346,24 @@ public class AArch64VaList implements VaList { } } - static AArch64VaList.Builder builder(SharedUtils.Allocator allocator) { - return new AArch64VaList.Builder(allocator); + static AArch64VaList.Builder builder(ResourceScope scope) { + return new AArch64VaList.Builder(scope); } - public static VaList ofAddress(MemoryAddress ma) { - return readFromSegment(ma.asSegmentRestricted(LAYOUT.byteSize())); + public static VaList ofAddress(MemoryAddress ma, ResourceScope scope) { + return readFromSegment(ma.asSegment(LAYOUT.byteSize(), scope)); } @Override - public boolean isAlive() { - return segment.isAlive(); - } - - @Override - public void close() { - segment.close(); - attachedSegments.forEach(MemorySegment::close); + public ResourceScope scope() { + return segment.scope(); } @Override public VaList copy() { - return copy(MemorySegment::allocateNative); - } - - @Override - public VaList copy(NativeScope scope) { - Objects.requireNonNull(scope); - return copy(SharedUtils.Allocator.ofScope(scope)); - } - - private VaList copy(SharedUtils.Allocator allocator) { - MemorySegment copy = allocator.allocate(LAYOUT); + MemorySegment copy = MemorySegment.allocateNative(LAYOUT, segment.scope()); copy.copyFrom(segment); - return new AArch64VaList(copy, gpRegsArea, fpRegsArea, List.of()); + return new AArch64VaList(copy, gpRegsArea, fpRegsArea); } @Override @@ -423,8 +397,8 @@ public class AArch64VaList implements VaList { + '}'; } - static class Builder implements VaList.Builder { - private final SharedUtils.Allocator allocator; + public static non-sealed class Builder implements VaList.Builder { + private final ResourceScope scope; private final MemorySegment gpRegs; private final MemorySegment fpRegs; @@ -432,10 +406,10 @@ public class AArch64VaList implements VaList { private long currentFPOffset = 0; private final List stackArgs = new ArrayList<>(); - Builder(SharedUtils.Allocator allocator) { - this.allocator = allocator; - this.gpRegs = allocator.allocate(LAYOUT_GP_REGS); - this.fpRegs = allocator.allocate(LAYOUT_FP_REGS); + Builder(ResourceScope scope) { + this.scope = scope; + this.gpRegs = MemorySegment.allocateNative(LAYOUT_GP_REGS, scope); + this.fpRegs = MemorySegment.allocateNative(LAYOUT_FP_REGS, scope); } @Override @@ -534,8 +508,8 @@ public class AArch64VaList implements VaList { return EMPTY; } + SegmentAllocator allocator = SegmentAllocator.arenaAllocator(scope); MemorySegment vaListSegment = allocator.allocate(LAYOUT); - List attachedSegments = new ArrayList<>(); MemoryAddress stackArgsPtr = MemoryAddress.NULL; if (!stackArgs.isEmpty()) { long stackArgsSize = stackArgs.stream() @@ -549,7 +523,6 @@ public class AArch64VaList implements VaList { writer.set(stackArgsSegment, arg.value); stackArgsSegment = stackArgsSegment.asSlice(alignedSize); } - attachedSegments.add(stackArgsSegment); } VH_gr_top.set(vaListSegment, gpRegs.asSlice(gpRegs.byteSize()).address()); @@ -558,16 +531,9 @@ public class AArch64VaList implements VaList { VH_gr_offs.set(vaListSegment, -MAX_GP_OFFSET); VH_vr_offs.set(vaListSegment, -MAX_FP_OFFSET); - attachedSegments.add(gpRegs); - attachedSegments.add(fpRegs); - assert gpRegs.ownerThread() == vaListSegment.ownerThread(); - assert fpRegs.ownerThread() == vaListSegment.ownerThread(); - return new AArch64VaList(vaListSegment, gpRegs, fpRegs, attachedSegments); + assert gpRegs.scope().ownerThread() == vaListSegment.scope().ownerThread(); + assert fpRegs.scope().ownerThread() == vaListSegment.scope().ownerThread(); + return new AArch64VaList(vaListSegment, gpRegs, fpRegs); } } - - private static MemorySegment handoffIfNeeded(MemorySegment segment, Thread thread) { - return segment.ownerThread() == thread ? - segment : segment.handoff(thread); - } } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/CallArranger.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/CallArranger.java index 4b19004ec50..cd02e266a75 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/CallArranger.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/aarch64/CallArranger.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2019, 2020, Arm Limited. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * @@ -25,13 +25,11 @@ */ package jdk.internal.foreign.abi.aarch64; -import jdk.incubator.foreign.Addressable; import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.GroupLayout; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; -import jdk.internal.foreign.PlatformLayouts; import jdk.internal.foreign.Utils; import jdk.internal.foreign.abi.CallingSequenceBuilder; import jdk.internal.foreign.abi.UpcallHandler; @@ -126,10 +124,10 @@ public class CallArranger { return new Bindings(csb.build(), returnInMemory); } - public static MethodHandle arrangeDowncall(Addressable addr, MethodType mt, FunctionDescriptor cDesc) { + public static MethodHandle arrangeDowncall(MethodType mt, FunctionDescriptor cDesc) { Bindings bindings = getBindings(mt, cDesc, false); - MethodHandle handle = new ProgrammableInvoker(C, addr, bindings.callingSequence).getBoundMethodHandle(); + MethodHandle handle = new ProgrammableInvoker(C, bindings.callingSequence).getBoundMethodHandle(); if (bindings.isInMemoryReturn) { handle = SharedUtils.adaptDowncallForIMR(handle, cDesc); @@ -142,10 +140,10 @@ public class CallArranger { Bindings bindings = getBindings(mt, cDesc, true); if (bindings.isInMemoryReturn) { - target = SharedUtils.adaptUpcallForIMR(target); + target = SharedUtils.adaptUpcallForIMR(target, true /* drop return, since we don't have bindings for it */); } - return new ProgrammableUpcallHandler(C, target, bindings.callingSequence); + return ProgrammableUpcallHandler.make(C, target, bindings.callingSequence); } private static boolean isInMemoryReturn(Optional returnLayout) { @@ -232,7 +230,7 @@ public class CallArranger { if (offset + STACK_SLOT_SIZE < layout.byteSize()) { bindings.dup(); } - Class type = SharedUtils.primitiveCarrierForSize(copy); + Class type = SharedUtils.primitiveCarrierForSize(copy, false); bindings.bufferLoad(offset, type) .vmStore(storage, type); offset += STACK_SLOT_SIZE; @@ -250,7 +248,7 @@ public class CallArranger { long copy = Math.min(layout.byteSize() - offset, STACK_SLOT_SIZE); VMStorage storage = storageCalculator.stackAlloc(copy, STACK_SLOT_SIZE); - Class type = SharedUtils.primitiveCarrierForSize(copy); + Class type = SharedUtils.primitiveCarrierForSize(copy, false); bindings.dup() .vmLoad(storage, type) .bufferStore(offset, type); @@ -291,7 +289,8 @@ public class CallArranger { while (offset < layout.byteSize()) { final long copy = Math.min(layout.byteSize() - offset, 8); VMStorage storage = regs[regIndex++]; - Class type = SharedUtils.primitiveCarrierForSize(copy); + boolean useFloat = storage.type() == StorageClasses.VECTOR; + Class type = SharedUtils.primitiveCarrierForSize(copy, useFloat); if (offset + copy < layout.byteSize()) { bindings.dup(); } @@ -324,7 +323,8 @@ public class CallArranger { for (int i = 0; i < group.memberLayouts().size(); i++) { VMStorage storage = regs[i]; final long size = group.memberLayouts().get(i).byteSize(); - Class type = SharedUtils.primitiveCarrierForSize(size); + boolean useFloat = storage.type() == StorageClasses.VECTOR; + Class type = SharedUtils.primitiveCarrierForSize(size, useFloat); if (i + 1 < group.memberLayouts().size()) { bindings.dup(); } @@ -393,7 +393,8 @@ public class CallArranger { final long copy = Math.min(layout.byteSize() - offset, 8); VMStorage storage = regs[regIndex++]; bindings.dup(); - Class type = SharedUtils.primitiveCarrierForSize(copy); + boolean useFloat = storage.type() == StorageClasses.VECTOR; + Class type = SharedUtils.primitiveCarrierForSize(copy, useFloat); bindings.vmLoad(storage, type) .bufferStore(offset, type); offset += copy; @@ -410,9 +411,6 @@ public class CallArranger { bindings.vmLoad(storage, long.class) .boxAddress() .toSegment(layout); - // ASSERT SCOPE OF BOXED ADDRESS HERE - // caveat. buffer should instead go out of scope after call - bindings.copy(layout); break; } case STRUCT_HFA: { @@ -426,7 +424,8 @@ public class CallArranger { for (int i = 0; i < group.memberLayouts().size(); i++) { VMStorage storage = regs[i]; final long size = group.memberLayouts().get(i).byteSize(); - Class type = SharedUtils.primitiveCarrierForSize(size); + boolean useFloat = storage.type() == StorageClasses.VECTOR; + Class type = SharedUtils.primitiveCarrierForSize(size, useFloat); bindings.dup() .vmLoad(storage, type) .bufferStore(offset, type); diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/CallArranger.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/CallArranger.java index d748f3e4145..f7f2ddd8139 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/CallArranger.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/CallArranger.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -25,13 +25,11 @@ */ package jdk.internal.foreign.abi.x64.sysv; -import jdk.incubator.foreign.Addressable; import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.GroupLayout; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; -import jdk.internal.foreign.PlatformLayouts; import jdk.internal.foreign.abi.CallingSequenceBuilder; import jdk.internal.foreign.abi.UpcallHandler; import jdk.internal.foreign.abi.ABIDescriptor; @@ -123,10 +121,10 @@ public class CallArranger { return new Bindings(csb.build(), returnInMemory, argCalc.storageCalculator.nVectorReg); } - public static MethodHandle arrangeDowncall(Addressable addr, MethodType mt, FunctionDescriptor cDesc) { + public static MethodHandle arrangeDowncall(MethodType mt, FunctionDescriptor cDesc) { Bindings bindings = getBindings(mt, cDesc, false); - MethodHandle handle = new ProgrammableInvoker(CSysV, addr, bindings.callingSequence).getBoundMethodHandle(); + MethodHandle handle = new ProgrammableInvoker(CSysV, bindings.callingSequence).getBoundMethodHandle(); handle = MethodHandles.insertArguments(handle, handle.type().parameterCount() - 1, bindings.nVectorArgs); if (bindings.isInMemoryReturn) { @@ -140,10 +138,10 @@ public class CallArranger { Bindings bindings = getBindings(mt, cDesc, true); if (bindings.isInMemoryReturn) { - target = SharedUtils.adaptUpcallForIMR(target); + target = SharedUtils.adaptUpcallForIMR(target, true /* drop return, since we don't have bindings for it */); } - return new ProgrammableUpcallHandler(CSysV, target, bindings.callingSequence); + return ProgrammableUpcallHandler.make(CSysV, target, bindings.callingSequence); } private static boolean isInMemoryReturn(Optional returnLayout) { @@ -270,10 +268,11 @@ public class CallArranger { while (offset < layout.byteSize()) { final long copy = Math.min(layout.byteSize() - offset, 8); VMStorage storage = regs[regIndex++]; - Class type = SharedUtils.primitiveCarrierForSize(copy); if (offset + copy < layout.byteSize()) { bindings.dup(); } + boolean useFloat = storage.type() == StorageClasses.VECTOR; + Class type = SharedUtils.primitiveCarrierForSize(copy, useFloat); bindings.bufferLoad(offset, type) .vmStore(storage, type); offset += copy; @@ -324,7 +323,8 @@ public class CallArranger { final long copy = Math.min(layout.byteSize() - offset, 8); VMStorage storage = regs[regIndex++]; bindings.dup(); - Class type = SharedUtils.primitiveCarrierForSize(copy); + boolean useFloat = storage.type() == StorageClasses.VECTOR; + Class type = SharedUtils.primitiveCarrierForSize(copy, useFloat); bindings.vmLoad(storage, type) .bufferStore(offset, type); offset += copy; diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVVaList.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVVaList.java index b6cce1b3d84..4a74ebfa0d5 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVVaList.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVVaList.java @@ -26,7 +26,6 @@ package jdk.internal.foreign.abi.x64.sysv; import jdk.incubator.foreign.*; -import jdk.internal.foreign.NativeMemorySegmentImpl; import jdk.internal.foreign.Utils; import jdk.internal.foreign.abi.SharedUtils; import jdk.internal.misc.Unsafe; @@ -42,11 +41,12 @@ import static jdk.internal.foreign.PlatformLayouts.SysV; import static jdk.incubator.foreign.CLinker.VaList; import static jdk.incubator.foreign.MemoryLayout.PathElement.groupElement; import static jdk.internal.foreign.abi.SharedUtils.SimpleVaArg; +import static jdk.internal.foreign.abi.SharedUtils.THROWING_ALLOCATOR; import static jdk.internal.foreign.abi.SharedUtils.checkCompatibleType; import static jdk.internal.foreign.abi.SharedUtils.vhPrimitiveOrAddress; // See https://software.intel.com/sites/default/files/article/402129/mpx-linux64-abi.pdf "3.5.7 Variable Argument Lists" -public class SysVVaList implements VaList { +public non-sealed class SysVVaList implements VaList { private static final Unsafe U = Unsafe.getUnsafe(); static final Class CARRIER = MemoryAddress.class; @@ -60,17 +60,17 @@ public class SysVVaList implements VaList { // /* size: 24, cachelines: 1, members: 4 */ // /* last cacheline: 24 bytes */ // }; - static final GroupLayout LAYOUT = MemoryLayout.ofStruct( + static final GroupLayout LAYOUT = MemoryLayout.structLayout( SysV.C_INT.withName("gp_offset"), SysV.C_INT.withName("fp_offset"), SysV.C_POINTER.withName("overflow_arg_area"), SysV.C_POINTER.withName("reg_save_area") ).withName("__va_list_tag"); - private static final MemoryLayout GP_REG = MemoryLayout.ofValueBits(64, ByteOrder.nativeOrder()); - private static final MemoryLayout FP_REG = MemoryLayout.ofValueBits(128, ByteOrder.nativeOrder()); + private static final MemoryLayout GP_REG = MemoryLayout.valueLayout(64, ByteOrder.nativeOrder()); + private static final MemoryLayout FP_REG = MemoryLayout.valueLayout(128, ByteOrder.nativeOrder()); - private static final GroupLayout LAYOUT_REG_SAVE_AREA = MemoryLayout.ofStruct( + private static final GroupLayout LAYOUT_REG_SAVE_AREA = MemoryLayout.structLayout( GP_REG.withName("%rdi"), GP_REG.withName("%rsi"), GP_REG.withName("%rdx"), @@ -117,30 +117,27 @@ public class SysVVaList implements VaList { private final MemorySegment segment; private final MemorySegment regSaveArea; - private final List attachedSegments; - private SysVVaList(MemorySegment segment, MemorySegment regSaveArea, List attachedSegments) { + private SysVVaList(MemorySegment segment, MemorySegment regSaveArea) { this.segment = segment; this.regSaveArea = regSaveArea; - this.attachedSegments = attachedSegments; } private static SysVVaList readFromSegment(MemorySegment segment) { MemorySegment regSaveArea = getRegSaveArea(segment); - return new SysVVaList(segment, regSaveArea, List.of(regSaveArea)); + return new SysVVaList(segment, regSaveArea); } private static MemoryAddress emptyListAddress() { long ptr = U.allocateMemory(LAYOUT.byteSize()); - MemorySegment base = MemoryAddress.ofLong(ptr) - .asSegmentRestricted(LAYOUT.byteSize(), () -> U.freeMemory(ptr), null) - .share(); - cleaner.register(SysVVaList.class, base::close); + MemorySegment base = MemoryAddress.ofLong(ptr).asSegment( + LAYOUT.byteSize(), () -> U.freeMemory(ptr), ResourceScope.newSharedScope()); + cleaner.register(SysVVaList.class, () -> base.scope().close()); VH_gp_offset.set(base, MAX_GP_OFFSET); VH_fp_offset.set(base, MAX_FP_OFFSET); VH_overflow_arg_area.set(base, MemoryAddress.NULL); VH_reg_save_area.set(base, MemoryAddress.NULL); - return base.withAccessModes(0).address(); + return base.address(); } public static VaList empty() { @@ -176,8 +173,8 @@ public class SysVVaList implements VaList { } private static MemorySegment getRegSaveArea(MemorySegment segment) { - return handoffIfNeeded(((MemoryAddress)VH_reg_save_area.get(segment)) - .asSegmentRestricted(LAYOUT_REG_SAVE_AREA.byteSize()), segment.ownerThread()); + return ((MemoryAddress)VH_reg_save_area.get(segment)).asSegment( + LAYOUT_REG_SAVE_AREA.byteSize(), segment.scope()); } private void preAlignStack(MemoryLayout layout) { @@ -211,21 +208,21 @@ public class SysVVaList implements VaList { } @Override - public MemorySegment vargAsSegment(MemoryLayout layout) { - return (MemorySegment) read(MemorySegment.class, layout); + public MemorySegment vargAsSegment(MemoryLayout layout, SegmentAllocator allocator) { + Objects.requireNonNull(allocator); + return (MemorySegment) read(MemorySegment.class, layout, allocator); } @Override - public MemorySegment vargAsSegment(MemoryLayout layout, NativeScope scope) { - Objects.requireNonNull(scope); - return (MemorySegment) read(MemorySegment.class, layout, SharedUtils.Allocator.ofScope(scope)); + public MemorySegment vargAsSegment(MemoryLayout layout, ResourceScope scope) { + return vargAsSegment(layout, SegmentAllocator.ofScope(scope)); } private Object read(Class carrier, MemoryLayout layout) { - return read(carrier, layout, MemorySegment::allocateNative); + return read(carrier, layout, THROWING_ALLOCATOR); } - private Object read(Class carrier, MemoryLayout layout, SharedUtils.Allocator allocator) { + private Object read(Class carrier, MemoryLayout layout, SegmentAllocator allocator) { Objects.requireNonNull(layout); checkCompatibleType(carrier, layout, SysVx64Linker.ADDRESS_SIZE); TypeClass typeClass = TypeClass.classifyLayout(layout); @@ -234,18 +231,16 @@ public class SysVVaList implements VaList { preAlignStack(layout); return switch (typeClass.kind()) { case STRUCT -> { - try (MemorySegment slice = handoffIfNeeded(stackPtr() - .asSegmentRestricted(layout.byteSize()), segment.ownerThread())) { - MemorySegment seg = allocator.allocate(layout); - seg.copyFrom(slice); - postAlignStack(layout); - yield seg; - } + MemorySegment slice = stackPtr().asSegment(layout.byteSize(), scope()); + MemorySegment seg = allocator.allocate(layout); + seg.copyFrom(slice); + postAlignStack(layout); + yield seg; } case POINTER, INTEGER, FLOAT -> { VarHandle reader = vhPrimitiveOrAddress(carrier, layout); - try (MemorySegment slice = handoffIfNeeded(stackPtr() - .asSegmentRestricted(layout.byteSize()), segment.ownerThread())) { + try (ResourceScope localScope = ResourceScope.newConfinedScope()) { + MemorySegment slice = stackPtr().asSegment(layout.byteSize(), localScope); Object res = reader.get(slice); postAlignStack(layout); yield res; @@ -305,40 +300,24 @@ public class SysVVaList implements VaList { } } - static SysVVaList.Builder builder(SharedUtils.Allocator allocator) { - return new SysVVaList.Builder(allocator); + static SysVVaList.Builder builder(ResourceScope scope) { + return new SysVVaList.Builder(scope); } - public static VaList ofAddress(MemoryAddress ma) { - return readFromSegment(ma.asSegmentRestricted(LAYOUT.byteSize())); + public static VaList ofAddress(MemoryAddress ma, ResourceScope scope) { + return readFromSegment(ma.asSegment(LAYOUT.byteSize(), scope)); } @Override - public boolean isAlive() { - return segment.isAlive(); - } - - @Override - public void close() { - segment.close(); - attachedSegments.forEach(MemorySegment::close); + public ResourceScope scope() { + return segment.scope(); } @Override public VaList copy() { - return copy(MemorySegment::allocateNative); - } - - @Override - public VaList copy(NativeScope scope) { - Objects.requireNonNull(scope); - return copy(SharedUtils.Allocator.ofScope(scope)); - } - - private VaList copy(SharedUtils.Allocator allocator) { - MemorySegment copy = allocator.allocate(LAYOUT); + MemorySegment copy = MemorySegment.allocateNative(LAYOUT, segment.scope()); copy.copyFrom(segment); - return new SysVVaList(copy, regSaveArea, List.of()); + return new SysVVaList(copy, regSaveArea); } @Override @@ -361,16 +340,16 @@ public class SysVVaList implements VaList { + '}'; } - static class Builder implements VaList.Builder { - private final SharedUtils.Allocator allocator; + public static non-sealed class Builder implements VaList.Builder { + private final ResourceScope scope; private final MemorySegment reg_save_area; private long currentGPOffset = 0; private long currentFPOffset = FP_OFFSET; private final List stackArgs = new ArrayList<>(); - public Builder(SharedUtils.Allocator allocator) { - this.allocator = allocator; - this.reg_save_area = allocator.allocate(LAYOUT_REG_SAVE_AREA); + public Builder(ResourceScope scope) { + this.scope = scope; + this.reg_save_area = MemorySegment.allocateNative(LAYOUT_REG_SAVE_AREA, scope); } @Override @@ -451,8 +430,8 @@ public class SysVVaList implements VaList { return EMPTY; } + SegmentAllocator allocator = SegmentAllocator.arenaAllocator(scope); MemorySegment vaListSegment = allocator.allocate(LAYOUT); - List attachedSegments = new ArrayList<>(); MemoryAddress stackArgsPtr = MemoryAddress.NULL; if (!stackArgs.isEmpty()) { long stackArgsSize = stackArgs.stream().reduce(0L, (acc, e) -> acc + e.layout.byteSize(), Long::sum); @@ -471,20 +450,13 @@ public class SysVVaList implements VaList { maOverflowArgArea = maOverflowArgArea.asSlice(arg.layout.byteSize()); } stackArgsPtr = stackArgsSegment.address(); - attachedSegments.add(stackArgsSegment); } VH_fp_offset.set(vaListSegment, (int) FP_OFFSET); VH_overflow_arg_area.set(vaListSegment, stackArgsPtr); VH_reg_save_area.set(vaListSegment, reg_save_area.address()); - attachedSegments.add(reg_save_area); - assert reg_save_area.ownerThread() == vaListSegment.ownerThread(); - return new SysVVaList(vaListSegment, reg_save_area, attachedSegments); + assert reg_save_area.scope().ownerThread() == vaListSegment.scope().ownerThread(); + return new SysVVaList(vaListSegment, reg_save_area); } } - - private static MemorySegment handoffIfNeeded(MemorySegment segment, Thread thread) { - return segment.ownerThread() == thread ? - segment : segment.handoff(thread); - } } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVx64Linker.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVx64Linker.java index 69a32bfdc30..10d4a61c11e 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVx64Linker.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVx64Linker.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -24,12 +24,13 @@ */ package jdk.internal.foreign.abi.x64.sysv; -import jdk.incubator.foreign.Addressable; + import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.MemoryAddress; -import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; -import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.ResourceScope; +import jdk.internal.foreign.AbstractCLinker; +import jdk.internal.foreign.ResourceScopeImpl; import jdk.internal.foreign.abi.SharedUtils; import jdk.internal.foreign.abi.UpcallStubs; @@ -37,15 +38,12 @@ import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; import java.util.Objects; -import java.util.Optional; import java.util.function.Consumer; -import static jdk.internal.foreign.PlatformLayouts.*; - /** * ABI implementation based on System V ABI AMD64 supplement v.0.99.6 */ -public class SysVx64Linker implements CLinker { +public final class SysVx64Linker extends AbstractCLinker { public static final int MAX_INTEGER_ARGUMENT_REGISTERS = 6; public static final int MAX_INTEGER_RETURN_REGISTERS = 2; public static final int MAX_VECTOR_ARGUMENT_REGISTERS = 8; @@ -64,8 +62,8 @@ public class SysVx64Linker implements CLinker { MethodHandles.Lookup lookup = MethodHandles.lookup(); MH_unboxVaList = lookup.findVirtual(VaList.class, "address", MethodType.methodType(MemoryAddress.class)); - MH_boxVaList = lookup.findStatic(SysVx64Linker.class, "newVaListOfAddress", - MethodType.methodType(VaList.class, MemoryAddress.class)); + MH_boxVaList = MethodHandles.insertArguments(lookup.findStatic(SysVx64Linker.class, "newVaListOfAddress", + MethodType.methodType(VaList.class, MemoryAddress.class, ResourceScope.class)), 1, ResourceScope.globalScope()); } catch (ReflectiveOperationException e) { throw new ExceptionInInitializerError(e); } @@ -78,33 +76,37 @@ public class SysVx64Linker implements CLinker { return instance; } - public static VaList newVaList(Consumer actions, SharedUtils.Allocator allocator) { - SysVVaList.Builder builder = SysVVaList.builder(allocator); + public static VaList newVaList(Consumer actions, ResourceScope scope) { + SysVVaList.Builder builder = SysVVaList.builder(scope); actions.accept(builder); return builder.build(); } @Override - public MethodHandle downcallHandle(Addressable symbol, MethodType type, FunctionDescriptor function) { - Objects.requireNonNull(symbol); + public final MethodHandle downcallHandle(MethodType type, FunctionDescriptor function) { Objects.requireNonNull(type); Objects.requireNonNull(function); MethodType llMt = SharedUtils.convertVaListCarriers(type, SysVVaList.CARRIER); - MethodHandle handle = CallArranger.arrangeDowncall(symbol, llMt, function); + MethodHandle handle = CallArranger.arrangeDowncall(llMt, function); + if (!type.returnType().equals(MemorySegment.class)) { + // not returning segment, just insert a throwing allocator + handle = MethodHandles.insertArguments(handle, 1, SharedUtils.THROWING_ALLOCATOR); + } handle = SharedUtils.unboxVaLists(type, handle, MH_unboxVaList); return handle; } @Override - public MemorySegment upcallStub(MethodHandle target, FunctionDescriptor function) { + public final MemoryAddress upcallStub(MethodHandle target, FunctionDescriptor function, ResourceScope scope) { + Objects.requireNonNull(scope); Objects.requireNonNull(target); Objects.requireNonNull(function); target = SharedUtils.boxVaLists(target, MH_boxVaList); - return UpcallStubs.upcallAddress(CallArranger.arrangeUpcall(target, target.type(), function)); + return UpcallStubs.upcallAddress(CallArranger.arrangeUpcall(target, target.type(), function), (ResourceScopeImpl) scope); } - public static VaList newVaListOfAddress(MemoryAddress ma) { - return SysVVaList.ofAddress(ma); + public static VaList newVaListOfAddress(MemoryAddress ma, ResourceScope scope) { + return SysVVaList.ofAddress(ma, scope); } public static VaList emptyVaList() { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/CallArranger.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/CallArranger.java index 8e8be4bc572..5ff77e3928a 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/CallArranger.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/CallArranger.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -24,13 +24,11 @@ */ package jdk.internal.foreign.abi.x64.windows; -import jdk.incubator.foreign.Addressable; import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.GroupLayout; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; -import jdk.internal.foreign.PlatformLayouts; import jdk.internal.foreign.Utils; import jdk.internal.foreign.abi.CallingSequenceBuilder; import jdk.internal.foreign.abi.UpcallHandler; @@ -124,10 +122,10 @@ public class CallArranger { return new Bindings(csb.csb.build(), returnInMemory); } - public static MethodHandle arrangeDowncall(Addressable addr, MethodType mt, FunctionDescriptor cDesc) { + public static MethodHandle arrangeDowncall(MethodType mt, FunctionDescriptor cDesc) { Bindings bindings = getBindings(mt, cDesc, false); - MethodHandle handle = new ProgrammableInvoker(CWindows, addr, bindings.callingSequence).getBoundMethodHandle(); + MethodHandle handle = new ProgrammableInvoker(CWindows, bindings.callingSequence).getBoundMethodHandle(); if (bindings.isInMemoryReturn) { handle = SharedUtils.adaptDowncallForIMR(handle, cDesc); @@ -140,10 +138,10 @@ public class CallArranger { Bindings bindings = getBindings(mt, cDesc, true); if (bindings.isInMemoryReturn) { - target = SharedUtils.adaptUpcallForIMR(target); + target = SharedUtils.adaptUpcallForIMR(target, false /* need the return value as well */); } - return new ProgrammableUpcallHandler(CWindows, target, bindings.callingSequence); + return ProgrammableUpcallHandler.make(CWindows, target, bindings.callingSequence); } private static boolean isInMemoryReturn(Optional returnLayout) { @@ -205,7 +203,7 @@ public class CallArranger { case STRUCT_REGISTER: { assert carrier == MemorySegment.class; VMStorage storage = storageCalculator.nextStorage(StorageClasses.INTEGER, layout); - Class type = SharedUtils.primitiveCarrierForSize(layout.byteSize()); + Class type = SharedUtils.primitiveCarrierForSize(layout.byteSize(), false); bindings.bufferLoad(0, type) .vmStore(storage, type); break; @@ -270,7 +268,7 @@ public class CallArranger { bindings.allocate(layout) .dup(); VMStorage storage = storageCalculator.nextStorage(StorageClasses.INTEGER, layout); - Class type = SharedUtils.primitiveCarrierForSize(layout.byteSize()); + Class type = SharedUtils.primitiveCarrierForSize(layout.byteSize(), false); bindings.vmLoad(storage, type) .bufferStore(0, type); break; @@ -281,9 +279,6 @@ public class CallArranger { bindings.vmLoad(storage, long.class) .boxAddress() .toSegment(layout); - // ASSERT SCOPE OF BOXED ADDRESS HERE - // caveat. buffer should instead go out of scope after call - bindings.copy(layout); break; } case POINTER: { diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/WinVaList.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/WinVaList.java index c6bf603be37..debdf9c6b5d 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/WinVaList.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/WinVaList.java @@ -27,6 +27,7 @@ package jdk.internal.foreign.abi.x64.windows; import jdk.incubator.foreign.*; import jdk.incubator.foreign.CLinker.VaList; +import jdk.internal.foreign.ResourceScopeImpl; import jdk.internal.foreign.abi.SharedUtils; import jdk.internal.foreign.abi.SharedUtils.SimpleVaArg; @@ -54,7 +55,7 @@ import static jdk.internal.foreign.PlatformLayouts.Win64.C_POINTER; // ? **(t**)((ap += sizeof(__int64)) - sizeof(__int64)) \ // : *(t* )((ap += sizeof(__int64)) - sizeof(__int64))) // -class WinVaList implements VaList { +public non-sealed class WinVaList implements VaList { public static final Class CARRIER = MemoryAddress.class; private static final long VA_SLOT_SIZE_BYTES = 8; private static final VarHandle VH_address = MemoryHandles.asAddressVarHandle(C_POINTER.varHandle(long.class)); @@ -62,13 +63,11 @@ class WinVaList implements VaList { private static final VaList EMPTY = new SharedUtils.EmptyVaList(MemoryAddress.NULL); private MemorySegment segment; - private final List attachedSegments; - private final MemorySegment livenessCheck; + private final ResourceScope scope; - private WinVaList(MemorySegment segment, List attachedSegments, MemorySegment livenessCheck) { + private WinVaList(MemorySegment segment, ResourceScope scope) { this.segment = segment; - this.attachedSegments = attachedSegments; - this.livenessCheck = livenessCheck; + this.scope = scope; } public static final VaList empty() { @@ -96,21 +95,21 @@ class WinVaList implements VaList { } @Override - public MemorySegment vargAsSegment(MemoryLayout layout) { - return (MemorySegment) read(MemorySegment.class, layout); + public MemorySegment vargAsSegment(MemoryLayout layout, SegmentAllocator allocator) { + Objects.requireNonNull(allocator); + return (MemorySegment) read(MemorySegment.class, layout, allocator); } @Override - public MemorySegment vargAsSegment(MemoryLayout layout, NativeScope scope) { - Objects.requireNonNull(scope); - return (MemorySegment) read(MemorySegment.class, layout, SharedUtils.Allocator.ofScope(scope)); + public MemorySegment vargAsSegment(MemoryLayout layout, ResourceScope scope) { + return vargAsSegment(layout, SegmentAllocator.ofScope(scope)); } private Object read(Class carrier, MemoryLayout layout) { - return read(carrier, layout, MemorySegment::allocateNative); + return read(carrier, layout, SharedUtils.THROWING_ALLOCATOR); } - private Object read(Class carrier, MemoryLayout layout, SharedUtils.Allocator allocator) { + private Object read(Class carrier, MemoryLayout layout, SegmentAllocator allocator) { Objects.requireNonNull(layout); SharedUtils.checkCompatibleType(carrier, layout, Windowsx64Linker.ADDRESS_SIZE); Object res; @@ -119,12 +118,10 @@ class WinVaList implements VaList { res = switch (typeClass) { case STRUCT_REFERENCE -> { MemoryAddress structAddr = (MemoryAddress) VH_address.get(segment); - try (MemorySegment struct = handoffIfNeeded(structAddr.asSegmentRestricted(layout.byteSize()), - segment.ownerThread())) { - MemorySegment seg = allocator.allocate(layout.byteSize()); - seg.copyFrom(struct); - yield seg; - } + MemorySegment struct = structAddr.asSegment(layout.byteSize(), scope()); + MemorySegment seg = allocator.allocate(layout); + seg.copyFrom(struct); + yield seg; } case STRUCT_REGISTER -> { MemorySegment struct = allocator.allocate(layout); @@ -148,36 +145,24 @@ class WinVaList implements VaList { segment = segment.asSlice(layouts.length * VA_SLOT_SIZE_BYTES); } - static WinVaList ofAddress(MemoryAddress addr) { - MemorySegment segment = addr.asSegmentRestricted(Long.MAX_VALUE); - return new WinVaList(segment, List.of(segment), null); + static WinVaList ofAddress(MemoryAddress addr, ResourceScope scope) { + MemorySegment segment = addr.asSegment(Long.MAX_VALUE, scope); + return new WinVaList(segment, scope); } - static Builder builder(SharedUtils.Allocator allocator) { - return new Builder(allocator); + static Builder builder(ResourceScope scope) { + return new Builder(scope); } @Override - public void close() { - if (livenessCheck != null) - livenessCheck.close(); - attachedSegments.forEach(MemorySegment::close); + public ResourceScope scope() { + return scope; } @Override public VaList copy() { - MemorySegment liveness = handoffIfNeeded(MemoryAddress.NULL.asSegmentRestricted(1), - segment.ownerThread()); - return new WinVaList(segment, List.of(), liveness); - } - - @Override - public VaList copy(NativeScope scope) { - Objects.requireNonNull(scope); - MemorySegment liveness = handoffIfNeeded(MemoryAddress.NULL.asSegmentRestricted(1), - segment.ownerThread()); - liveness = liveness.handoff(scope); - return new WinVaList(segment, List.of(), liveness); + ((ResourceScopeImpl)scope).checkValidStateSlow(); + return new WinVaList(segment, scope); } @Override @@ -185,20 +170,14 @@ class WinVaList implements VaList { return segment.address(); } - @Override - public boolean isAlive() { - if (livenessCheck != null) - return livenessCheck.isAlive(); - return segment.isAlive(); - } + public static non-sealed class Builder implements VaList.Builder { - static class Builder implements VaList.Builder { - - private final SharedUtils.Allocator allocator; + private final ResourceScope scope; private final List args = new ArrayList<>(); - public Builder(SharedUtils.Allocator allocator) { - this.allocator = allocator; + public Builder(ResourceScope scope) { + ((ResourceScopeImpl)scope).checkValidStateSlow(); + this.scope = scope; } private Builder arg(Class carrier, MemoryLayout layout, Object value) { @@ -238,6 +217,7 @@ class WinVaList implements VaList { if (args.isEmpty()) { return EMPTY; } + SegmentAllocator allocator = SegmentAllocator.arenaAllocator(scope); MemorySegment segment = allocator.allocate(VA_SLOT_SIZE_BYTES * args.size()); List attachedSegments = new ArrayList<>(); attachedSegments.add(segment); @@ -267,12 +247,7 @@ class WinVaList implements VaList { cursor = cursor.asSlice(VA_SLOT_SIZE_BYTES); } - return new WinVaList(segment, attachedSegments, null); + return new WinVaList(segment, scope); } } - - private static MemorySegment handoffIfNeeded(MemorySegment segment, Thread thread) { - return segment.ownerThread() == thread ? - segment : segment.handoff(thread); - } } diff --git a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/Windowsx64Linker.java b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/Windowsx64Linker.java index 4978469c6c3..d40aabd1e7d 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/Windowsx64Linker.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/abi/x64/windows/Windowsx64Linker.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -24,12 +24,12 @@ */ package jdk.internal.foreign.abi.x64.windows; -import jdk.incubator.foreign.Addressable; import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.MemoryAddress; -import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; -import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.ResourceScope; +import jdk.internal.foreign.AbstractCLinker; +import jdk.internal.foreign.ResourceScopeImpl; import jdk.internal.foreign.abi.SharedUtils; import jdk.internal.foreign.abi.UpcallStubs; @@ -39,12 +39,10 @@ import java.lang.invoke.MethodType; import java.util.Objects; import java.util.function.Consumer; -import static jdk.internal.foreign.PlatformLayouts.*; - /** * ABI implementation based on Windows ABI AMD64 supplement v.0.99.6 */ -public class Windowsx64Linker implements CLinker { +public final class Windowsx64Linker extends AbstractCLinker { public static final int MAX_INTEGER_ARGUMENT_REGISTERS = 4; public static final int MAX_INTEGER_RETURN_REGISTERS = 1; @@ -65,8 +63,8 @@ public class Windowsx64Linker implements CLinker { MethodHandles.Lookup lookup = MethodHandles.lookup(); MH_unboxVaList = lookup.findVirtual(VaList.class, "address", MethodType.methodType(MemoryAddress.class)); - MH_boxVaList = lookup.findStatic(Windowsx64Linker.class, "newVaListOfAddress", - MethodType.methodType(VaList.class, MemoryAddress.class)); + MH_boxVaList = MethodHandles.insertArguments(lookup.findStatic(Windowsx64Linker.class, "newVaListOfAddress", + MethodType.methodType(VaList.class, MemoryAddress.class, ResourceScope.class)), 1, ResourceScope.globalScope()); } catch (ReflectiveOperationException e) { throw new ExceptionInInitializerError(e); } @@ -79,33 +77,37 @@ public class Windowsx64Linker implements CLinker { return instance; } - public static VaList newVaList(Consumer actions, SharedUtils.Allocator allocator) { - WinVaList.Builder builder = WinVaList.builder(allocator); + public static VaList newVaList(Consumer actions, ResourceScope scope) { + WinVaList.Builder builder = WinVaList.builder(scope); actions.accept(builder); return builder.build(); } @Override - public MethodHandle downcallHandle(Addressable symbol, MethodType type, FunctionDescriptor function) { - Objects.requireNonNull(symbol); + public final MethodHandle downcallHandle(MethodType type, FunctionDescriptor function) { Objects.requireNonNull(type); Objects.requireNonNull(function); MethodType llMt = SharedUtils.convertVaListCarriers(type, WinVaList.CARRIER); - MethodHandle handle = CallArranger.arrangeDowncall(symbol, llMt, function); + MethodHandle handle = CallArranger.arrangeDowncall(llMt, function); + if (!type.returnType().equals(MemorySegment.class)) { + // not returning segment, just insert a throwing allocator + handle = MethodHandles.insertArguments(handle, 1, SharedUtils.THROWING_ALLOCATOR); + } handle = SharedUtils.unboxVaLists(type, handle, MH_unboxVaList); return handle; } @Override - public MemorySegment upcallStub(MethodHandle target, FunctionDescriptor function) { + public final MemoryAddress upcallStub(MethodHandle target, FunctionDescriptor function, ResourceScope scope) { + Objects.requireNonNull(scope); Objects.requireNonNull(target); Objects.requireNonNull(function); target = SharedUtils.boxVaLists(target, MH_boxVaList); - return UpcallStubs.upcallAddress(CallArranger.arrangeUpcall(target, target.type(), function)); + return UpcallStubs.upcallAddress(CallArranger.arrangeUpcall(target, target.type(), function), (ResourceScopeImpl) scope); } - public static VaList newVaListOfAddress(MemoryAddress ma) { - return WinVaList.ofAddress(ma); + public static VaList newVaListOfAddress(MemoryAddress ma, ResourceScope scope) { + return WinVaList.ofAddress(ma, scope); } public static VaList emptyVaList() { diff --git a/test/hotspot/jtreg/gc/shenandoah/compiler/TestLinkToNativeRBP.java b/test/hotspot/jtreg/gc/shenandoah/compiler/TestLinkToNativeRBP.java index 67db793df3f..ab58cdec657 100644 --- a/test/hotspot/jtreg/gc/shenandoah/compiler/TestLinkToNativeRBP.java +++ b/test/hotspot/jtreg/gc/shenandoah/compiler/TestLinkToNativeRBP.java @@ -32,7 +32,7 @@ * * @modules jdk.incubator.foreign * - * @run main/othervm -Dforeign.restricted=permit -XX:+UnlockDiagnosticVMOptions + * @run main/othervm --enable-native-access=ALL-UNNAMED -XX:+UnlockDiagnosticVMOptions * -XX:+UseShenandoahGC -XX:ShenandoahGCHeuristics=aggressive TestLinkToNativeRBP * */ diff --git a/test/jdk/TEST.groups b/test/jdk/TEST.groups index bf5ae7e7eb4..dd5b089e9b0 100644 --- a/test/jdk/TEST.groups +++ b/test/jdk/TEST.groups @@ -333,7 +333,8 @@ jdk_svc = \ :svc_tools jdk_foreign = \ - java/foreign + java/foreign \ + -java/foreign/TestMatrix.java jdk_vector = \ jdk/incubator/vector diff --git a/test/jdk/java/foreign/CallGeneratorHelper.java b/test/jdk/java/foreign/CallGeneratorHelper.java index f9eced368d7..cc90a01dd9b 100644 --- a/test/jdk/java/foreign/CallGeneratorHelper.java +++ b/test/jdk/java/foreign/CallGeneratorHelper.java @@ -26,6 +26,8 @@ import jdk.incubator.foreign.GroupLayout; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; import jdk.incubator.foreign.ValueLayout; import java.lang.invoke.VarHandle; @@ -43,12 +45,12 @@ import static org.testng.Assert.*; public class CallGeneratorHelper extends NativeTestHelper { + static SegmentAllocator IMPLICIT_ALLOCATOR = (size, align) -> MemorySegment.allocateNative(size, align, ResourceScope.newImplicitScope()); + static final int MAX_FIELDS = 3; static final int MAX_PARAMS = 3; static final int CHUNK_SIZE = 600; - static int functions = 0; - public static void assertStructEquals(MemorySegment actual, MemorySegment expected, MemoryLayout layout) { assertEquals(actual.byteSize(), expected.byteSize()); GroupLayout g = (GroupLayout) layout; @@ -139,13 +141,13 @@ public class CallGeneratorHelper extends NativeTestHelper { MemoryLayout l = field.layout(); long padding = offset % l.bitSize(); if (padding != 0) { - layouts.add(MemoryLayout.ofPaddingBits(padding)); + layouts.add(MemoryLayout.paddingLayout(padding)); offset += padding; } layouts.add(l.withName("field" + offset)); offset += l.bitSize(); } - return MemoryLayout.ofStruct(layouts.toArray(new MemoryLayout[0])); + return MemoryLayout.structLayout(layouts.toArray(new MemoryLayout[0])); } else { return layout; } @@ -182,6 +184,7 @@ public class CallGeneratorHelper extends NativeTestHelper { @DataProvider(name = "functions") public static Object[][] functions() { + int functions = 0; List downcalls = new ArrayList<>(); for (Ret r : Ret.values()) { for (int i = 0; i <= MAX_PARAMS; i++) { @@ -193,16 +196,18 @@ public class CallGeneratorHelper extends NativeTestHelper { for (int j = 1; j <= MAX_FIELDS; j++) { for (List fields : StructFieldType.perms(j)) { String structCode = sigCode(fields); + int count = functions; int fCode = functions++ / CHUNK_SIZE; String fName = String.format("f%d_%s_%s_%s", fCode, retCode, sigCode, structCode); - downcalls.add(new Object[] { fName, r, ptypes, fields }); + downcalls.add(new Object[] { count, fName, r, ptypes, fields }); } } } else { String structCode = sigCode(List.of()); + int count = functions; int fCode = functions++ / CHUNK_SIZE; String fName = String.format("f%d_%s_%s_%s", fCode, retCode, sigCode, structCode); - downcalls.add(new Object[] { fName, r, ptypes, List.of() }); + downcalls.add(new Object[] { count, fName, r, ptypes, List.of() }); } } } @@ -357,19 +362,17 @@ public class CallGeneratorHelper extends NativeTestHelper { //helper methods @SuppressWarnings("unchecked") - static Object makeArg(MemoryLayout layout, List> checks, boolean check, List segments) throws ReflectiveOperationException { + static Object makeArg(MemoryLayout layout, List> checks, boolean check) throws ReflectiveOperationException { if (layout instanceof GroupLayout) { - MemorySegment segment = MemorySegment.allocateNative(layout); - initStruct(segment, (GroupLayout)layout, checks, check, segments); - segments.add(segment); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); + initStruct(segment, (GroupLayout)layout, checks, check); return segment; } else if (isPointer(layout)) { - MemorySegment segment = MemorySegment.allocateNative(1); - segments.add(segment); + MemorySegment segment = MemorySegment.allocateNative(1, ResourceScope.newImplicitScope()); if (check) { checks.add(o -> { try { - assertEquals((MemoryAddress)o, segment.address()); + assertEquals(o, segment.address()); } catch (Throwable ex) { throw new IllegalStateException(ex); } @@ -398,12 +401,12 @@ public class CallGeneratorHelper extends NativeTestHelper { } } - static void initStruct(MemorySegment str, GroupLayout g, List> checks, boolean check, List segments) throws ReflectiveOperationException { + static void initStruct(MemorySegment str, GroupLayout g, List> checks, boolean check) throws ReflectiveOperationException { for (MemoryLayout l : g.memberLayouts()) { if (l.isPadding()) continue; VarHandle accessor = g.varHandle(structFieldCarrier(l), MemoryLayout.PathElement.groupElement(l.name().get())); List> fieldsCheck = new ArrayList<>(); - Object value = makeArg(l, fieldsCheck, check, segments); + Object value = makeArg(l, fieldsCheck, check); if (isPointer(l)) { value = ((MemoryAddress)value).toRawLongValue(); } diff --git a/test/jdk/java/foreign/NativeTestHelper.java b/test/jdk/java/foreign/NativeTestHelper.java index 9eba210d330..9b57beac732 100644 --- a/test/jdk/java/foreign/NativeTestHelper.java +++ b/test/jdk/java/foreign/NativeTestHelper.java @@ -24,6 +24,9 @@ import jdk.incubator.foreign.CLinker; import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; public class NativeTestHelper { @@ -39,4 +42,38 @@ public class NativeTestHelper { public static boolean isPointer(MemoryLayout layout) { return kind(layout).isPointer(); } + + public static class NativeScope implements SegmentAllocator, AutoCloseable { + final ResourceScope resourceScope; + final ResourceScope.Handle scopeHandle; + final SegmentAllocator allocator; + + long allocatedBytes = 0; + + public NativeScope() { + this.resourceScope = ResourceScope.newConfinedScope(); + this.scopeHandle = resourceScope.acquire(); + this.allocator = SegmentAllocator.arenaAllocator(resourceScope); + } + + @Override + public MemorySegment allocate(long bytesSize, long bytesAlignment) { + allocatedBytes += bytesSize; + return allocator.allocate(bytesSize, bytesAlignment); + } + + public ResourceScope scope() { + return resourceScope; + } + + public long allocatedBytes() { + return allocatedBytes; + } + + @Override + public void close() { + resourceScope.release(scopeHandle); + resourceScope.close(); + } + } } diff --git a/test/jdk/java/foreign/SafeFunctionAccessTest.java b/test/jdk/java/foreign/SafeFunctionAccessTest.java new file mode 100644 index 00000000000..979a140bd41 --- /dev/null +++ b/test/jdk/java/foreign/SafeFunctionAccessTest.java @@ -0,0 +1,81 @@ +/* + * Copyright (c) 2021, 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 + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @run testng/othervm --enable-native-access=ALL-UNNAMED SafeFunctionAccessTest + */ + +import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.FunctionDescriptor; +import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodType; + +import org.testng.annotations.*; +import static org.testng.Assert.*; + +public class SafeFunctionAccessTest { + + LibraryLookup lookup = LibraryLookup.ofLibrary("SafeAccess"); + + static MemoryLayout POINT = MemoryLayout.structLayout( + CLinker.C_INT, CLinker.C_INT + ); + + @Test(expectedExceptions = IllegalStateException.class) + public void testClosedStruct() throws Throwable { + MemorySegment segment; + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + segment = MemorySegment.allocateNative(POINT, scope); + } + assertFalse(segment.scope().isAlive()); + MethodHandle handle = CLinker.getInstance().downcallHandle( + lookup.lookup("struct_func").get(), + MethodType.methodType(void.class, MemorySegment.class), + FunctionDescriptor.ofVoid(POINT)); + + handle.invokeExact(segment); + } + + @Test(expectedExceptions = IllegalStateException.class) + public void testClosedPointer() throws Throwable { + MemoryAddress address; + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + address = MemorySegment.allocateNative(POINT, scope).address(); + } + assertFalse(address.scope().isAlive()); + MethodHandle handle = CLinker.getInstance().downcallHandle( + lookup.lookup("addr_func").get(), + MethodType.methodType(void.class, MemoryAddress.class), + FunctionDescriptor.ofVoid(CLinker.C_POINTER)); + + handle.invokeExact(address); + } +} diff --git a/test/jdk/java/foreign/StdLibTest.java b/test/jdk/java/foreign/StdLibTest.java index 3dc6046e4da..945380e68e4 100644 --- a/test/jdk/java/foreign/StdLibTest.java +++ b/test/jdk/java/foreign/StdLibTest.java @@ -24,7 +24,7 @@ /* * @test * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" - * @run testng/othervm -Dforeign.restricted=permit StdLibTest + * @run testng/othervm --enable-native-access=ALL-UNNAMED StdLibTest */ import java.lang.invoke.MethodHandle; @@ -40,7 +40,8 @@ import java.util.Collections; import java.util.LinkedHashSet; import java.util.List; import java.util.Set; -import java.util.function.Consumer; +import java.util.function.BiConsumer; +import java.util.function.Function; import java.util.stream.Collectors; import java.util.stream.Stream; @@ -191,7 +192,7 @@ public class StdLibTest { MethodType.methodType(int.class, MemoryAddress.class, VaList.class), FunctionDescriptor.of(C_INT, C_POINTER, C_VA_LIST)); - final static LibraryLookup.Symbol printfAddr = lookup.lookup("printf").get(); + final static MemoryAddress printfAddr = lookup.lookup("printf").get(); final static FunctionDescriptor printfBase = FunctionDescriptor.of(C_INT, C_POINTER); @@ -206,38 +207,43 @@ public class StdLibTest { } String strcat(String s1, String s2) throws Throwable { - try (MemorySegment buf = MemorySegment.allocateNative(s1.length() + s2.length() + 1) ; - MemorySegment other = toCString(s2)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment buf = MemorySegment.allocateNative(s1.length() + s2.length() + 1, scope); + MemorySegment other = toCString(s2, scope); char[] chars = s1.toCharArray(); for (long i = 0 ; i < chars.length ; i++) { setByteAtOffset(buf, i, (byte)chars[(int)i]); } setByteAtOffset(buf, chars.length, (byte)'\0'); - return toJavaStringRestricted(((MemoryAddress)strcat.invokeExact(buf.address(), other.address()))); + return toJavaString(((MemoryAddress)strcat.invokeExact(buf.address(), other.address()))); } } int strcmp(String s1, String s2) throws Throwable { - try (MemorySegment ns1 = toCString(s1) ; - MemorySegment ns2 = toCString(s2)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment ns1 = toCString(s1, scope); + MemorySegment ns2 = toCString(s2, scope); return (int)strcmp.invokeExact(ns1.address(), ns2.address()); } } int puts(String msg) throws Throwable { - try (MemorySegment s = toCString(msg)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s = toCString(msg, scope); return (int)puts.invokeExact(s.address()); } } int strlen(String msg) throws Throwable { - try (MemorySegment s = toCString(msg)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s = toCString(msg, scope); return (int)strlen.invokeExact(s.address()); } } Tm gmtime(long arg) throws Throwable { - try (MemorySegment time = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment time = MemorySegment.allocateNative(8, scope); setLong(time, arg); return new Tm((MemoryAddress)gmtime.invokeExact(time.address())); } @@ -251,7 +257,7 @@ public class StdLibTest { static final long SIZE = 56; Tm(MemoryAddress addr) { - this.base = addr.asSegmentRestricted(SIZE); + this.base = addr.asSegment(SIZE, ResourceScope.globalScope()); } int sec() { @@ -286,15 +292,14 @@ public class StdLibTest { int[] qsort(int[] arr) throws Throwable { //init native array - try (NativeScope scope = NativeScope.unboundedScope()) { - - MemorySegment nativeArr = scope.allocateArray(C_INT, arr); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + SegmentAllocator allocator = SegmentAllocator.ofScope(scope); + MemorySegment nativeArr = allocator.allocateArray(C_INT, arr); //call qsort - MemorySegment qsortUpcallStub = abi.upcallStub(qsortCompar.bindTo(nativeArr), qsortComparFunction); - qsortUpcallStub = qsortUpcallStub.handoff(scope); + MemoryAddress qsortUpcallStub = abi.upcallStub(qsortCompar.bindTo(nativeArr), qsortComparFunction, scope); - qsort.invokeExact(nativeArr.address(), (long)arr.length, C_INT.byteSize(), qsortUpcallStub.address()); + qsort.invokeExact(nativeArr.address(), (long)arr.length, C_INT.byteSize(), qsortUpcallStub); //convert back to Java array return nativeArr.toIntArray(); @@ -311,23 +316,18 @@ public class StdLibTest { } int printf(String format, List args) throws Throwable { - try (MemorySegment formatStr = toCString(format)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment formatStr = toCString(format, scope); return (int)specializedPrintf(args).invokeExact(formatStr.address(), - args.stream().map(a -> a.nativeValue).toArray()); + args.stream().map(a -> a.nativeValue(scope)).toArray()); } } int vprintf(String format, List args) throws Throwable { - try (MemorySegment formatStr = toCString(format)) { - VaList vaList = VaList.make(b -> args.forEach(a -> a.accept(b))); - int result = (int)vprintf.invokeExact(formatStr.address(), vaList); - try { - vaList.close(); - } - catch (UnsupportedOperationException e) { - assertEquals(e.getMessage(), "Empty VaList"); - } - return result; + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment formatStr = toCString(format, scope); + VaList vaList = VaList.make(b -> args.forEach(a -> a.accept(b, scope)), scope); + return (int)vprintf.invokeExact(formatStr.address(), vaList); } } @@ -398,39 +398,43 @@ public class StdLibTest { .toArray(Object[][]::new); } - enum PrintfArg implements Consumer { + enum PrintfArg implements BiConsumer { - INTEGRAL(int.class, asVarArg(C_INT), "%d", 42, 42, VaList.Builder::vargFromInt), - STRING(MemoryAddress.class, asVarArg(C_POINTER), "%s", toCString("str").address(), "str", VaList.Builder::vargFromAddress), - CHAR(byte.class, asVarArg(C_CHAR), "%c", (byte) 'h', 'h', (builder, layout, value) -> builder.vargFromInt(C_INT, (int)value)), - DOUBLE(double.class, asVarArg(C_DOUBLE), "%.4f", 1.2345d, 1.2345d, VaList.Builder::vargFromDouble); + INTEGRAL(int.class, asVarArg(C_INT), "%d", scope -> 42, 42, VaList.Builder::vargFromInt), + STRING(MemoryAddress.class, asVarArg(C_POINTER), "%s", scope -> toCString("str", scope).address(), "str", VaList.Builder::vargFromAddress), + CHAR(byte.class, asVarArg(C_CHAR), "%c", scope -> (byte) 'h', 'h', (builder, layout, value) -> builder.vargFromInt(C_INT, (int)value)), + DOUBLE(double.class, asVarArg(C_DOUBLE), "%.4f", scope ->1.2345d, 1.2345d, VaList.Builder::vargFromDouble); final Class carrier; final ValueLayout layout; final String format; - final Object nativeValue; + final Function nativeValueFactory; final Object javaValue; @SuppressWarnings("rawtypes") final VaListBuilderCall builderCall; - PrintfArg(Class carrier, ValueLayout layout, String format, Z nativeValue, Object javaValue, VaListBuilderCall builderCall) { + PrintfArg(Class carrier, ValueLayout layout, String format, Function nativeValueFactory, Object javaValue, VaListBuilderCall builderCall) { this.carrier = carrier; this.layout = layout; this.format = format; - this.nativeValue = nativeValue; + this.nativeValueFactory = nativeValueFactory; this.javaValue = javaValue; this.builderCall = builderCall; } @Override @SuppressWarnings("unchecked") - public void accept(VaList.Builder builder) { - builderCall.build(builder, layout, nativeValue); + public void accept(VaList.Builder builder, ResourceScope scope) { + builderCall.build(builder, layout, nativeValueFactory.apply(scope)); } interface VaListBuilderCall { void build(VaList.Builder builder, ValueLayout layout, V value); } + + public Object nativeValue(ResourceScope scope) { + return nativeValueFactory.apply(scope); + } } static Set> perms(int count, Z[] arr) { diff --git a/test/jdk/java/foreign/TestAdaptVarHandles.java b/test/jdk/java/foreign/TestAdaptVarHandles.java index 456a4e1a282..de88626948a 100644 --- a/test/jdk/java/foreign/TestAdaptVarHandles.java +++ b/test/jdk/java/foreign/TestAdaptVarHandles.java @@ -30,11 +30,11 @@ * @run testng/othervm -Djava.lang.invoke.VarHandle.VAR_HANDLE_GUARDS=false -Djava.lang.invoke.VarHandle.VAR_HANDLE_IDENTITY_ADAPT=true -Xverify:all TestAdaptVarHandles */ -import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.ValueLayout; import org.testng.annotations.*; import static org.testng.Assert.*; @@ -86,7 +86,7 @@ public class TestAdaptVarHandles { } } - static final VarHandle intHandleIndexed = MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT) + static final VarHandle intHandleIndexed = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT) .varHandle(int.class, MemoryLayout.PathElement.sequenceElement()); static final VarHandle intHandle = MemoryLayouts.JAVA_INT.varHandle(int.class); @@ -96,7 +96,7 @@ public class TestAdaptVarHandles { @Test public void testFilterValue() throws Throwable { ValueLayout layout = MemoryLayouts.JAVA_INT; - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); VarHandle intHandle = layout.varHandle(int.class); VarHandle i2SHandle = MemoryHandles.filterValue(intHandle, S2I, I2S); i2SHandle.set(segment, "1"); @@ -115,7 +115,7 @@ public class TestAdaptVarHandles { @Test public void testFilterValueComposite() throws Throwable { ValueLayout layout = MemoryLayouts.JAVA_INT; - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); VarHandle intHandle = layout.varHandle(int.class); MethodHandle CTX_S2I = MethodHandles.dropArguments(S2I, 0, String.class, String.class); VarHandle i2SHandle = MemoryHandles.filterValue(intHandle, CTX_S2I, CTX_I2S); @@ -136,7 +136,7 @@ public class TestAdaptVarHandles { @Test public void testFilterValueLoose() throws Throwable { ValueLayout layout = MemoryLayouts.JAVA_INT; - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); VarHandle intHandle = layout.varHandle(int.class); VarHandle i2SHandle = MemoryHandles.filterValue(intHandle, O2I, I2O); i2SHandle.set(segment, "1"); @@ -204,7 +204,7 @@ public class TestAdaptVarHandles { @Test public void testFilterCoordinates() throws Throwable { ValueLayout layout = MemoryLayouts.JAVA_INT; - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); VarHandle intHandle_longIndex = MemoryHandles.filterCoordinates(intHandleIndexed, 0, BASE_ADDR, S2L); intHandle_longIndex.set(segment, "0", 1); int oldValue = (int)intHandle_longIndex.getAndAdd(segment, "0", 42); @@ -247,7 +247,7 @@ public class TestAdaptVarHandles { @Test public void testInsertCoordinates() throws Throwable { ValueLayout layout = MemoryLayouts.JAVA_INT; - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); VarHandle intHandle_longIndex = MemoryHandles.insertCoordinates(intHandleIndexed, 0, segment, 0L); intHandle_longIndex.set(1); int oldValue = (int)intHandle_longIndex.getAndAdd(42); @@ -285,7 +285,7 @@ public class TestAdaptVarHandles { @Test public void testPermuteCoordinates() throws Throwable { ValueLayout layout = MemoryLayouts.JAVA_INT; - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); VarHandle intHandle_swap = MemoryHandles.permuteCoordinates(intHandleIndexed, List.of(long.class, MemorySegment.class), 1, 0); intHandle_swap.set(0L, segment, 1); @@ -324,7 +324,7 @@ public class TestAdaptVarHandles { @Test public void testCollectCoordinates() throws Throwable { ValueLayout layout = MemoryLayouts.JAVA_INT; - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); VarHandle intHandle_sum = MemoryHandles.collectCoordinates(intHandleIndexed, 1, SUM_OFFSETS); intHandle_sum.set(segment, -2L, 2L, 1); int oldValue = (int)intHandle_sum.getAndAdd(segment, -2L, 2L, 42); @@ -367,7 +367,7 @@ public class TestAdaptVarHandles { @Test public void testDropCoordinates() throws Throwable { ValueLayout layout = MemoryLayouts.JAVA_INT; - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); VarHandle intHandle_dummy = MemoryHandles.dropCoordinates(intHandleIndexed, 1, float.class, String.class); intHandle_dummy.set(segment, 1f, "hello", 0L, 1); int oldValue = (int)intHandle_dummy.getAndAdd(segment, 1f, "hello", 0L, 42); diff --git a/test/jdk/java/foreign/TestAddressHandle.java b/test/jdk/java/foreign/TestAddressHandle.java index 13abca5420a..de9bde07a5b 100644 --- a/test/jdk/java/foreign/TestAddressHandle.java +++ b/test/jdk/java/foreign/TestAddressHandle.java @@ -62,7 +62,8 @@ public class TestAddressHandle { @Test(dataProvider = "addressHandles") public void testAddressHandle(VarHandle addrHandle, int byteSize) { VarHandle longHandle = MemoryLayouts.JAVA_LONG.varHandle(long.class); - try (MemorySegment segment = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(8, scope); MemorySegment target = ByteOrder.nativeOrder() == ByteOrder.BIG_ENDIAN ? segment.asSlice(8 - byteSize) : segment; @@ -78,7 +79,8 @@ public class TestAddressHandle { @Test(dataProvider = "addressHandles") public void testNull(VarHandle addrHandle, int byteSize) { VarHandle longHandle = MemoryLayouts.JAVA_LONG.varHandle(long.class); - try (MemorySegment segment = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(8, scope); longHandle.set(segment, 0L); MemoryAddress address = (MemoryAddress)addrHandle.get(segment); assertTrue(address == MemoryAddress.NULL); diff --git a/test/jdk/java/foreign/TestArrays.java b/test/jdk/java/foreign/TestArrays.java index b8be313ba4d..451beeb4544 100644 --- a/test/jdk/java/foreign/TestArrays.java +++ b/test/jdk/java/foreign/TestArrays.java @@ -24,7 +24,7 @@ /* * @test - * @run testng/othervm -Dforeign.restricted=permit TestArrays + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestArrays */ import jdk.incubator.foreign.MemoryAddress; @@ -32,6 +32,7 @@ import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayout.PathElement; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; import java.lang.invoke.VarHandle; @@ -42,36 +43,35 @@ import java.util.function.Function; import org.testng.annotations.*; -import static jdk.incubator.foreign.MemorySegment.READ; import static org.testng.Assert.*; public class TestArrays { - static SequenceLayout bytes = MemoryLayout.ofSequence(100, + static SequenceLayout bytes = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_BYTE ); - static SequenceLayout chars = MemoryLayout.ofSequence(100, + static SequenceLayout chars = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_CHAR ); - static SequenceLayout shorts = MemoryLayout.ofSequence(100, + static SequenceLayout shorts = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_SHORT ); - static SequenceLayout ints = MemoryLayout.ofSequence(100, + static SequenceLayout ints = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_INT ); - static SequenceLayout floats = MemoryLayout.ofSequence(100, + static SequenceLayout floats = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_FLOAT ); - static SequenceLayout longs = MemoryLayout.ofSequence(100, + static SequenceLayout longs = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_LONG ); - static SequenceLayout doubles = MemoryLayout.ofSequence(100, + static SequenceLayout doubles = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_DOUBLE ); @@ -101,27 +101,27 @@ public class TestArrays { @Test(dataProvider = "arrays") public void testArrays(Consumer init, Consumer checker, MemoryLayout layout) { - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { - init.accept(segment); - checker.accept(segment); - } + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); + init.accept(segment); + assertFalse(segment.isReadOnly()); + checker.accept(segment); } @Test(dataProvider = "elemLayouts", - expectedExceptions = UnsupportedOperationException.class) + expectedExceptions = IllegalStateException.class) public void testTooBigForArray(MemoryLayout layout, Function arrayFactory) { - MemoryLayout seq = MemoryLayout.ofSequence((Integer.MAX_VALUE * layout.byteSize()) + 1, layout); + MemoryLayout seq = MemoryLayout.sequenceLayout((Integer.MAX_VALUE * layout.byteSize()) + 1, layout); //do not really allocate here, as it's way too much memory - try (MemorySegment segment = MemoryAddress.NULL.asSegmentRestricted(seq.byteSize())) { - arrayFactory.apply(segment); - } + MemorySegment segment = MemoryAddress.NULL.asSegment(seq.byteSize(), ResourceScope.globalScope()); + arrayFactory.apply(segment); } @Test(dataProvider = "elemLayouts", - expectedExceptions = UnsupportedOperationException.class) + expectedExceptions = IllegalStateException.class) public void testBadSize(MemoryLayout layout, Function arrayFactory) { - if (layout.byteSize() == 1) throw new UnsupportedOperationException(); //make it fail - try (MemorySegment segment = MemorySegment.allocateNative(layout.byteSize() + 1)) { + if (layout.byteSize() == 1) throw new IllegalStateException(); //make it fail + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout.byteSize() + 1, layout.byteSize(), scope); arrayFactory.apply(segment); } } @@ -129,27 +129,11 @@ public class TestArrays { @Test(dataProvider = "elemLayouts", expectedExceptions = IllegalStateException.class) public void testArrayFromClosedSegment(MemoryLayout layout, Function arrayFactory) { - MemorySegment segment = MemorySegment.allocateNative(layout); - segment.close(); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newConfinedScope()); + segment.scope().close(); arrayFactory.apply(segment); } - @Test(dataProvider = "elemLayouts", - expectedExceptions = UnsupportedOperationException.class) - public void testArrayFromHeapSegmentWithoutAccess(MemoryLayout layout, Function arrayFactory) { - MemorySegment segment = MemorySegment.ofArray(new byte[(int)layout.byteSize()]); - segment = segment.withAccessModes(MemorySegment.ALL_ACCESS & ~READ); - arrayFactory.apply(segment); - } - - @Test(dataProvider = "elemLayouts", - expectedExceptions = UnsupportedOperationException.class) - public void testArrayFromNativeSegmentWithoutAccess(MemoryLayout layout, Function arrayFactory) { - try (MemorySegment segment = MemorySegment.allocateNative(layout).withAccessModes(MemorySegment.ALL_ACCESS & ~READ)) { - arrayFactory.apply(segment); - } - } - @DataProvider(name = "arrays") public Object[][] nativeAccessOps() { Consumer byteInitializer = diff --git a/test/jdk/java/foreign/TestByteBuffer.java b/test/jdk/java/foreign/TestByteBuffer.java index ef9c559b75a..cdc9addb381 100644 --- a/test/jdk/java/foreign/TestByteBuffer.java +++ b/test/jdk/java/foreign/TestByteBuffer.java @@ -25,17 +25,16 @@ * @test * @modules java.base/sun.nio.ch * jdk.incubator.foreign/jdk.internal.foreign - * @run testng/othervm -Dforeign.restricted=permit TestByteBuffer + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestByteBuffer */ - -import jdk.incubator.foreign.MappedMemorySegments; import jdk.incubator.foreign.MemoryAccess; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemorySegment; import jdk.incubator.foreign.MemoryLayout.PathElement; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; import java.io.File; @@ -43,6 +42,7 @@ import java.io.IOException; import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.VarHandle; +import java.lang.ref.Cleaner; import java.lang.ref.WeakReference; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; @@ -63,6 +63,7 @@ import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.StandardOpenOption; import java.util.ArrayList; +import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.Map; @@ -80,7 +81,7 @@ import jdk.internal.foreign.NativeMemorySegmentImpl; import org.testng.SkipException; import org.testng.annotations.*; import sun.nio.ch.DirectBuffer; -import static jdk.incubator.foreign.MemorySegment.*; + import static org.testng.Assert.*; public class TestByteBuffer { @@ -99,37 +100,37 @@ public class TestByteBuffer { } } - static SequenceLayout tuples = MemoryLayout.ofSequence(500, - MemoryLayout.ofStruct( + static SequenceLayout tuples = MemoryLayout.sequenceLayout(500, + MemoryLayout.structLayout( MemoryLayouts.BITS_32_BE.withName("index"), MemoryLayouts.BITS_32_BE.withName("value") )); - static SequenceLayout bytes = MemoryLayout.ofSequence(100, + static SequenceLayout bytes = MemoryLayout.sequenceLayout(100, MemoryLayouts.BITS_8_BE ); - static SequenceLayout chars = MemoryLayout.ofSequence(100, + static SequenceLayout chars = MemoryLayout.sequenceLayout(100, MemoryLayouts.BITS_16_BE ); - static SequenceLayout shorts = MemoryLayout.ofSequence(100, + static SequenceLayout shorts = MemoryLayout.sequenceLayout(100, MemoryLayouts.BITS_16_BE ); - static SequenceLayout ints = MemoryLayout.ofSequence(100, + static SequenceLayout ints = MemoryLayout.sequenceLayout(100, MemoryLayouts.BITS_32_BE ); - static SequenceLayout floats = MemoryLayout.ofSequence(100, + static SequenceLayout floats = MemoryLayout.sequenceLayout(100, MemoryLayouts.BITS_32_BE ); - static SequenceLayout longs = MemoryLayout.ofSequence(100, + static SequenceLayout longs = MemoryLayout.sequenceLayout(100, MemoryLayouts.BITS_64_BE ); - static SequenceLayout doubles = MemoryLayout.ofSequence(100, + static SequenceLayout doubles = MemoryLayout.sequenceLayout(100, MemoryLayouts.BITS_64_BE ); @@ -186,7 +187,8 @@ public class TestByteBuffer { @Test public void testOffheap() { - try (MemorySegment segment = MemorySegment.allocateNative(tuples)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(tuples, scope); initTuples(segment, tuples.elementCount().getAsLong()); ByteBuffer bb = segment.asByteBuffer(); @@ -230,14 +232,14 @@ public class TestByteBuffer { @Test public void testDefaultAccessModesMappedSegment() throws Throwable { - try (MemorySegment segment = MemorySegment.mapFile(tempPath, 0L, 8, FileChannel.MapMode.READ_WRITE)) { - assertTrue(segment.hasAccessModes(ALL_ACCESS)); - assertEquals(segment.accessModes(), ALL_ACCESS); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.mapFile(tempPath, 0L, 8, FileChannel.MapMode.READ_WRITE, scope); + assertFalse(segment.isReadOnly()); } - try (MemorySegment segment = MemorySegment.mapFile(tempPath, 0L, 8, FileChannel.MapMode.READ_ONLY)) { - assertTrue(segment.hasAccessModes(ALL_ACCESS & ~WRITE)); - assertEquals(segment.accessModes(), ALL_ACCESS & ~WRITE); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.mapFile(tempPath, 0L, 8, FileChannel.MapMode.READ_ONLY, scope); + assertTrue(segment.isReadOnly()); } } @@ -247,27 +249,29 @@ public class TestByteBuffer { f.createNewFile(); f.deleteOnExit(); - //write to channel - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, tuples.byteSize(), FileChannel.MapMode.READ_WRITE)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + //write to channel + MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, tuples.byteSize(), FileChannel.MapMode.READ_WRITE, scope); initTuples(segment, tuples.elementCount().getAsLong()); - MappedMemorySegments.force(segment); + segment.force(); } - //read from channel - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, tuples.byteSize(), FileChannel.MapMode.READ_ONLY)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + //read from channel + MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, tuples.byteSize(), FileChannel.MapMode.READ_ONLY, scope); checkTuples(segment, segment.asByteBuffer(), tuples.elementCount().getAsLong()); } } - @Test(dataProvider = "mappedOps", expectedExceptions = IllegalStateException.class) + @Test(dataProvider = "mappedOps", expectedExceptions = UnsupportedOperationException.class) public void testMappedSegmentOperations(MappedSegmentOp mappedBufferOp) throws Throwable { File f = new File("test3.out"); f.createNewFile(); f.deleteOnExit(); - MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, 8, FileChannel.MapMode.READ_WRITE); + MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, 8, FileChannel.MapMode.READ_WRITE, ResourceScope.newImplicitScope()); assertTrue(segment.isMapped()); - segment.close(); + segment.scope().close(); mappedBufferOp.apply(segment); } @@ -281,17 +285,19 @@ public class TestByteBuffer { // write one at a time for (int i = 0 ; i < tuples.byteSize() ; i += tupleLayout.byteSize()) { - //write to channel - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), i, tuples.byteSize(), FileChannel.MapMode.READ_WRITE)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + //write to channel + MemorySegment segment = MemorySegment.mapFile(f.toPath(), i, tuples.byteSize(), FileChannel.MapMode.READ_WRITE, scope); initTuples(segment, 1); - MappedMemorySegments.force(segment); + segment.force(); } } // check one at a time for (int i = 0 ; i < tuples.byteSize() ; i += tupleLayout.byteSize()) { - //read from channel - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, tuples.byteSize(), FileChannel.MapMode.READ_ONLY)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + //read from channel + MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, tuples.byteSize(), FileChannel.MapMode.READ_ONLY, scope); checkTuples(segment, segment.asByteBuffer(), 1); } } @@ -309,14 +315,15 @@ public class TestByteBuffer { f.createNewFile(); f.deleteOnExit(); - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0, LARGE_SIZE, FileChannel.MapMode.READ_WRITE)) { - MappedMemorySegments.isLoaded(segment); - MappedMemorySegments.load(segment); - MappedMemorySegments.isLoaded(segment); - MappedMemorySegments.force(segment); - MappedMemorySegments.isLoaded(segment); - MappedMemorySegments.unload(segment); - MappedMemorySegments.isLoaded(segment); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0, LARGE_SIZE, FileChannel.MapMode.READ_WRITE, scope); + segment.isLoaded(); + segment.load(); + segment.isLoaded(); + segment.force(); + segment.isLoaded(); + segment.unload(); + segment.isLoaded(); } } @@ -342,7 +349,8 @@ public class TestByteBuffer { @Test(dataProvider = "bufferOps") public void testScopedBuffer(Function bufferFactory, @NoInjection Method method, Object[] args) { Buffer bb; - try (MemorySegment segment = MemorySegment.allocateNative(bytes)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(bytes, scope); bb = bufferFactory.apply(segment.asByteBuffer()); } //outside of scope!! @@ -353,7 +361,7 @@ public class TestByteBuffer { Throwable cause = ex.getCause(); if (cause instanceof IllegalStateException) { //all get/set buffer operation should fail because of the scope check - assertTrue(ex.getCause().getMessage().contains("already closed")); + assertTrue(ex.getCause().getMessage().contains("Already closed")); } else { //all other exceptions were unexpected - fail fail("Unexpected exception", cause); @@ -367,7 +375,8 @@ public class TestByteBuffer { @Test(dataProvider = "bufferHandleOps") public void testScopedBufferAndVarHandle(VarHandle bufferHandle) { ByteBuffer bb; - try (MemorySegment segment = MemorySegment.allocateNative(bytes)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(bytes, scope); bb = segment.asByteBuffer(); for (Map.Entry e : varHandleMembers(bb, bufferHandle).entrySet()) { MethodHandle handle = e.getKey().bindTo(bufferHandle) @@ -389,7 +398,7 @@ public class TestByteBuffer { handle.invoke(e.getValue()); fail(); } catch (IllegalStateException ex) { - assertTrue(ex.getMessage().contains("already closed")); + assertTrue(ex.getMessage().contains("Already closed")); } catch (UnsupportedOperationException ex) { //skip } catch (Throwable ex) { @@ -400,7 +409,8 @@ public class TestByteBuffer { @Test(dataProvider = "bufferOps") public void testDirectBuffer(Function bufferFactory, @NoInjection Method method, Object[] args) { - try (MemorySegment segment = MemorySegment.allocateNative(bytes)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(bytes, scope); Buffer bb = bufferFactory.apply(segment.asByteBuffer()); assertTrue(bb.isDirect()); DirectBuffer directBuffer = ((DirectBuffer)bb); @@ -412,7 +422,8 @@ public class TestByteBuffer { @Test(dataProvider="resizeOps") public void testResizeOffheap(Consumer checker, Consumer initializer, SequenceLayout seq) { - try (MemorySegment segment = MemorySegment.allocateNative(seq)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(seq, scope); initializer.accept(segment); checker.accept(segment); } @@ -449,7 +460,8 @@ public class TestByteBuffer { @Test(dataProvider="resizeOps") public void testResizeRoundtripNative(Consumer checker, Consumer initializer, SequenceLayout seq) { - try (MemorySegment segment = MemorySegment.allocateNative(seq)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(seq, scope); initializer.accept(segment); MemorySegment second = MemorySegment.ofByteBuffer(segment.asByteBuffer()); checker.accept(second); @@ -459,18 +471,17 @@ public class TestByteBuffer { @Test(expectedExceptions = IllegalStateException.class) public void testBufferOnClosedScope() { MemorySegment leaked; - try (MemorySegment segment = MemorySegment.allocateNative(bytes)) { - leaked = segment; + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + leaked = MemorySegment.allocateNative(bytes, scope); } ByteBuffer byteBuffer = leaked.asByteBuffer(); // ok byteBuffer.get(); // should throw } - @Test(expectedExceptions = UnsupportedOperationException.class) + @Test(expectedExceptions = IllegalStateException.class) public void testTooBigForByteBuffer() { - try (MemorySegment segment = MemoryAddress.NULL.asSegmentRestricted(Integer.MAX_VALUE + 10L)) { - segment.asByteBuffer(); - } + MemorySegment segment = MemoryAddress.NULL.asSegment(Integer.MAX_VALUE + 10L, ResourceScope.globalScope()); + segment.asByteBuffer(); } @Test(expectedExceptions = IllegalArgumentException.class) @@ -478,7 +489,7 @@ public class TestByteBuffer { File f = new File("testNeg1.out"); f.createNewFile(); f.deleteOnExit(); - MemorySegment.mapFile(f.toPath(), 0L, -1, FileChannel.MapMode.READ_WRITE); + MemorySegment.mapFile(f.toPath(), 0L, -1, FileChannel.MapMode.READ_WRITE, ResourceScope.newImplicitScope()); } @Test(expectedExceptions = IllegalArgumentException.class) @@ -486,7 +497,7 @@ public class TestByteBuffer { File f = new File("testNeg2.out"); f.createNewFile(); f.deleteOnExit(); - MemorySegment.mapFile(f.toPath(), -1, 1, FileChannel.MapMode.READ_WRITE); + MemorySegment.mapFile(f.toPath(), -1, 1, FileChannel.MapMode.READ_WRITE, ResourceScope.newImplicitScope()); } @Test @@ -497,15 +508,17 @@ public class TestByteBuffer { int SIZE = Byte.MAX_VALUE; - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0, SIZE, FileChannel.MapMode.READ_WRITE)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0, SIZE, FileChannel.MapMode.READ_WRITE, scope); for (byte offset = 0; offset < SIZE; offset++) { MemoryAccess.setByteAtOffset(segment, offset, offset); } - MappedMemorySegments.force(segment); + segment.force(); } for (int offset = 0 ; offset < SIZE ; offset++) { - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), offset, SIZE - offset, FileChannel.MapMode.READ_ONLY)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.mapFile(f.toPath(), offset, SIZE - offset, FileChannel.MapMode.READ_ONLY, scope); assertEquals(MemoryAccess.getByte(segment), offset); } } @@ -517,39 +530,42 @@ public class TestByteBuffer { f.createNewFile(); f.deleteOnExit(); //RW - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, 0L, FileChannel.MapMode.READ_WRITE)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, 0L, FileChannel.MapMode.READ_WRITE, scope); assertEquals(segment.byteSize(), 0); assertEquals(segment.isMapped(), true); - assertTrue((segment.accessModes() & (READ | WRITE)) == (READ | WRITE)); - MappedMemorySegments.force(segment); - MappedMemorySegments.load(segment); - MappedMemorySegments.isLoaded(segment); - MappedMemorySegments.unload(segment); + assertFalse(segment.isReadOnly()); + segment.force(); + segment.load(); + segment.isLoaded(); + segment.unload(); } //RO - try (MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, 0L, FileChannel.MapMode.READ_ONLY)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.mapFile(f.toPath(), 0L, 0L, FileChannel.MapMode.READ_ONLY, scope); assertEquals(segment.byteSize(), 0); assertEquals(segment.isMapped(), true); - assertTrue((segment.accessModes() & (READ | WRITE)) == READ); - MappedMemorySegments.force(segment); - MappedMemorySegments.load(segment); - MappedMemorySegments.isLoaded(segment); - MappedMemorySegments.unload(segment); + assertTrue(segment.isReadOnly()); + segment.force(); + segment.load(); + segment.isLoaded(); + segment.unload(); } } @Test(expectedExceptions = IllegalArgumentException.class) public void testMapCustomPath() throws IOException { Path path = Path.of(URI.create("jrt:/")); - MemorySegment.mapFile(path, 0L, 0L, FileChannel.MapMode.READ_WRITE); + MemorySegment.mapFile(path, 0L, 0L, FileChannel.MapMode.READ_WRITE, ResourceScope.newImplicitScope()); } @Test(dataProvider="resizeOps") public void testCopyHeapToNative(Consumer checker, Consumer initializer, SequenceLayout seq) { checkByteArrayAlignment(seq.elementLayout()); int bytes = (int)seq.byteSize(); - try (MemorySegment nativeArray = MemorySegment.allocateNative(bytes); - MemorySegment heapArray = MemorySegment.ofArray(new byte[bytes])) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment nativeArray = MemorySegment.allocateNative(bytes, 1, scope); + MemorySegment heapArray = MemorySegment.ofArray(new byte[bytes]); initializer.accept(heapArray); nativeArray.copyFrom(heapArray); checker.accept(nativeArray); @@ -560,8 +576,9 @@ public class TestByteBuffer { public void testCopyNativeToHeap(Consumer checker, Consumer initializer, SequenceLayout seq) { checkByteArrayAlignment(seq.elementLayout()); int bytes = (int)seq.byteSize(); - try (MemorySegment nativeArray = MemorySegment.allocateNative(seq); - MemorySegment heapArray = MemorySegment.ofArray(new byte[bytes])) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment nativeArray = MemorySegment.allocateNative(seq, scope); + MemorySegment heapArray = MemorySegment.ofArray(new byte[bytes]); initializer.accept(nativeArray); heapArray.copyFrom(nativeArray); checker.accept(heapArray); @@ -571,111 +588,109 @@ public class TestByteBuffer { @Test public void testDefaultAccessModesOfBuffer() { ByteBuffer rwBuffer = ByteBuffer.wrap(new byte[4]); - try (MemorySegment segment = MemorySegment.ofByteBuffer(rwBuffer)) { - assertTrue(segment.hasAccessModes(ALL_ACCESS)); - assertEquals(segment.accessModes(), ALL_ACCESS); + { + MemorySegment segment = MemorySegment.ofByteBuffer(rwBuffer); + assertFalse(segment.isReadOnly()); } - ByteBuffer roBuffer = rwBuffer.asReadOnlyBuffer(); - try (MemorySegment segment = MemorySegment.ofByteBuffer(roBuffer)) { - assertTrue(segment.hasAccessModes(ALL_ACCESS & ~WRITE)); - assertEquals(segment.accessModes(), ALL_ACCESS & ~WRITE); + { + ByteBuffer roBuffer = rwBuffer.asReadOnlyBuffer(); + MemorySegment segment = MemorySegment.ofByteBuffer(roBuffer); + assertTrue(segment.isReadOnly()); } } @Test(dataProvider="bufferSources") public void testBufferToSegment(ByteBuffer bb, Predicate segmentChecker) { MemorySegment segment = MemorySegment.ofByteBuffer(bb); - assertEquals(segment.hasAccessModes(MemorySegment.WRITE), !bb.isReadOnly()); + assertEquals(segment.isReadOnly(), bb.isReadOnly()); assertTrue(segmentChecker.test(segment)); assertTrue(segmentChecker.test(segment.asSlice(0, segment.byteSize()))); - assertTrue(segmentChecker.test(segment.withAccessModes(MemorySegment.READ))); assertEquals(bb.capacity(), segment.byteSize()); //another round trip segment = MemorySegment.ofByteBuffer(segment.asByteBuffer()); - assertEquals(segment.hasAccessModes(MemorySegment.WRITE), !bb.isReadOnly()); + assertEquals(segment.isReadOnly(), bb.isReadOnly()); assertTrue(segmentChecker.test(segment)); assertTrue(segmentChecker.test(segment.asSlice(0, segment.byteSize()))); - assertTrue(segmentChecker.test(segment.withAccessModes(MemorySegment.READ))); assertEquals(bb.capacity(), segment.byteSize()); } @Test(dataProvider="bufferSources") public void bufferProperties(ByteBuffer bb, Predicate _unused) { - try (MemorySegment segment = MemorySegment.ofByteBuffer(bb)) { - ByteBuffer buffer = segment.asByteBuffer(); - assertEquals(buffer.position(), 0); - assertEquals(buffer.capacity(), segment.byteSize()); - assertEquals(buffer.limit(), segment.byteSize()); - } + MemorySegment segment = MemorySegment.ofByteBuffer(bb); + ByteBuffer buffer = segment.asByteBuffer(); + assertEquals(buffer.position(), 0); + assertEquals(buffer.capacity(), segment.byteSize()); + assertEquals(buffer.limit(), segment.byteSize()); } @Test public void testRoundTripAccess() { - try(MemorySegment ms = MemorySegment.allocateNative(4)) { - MemorySegment msNoAccess = ms.withAccessModes(MemorySegment.READ); // READ is required to make BB + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment ms = MemorySegment.allocateNative(4, 1, scope); + MemorySegment msNoAccess = ms.asReadOnly(); MemorySegment msRoundTrip = MemorySegment.ofByteBuffer(msNoAccess.asByteBuffer()); - assertEquals(msNoAccess.accessModes(), msRoundTrip.accessModes()); + assertEquals(msNoAccess.isReadOnly(), msRoundTrip.isReadOnly()); } } @Test(expectedExceptions = IllegalStateException.class) public void testDeadAccessOnClosedBufferSegment() { - MemorySegment s1 = MemorySegment.allocateNative(MemoryLayouts.JAVA_INT); + MemorySegment s1 = MemorySegment.allocateNative(MemoryLayouts.JAVA_INT, ResourceScope.newConfinedScope()); MemorySegment s2 = MemorySegment.ofByteBuffer(s1.asByteBuffer()); - s1.close(); // memory freed + // memory freed + s1.scope().close(); MemoryAccess.setInt(s2, 10); // Dead access! } - @Test(expectedExceptions = UnsupportedOperationException.class) - public void testIOOnSharedSegmentBuffer() throws IOException { + @Test(dataProvider = "allScopes") + public void testIOOnSegmentBuffer(Supplier scopeSupplier) throws IOException { File tmp = File.createTempFile("tmp", "txt"); tmp.deleteOnExit(); - try (FileChannel channel = FileChannel.open(tmp.toPath(), StandardOpenOption.WRITE)) { - MemorySegment segment = MemorySegment.allocateNative(10).share(); + ResourceScope scope; + try (FileChannel channel = FileChannel.open(tmp.toPath(), StandardOpenOption.READ, StandardOpenOption.WRITE) ; + ResourceScope scp = closeableScopeOrNull(scope = scopeSupplier.get())) { + MemorySegment segment = MemorySegment.allocateNative(10, 1, scope); for (int i = 0; i < 10; i++) { MemoryAccess.setByteAtOffset(segment, i, (byte) i); } ByteBuffer bb = segment.asByteBuffer(); - segment.close(); - channel.write(bb); + assertEquals(channel.write(bb), 10); + segment.fill((byte)0x00); + assertEquals(bb.clear(), ByteBuffer.wrap(new byte[] {0, 0, 0, 0, 0, 0, 0, 0, 0, 0})); + assertEquals(channel.position(0).read(bb.clear()), 10); + assertEquals(bb.flip(), ByteBuffer.wrap(new byte[] {0, 1, 2, 3, 4, 5, 6, 7, 8, 9})); } } - @Test(expectedExceptions = IllegalStateException.class) - public void testIOOnClosedConfinedSegmentBuffer() throws IOException { + static final Class ISE = IllegalStateException.class; + + @Test(dataProvider = "closeableScopes") + public void testIOOnClosedSegmentBuffer(Supplier scopeSupplier) throws IOException { File tmp = File.createTempFile("tmp", "txt"); tmp.deleteOnExit(); - try (FileChannel channel = FileChannel.open(tmp.toPath(), StandardOpenOption.WRITE)) { - MemorySegment segment = MemorySegment.allocateNative(10); + try (FileChannel channel = FileChannel.open(tmp.toPath(), StandardOpenOption.READ, StandardOpenOption.WRITE)) { + MemorySegment segment = MemorySegment.allocateNative(10, scopeSupplier.get()); for (int i = 0; i < 10; i++) { MemoryAccess.setByteAtOffset(segment, i, (byte) i); } ByteBuffer bb = segment.asByteBuffer(); - segment.close(); - channel.write(bb); + segment.scope().close(); + assertThrows(ISE, () -> channel.read(bb)); + assertThrows(ISE, () -> channel.read(new ByteBuffer[] {bb})); + assertThrows(ISE, () -> channel.read(new ByteBuffer[] {bb}, 0, 1)); + assertThrows(ISE, () -> channel.write(bb)); + assertThrows(ISE, () -> channel.write(new ByteBuffer[] {bb})); + assertThrows(ISE, () -> channel.write(new ByteBuffer[] {bb}, 0 ,1)); } } @Test - public void testIOOnConfinedSegment() throws IOException { - File tmp = File.createTempFile("tmp", "txt"); - tmp.deleteOnExit(); - try (FileChannel channel = FileChannel.open(tmp.toPath(), StandardOpenOption.WRITE)) { - MemorySegment segment = MemorySegment.allocateNative(10); - for (int i = 0; i < 10; i++) { - MemoryAccess.setByteAtOffset(segment, i, (byte) i); - } - ByteBuffer bb = segment.asByteBuffer(); - channel.write(bb); - } - } - - @Test(dataProvider="segments") - public void buffersAndArraysFromSlices(Supplier segmentSupplier) { - try (MemorySegment segment = segmentSupplier.get()) { + public void buffersAndArraysFromSlices() { + try (ResourceScope scope = ResourceScope.newSharedScope()) { + MemorySegment segment = MemorySegment.allocateNative(16, scope); int newSize = 8; var slice = segment.asSlice(4, newSize); @@ -690,9 +705,10 @@ public class TestByteBuffer { } } - @Test(dataProvider="segments") - public void viewsFromSharedSegment(Supplier segmentSupplier) { - try (MemorySegment segment = segmentSupplier.get().share()) { + @Test + public void viewsFromSharedSegment() { + try (ResourceScope scope = ResourceScope.newSharedScope()) { + MemorySegment segment = MemorySegment.allocateNative(16, scope); var byteBuffer = segment.asByteBuffer(); byteBuffer.asReadOnlyBuffer(); byteBuffer.slice(0, 8); @@ -702,11 +718,42 @@ public class TestByteBuffer { @DataProvider(name = "segments") public static Object[][] segments() throws Throwable { return new Object[][] { - { (Supplier) () -> MemorySegment.allocateNative(16) }, + { (Supplier) () -> MemorySegment.allocateNative(16, ResourceScope.newImplicitScope()) }, { (Supplier) () -> MemorySegment.ofArray(new byte[16]) } }; } + @DataProvider(name = "closeableScopes") + public static Object[][] closeableScopes() { + return new Object[][] { + { (Supplier) () -> ResourceScope.newSharedScope() }, + { (Supplier) () -> ResourceScope.newConfinedScope() }, + { (Supplier) () -> ResourceScope.newSharedScope(Cleaner.create()) }, + { (Supplier) () -> ResourceScope.newConfinedScope(Cleaner.create()) } + }; + } + + @DataProvider(name = "implicitScopes") + public static Object[][] implicitScopes() { + return new Object[][] { + { (Supplier) ResourceScope::newImplicitScope }, + { (Supplier) ResourceScope::globalScope }, + }; + } + + @DataProvider(name = "allScopes") + public static Object[][] allScopes() { + return Stream.of(implicitScopes(), closeableScopes()) + .flatMap(Arrays::stream) + .toArray(Object[][]::new); + } + + static ResourceScope closeableScopeOrNull(ResourceScope scope) { + if (scope.isImplicit()) + return null; + return scope; + } + @DataProvider(name = "bufferOps") public static Object[][] bufferOps() throws Throwable { List args = new ArrayList<>(); @@ -891,10 +938,10 @@ public class TestByteBuffer { } enum MappedSegmentOp { - LOAD(MappedMemorySegments::load), - UNLOAD(MappedMemorySegments::unload), - IS_LOADED(MappedMemorySegments::isLoaded), - FORCE(MappedMemorySegments::force), + LOAD(MemorySegment::load), + UNLOAD(MemorySegment::unload), + IS_LOADED(MemorySegment::isLoaded), + FORCE(MemorySegment::force), BUFFER_LOAD(m -> ((MappedByteBuffer)m.asByteBuffer()).load()), BUFFER_IS_LOADED(m -> ((MappedByteBuffer)m.asByteBuffer()).isLoaded()), BUFFER_FORCE(m -> ((MappedByteBuffer)m.asByteBuffer()).force()); diff --git a/test/jdk/java/foreign/TestCleaner.java b/test/jdk/java/foreign/TestCleaner.java deleted file mode 100644 index 8ec8e9e0c72..00000000000 --- a/test/jdk/java/foreign/TestCleaner.java +++ /dev/null @@ -1,185 +0,0 @@ -/* - * 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 java.base/jdk.internal.ref - * jdk.incubator.foreign/jdk.incubator.foreign - * @run testng/othervm -Dforeign.restricted=permit TestCleaner - */ - -import jdk.incubator.foreign.MemorySegment; -import java.lang.ref.Cleaner; -import jdk.internal.ref.CleanerFactory; - -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; -import static org.testng.Assert.*; - -import java.util.ArrayList; -import java.util.List; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.function.Function; -import java.util.function.Supplier; -import java.util.stream.Stream; - -public class TestCleaner { - - static class SegmentState { - private AtomicInteger cleanupCalls = new AtomicInteger(0); - - void cleanup() { - cleanupCalls.incrementAndGet(); - } - - int cleanupCalls() { - return cleanupCalls.get(); - } - } - - @Test(dataProvider = "cleaners") - public void testAtMostOnce(RegisterKind registerKind, Supplier cleanerFactory, SegmentFunction segmentFunction) { - SegmentState segmentState = new SegmentState(); - MemorySegment root = MemorySegment.allocateNative(10).share(); - MemorySegment segment = root.address().asSegmentRestricted(10, () -> { - root.close(); - segmentState.cleanup(); - }, null); - - if (registerKind == RegisterKind.BEFORE) { - // register cleaners before - segment = segment.registerCleaner(cleanerFactory.get()); - } - - kickGCAndCheck(segmentState, segment); - - segment = segmentFunction.apply(segment); - - kickGCAndCheck(segmentState, segment); - - if (segment.isAlive() && registerKind == RegisterKind.AFTER) { - // register cleaners after - segment = segment.registerCleaner(cleanerFactory.get()); - } - - kickGCAndCheck(segmentState, segment); - segment = null; - while (segmentState.cleanupCalls() == 0) { - byte[] b = new byte[100]; - System.gc(); - try { - Thread.sleep(10); - } catch (InterruptedException ex) { - throw new AssertionError(ex); - } - } - assertEquals(segmentState.cleanupCalls(), 1); - } - - private void kickGCAndCheck(SegmentState segmentState, MemorySegment segment) { - for (int i = 0 ; i < 100 ; i++) { - byte[] b = new byte[100]; - System.gc(); - Thread.onSpinWait(); - } - //check that cleanup has not been called by any cleaner yet! - assertEquals(segmentState.cleanupCalls(), segment.isAlive() ? 0 : 1); - } - - @Test(dataProvider = "segmentFunctions") - public void testBadDoubleRegister(Supplier cleanerFactory, SegmentFunction segmentFunction) { - MemorySegment segment = MemorySegment.allocateNative(10); - segment = segment.registerCleaner(cleanerFactory.get()); - segment = segmentFunction.apply(segment); - try { - segment.registerCleaner(cleanerFactory.get()); // error here! - fail(); - } catch (IllegalStateException ex) { - if (!segment.isAlive()) { - assertTrue(ex.getMessage().contains("This segment is already closed")); - } else { - assertTrue(ex.getMessage().contains("Already registered with a cleaner")); - } - } - } - - enum SegmentFunction implements Function { - IDENTITY(Function.identity()), - CLOSE(s -> { s.close(); return s; }), - SHARE(s -> { return s.share(); }); - - private final Function segmentFunction; - - SegmentFunction(Function segmentFunction) { - this.segmentFunction = segmentFunction; - } - - @Override - public MemorySegment apply(MemorySegment segment) { - return segmentFunction.apply(segment); - } - } - - @DataProvider - static Object[][] segmentFunctions() { - Supplier[] cleaners = { - (Supplier)Cleaner::create, - (Supplier)CleanerFactory::cleaner - }; - - SegmentFunction[] segmentFunctions = SegmentFunction.values(); - Object[][] data = new Object[cleaners.length * segmentFunctions.length][3]; - - for (int cleaner = 0 ; cleaner < cleaners.length ; cleaner++) { - for (int segmentFunction = 0 ; segmentFunction < segmentFunctions.length ; segmentFunction++) { - data[cleaner + (cleaners.length * segmentFunction)] = - new Object[] { cleaners[cleaner], segmentFunctions[segmentFunction] }; - } - } - - return data; - } - - enum RegisterKind { - BEFORE, - AFTER; - } - - @DataProvider - static Object[][] cleaners() { - Supplier[] cleaners = { - (Supplier)Cleaner::create, - (Supplier)CleanerFactory::cleaner - }; - - List data = new ArrayList<>(); - for (RegisterKind kind : RegisterKind.values()) { - for (Object cleaner : cleaners) { - for (SegmentFunction segmentFunction : SegmentFunction.values()) { - data.add(new Object[] {kind, cleaner, segmentFunction}); - } - } - } - return data.toArray(Object[][]::new); - } -} diff --git a/test/jdk/java/foreign/TestCondy.java b/test/jdk/java/foreign/TestCondy.java index aa4eb20d803..0f262214799 100644 --- a/test/jdk/java/foreign/TestCondy.java +++ b/test/jdk/java/foreign/TestCondy.java @@ -69,12 +69,12 @@ public class TestCondy { testValues.addAll(Arrays.asList(constants)); - testValues.add(MemoryLayout.ofStruct(constants)); - testValues.add(MemoryLayout.ofUnion(constants)); + testValues.add(MemoryLayout.structLayout(constants)); + testValues.add(MemoryLayout.unionLayout(constants)); for (MemoryLayout ml : constants) { - testValues.add(MemoryLayout.ofSequence(ml)); - testValues.add(MemoryLayout.ofSequence(10, ml)); + testValues.add(MemoryLayout.sequenceLayout(ml)); + testValues.add(MemoryLayout.sequenceLayout(10, ml)); } testValues.add(FunctionDescriptor.ofVoid(constants)); diff --git a/test/jdk/java/foreign/TestDowncall.java b/test/jdk/java/foreign/TestDowncall.java index 4b0d344930f..40cc6cb45fa 100644 --- a/test/jdk/java/foreign/TestDowncall.java +++ b/test/jdk/java/foreign/TestDowncall.java @@ -29,30 +29,14 @@ * @build NativeTestHelper CallGeneratorHelper TestDowncall * * @run testng/othervm - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false - * TestDowncall - * @run testng/othervm - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false - * TestDowncall - * @run testng/othervm - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true - * TestDowncall - * @run testng/othervm - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * --enable-native-access=ALL-UNNAMED * TestDowncall */ import jdk.incubator.foreign.CLinker; import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import java.lang.invoke.MethodHandle; @@ -62,27 +46,72 @@ import java.util.List; import java.util.function.Consumer; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.SegmentAllocator; import org.testng.annotations.*; +import static org.testng.Assert.*; public class TestDowncall extends CallGeneratorHelper { static LibraryLookup lib = LibraryLookup.ofLibrary("TestDowncall"); static CLinker abi = CLinker.getInstance(); + @Test(dataProvider="functions", dataProviderClass=CallGeneratorHelper.class) + public void testDowncall(int count, String fName, Ret ret, List paramTypes, List fields) throws Throwable { + List> checks = new ArrayList<>(); + MemoryAddress addr = lib.lookup(fName).get(); + MethodType mt = methodType(ret, paramTypes, fields); + FunctionDescriptor descriptor = function(ret, paramTypes, fields); + Object[] args = makeArgs(paramTypes, fields, checks); + try (NativeScope scope = new NativeScope()) { + boolean needsScope = mt.returnType().equals(MemorySegment.class); + Object res = doCall(addr, scope, mt, descriptor, args); + if (ret == Ret.NON_VOID) { + checks.forEach(c -> c.accept(res)); + if (needsScope) { + // check that return struct has indeed been allocated in the native scope + assertEquals(((MemorySegment) res).scope(), scope.scope()); + assertEquals(scope.allocatedBytes(), descriptor.returnLayout().get().byteSize()); + } else { + // if here, there should be no allocation through the scope! + assertEquals(scope.allocatedBytes(), 0L); + } + } else { + // if here, there should be no allocation through the scope! + assertEquals(scope.allocatedBytes(), 0L); + } + } + } @Test(dataProvider="functions", dataProviderClass=CallGeneratorHelper.class) - public void testDowncall(String fName, Ret ret, List paramTypes, List fields) throws Throwable { + public void testDowncallNoScope(int count, String fName, Ret ret, List paramTypes, List fields) throws Throwable { List> checks = new ArrayList<>(); - List segments = new ArrayList<>(); - LibraryLookup.Symbol addr = lib.lookup(fName).get(); - MethodHandle mh = abi.downcallHandle(addr, methodType(ret, paramTypes, fields), function(ret, paramTypes, fields)); - Object[] args = makeArgs(paramTypes, fields, checks, segments); - mh = mh.asSpreader(Object[].class, paramTypes.size()); - Object res = mh.invoke(args); + MemoryAddress addr = lib.lookup(fName).get(); + MethodType mt = methodType(ret, paramTypes, fields); + FunctionDescriptor descriptor = function(ret, paramTypes, fields); + Object[] args = makeArgs(paramTypes, fields, checks); + boolean needsScope = mt.returnType().equals(MemorySegment.class); + if (count % 100 == 0) { + System.gc(); + } + Object res = doCall(addr, IMPLICIT_ALLOCATOR, mt, descriptor, args); if (ret == Ret.NON_VOID) { checks.forEach(c -> c.accept(res)); + if (needsScope) { + // check that return struct has indeed been allocated in the default scope + try { + ((MemorySegment)res).scope().close(); // should throw + fail("Expected exception!"); + } catch (UnsupportedOperationException ex) { + // ok + } + } } - segments.forEach(MemorySegment::close); + } + + Object doCall(MemoryAddress addr, SegmentAllocator allocator, MethodType type, FunctionDescriptor descriptor, Object[] args) throws Throwable { + MethodHandle mh = abi.downcallHandle(addr, allocator, type, descriptor); + Object res = mh.invokeWithArguments(args); + return res; } static MethodType methodType(Ret ret, List params, List fields) { @@ -101,10 +130,10 @@ public class TestDowncall extends CallGeneratorHelper { FunctionDescriptor.of(paramLayouts[0], paramLayouts); } - static Object[] makeArgs(List params, List fields, List> checks, List segments) throws ReflectiveOperationException { + static Object[] makeArgs(List params, List fields, List> checks) throws ReflectiveOperationException { Object[] args = new Object[params.size()]; for (int i = 0 ; i < params.size() ; i++) { - args[i] = makeArg(params.get(i).layout(fields), checks, i == 0, segments); + args[i] = makeArg(params.get(i).layout(fields), checks, i == 0); } return args; } diff --git a/test/jdk/java/foreign/TestFree.java b/test/jdk/java/foreign/TestFree.java index a8a32c50962..ad26e1eb571 100644 --- a/test/jdk/java/foreign/TestFree.java +++ b/test/jdk/java/foreign/TestFree.java @@ -26,28 +26,30 @@ * @test * @bug 8248421 * @summary SystemCLinker should have a way to free memory allocated outside Java - * @run testng/othervm -Dforeign.restricted=permit TestFree + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestFree */ import jdk.incubator.foreign.MemoryAccess; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; + import static jdk.incubator.foreign.CLinker.*; import static org.testng.Assert.assertEquals; public class TestFree { - private static MemorySegment asArrayRestricted(MemoryAddress addr, MemoryLayout layout, int numElements) { - return addr.asSegmentRestricted(numElements * layout.byteSize()); + private static MemorySegment asArray(MemoryAddress addr, MemoryLayout layout, int numElements) { + return addr.asSegment(numElements * layout.byteSize(), ResourceScope.globalScope()); } public void test() throws Throwable { String str = "hello world"; - MemoryAddress addr = allocateMemoryRestricted(str.length() + 1); - MemorySegment seg = asArrayRestricted(addr, C_CHAR, str.length() + 1); + MemoryAddress addr = allocateMemory(str.length() + 1); + MemorySegment seg = asArray(addr, C_CHAR, str.length() + 1); seg.copyFrom(MemorySegment.ofArray(str.getBytes())); MemoryAccess.setByteAtOffset(seg, str.length(), (byte)0); assertEquals(str, toJavaString(seg)); - freeMemoryRestricted(addr); + freeMemory(addr); } } diff --git a/test/jdk/java/foreign/TestHandshake.java b/test/jdk/java/foreign/TestHandshake.java index fa5f79dfe7b..ee9b7394594 100644 --- a/test/jdk/java/foreign/TestHandshake.java +++ b/test/jdk/java/foreign/TestHandshake.java @@ -42,8 +42,10 @@ import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.ThreadLocalRandom; import java.util.concurrent.TimeUnit; +import java.util.concurrent.atomic.AtomicBoolean; import java.util.concurrent.atomic.AtomicLong; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; import static org.testng.Assert.*; @@ -53,30 +55,33 @@ public class TestHandshake { static final int ITERATIONS = 5; static final int SEGMENT_SIZE = 1_000_000; static final int MAX_DELAY_MILLIS = 500; - static final int MAX_EXECUTOR_WAIT_SECONDS = 10; + static final int MAX_EXECUTOR_WAIT_SECONDS = 20; static final int MAX_THREAD_SPIN_WAIT_MILLIS = 200; static final int NUM_ACCESSORS = Math.min(10, Runtime.getRuntime().availableProcessors()); static final AtomicLong start = new AtomicLong(); + static final AtomicBoolean started = new AtomicBoolean(); @Test(dataProvider = "accessors") public void testHandshake(String testName, AccessorFactory accessorFactory) throws InterruptedException { for (int it = 0 ; it < ITERATIONS ; it++) { - MemorySegment segment = MemorySegment.allocateNative(SEGMENT_SIZE).share(); + ResourceScope scope = ResourceScope.newSharedScope(); + MemorySegment segment = MemorySegment.allocateNative(SEGMENT_SIZE, 1, scope); System.out.println("ITERATION " + it); ExecutorService accessExecutor = Executors.newCachedThreadPool(); start.set(System.currentTimeMillis()); + started.set(false); for (int i = 0; i < NUM_ACCESSORS ; i++) { accessExecutor.execute(accessorFactory.make(i, segment)); } int delay = ThreadLocalRandom.current().nextInt(MAX_DELAY_MILLIS); System.out.println("Starting handshaker with delay set to " + delay + " millis"); Thread.sleep(delay); - accessExecutor.execute(new Handshaker(segment)); + accessExecutor.execute(new Handshaker(scope)); accessExecutor.shutdown(); assertTrue(accessExecutor.awaitTermination(MAX_EXECUTOR_WAIT_SECONDS, TimeUnit.SECONDS)); - assertTrue(!segment.isAlive()); + assertTrue(!segment.scope().isAlive()); } } @@ -91,27 +96,25 @@ public class TestHandshake { @Override public final void run() { - outer: while (segment.isAlive()) { + start("\"Accessor #\" + id"); + outer: while (segment.scope().isAlive()) { try { doAccess(); } catch (IllegalStateException ex) { long delay = System.currentTimeMillis() - start.get(); - System.out.println("Accessor #" + id + " suspending - delay (ms): " + delay); + System.out.println("Accessor #" + id + " suspending - elapsed (ms): " + delay); backoff(); delay = System.currentTimeMillis() - start.get(); - System.out.println("Accessor #" + id + " resuming - delay (ms): " + delay); + System.out.println("Accessor #" + id + " resuming - elapsed (ms): " + delay); continue outer; } } long delay = System.currentTimeMillis() - start.get(); - System.out.println("Accessor #" + id + " terminated - delay (ms): " + delay); - cleanup(); + System.out.println("Accessor #" + id + " terminated - elapsed (ms): " + delay); } abstract void doAccess(); - void cleanup() {} - private void backoff() { try { Thread.sleep(ThreadLocalRandom.current().nextInt(MAX_THREAD_SPIN_WAIT_MILLIS)); @@ -121,6 +124,13 @@ public class TestHandshake { } } + static void start(String name) { + if (started.compareAndSet(false, true)) { + long delay = System.currentTimeMillis() - start.get(); + System.out.println("Started first thread: " + name + " ; elapsed (ms): " + delay); + } + } + static abstract class AbstractBufferAccessor extends AbstractSegmentAccessor { final ByteBuffer bb; @@ -181,7 +191,7 @@ public class TestHandshake { SegmentMismatchAccessor(int id, MemorySegment segment) { super(id, segment); - this.copy = MemorySegment.allocateNative(SEGMENT_SIZE).share(); + this.copy = MemorySegment.allocateNative(SEGMENT_SIZE, 1, segment.scope()); copy.copyFrom(segment); MemoryAccess.setByteAtOffset(copy, ThreadLocalRandom.current().nextInt(SEGMENT_SIZE), (byte)42); } @@ -190,11 +200,6 @@ public class TestHandshake { public void doAccess() { segment.mismatch(copy); } - - @Override - void cleanup() { - copy.close(); - } } static class BufferAccessor extends AbstractBufferAccessor { @@ -231,24 +236,25 @@ public class TestHandshake { static class Handshaker implements Runnable { - final MemorySegment segment; + final ResourceScope scope; - Handshaker(MemorySegment segment) { - this.segment = segment; + Handshaker(ResourceScope scope) { + this.scope = scope; } @Override public void run() { + start("Handshaker"); while (true) { try { - segment.close(); + scope.close(); break; } catch (IllegalStateException ex) { Thread.onSpinWait(); } } long delay = System.currentTimeMillis() - start.get(); - System.out.println("Segment closed - delay (ms): " + delay); + System.out.println("Segment closed - elapsed (ms): " + delay); } } diff --git a/test/jdk/java/foreign/TestIllegalLink.java b/test/jdk/java/foreign/TestIllegalLink.java index 8e7795168aa..46a77d6668c 100644 --- a/test/jdk/java/foreign/TestIllegalLink.java +++ b/test/jdk/java/foreign/TestIllegalLink.java @@ -25,7 +25,7 @@ /* * @test * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" - * @run testng/othervm -Dforeign.restricted=permit TestIllegalLink + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestIllegalLink */ import jdk.incubator.foreign.CLinker; @@ -73,7 +73,7 @@ public class TestIllegalLink { }, { MethodType.methodType(void.class, int.class), - FunctionDescriptor.ofVoid(MemoryLayout.ofPaddingBits(32)), + FunctionDescriptor.ofVoid(MemoryLayout.paddingLayout(32)), "Expected a ValueLayout" }, { @@ -88,7 +88,7 @@ public class TestIllegalLink { }, { MethodType.methodType(void.class, MemoryAddress.class), - FunctionDescriptor.ofVoid(MemoryLayout.ofPaddingBits(64)), + FunctionDescriptor.ofVoid(MemoryLayout.paddingLayout(64)), "Expected a ValueLayout" }, { diff --git a/test/jdk/java/foreign/TestIntrinsics.java b/test/jdk/java/foreign/TestIntrinsics.java index 01bf3c9212a..7f5027d8885 100644 --- a/test/jdk/java/foreign/TestIntrinsics.java +++ b/test/jdk/java/foreign/TestIntrinsics.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -27,7 +27,7 @@ * @run testng/othervm * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true - * -Dforeign.restricted=permit + * --enable-native-access=ALL-UNNAMED * -Xbatch * TestIntrinsics */ @@ -41,6 +41,7 @@ import java.lang.invoke.MethodType; import java.util.ArrayList; import java.util.List; +import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import org.testng.annotations.*; @@ -54,14 +55,8 @@ public class TestIntrinsics { static final CLinker abi = CLinker.getInstance(); static final LibraryLookup lookup = LibraryLookup.ofLibrary("Intrinsics"); - private static MethodHandle linkIndentity(String name, Class carrier, MemoryLayout layout, boolean trivial) { - LibraryLookup.Symbol ma = lookup.lookup(name).orElseThrow(); - MethodType mt = methodType(carrier, carrier); - FunctionDescriptor fd = FunctionDescriptor.of(layout, layout); - if (trivial) { - fd = fd.withAttribute(TRIVIAL_ATTRIBUTE_NAME, true); - } - return abi.downcallHandle(ma, mt, fd); + private interface RunnableX { + void run() throws Throwable; } @Test(dataProvider = "tests") @@ -71,45 +66,50 @@ public class TestIntrinsics { } } - private interface RunnableX { - void run() throws Throwable; - } - - private interface AddTest { - void add(MethodHandle target, Object expectedResult, Object... args); - } - @DataProvider public Object[][] tests() { List testsList = new ArrayList<>(); + + interface AddTest { + void add(MethodHandle target, Object expectedResult, Object... args); + } + AddTest tests = (mh, expectedResult, args) -> testsList.add(() -> { Object actual = mh.invokeWithArguments(args); assertEquals(actual, expectedResult); }); + interface AddIdentity { + void add(String name, Class carrier, MemoryLayout layout, Object arg); + } + + AddIdentity addIdentity = (name, carrier, layout, arg) -> { + MemoryAddress ma = lookup.lookup(name).orElseThrow(); + MethodType mt = methodType(carrier, carrier); + FunctionDescriptor fd = FunctionDescriptor.of(layout, layout); + + tests.add(abi.downcallHandle(ma, mt, fd), arg, arg); + tests.add(abi.downcallHandle(ma, mt, fd.withAttribute(TRIVIAL_ATTRIBUTE_NAME, true)), arg, arg); + tests.add(abi.downcallHandle(mt, fd), arg, ma, arg); + }; + { // empty - LibraryLookup.Symbol ma = lookup.lookup("empty").orElseThrow(); + MemoryAddress ma = lookup.lookup("empty").orElseThrow(); MethodType mt = methodType(void.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid(); tests.add(abi.downcallHandle(ma, mt, fd), null); tests.add(abi.downcallHandle(ma, mt, fd.withAttribute(TRIVIAL_ATTRIBUTE_NAME, true)), null); } - tests.add(linkIndentity("identity_char", byte.class, C_CHAR, false), (byte) 10, (byte) 10); - tests.add(linkIndentity("identity_char", byte.class, C_CHAR, true), (byte) 10, (byte) 10); - tests.add(linkIndentity("identity_short", short.class, C_SHORT, false), (short) 10, (short) 10); - tests.add(linkIndentity("identity_short", short.class, C_SHORT, true), (short) 10, (short) 10); - tests.add(linkIndentity("identity_int", int.class, C_INT, false), 10, 10); - tests.add(linkIndentity("identity_int", int.class, C_INT, true), 10, 10); - tests.add(linkIndentity("identity_long", long.class, C_LONG_LONG, false), 10L, 10L); - tests.add(linkIndentity("identity_long", long.class, C_LONG_LONG, true), 10L, 10L); - tests.add(linkIndentity("identity_float", float.class, C_FLOAT, false), 10F, 10F); - tests.add(linkIndentity("identity_float", float.class, C_FLOAT, true), 10F, 10F); - tests.add(linkIndentity("identity_double", double.class, C_DOUBLE, false), 10D, 10D); - tests.add(linkIndentity("identity_double", double.class, C_DOUBLE, true), 10D, 10D); + addIdentity.add("identity_char", byte.class, C_CHAR, (byte) 10); + addIdentity.add("identity_short", short.class, C_SHORT, (short) 10); + addIdentity.add("identity_int", int.class, C_INT, 10); + addIdentity.add("identity_long", long.class, C_LONG_LONG, 10L); + addIdentity.add("identity_float", float.class, C_FLOAT, 10F); + addIdentity.add("identity_double", double.class, C_DOUBLE, 10D); { // identity_va - LibraryLookup.Symbol ma = lookup.lookup("identity_va").orElseThrow(); + MemoryAddress ma = lookup.lookup("identity_va").orElseThrow(); MethodType mt = methodType(int.class, int.class, double.class, int.class, float.class, long.class); FunctionDescriptor fd = FunctionDescriptor.of(C_INT, C_INT, asVarArg(C_DOUBLE), asVarArg(C_INT), asVarArg(C_FLOAT), asVarArg(C_LONG_LONG)); @@ -124,7 +124,7 @@ public class TestIntrinsics { C_SHORT, C_SHORT); Object[] args = {1, 10D, 2L, 3F, (byte) 0, (short) 13, 'a'}; for (int i = 0; i < args.length; i++) { - LibraryLookup.Symbol ma = lookup.lookup("invoke_high_arity" + i).orElseThrow(); + MemoryAddress ma = lookup.lookup("invoke_high_arity" + i).orElseThrow(); MethodType mt = baseMT.changeReturnType(baseMT.parameterType(i)); FunctionDescriptor fd = baseFD.withReturnLayout(baseFD.argumentLayouts().get(i)); Object expected = args[i]; diff --git a/test/jdk/java/foreign/TestLayoutConstants.java b/test/jdk/java/foreign/TestLayoutConstants.java index 904d03eccee..2ea29679234 100644 --- a/test/jdk/java/foreign/TestLayoutConstants.java +++ b/test/jdk/java/foreign/TestLayoutConstants.java @@ -67,43 +67,43 @@ public class TestLayoutConstants { return new Object[][] { //padding { MemoryLayouts.PAD_32 }, - { MemoryLayout.ofSequence(MemoryLayouts.PAD_32) }, - { MemoryLayout.ofSequence(5, MemoryLayouts.PAD_32) }, - { MemoryLayout.ofStruct(MemoryLayouts.PAD_32, MemoryLayouts.PAD_32) }, - { MemoryLayout.ofUnion(MemoryLayouts.PAD_32, MemoryLayouts.PAD_32) }, + { MemoryLayout.sequenceLayout(MemoryLayouts.PAD_32) }, + { MemoryLayout.sequenceLayout(5, MemoryLayouts.PAD_32) }, + { MemoryLayout.structLayout(MemoryLayouts.PAD_32, MemoryLayouts.PAD_32) }, + { MemoryLayout.unionLayout(MemoryLayouts.PAD_32, MemoryLayouts.PAD_32) }, //values, big endian { MemoryLayouts.BITS_32_BE }, - { MemoryLayout.ofStruct( + { MemoryLayout.structLayout( MemoryLayouts.BITS_32_BE, MemoryLayouts.BITS_32_BE) }, - { MemoryLayout.ofUnion( + { MemoryLayout.unionLayout( MemoryLayouts.BITS_32_BE, MemoryLayouts.BITS_32_BE) }, //values, little endian { MemoryLayouts.BITS_32_LE }, - { MemoryLayout.ofStruct( + { MemoryLayout.structLayout( MemoryLayouts.BITS_32_LE, MemoryLayouts.BITS_32_LE) }, - { MemoryLayout.ofUnion( + { MemoryLayout.unionLayout( MemoryLayouts.BITS_32_LE, MemoryLayouts.BITS_32_LE) }, //deeply nested - { MemoryLayout.ofStruct( + { MemoryLayout.structLayout( MemoryLayouts.PAD_16, - MemoryLayout.ofStruct( + MemoryLayout.structLayout( MemoryLayouts.PAD_8, MemoryLayouts.BITS_32_BE)) }, - { MemoryLayout.ofUnion( + { MemoryLayout.unionLayout( MemoryLayouts.PAD_16, - MemoryLayout.ofStruct( + MemoryLayout.structLayout( MemoryLayouts.PAD_8, MemoryLayouts.BITS_32_BE)) }, - { MemoryLayout.ofSequence( - MemoryLayout.ofStruct( + { MemoryLayout.sequenceLayout( + MemoryLayout.structLayout( MemoryLayouts.PAD_8, MemoryLayouts.BITS_32_BE)) }, - { MemoryLayout.ofSequence(5, - MemoryLayout.ofStruct( + { MemoryLayout.sequenceLayout(5, + MemoryLayout.structLayout( MemoryLayouts.PAD_8, MemoryLayouts.BITS_32_BE)) }, { MemoryLayouts.BITS_32_LE.withName("myInt") }, diff --git a/test/jdk/java/foreign/TestLayoutEquality.java b/test/jdk/java/foreign/TestLayoutEquality.java index 3549d91c33f..8eb59ec340e 100644 --- a/test/jdk/java/foreign/TestLayoutEquality.java +++ b/test/jdk/java/foreign/TestLayoutEquality.java @@ -45,7 +45,7 @@ public class TestLayoutEquality { @Test(dataProvider = "layoutConstants") public void testReconstructedEquality(ValueLayout layout) { - ValueLayout newLayout = MemoryLayout.ofValueBits(layout.bitSize(), layout.order()); + ValueLayout newLayout = MemoryLayout.valueLayout(layout.bitSize(), layout.order()); // properties should be equal assertEquals(newLayout.bitSize(), layout.bitSize()); diff --git a/test/jdk/java/foreign/TestLayoutPaths.java b/test/jdk/java/foreign/TestLayoutPaths.java index 39291f6437b..1cf3798c249 100644 --- a/test/jdk/java/foreign/TestLayoutPaths.java +++ b/test/jdk/java/foreign/TestLayoutPaths.java @@ -31,8 +31,11 @@ import jdk.incubator.foreign.GroupLayout; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayout.PathElement; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; +import org.testng.SkipException; import org.testng.annotations.*; import java.lang.invoke.MethodHandle; @@ -49,61 +52,61 @@ public class TestLayoutPaths { @Test(expectedExceptions = IllegalArgumentException.class) public void testBadBitSelectFromSeq() { - SequenceLayout seq = MemoryLayout.ofSequence(JAVA_INT); - seq.bitOffset(PathElement.groupElement("foo")); + SequenceLayout seq = MemoryLayout.sequenceLayout(JAVA_INT); + seq.bitOffset(groupElement("foo")); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadByteSelectFromSeq() { - SequenceLayout seq = MemoryLayout.ofSequence(JAVA_INT); - seq.byteOffset(PathElement.groupElement("foo")); + SequenceLayout seq = MemoryLayout.sequenceLayout(JAVA_INT); + seq.byteOffset(groupElement("foo")); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadBitSelectFromStruct() { - GroupLayout g = MemoryLayout.ofStruct(JAVA_INT); + GroupLayout g = MemoryLayout.structLayout(JAVA_INT); g.bitOffset(sequenceElement()); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadByteSelectFromStruct() { - GroupLayout g = MemoryLayout.ofStruct(JAVA_INT); + GroupLayout g = MemoryLayout.structLayout(JAVA_INT); g.byteOffset(sequenceElement()); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadBitSelectFromValue() { - SequenceLayout seq = MemoryLayout.ofSequence(JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(JAVA_INT); seq.bitOffset(sequenceElement(), sequenceElement()); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadByteSelectFromValue() { - SequenceLayout seq = MemoryLayout.ofSequence(JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(JAVA_INT); seq.byteOffset(sequenceElement(), sequenceElement()); } @Test(expectedExceptions = IllegalArgumentException.class) public void testUnknownBitStructField() { - GroupLayout g = MemoryLayout.ofStruct(JAVA_INT); - g.bitOffset(PathElement.groupElement("foo")); + GroupLayout g = MemoryLayout.structLayout(JAVA_INT); + g.bitOffset(groupElement("foo")); } @Test(expectedExceptions = IllegalArgumentException.class) public void testUnknownByteStructField() { - GroupLayout g = MemoryLayout.ofStruct(JAVA_INT); - g.byteOffset(PathElement.groupElement("foo")); + GroupLayout g = MemoryLayout.structLayout(JAVA_INT); + g.byteOffset(groupElement("foo")); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBitOutOfBoundsSeqIndex() { - SequenceLayout seq = MemoryLayout.ofSequence(5, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, JAVA_INT); seq.bitOffset(sequenceElement(6)); } @Test(expectedExceptions = IllegalArgumentException.class) public void testByteOutOfBoundsSeqIndex() { - SequenceLayout seq = MemoryLayout.ofSequence(5, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, JAVA_INT); seq.byteOffset(sequenceElement(6)); } @@ -114,19 +117,19 @@ public class TestLayoutPaths { @Test(expectedExceptions = IllegalArgumentException.class) public void testBitNegativeSeqIndex() { - SequenceLayout seq = MemoryLayout.ofSequence(5, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, JAVA_INT); seq.bitOffset(sequenceElement(-2)); } @Test(expectedExceptions = IllegalArgumentException.class) public void testByteNegativeSeqIndex() { - SequenceLayout seq = MemoryLayout.ofSequence(5, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, JAVA_INT); seq.byteOffset(sequenceElement(-2)); } @Test(expectedExceptions = IllegalArgumentException.class) public void testOutOfBoundsSeqRange() { - SequenceLayout seq = MemoryLayout.ofSequence(5, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, JAVA_INT); seq.bitOffset(sequenceElement(6, 2)); } @@ -137,88 +140,88 @@ public class TestLayoutPaths { @Test(expectedExceptions = IllegalArgumentException.class) public void testBitNegativeSeqRange() { - SequenceLayout seq = MemoryLayout.ofSequence(5, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, JAVA_INT); seq.bitOffset(sequenceElement(-2, 2)); } @Test(expectedExceptions = IllegalArgumentException.class) public void testByteNegativeSeqRange() { - SequenceLayout seq = MemoryLayout.ofSequence(5, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, JAVA_INT); seq.byteOffset(sequenceElement(-2, 2)); } @Test(expectedExceptions = IllegalArgumentException.class) public void testIncompleteAccess() { - SequenceLayout seq = MemoryLayout.ofSequence(5, MemoryLayout.ofStruct(JAVA_INT)); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, MemoryLayout.structLayout(JAVA_INT)); seq.varHandle(int.class, sequenceElement()); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBitOffsetHandleBadRange() { - SequenceLayout seq = MemoryLayout.ofSequence(5, MemoryLayout.ofStruct(JAVA_INT)); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, MemoryLayout.structLayout(JAVA_INT)); seq.bitOffsetHandle(sequenceElement(0, 1)); // ranges not accepted } @Test(expectedExceptions = IllegalArgumentException.class) public void testByteOffsetHandleBadRange() { - SequenceLayout seq = MemoryLayout.ofSequence(5, MemoryLayout.ofStruct(JAVA_INT)); + SequenceLayout seq = MemoryLayout.sequenceLayout(5, MemoryLayout.structLayout(JAVA_INT)); seq.byteOffsetHandle(sequenceElement(0, 1)); // ranges not accepted } @Test(expectedExceptions = UnsupportedOperationException.class) public void testBadMultiple() { - GroupLayout g = MemoryLayout.ofStruct(MemoryLayout.ofPaddingBits(3), JAVA_INT.withName("foo")); - g.byteOffset(PathElement.groupElement("foo")); + GroupLayout g = MemoryLayout.structLayout(MemoryLayout.paddingLayout(3), JAVA_INT.withName("foo")); + g.byteOffset(groupElement("foo")); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testBitOffsetBadUnboundedSequenceTraverse() { - MemoryLayout layout = MemoryLayout.ofSequence(MemoryLayout.ofSequence(JAVA_INT)); + MemoryLayout layout = MemoryLayout.sequenceLayout(MemoryLayout.sequenceLayout(JAVA_INT)); layout.bitOffset(sequenceElement(1), sequenceElement(0)); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testByteOffsetBadUnboundedSequenceTraverse() { - MemoryLayout layout = MemoryLayout.ofSequence(MemoryLayout.ofSequence(JAVA_INT)); + MemoryLayout layout = MemoryLayout.sequenceLayout(MemoryLayout.sequenceLayout(JAVA_INT)); layout.byteOffset(sequenceElement(1), sequenceElement(0)); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testBitOffsetHandleBadUnboundedSequenceTraverse() { - MemoryLayout layout = MemoryLayout.ofSequence(MemoryLayout.ofSequence(JAVA_INT)); + MemoryLayout layout = MemoryLayout.sequenceLayout(MemoryLayout.sequenceLayout(JAVA_INT)); layout.bitOffsetHandle(sequenceElement(1), sequenceElement(0)); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testByteOffsetHandleBadUnboundedSequenceTraverse() { - MemoryLayout layout = MemoryLayout.ofSequence(MemoryLayout.ofSequence(JAVA_INT)); + MemoryLayout layout = MemoryLayout.sequenceLayout(MemoryLayout.sequenceLayout(JAVA_INT)); layout.byteOffsetHandle(sequenceElement(1), sequenceElement(0)); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testBadByteOffsetNoMultipleOf8() { - MemoryLayout layout = MemoryLayout.ofStruct(MemoryLayout.ofPaddingBits(7), JAVA_INT.withName("x")); + MemoryLayout layout = MemoryLayout.structLayout(MemoryLayout.paddingLayout(7), JAVA_INT.withName("x")); layout.byteOffset(groupElement("x")); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testBadByteOffsetHandleNoMultipleOf8() throws Throwable { - MemoryLayout layout = MemoryLayout.ofStruct(MemoryLayout.ofPaddingBits(7), JAVA_INT.withName("x")); + MemoryLayout layout = MemoryLayout.structLayout(MemoryLayout.paddingLayout(7), JAVA_INT.withName("x")); MethodHandle handle = layout.byteOffsetHandle(groupElement("x")); handle.invoke(); } @Test public void testBadContainerAlign() { - GroupLayout g = MemoryLayout.ofStruct(JAVA_INT.withBitAlignment(16).withName("foo")).withBitAlignment(8); + GroupLayout g = MemoryLayout.structLayout(JAVA_INT.withBitAlignment(16).withName("foo")).withBitAlignment(8); try { - g.bitOffset(PathElement.groupElement("foo")); - g.byteOffset(PathElement.groupElement("foo")); + g.bitOffset(groupElement("foo")); + g.byteOffset(groupElement("foo")); } catch (Throwable ex) { throw new AssertionError(ex); // should be ok! } try { - g.varHandle(int.class, PathElement.groupElement("foo")); //ok + g.varHandle(int.class, groupElement("foo")); //ok assertTrue(false); //should fail! } catch (UnsupportedOperationException ex) { //ok @@ -229,15 +232,15 @@ public class TestLayoutPaths { @Test public void testBadAlignOffset() { - GroupLayout g = MemoryLayout.ofStruct(MemoryLayouts.PAD_8, JAVA_INT.withBitAlignment(16).withName("foo")); + GroupLayout g = MemoryLayout.structLayout(MemoryLayouts.PAD_8, JAVA_INT.withBitAlignment(16).withName("foo")); try { - g.bitOffset(PathElement.groupElement("foo")); - g.byteOffset(PathElement.groupElement("foo")); + g.bitOffset(groupElement("foo")); + g.byteOffset(groupElement("foo")); } catch (Throwable ex) { throw new AssertionError(ex); // should be ok! } try { - g.varHandle(int.class, PathElement.groupElement("foo")); //ok + g.varHandle(int.class, groupElement("foo")); //ok assertTrue(false); //should fail! } catch (UnsupportedOperationException ex) { //ok @@ -248,7 +251,7 @@ public class TestLayoutPaths { @Test public void testBadSequencePathInOffset() { - SequenceLayout seq = MemoryLayout.ofSequence(10, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(10, JAVA_INT); // bad path elements for (PathElement e : List.of( sequenceElement(), sequenceElement(0, 2) )) { try { @@ -268,7 +271,7 @@ public class TestLayoutPaths { @Test public void testBadSequencePathInSelect() { - SequenceLayout seq = MemoryLayout.ofSequence(10, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(10, JAVA_INT); for (PathElement e : List.of( sequenceElement(0), sequenceElement(0, 2) )) { try { seq.select(e); @@ -281,7 +284,7 @@ public class TestLayoutPaths { @Test public void testBadSequencePathInMap() { - SequenceLayout seq = MemoryLayout.ofSequence(10, JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(10, JAVA_INT); for (PathElement e : List.of( sequenceElement(0), sequenceElement(0, 2) )) { try { seq.map(l -> l, e); @@ -295,7 +298,7 @@ public class TestLayoutPaths { @Test public void testStructPaths() { long[] offsets = { 0, 8, 24, 56 }; - GroupLayout g = MemoryLayout.ofStruct( + GroupLayout g = MemoryLayout.structLayout( MemoryLayouts.JAVA_BYTE.withName("1"), MemoryLayouts.JAVA_CHAR.withName("2"), MemoryLayouts.JAVA_FLOAT.withName("3"), @@ -305,23 +308,23 @@ public class TestLayoutPaths { // test select for (int i = 1 ; i <= 4 ; i++) { - MemoryLayout selected = g.select(PathElement.groupElement(String.valueOf(i))); + MemoryLayout selected = g.select(groupElement(String.valueOf(i))); assertTrue(selected == g.memberLayouts().get(i - 1)); } // test offset for (int i = 1 ; i <= 4 ; i++) { - long bitOffset = g.bitOffset(PathElement.groupElement(String.valueOf(i))); + long bitOffset = g.bitOffset(groupElement(String.valueOf(i))); assertEquals(offsets[i - 1], bitOffset); - long byteOffset = g.byteOffset(PathElement.groupElement(String.valueOf(i))); + long byteOffset = g.byteOffset(groupElement(String.valueOf(i))); assertEquals((offsets[i - 1]) >>> 3, byteOffset); } // test map for (int i = 1 ; i <= 4 ; i++) { - GroupLayout g2 = (GroupLayout)g.map(l -> MemoryLayouts.JAVA_DOUBLE, PathElement.groupElement(String.valueOf(i))); + GroupLayout g2 = (GroupLayout)g.map(l -> MemoryLayouts.JAVA_DOUBLE, groupElement(String.valueOf(i))); assertTrue(g2.isStruct()); for (int j = 0 ; j < 4 ; j++) { if (j == i - 1) { @@ -336,7 +339,7 @@ public class TestLayoutPaths { @Test public void testUnionPaths() { long[] offsets = { 0, 0, 0, 0 }; - GroupLayout g = MemoryLayout.ofUnion( + GroupLayout g = MemoryLayout.unionLayout( MemoryLayouts.JAVA_BYTE.withName("1"), MemoryLayouts.JAVA_CHAR.withName("2"), MemoryLayouts.JAVA_FLOAT.withName("3"), @@ -346,23 +349,23 @@ public class TestLayoutPaths { // test select for (int i = 1 ; i <= 4 ; i++) { - MemoryLayout selected = g.select(PathElement.groupElement(String.valueOf(i))); + MemoryLayout selected = g.select(groupElement(String.valueOf(i))); assertTrue(selected == g.memberLayouts().get(i - 1)); } // test offset for (int i = 1 ; i <= 4 ; i++) { - long bitOffset = g.bitOffset(PathElement.groupElement(String.valueOf(i))); + long bitOffset = g.bitOffset(groupElement(String.valueOf(i))); assertEquals(offsets[i - 1], bitOffset); - long byteOffset = g.byteOffset(PathElement.groupElement(String.valueOf(i))); + long byteOffset = g.byteOffset(groupElement(String.valueOf(i))); assertEquals((offsets[i - 1]) >>> 3, byteOffset); } // test map for (int i = 1 ; i <= 4 ; i++) { - GroupLayout g2 = (GroupLayout)g.map(l -> MemoryLayouts.JAVA_DOUBLE, PathElement.groupElement(String.valueOf(i))); + GroupLayout g2 = (GroupLayout)g.map(l -> MemoryLayouts.JAVA_DOUBLE, groupElement(String.valueOf(i))); assertTrue(g2.isUnion()); for (int j = 0 ; j < 4 ; j++) { if (j == i - 1) { @@ -377,7 +380,7 @@ public class TestLayoutPaths { @Test public void testSequencePaths() { long[] offsets = { 0, 8, 16, 24 }; - SequenceLayout g = MemoryLayout.ofSequence(4, MemoryLayouts.JAVA_BYTE); + SequenceLayout g = MemoryLayout.sequenceLayout(4, MemoryLayouts.JAVA_BYTE); // test select @@ -399,7 +402,7 @@ public class TestLayoutPaths { assertTrue(seq2.elementLayout() == MemoryLayouts.JAVA_DOUBLE); } - @Test(dataProvider = "offsetHandleCases") + @Test(dataProvider = "testLayouts") public void testOffsetHandle(MemoryLayout layout, PathElement[] pathElements, long[] indexes, long expectedBitOffset) throws Throwable { MethodHandle bitOffsetHandle = layout.bitOffsetHandle(pathElements); @@ -415,46 +418,46 @@ public class TestLayoutPaths { } @DataProvider - public static Object[][] offsetHandleCases() { + public static Object[][] testLayouts() { List testCases = new ArrayList<>(); testCases.add(new Object[] { - MemoryLayout.ofSequence(10, JAVA_INT), + MemoryLayout.sequenceLayout(10, JAVA_INT), new PathElement[] { sequenceElement() }, new long[] { 4 }, JAVA_INT.bitSize() * 4 }); testCases.add(new Object[] { - MemoryLayout.ofSequence(10, MemoryLayout.ofStruct(JAVA_INT, JAVA_INT.withName("y"))), + MemoryLayout.sequenceLayout(10, MemoryLayout.structLayout(JAVA_INT, JAVA_INT.withName("y"))), new PathElement[] { sequenceElement(), groupElement("y") }, new long[] { 4 }, (JAVA_INT.bitSize() * 2) * 4 + JAVA_INT.bitSize() }); testCases.add(new Object[] { - MemoryLayout.ofSequence(10, MemoryLayout.ofStruct(MemoryLayout.ofPaddingBits(5), JAVA_INT.withName("y"))), + MemoryLayout.sequenceLayout(10, MemoryLayout.structLayout(MemoryLayout.paddingLayout(5), JAVA_INT.withName("y"))), new PathElement[] { sequenceElement(), groupElement("y") }, new long[] { 4 }, (JAVA_INT.bitSize() + 5) * 4 + 5 }); testCases.add(new Object[] { - MemoryLayout.ofSequence(10, JAVA_INT), + MemoryLayout.sequenceLayout(10, JAVA_INT), new PathElement[] { sequenceElement() }, new long[] { 4 }, JAVA_INT.bitSize() * 4 }); testCases.add(new Object[] { - MemoryLayout.ofStruct( - MemoryLayout.ofSequence(10, JAVA_INT).withName("data") + MemoryLayout.structLayout( + MemoryLayout.sequenceLayout(10, JAVA_INT).withName("data") ), new PathElement[] { groupElement("data"), sequenceElement() }, new long[] { 4 }, JAVA_INT.bitSize() * 4 }); - MemoryLayout complexLayout = MemoryLayout.ofStruct( - MemoryLayout.ofSequence(10, - MemoryLayout.ofSequence(10, - MemoryLayout.ofStruct( + MemoryLayout complexLayout = MemoryLayout.structLayout( + MemoryLayout.sequenceLayout(10, + MemoryLayout.sequenceLayout(10, + MemoryLayout.structLayout( JAVA_INT.withName("x"), JAVA_INT.withName("y") ) @@ -490,5 +493,73 @@ public class TestLayoutPaths { return testCases.toArray(Object[][]::new); } + @Test(dataProvider = "testLayouts") + public void testSliceHandle(MemoryLayout layout, PathElement[] pathElements, long[] indexes, + long expectedBitOffset) throws Throwable { + if (expectedBitOffset % 8 != 0) + throw new SkipException("Offset not a multiple of 8"); + + MemoryLayout selected = layout.select(pathElements); + MethodHandle sliceHandle = layout.sliceHandle(pathElements); + sliceHandle = sliceHandle.asSpreader(long[].class, indexes.length); + + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); + MemorySegment slice = (MemorySegment) sliceHandle.invokeExact(segment, indexes); + assertEquals(slice.address().segmentOffset(segment), expectedBitOffset / 8); + assertEquals(slice.byteSize(), selected.byteSize()); + } + } + + @Test(expectedExceptions = UnsupportedOperationException.class) + public void testSliceHandleUOEInvalidSize() { + MemoryLayout layout = MemoryLayout.structLayout( + MemoryLayout.valueLayout(32, ByteOrder.nativeOrder()).withName("x"), + MemoryLayout.valueLayout(31, ByteOrder.nativeOrder()).withName("y") // size not a multiple of 8 + ); + + layout.sliceHandle(groupElement("y")); // should throw + } + + @Test(expectedExceptions = UnsupportedOperationException.class) + public void testSliceHandleUOEInvalidOffsetEager() throws Throwable { + MemoryLayout layout = MemoryLayout.structLayout( + MemoryLayout.paddingLayout(5), + MemoryLayout.valueLayout(32, ByteOrder.nativeOrder()).withName("y") // offset not a multiple of 8 + ); + + layout.sliceHandle(groupElement("y")); // should throw + } + + @Test(expectedExceptions = UnsupportedOperationException.class) + public void testSliceHandleUOEInvalidOffsetLate() throws Throwable { + MemoryLayout layout = MemoryLayout.sequenceLayout(3, + MemoryLayout.structLayout( + MemoryLayout.paddingLayout(4), + MemoryLayout.valueLayout(32, ByteOrder.nativeOrder()).withName("y") // offset not a multiple of 8 + ) + ); + + MethodHandle sliceHandle; + try { + sliceHandle = layout.sliceHandle(sequenceElement(), groupElement("y")); // should work + } catch (UnsupportedOperationException uoe) { + fail("Unexpected exception", uoe); + return; + } + + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); + + try { + sliceHandle.invokeExact(segment, 1); // should work + } catch (UnsupportedOperationException uoe) { + fail("Unexpected exception", uoe); + return; + } + + sliceHandle.invokeExact(segment, 0); // should throw + } + } } diff --git a/test/jdk/java/foreign/TestLayouts.java b/test/jdk/java/foreign/TestLayouts.java index d1b2f29b23d..006163d3d50 100644 --- a/test/jdk/java/foreign/TestLayouts.java +++ b/test/jdk/java/foreign/TestLayouts.java @@ -50,17 +50,18 @@ public class TestLayouts { @Test public void testVLAInStruct() { - MemoryLayout layout = MemoryLayout.ofStruct( + MemoryLayout layout = MemoryLayout.structLayout( MemoryLayouts.JAVA_INT.withName("size"), - MemoryLayout.ofPaddingBits(32), - MemoryLayout.ofSequence(MemoryLayouts.JAVA_DOUBLE).withName("arr")); + MemoryLayout.paddingLayout(32), + MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_DOUBLE).withName("arr")); assertFalse(layout.hasSize()); VarHandle size_handle = layout.varHandle(int.class, MemoryLayout.PathElement.groupElement("size")); VarHandle array_elem_handle = layout.varHandle(double.class, MemoryLayout.PathElement.groupElement("arr"), MemoryLayout.PathElement.sequenceElement()); - try (MemorySegment segment = MemorySegment.allocateNative( - layout.map(l -> ((SequenceLayout)l).withElementCount(4), MemoryLayout.PathElement.groupElement("arr")))) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative( + layout.map(l -> ((SequenceLayout)l).withElementCount(4), MemoryLayout.PathElement.groupElement("arr")), scope); size_handle.set(segment, 4); for (int i = 0 ; i < 4 ; i++) { array_elem_handle.set(segment, i, (double)i); @@ -75,18 +76,19 @@ public class TestLayouts { @Test public void testVLAInSequence() { - MemoryLayout layout = MemoryLayout.ofStruct( + MemoryLayout layout = MemoryLayout.structLayout( MemoryLayouts.JAVA_INT.withName("size"), - MemoryLayout.ofPaddingBits(32), - MemoryLayout.ofSequence(1, MemoryLayout.ofSequence(MemoryLayouts.JAVA_DOUBLE)).withName("arr")); + MemoryLayout.paddingLayout(32), + MemoryLayout.sequenceLayout(1, MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_DOUBLE)).withName("arr")); assertFalse(layout.hasSize()); VarHandle size_handle = layout.varHandle(int.class, MemoryLayout.PathElement.groupElement("size")); VarHandle array_elem_handle = layout.varHandle(double.class, MemoryLayout.PathElement.groupElement("arr"), MemoryLayout.PathElement.sequenceElement(0), MemoryLayout.PathElement.sequenceElement()); - try (MemorySegment segment = MemorySegment.allocateNative( - layout.map(l -> ((SequenceLayout)l).withElementCount(4), MemoryLayout.PathElement.groupElement("arr"), MemoryLayout.PathElement.sequenceElement()))) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative( + layout.map(l -> ((SequenceLayout)l).withElementCount(4), MemoryLayout.PathElement.groupElement("arr"), MemoryLayout.PathElement.sequenceElement()), scope); size_handle.set(segment, 4); for (int i = 0 ; i < 4 ; i++) { array_elem_handle.set(segment, i, (double)i); @@ -101,8 +103,9 @@ public class TestLayouts { @Test public void testIndexedSequencePath() { - MemoryLayout seq = MemoryLayout.ofSequence(10, MemoryLayouts.JAVA_INT); - try (MemorySegment segment = MemorySegment.allocateNative(seq)) { + MemoryLayout seq = MemoryLayout.sequenceLayout(10, MemoryLayouts.JAVA_INT); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(seq, scope); VarHandle indexHandle = seq.varHandle(int.class, MemoryLayout.PathElement.sequenceElement()); // init segment for (int i = 0 ; i < 10 ; i++) { @@ -140,31 +143,31 @@ public class TestLayouts { @Test(expectedExceptions = IllegalArgumentException.class) public void testBadUnboundSequenceLayoutResize() { - SequenceLayout seq = MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT); seq.withElementCount(-1); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadBoundSequenceLayoutResize() { - SequenceLayout seq = MemoryLayout.ofSequence(10, MemoryLayouts.JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(10, MemoryLayouts.JAVA_INT); seq.withElementCount(-1); } @Test public void testEmptyGroup() { - MemoryLayout struct = MemoryLayout.ofStruct(); + MemoryLayout struct = MemoryLayout.structLayout(); assertEquals(struct.bitSize(), 0); assertEquals(struct.bitAlignment(), 1); - MemoryLayout union = MemoryLayout.ofUnion(); + MemoryLayout union = MemoryLayout.unionLayout(); assertEquals(union.bitSize(), 0); assertEquals(union.bitAlignment(), 1); } @Test public void testStructSizeAndAlign() { - MemoryLayout struct = MemoryLayout.ofStruct( - MemoryLayout.ofPaddingBits(8), + MemoryLayout struct = MemoryLayout.structLayout( + MemoryLayout.paddingLayout(8), MemoryLayouts.JAVA_BYTE, MemoryLayouts.JAVA_CHAR, MemoryLayouts.JAVA_INT, @@ -176,26 +179,26 @@ public class TestLayouts { @Test(dataProvider="basicLayouts") public void testPaddingNoAlign(MemoryLayout layout) { - assertEquals(MemoryLayout.ofPaddingBits(layout.bitSize()).bitAlignment(), 1); + assertEquals(MemoryLayout.paddingLayout(layout.bitSize()).bitAlignment(), 1); } @Test(dataProvider="basicLayouts") public void testStructPaddingAndAlign(MemoryLayout layout) { - MemoryLayout struct = MemoryLayout.ofStruct( - layout, MemoryLayout.ofPaddingBits(128 - layout.bitSize())); + MemoryLayout struct = MemoryLayout.structLayout( + layout, MemoryLayout.paddingLayout(128 - layout.bitSize())); assertEquals(struct.bitAlignment(), layout.bitAlignment()); } @Test(dataProvider="basicLayouts") public void testUnionPaddingAndAlign(MemoryLayout layout) { - MemoryLayout struct = MemoryLayout.ofUnion( - layout, MemoryLayout.ofPaddingBits(128 - layout.bitSize())); + MemoryLayout struct = MemoryLayout.unionLayout( + layout, MemoryLayout.paddingLayout(128 - layout.bitSize())); assertEquals(struct.bitAlignment(), layout.bitAlignment()); } @Test public void testUnionSizeAndAlign() { - MemoryLayout struct = MemoryLayout.ofUnion( + MemoryLayout struct = MemoryLayout.unionLayout( MemoryLayouts.JAVA_BYTE, MemoryLayouts.JAVA_CHAR, MemoryLayouts.JAVA_INT, @@ -237,15 +240,15 @@ public class TestLayouts { @DataProvider(name = "unboundLayouts") public Object[][] unboundLayouts() { return new Object[][] { - { MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT), 32 }, - { MemoryLayout.ofSequence(MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT)), 32 }, - { MemoryLayout.ofSequence(4, MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT)), 32 }, - { MemoryLayout.ofStruct(MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT)), 32 }, - { MemoryLayout.ofStruct(MemoryLayout.ofSequence(MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT))), 32 }, - { MemoryLayout.ofStruct(MemoryLayout.ofSequence(4, MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT))), 32 }, - { MemoryLayout.ofUnion(MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT)), 32 }, - { MemoryLayout.ofUnion(MemoryLayout.ofSequence(MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT))), 32 }, - { MemoryLayout.ofUnion(MemoryLayout.ofSequence(4, MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT))), 32 }, + { MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT), 32 }, + { MemoryLayout.sequenceLayout(MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT)), 32 }, + { MemoryLayout.sequenceLayout(4, MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT)), 32 }, + { MemoryLayout.structLayout(MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT)), 32 }, + { MemoryLayout.structLayout(MemoryLayout.sequenceLayout(MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT))), 32 }, + { MemoryLayout.structLayout(MemoryLayout.sequenceLayout(4, MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT))), 32 }, + { MemoryLayout.unionLayout(MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT)), 32 }, + { MemoryLayout.unionLayout(MemoryLayout.sequenceLayout(MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT))), 32 }, + { MemoryLayout.unionLayout(MemoryLayout.sequenceLayout(4, MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT))), 32 }, }; } @@ -268,10 +271,10 @@ public class TestLayouts { } enum SizedLayoutFactory { - VALUE_LE(size -> MemoryLayout.ofValueBits(size, ByteOrder.LITTLE_ENDIAN)), - VALUE_BE(size -> MemoryLayout.ofValueBits(size, ByteOrder.BIG_ENDIAN)), - PADDING(MemoryLayout::ofPaddingBits), - SEQUENCE(size -> MemoryLayout.ofSequence(size, MemoryLayouts.PAD_8)); + VALUE_LE(size -> MemoryLayout.valueLayout(size, ByteOrder.LITTLE_ENDIAN)), + VALUE_BE(size -> MemoryLayout.valueLayout(size, ByteOrder.BIG_ENDIAN)), + PADDING(MemoryLayout::paddingLayout), + SEQUENCE(size -> MemoryLayout.sequenceLayout(size, MemoryLayouts.PAD_8)); private final LongFunction factory; @@ -288,9 +291,9 @@ public class TestLayouts { VALUE_LE(MemoryLayouts.BITS_8_LE), VALUE_BE(MemoryLayouts.BITS_8_BE), PADDING(MemoryLayouts.PAD_8), - SEQUENCE(MemoryLayout.ofSequence(1, MemoryLayouts.PAD_8)), - STRUCT(MemoryLayout.ofStruct(MemoryLayouts.PAD_8, MemoryLayouts.PAD_8)), - UNION(MemoryLayout.ofUnion(MemoryLayouts.PAD_8, MemoryLayouts.PAD_8)); + SEQUENCE(MemoryLayout.sequenceLayout(1, MemoryLayouts.PAD_8)), + STRUCT(MemoryLayout.structLayout(MemoryLayouts.PAD_8, MemoryLayouts.PAD_8)), + UNION(MemoryLayout.unionLayout(MemoryLayouts.PAD_8, MemoryLayouts.PAD_8)); final MemoryLayout layout; @@ -316,15 +319,15 @@ public class TestLayouts { } //add basic layouts wrapped in a sequence with given size for (MemoryLayout l : basicLayouts) { - layoutsAndAlignments[i++] = new Object[] { MemoryLayout.ofSequence(4, l), l.bitAlignment() }; + layoutsAndAlignments[i++] = new Object[] { MemoryLayout.sequenceLayout(4, l), l.bitAlignment() }; } //add basic layouts wrapped in a struct for (MemoryLayout l : basicLayouts) { - layoutsAndAlignments[i++] = new Object[] { MemoryLayout.ofStruct(l), l.bitAlignment() }; + layoutsAndAlignments[i++] = new Object[] { MemoryLayout.structLayout(l), l.bitAlignment() }; } //add basic layouts wrapped in a union for (MemoryLayout l : basicLayouts) { - layoutsAndAlignments[i++] = new Object[] { MemoryLayout.ofUnion(l), l.bitAlignment() }; + layoutsAndAlignments[i++] = new Object[] { MemoryLayout.unionLayout(l), l.bitAlignment() }; } return layoutsAndAlignments; } diff --git a/test/jdk/java/foreign/TestLibraryLookup.java b/test/jdk/java/foreign/TestLibraryLookup.java index 765e0957b5c..f9d03a0cba5 100644 --- a/test/jdk/java/foreign/TestLibraryLookup.java +++ b/test/jdk/java/foreign/TestLibraryLookup.java @@ -25,11 +25,15 @@ * @test * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" * @modules jdk.incubator.foreign/jdk.internal.foreign - * @run testng/othervm -Dforeign.restricted=permit TestLibraryLookup + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestLibraryLookup */ import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryAccess; import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemoryLayouts; +import jdk.incubator.foreign.MemorySegment; import jdk.internal.foreign.LibrariesHelper; import org.testng.annotations.Test; @@ -66,10 +70,9 @@ public class TestLibraryLookup { @Test public void testSimpleLookup() throws Throwable { - LibraryLookup.Symbol symbol = null; + MemoryAddress symbol = null; LibraryLookup lookup = LibraryLookup.ofLibrary("LookupTest"); symbol = lookup.lookup("f").get(); - assertEquals(symbol.name(), "f"); assertEquals(LibrariesHelper.numLoadedLibraries(), 1); lookup = null; symbol = null; @@ -78,7 +81,7 @@ public class TestLibraryLookup { @Test public void testInvalidSymbolLookup() throws Throwable { - LibraryLookup.Symbol symbol = null; + MemoryAddress symbol = null; LibraryLookup lookup = LibraryLookup.ofLibrary("LookupTest"); assertTrue(lookup.lookup("nonExistent").isEmpty()); assertEquals(LibrariesHelper.numLoadedLibraries(), 1); @@ -87,13 +90,39 @@ public class TestLibraryLookup { waitUnload(); } + @Test + public void testVariableSymbolLookup() throws Throwable { + LibraryLookup lookup = LibraryLookup.ofLibrary("LookupTest"); + MemorySegment segment = lookup.lookup("c", MemoryLayouts.JAVA_INT).get(); + assertEquals(MemoryAccess.getInt(segment), 42); + lookup = null; + segment = null; + waitUnload(); + } + + @Test + public void testBadVariableSymbolLookup() { + LibraryLookup lookup = LibraryLookup.ofLibrary("LookupTest"); + try { + MemoryLayout layout = MemoryLayouts.JAVA_INT.withBitAlignment(1 << 16); + MemorySegment segment = lookup.lookup("c", layout).get(); + // no exception, check that address is aligned + if ((segment.address().toRawLongValue() % layout.byteAlignment()) != 0) { + fail("Unaligned address"); + } + } catch (IllegalArgumentException ex) { + // ok, means address was not aligned + } + + } + @Test public void testMultiLookupSameLoader() throws Throwable { - List symbols = new ArrayList<>(); + List symbols = new ArrayList<>(); List lookups = new ArrayList<>(); for (int i = 0 ; i < 5 ; i++) { LibraryLookup lookup = LibraryLookup.ofLibrary("LookupTest"); - LibraryLookup.Symbol symbol = lookup.lookup("f").get(); + MemoryAddress symbol = lookup.lookup("f").get(); lookups.add(lookup); symbols.add(symbol); assertEquals(LibrariesHelper.numLoadedLibraries(), 1); @@ -151,7 +180,7 @@ public class TestLibraryLookup { static class Holder { public static LibraryLookup lookup; - public static LibraryLookup.Symbol symbol; + public static MemoryAddress symbol; static { try { diff --git a/test/jdk/java/foreign/TestMatrix.java b/test/jdk/java/foreign/TestMatrix.java new file mode 100644 index 00000000000..0c9f4f0e3a0 --- /dev/null +++ b/test/jdk/java/foreign/TestMatrix.java @@ -0,0 +1,259 @@ +/* + * @test + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @modules jdk.incubator.foreign/jdk.internal.foreign + * @build NativeTestHelper CallGeneratorHelper TestUpcallHighArity TestUpcall TestDowncall + * + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcallHighArity + * + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcallHighArity + * + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcallHighArity + * + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcallHighArity + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcallHighArity + * + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * TestDowncall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * TestDowncall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * TestDowncall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * TestDowncall + * + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcall + * + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcall + * + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=true + * TestUpcall + * + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcall + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_SPEC=false + * -Djdk.internal.foreign.ProgrammableUpcallHandler.USE_INTRINSICS=false + * TestUpcall + */ diff --git a/test/jdk/java/foreign/TestMemoryAccess.java b/test/jdk/java/foreign/TestMemoryAccess.java index bcaf2b1756f..d843c7b0b39 100644 --- a/test/jdk/java/foreign/TestMemoryAccess.java +++ b/test/jdk/java/foreign/TestMemoryAccess.java @@ -34,6 +34,7 @@ import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayout.PathElement; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; import jdk.incubator.foreign.ValueLayout; @@ -53,38 +54,39 @@ public class TestMemoryAccess { @Test(dataProvider = "elements") public void testPaddedAccessByName(Function viewFactory, MemoryLayout elemLayout, Class carrier, Checker checker) { - GroupLayout layout = MemoryLayout.ofStruct(MemoryLayout.ofPaddingBits(elemLayout.bitSize()), elemLayout.withName("elem")); + GroupLayout layout = MemoryLayout.structLayout(MemoryLayout.paddingLayout(elemLayout.bitSize()), elemLayout.withName("elem")); testAccessInternal(viewFactory, layout, layout.varHandle(carrier, PathElement.groupElement("elem")), checker); } @Test(dataProvider = "elements") public void testPaddedAccessByIndexSeq(Function viewFactory, MemoryLayout elemLayout, Class carrier, Checker checker) { - SequenceLayout layout = MemoryLayout.ofSequence(2, elemLayout); + SequenceLayout layout = MemoryLayout.sequenceLayout(2, elemLayout); testAccessInternal(viewFactory, layout, layout.varHandle(carrier, PathElement.sequenceElement(1)), checker); } @Test(dataProvider = "arrayElements") public void testArrayAccess(Function viewFactory, MemoryLayout elemLayout, Class carrier, ArrayChecker checker) { - SequenceLayout seq = MemoryLayout.ofSequence(10, elemLayout.withName("elem")); + SequenceLayout seq = MemoryLayout.sequenceLayout(10, elemLayout.withName("elem")); testArrayAccessInternal(viewFactory, seq, seq.varHandle(carrier, PathElement.sequenceElement()), checker); } @Test(dataProvider = "arrayElements") public void testPaddedArrayAccessByName(Function viewFactory, MemoryLayout elemLayout, Class carrier, ArrayChecker checker) { - SequenceLayout seq = MemoryLayout.ofSequence(10, MemoryLayout.ofStruct(MemoryLayout.ofPaddingBits(elemLayout.bitSize()), elemLayout.withName("elem"))); + SequenceLayout seq = MemoryLayout.sequenceLayout(10, MemoryLayout.structLayout(MemoryLayout.paddingLayout(elemLayout.bitSize()), elemLayout.withName("elem"))); testArrayAccessInternal(viewFactory, seq, seq.varHandle(carrier, MemoryLayout.PathElement.sequenceElement(), MemoryLayout.PathElement.groupElement("elem")), checker); } @Test(dataProvider = "arrayElements") public void testPaddedArrayAccessByIndexSeq(Function viewFactory, MemoryLayout elemLayout, Class carrier, ArrayChecker checker) { - SequenceLayout seq = MemoryLayout.ofSequence(10, MemoryLayout.ofSequence(2, elemLayout)); + SequenceLayout seq = MemoryLayout.sequenceLayout(10, MemoryLayout.sequenceLayout(2, elemLayout)); testArrayAccessInternal(viewFactory, seq, seq.varHandle(carrier, PathElement.sequenceElement(), MemoryLayout.PathElement.sequenceElement(1)), checker); } private void testAccessInternal(Function viewFactory, MemoryLayout layout, VarHandle handle, Checker checker) { MemorySegment outer_segment; - try (MemorySegment segment = viewFactory.apply(MemorySegment.allocateNative(layout))) { - boolean isRO = !segment.hasAccessModes(MemorySegment.WRITE); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = viewFactory.apply(MemorySegment.allocateNative(layout, scope)); + boolean isRO = segment.isReadOnly(); try { checker.check(handle, segment); if (isRO) { @@ -114,8 +116,9 @@ public class TestMemoryAccess { private void testArrayAccessInternal(Function viewFactory, SequenceLayout seq, VarHandle handle, ArrayChecker checker) { MemorySegment outer_segment; - try (MemorySegment segment = viewFactory.apply(MemorySegment.allocateNative(seq))) { - boolean isRO = !segment.hasAccessModes(MemorySegment.WRITE); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = viewFactory.apply(MemorySegment.allocateNative(seq, scope)); + boolean isRO = segment.isReadOnly(); try { for (int i = 0; i < seq.elementCount().getAsLong(); i++) { checker.check(handle, segment, i); @@ -147,16 +150,16 @@ public class TestMemoryAccess { @Test(dataProvider = "matrixElements") public void testMatrixAccess(Function viewFactory, MemoryLayout elemLayout, Class carrier, MatrixChecker checker) { - SequenceLayout seq = MemoryLayout.ofSequence(20, - MemoryLayout.ofSequence(10, elemLayout.withName("elem"))); + SequenceLayout seq = MemoryLayout.sequenceLayout(20, + MemoryLayout.sequenceLayout(10, elemLayout.withName("elem"))); testMatrixAccessInternal(viewFactory, seq, seq.varHandle(carrier, PathElement.sequenceElement(), PathElement.sequenceElement()), checker); } @Test(dataProvider = "matrixElements") public void testPaddedMatrixAccessByName(Function viewFactory, MemoryLayout elemLayout, Class carrier, MatrixChecker checker) { - SequenceLayout seq = MemoryLayout.ofSequence(20, - MemoryLayout.ofSequence(10, MemoryLayout.ofStruct(MemoryLayout.ofPaddingBits(elemLayout.bitSize()), elemLayout.withName("elem")))); + SequenceLayout seq = MemoryLayout.sequenceLayout(20, + MemoryLayout.sequenceLayout(10, MemoryLayout.structLayout(MemoryLayout.paddingLayout(elemLayout.bitSize()), elemLayout.withName("elem")))); testMatrixAccessInternal(viewFactory, seq, seq.varHandle(carrier, PathElement.sequenceElement(), PathElement.sequenceElement(), PathElement.groupElement("elem")), @@ -165,8 +168,8 @@ public class TestMemoryAccess { @Test(dataProvider = "matrixElements") public void testPaddedMatrixAccessByIndexSeq(Function viewFactory, MemoryLayout elemLayout, Class carrier, MatrixChecker checker) { - SequenceLayout seq = MemoryLayout.ofSequence(20, - MemoryLayout.ofSequence(10, MemoryLayout.ofSequence(2, elemLayout))); + SequenceLayout seq = MemoryLayout.sequenceLayout(20, + MemoryLayout.sequenceLayout(10, MemoryLayout.sequenceLayout(2, elemLayout))); testMatrixAccessInternal(viewFactory, seq, seq.varHandle(carrier, PathElement.sequenceElement(), PathElement.sequenceElement(), PathElement.sequenceElement(1)), @@ -182,8 +185,9 @@ public class TestMemoryAccess { private void testMatrixAccessInternal(Function viewFactory, SequenceLayout seq, VarHandle handle, MatrixChecker checker) { MemorySegment outer_segment; - try (MemorySegment segment = viewFactory.apply(MemorySegment.allocateNative(seq))) { - boolean isRO = !segment.hasAccessModes(MemorySegment.WRITE); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = viewFactory.apply(MemorySegment.allocateNative(seq, scope)); + boolean isRO = segment.isReadOnly(); try { for (int i = 0; i < seq.elementCount().getAsLong(); i++) { for (int j = 0; j < ((SequenceLayout) seq.elementLayout()).elementCount().getAsLong(); j++) { @@ -217,7 +221,7 @@ public class TestMemoryAccess { } static Function ID = Function.identity(); - static Function IMMUTABLE = ms -> ms.withAccessModes(MemorySegment.READ | MemorySegment.CLOSE); + static Function IMMUTABLE = MemorySegment::asReadOnly; @DataProvider(name = "elements") public Object[][] createData() { diff --git a/test/jdk/java/foreign/TestMemoryAlignment.java b/test/jdk/java/foreign/TestMemoryAlignment.java index 2b6ee427151..93f218245c8 100644 --- a/test/jdk/java/foreign/TestMemoryAlignment.java +++ b/test/jdk/java/foreign/TestMemoryAlignment.java @@ -32,6 +32,7 @@ import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.GroupLayout; import jdk.incubator.foreign.MemoryLayout.PathElement; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; import jdk.incubator.foreign.ValueLayout; import java.lang.invoke.VarHandle; @@ -49,7 +50,8 @@ public class TestMemoryAlignment { ValueLayout aligned = layout.withBitAlignment(align); assertEquals(aligned.bitAlignment(), align); //unreasonable alignment here, to make sure access throws VarHandle vh = aligned.varHandle(int.class); - try (MemorySegment segment = MemorySegment.allocateNative(aligned)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(aligned, scope); vh.set(segment, -42); int val = (int)vh.get(segment); assertEquals(val, -42); @@ -61,10 +63,11 @@ public class TestMemoryAlignment { ValueLayout layout = MemoryLayouts.BITS_32_BE; assertEquals(layout.bitAlignment(), 32); ValueLayout aligned = layout.withBitAlignment(align); - MemoryLayout alignedGroup = MemoryLayout.ofStruct(MemoryLayouts.PAD_8, aligned); + MemoryLayout alignedGroup = MemoryLayout.structLayout(MemoryLayouts.PAD_8, aligned); assertEquals(alignedGroup.bitAlignment(), align); VarHandle vh = aligned.varHandle(int.class); - try (MemorySegment segment = MemorySegment.allocateNative(alignedGroup)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(alignedGroup, scope); vh.set(segment.asSlice(1L), -42); assertEquals(align, 8); //this is the only case where access is aligned } catch (IllegalStateException ex) { @@ -76,7 +79,7 @@ public class TestMemoryAlignment { public void testUnalignedPath(long align) { MemoryLayout layout = MemoryLayouts.BITS_32_BE; MemoryLayout aligned = layout.withBitAlignment(align).withName("value"); - GroupLayout alignedGroup = MemoryLayout.ofStruct(MemoryLayouts.PAD_8, aligned); + GroupLayout alignedGroup = MemoryLayout.structLayout(MemoryLayouts.PAD_8, aligned); try { alignedGroup.varHandle(int.class, PathElement.groupElement("value")); assertEquals(align, 8); //this is the only case where path is aligned @@ -87,10 +90,11 @@ public class TestMemoryAlignment { @Test(dataProvider = "alignments") public void testUnalignedSequence(long align) { - SequenceLayout layout = MemoryLayout.ofSequence(5, MemoryLayouts.BITS_32_BE.withBitAlignment(align)); + SequenceLayout layout = MemoryLayout.sequenceLayout(5, MemoryLayouts.BITS_32_BE.withBitAlignment(align)); try { VarHandle vh = layout.varHandle(int.class, PathElement.sequenceElement()); - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); for (long i = 0 ; i < 5 ; i++) { vh.set(segment, i, -42); } @@ -106,14 +110,15 @@ public class TestMemoryAlignment { ValueLayout vShort = MemoryLayouts.BITS_16_BE; ValueLayout vInt = MemoryLayouts.BITS_32_BE; //mimic pragma pack(1) - GroupLayout g = MemoryLayout.ofStruct(vChar.withBitAlignment(8).withName("a"), + GroupLayout g = MemoryLayout.structLayout(vChar.withBitAlignment(8).withName("a"), vShort.withBitAlignment(8).withName("b"), vInt.withBitAlignment(8).withName("c")); assertEquals(g.bitAlignment(), 8); VarHandle vh_c = g.varHandle(byte.class, PathElement.groupElement("a")); VarHandle vh_s = g.varHandle(short.class, PathElement.groupElement("b")); VarHandle vh_i = g.varHandle(int.class, PathElement.groupElement("c")); - try (MemorySegment segment = MemorySegment.allocateNative(g)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(g, scope); vh_c.set(segment, Byte.MIN_VALUE); assertEquals(vh_c.get(segment), Byte.MIN_VALUE); vh_s.set(segment, Short.MIN_VALUE); diff --git a/test/jdk/java/foreign/TestMemoryCopy.java b/test/jdk/java/foreign/TestMemoryCopy.java index 4e5bfc974b9..cba9186f124 100644 --- a/test/jdk/java/foreign/TestMemoryCopy.java +++ b/test/jdk/java/foreign/TestMemoryCopy.java @@ -29,6 +29,7 @@ import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; @@ -64,7 +65,7 @@ public class TestMemoryCopy { static class SegmentSlice { enum Kind { - NATIVE(MemorySegment::allocateNative), + NATIVE(i -> MemorySegment.allocateNative(i, ResourceScope.newImplicitScope())), ARRAY(i -> MemorySegment.ofArray(new byte[i])); final IntFunction segmentFactory; diff --git a/test/jdk/java/foreign/TestMemoryHandleAsUnsigned.java b/test/jdk/java/foreign/TestMemoryHandleAsUnsigned.java index a0678a7083b..d4a6e536234 100644 --- a/test/jdk/java/foreign/TestMemoryHandleAsUnsigned.java +++ b/test/jdk/java/foreign/TestMemoryHandleAsUnsigned.java @@ -22,17 +22,17 @@ * */ -import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayout.PathElement; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; import java.lang.invoke.VarHandle; -import java.nio.ByteOrder; import java.util.Arrays; import java.util.stream.IntStream; import java.util.stream.LongStream; + +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.*; import static java.nio.ByteOrder.BIG_ENDIAN; import static org.testng.Assert.*; @@ -58,7 +58,8 @@ public class TestMemoryHandleAsUnsigned { VarHandle byteHandle = layout.varHandle(byte.class); VarHandle intHandle = MemoryHandles.asUnsigned(byteHandle, int.class); - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); intHandle.set(segment, intValue); int expectedIntValue = Byte.toUnsignedInt(byteValue); assertEquals((int) intHandle.get(segment), expectedIntValue); @@ -80,7 +81,8 @@ public class TestMemoryHandleAsUnsigned { VarHandle byteHandle = layout.varHandle(byte.class); VarHandle longHandle = MemoryHandles.asUnsigned(byteHandle, long.class); - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); longHandle.set(segment, longValue); long expectedLongValue = Byte.toUnsignedLong(byteValue); assertEquals((long) longHandle.get(segment), expectedLongValue); @@ -102,7 +104,8 @@ public class TestMemoryHandleAsUnsigned { VarHandle shortHandle = layout.varHandle(short.class); VarHandle intHandle = MemoryHandles.asUnsigned(shortHandle, int.class); - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); intHandle.set(segment, intValue); int expectedIntValue = Short.toUnsignedInt(shortValue); assertEquals((int) intHandle.get(segment), expectedIntValue); @@ -124,7 +127,8 @@ public class TestMemoryHandleAsUnsigned { VarHandle shortHandle = layout.varHandle(short.class); VarHandle longHandle = MemoryHandles.asUnsigned(shortHandle, long.class); - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); longHandle.set(segment, longValue); long expectedLongValue = Short.toUnsignedLong(shortValue); assertEquals((long) longHandle.get(segment), expectedLongValue); @@ -150,7 +154,8 @@ public class TestMemoryHandleAsUnsigned { VarHandle intHandle = layout.varHandle(int.class); VarHandle longHandle = MemoryHandles.asUnsigned(intHandle, long.class); - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); longHandle.set(segment, longValue); long expectedLongValue = Integer.toUnsignedLong(intValue); assertEquals((long) longHandle.get(segment), expectedLongValue); @@ -160,11 +165,12 @@ public class TestMemoryHandleAsUnsigned { @Test public void testCoordinatesSequenceLayout() { - MemoryLayout layout = MemoryLayout.ofSequence(2, MemoryLayouts.BITS_8_BE); + MemoryLayout layout = MemoryLayout.sequenceLayout(2, MemoryLayouts.BITS_8_BE); VarHandle byteHandle = layout.varHandle(byte.class, PathElement.sequenceElement()); VarHandle intHandle = MemoryHandles.asUnsigned(byteHandle, int.class); - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); intHandle.set(segment, 0L, (int) -1); assertEquals((int) intHandle.get(segment, 0L), 255); intHandle.set(segment, 1L, (int) 200); @@ -178,13 +184,13 @@ public class TestMemoryHandleAsUnsigned { MemorySegment segment = MemorySegment.ofArray(arr); { - VarHandle byteHandle = MemoryLayout.ofSequence(MemoryLayouts.JAVA_BYTE) + VarHandle byteHandle = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_BYTE) .varHandle(byte.class, PathElement.sequenceElement()); VarHandle intHandle = MemoryHandles.asUnsigned(byteHandle, int.class); assertEquals((int) intHandle.get(segment, 2L), 129); } { - VarHandle byteHandle = MemoryLayout.ofSequence(MemoryLayouts.JAVA_BYTE) + VarHandle byteHandle = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_BYTE) .varHandle(byte.class, PathElement.sequenceElement()); VarHandle intHandle = MemoryHandles.asUnsigned(byteHandle, int.class); assertEquals((int) intHandle.get(segment, 2L), 129); diff --git a/test/jdk/java/foreign/TestMismatch.java b/test/jdk/java/foreign/TestMismatch.java index 7d1401a3cf5..5f4dd4c3195 100644 --- a/test/jdk/java/foreign/TestMismatch.java +++ b/test/jdk/java/foreign/TestMismatch.java @@ -31,13 +31,13 @@ import java.util.ArrayList; import java.util.List; import java.util.concurrent.atomic.AtomicReference; import java.util.function.IntFunction; -import jdk.incubator.foreign.MemoryAddress; + import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; import static java.lang.System.out; -import static jdk.incubator.foreign.MemorySegment.READ; import static org.testng.Assert.assertEquals; import static org.testng.Assert.assertThrows; @@ -101,7 +101,8 @@ public class TestMismatch { public void testEmpty() { var s1 = MemorySegment.ofArray(new byte[0]); assertEquals(s1.mismatch(s1), -1); - try (var nativeSegment = MemorySegment.allocateNative(4)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + var nativeSegment = MemorySegment.allocateNative(4, 4, scope); var s2 = nativeSegment.asSlice(0, 0); assertEquals(s1.mismatch(s2), -1); assertEquals(s2.mismatch(s1), -1); @@ -112,8 +113,9 @@ public class TestMismatch { public void testLarge() { // skip if not on 64 bits if (MemoryLayouts.ADDRESS.byteSize() > 32) { - try (var s1 = MemorySegment.allocateNative((long) Integer.MAX_VALUE + 10L); - var s2 = MemorySegment.allocateNative((long) Integer.MAX_VALUE + 10L)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + var s1 = MemorySegment.allocateNative((long) Integer.MAX_VALUE + 10L, 8, scope); + var s2 = MemorySegment.allocateNative((long) Integer.MAX_VALUE + 10L, 8, scope); assertEquals(s1.mismatch(s1), -1); assertEquals(s1.mismatch(s2), -1); assertEquals(s2.mismatch(s1), -1); @@ -149,69 +151,61 @@ public class TestMismatch { @Test public void testClosed() { - var s1 = MemorySegment.ofArray(new byte[4]); - var s2 = MemorySegment.ofArray(new byte[4]); - s1.close(); + MemorySegment s1, s2; + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + s1 = MemorySegment.allocateNative(4, 1, scope); + s2 = MemorySegment.allocateNative(4, 1, scope); + } assertThrows(ISE, () -> s1.mismatch(s1)); assertThrows(ISE, () -> s1.mismatch(s2)); assertThrows(ISE, () -> s2.mismatch(s1)); } - @Test - public void testInsufficientAccessModes() { - var s1 = MemorySegment.ofArray(new byte[4]); - var s2 = MemorySegment.ofArray(new byte[4]); - var s1WithoutRead = s1.withAccessModes(s1.accessModes() & ~READ); - var s2WithoutRead = s2.withAccessModes(s2.accessModes() & ~READ); - - assertThrows(UOE, () -> s1.mismatch(s2WithoutRead)); - assertThrows(UOE, () -> s1WithoutRead.mismatch(s2)); - assertThrows(UOE, () -> s1WithoutRead.mismatch(s2WithoutRead)); - } - @Test public void testThreadAccess() throws Exception { - var segment = MemorySegment.ofArray(new byte[4]); - { - AtomicReference exception = new AtomicReference<>(); - Runnable action = () -> { - try { - MemorySegment.ofArray(new byte[4]).mismatch(segment); - } catch (RuntimeException e) { - exception.set(e); - } - }; - Thread thread = new Thread(action); - thread.start(); - thread.join(); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + var segment = MemorySegment.allocateNative(4, 1, scope); + { + AtomicReference exception = new AtomicReference<>(); + Runnable action = () -> { + try { + MemorySegment.ofArray(new byte[4]).mismatch(segment); + } catch (RuntimeException e) { + exception.set(e); + } + }; + Thread thread = new Thread(action); + thread.start(); + thread.join(); - RuntimeException e = exception.get(); - if (!(e instanceof IllegalStateException)) { - throw e; + RuntimeException e = exception.get(); + if (!(e instanceof IllegalStateException)) { + throw e; + } } - } - { - AtomicReference exception = new AtomicReference<>(); - Runnable action = () -> { - try { - segment.mismatch(MemorySegment.ofArray(new byte[4])); - } catch (RuntimeException e) { - exception.set(e); - } - }; - Thread thread = new Thread(action); - thread.start(); - thread.join(); + { + AtomicReference exception = new AtomicReference<>(); + Runnable action = () -> { + try { + segment.mismatch(MemorySegment.ofArray(new byte[4])); + } catch (RuntimeException e) { + exception.set(e); + } + }; + Thread thread = new Thread(action); + thread.start(); + thread.join(); - RuntimeException e = exception.get(); - if (!(e instanceof IllegalStateException)) { - throw e; + RuntimeException e = exception.get(); + if (!(e instanceof IllegalStateException)) { + throw e; + } } } } enum SegmentKind { - NATIVE(MemorySegment::allocateNative), + NATIVE(i -> MemorySegment.allocateNative(i, ResourceScope.newImplicitScope())), ARRAY(i -> MemorySegment.ofArray(new byte[i])); final IntFunction segmentFactory; diff --git a/test/jdk/java/foreign/TestNative.java b/test/jdk/java/foreign/TestNative.java index c8114e7426e..443c13b11d5 100644 --- a/test/jdk/java/foreign/TestNative.java +++ b/test/jdk/java/foreign/TestNative.java @@ -26,7 +26,7 @@ * @test * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" * @modules jdk.incubator.foreign/jdk.internal.foreign - * @run testng/othervm -Dforeign.restricted=permit TestNative + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestNative */ import jdk.incubator.foreign.CLinker; @@ -36,6 +36,7 @@ import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayout.PathElement; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; @@ -54,36 +55,36 @@ import java.util.function.BiConsumer; import java.util.function.BiFunction; import java.util.function.Consumer; import java.util.function.Function; -import static jdk.incubator.foreign.MemorySegment.*; + import static org.testng.Assert.*; public class TestNative { - static SequenceLayout bytes = MemoryLayout.ofSequence(100, + static SequenceLayout bytes = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_BYTE.withOrder(ByteOrder.nativeOrder()) ); - static SequenceLayout chars = MemoryLayout.ofSequence(100, + static SequenceLayout chars = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_CHAR.withOrder(ByteOrder.nativeOrder()) ); - static SequenceLayout shorts = MemoryLayout.ofSequence(100, + static SequenceLayout shorts = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_SHORT.withOrder(ByteOrder.nativeOrder()) ); - static SequenceLayout ints = MemoryLayout.ofSequence(100, + static SequenceLayout ints = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_INT.withOrder(ByteOrder.nativeOrder()) ); - static SequenceLayout floats = MemoryLayout.ofSequence(100, + static SequenceLayout floats = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_FLOAT.withOrder(ByteOrder.nativeOrder()) ); - static SequenceLayout longs = MemoryLayout.ofSequence(100, + static SequenceLayout longs = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_LONG.withOrder(ByteOrder.nativeOrder()) ); - static SequenceLayout doubles = MemoryLayout.ofSequence(100, + static SequenceLayout doubles = MemoryLayout.sequenceLayout(100, MemoryLayouts.JAVA_DOUBLE.withOrder(ByteOrder.nativeOrder()) ); @@ -144,16 +145,17 @@ public class TestNative { public static native long getCapacity(Buffer buffer); public static MemoryAddress allocate(int size) { - return CLinker.allocateMemoryRestricted(size); + return CLinker.allocateMemory(size); } public static void free(MemoryAddress addr) { - CLinker.freeMemoryRestricted(addr); + CLinker.freeMemory(addr); } @Test(dataProvider="nativeAccessOps") public void testNativeAccess(Consumer checker, Consumer initializer, SequenceLayout seq) { - try (MemorySegment segment = MemorySegment.allocateNative(seq)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(seq, scope); initializer.accept(segment); checker.accept(segment); } @@ -162,7 +164,8 @@ public class TestNative { @Test(dataProvider="buffers") public void testNativeCapacity(Function bufferFunction, int elemSize) { int capacity = (int)doubles.byteSize(); - try (MemorySegment segment = MemorySegment.allocateNative(doubles)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(doubles, scope); ByteBuffer bb = segment.asByteBuffer(); Buffer buf = bufferFunction.apply(bb); int expected = capacity / elemSize; @@ -174,33 +177,34 @@ public class TestNative { @Test public void testDefaultAccessModes() { MemoryAddress addr = allocate(12); - MemorySegment mallocSegment = addr.asSegmentRestricted(12, () -> free(addr), null); - try (MemorySegment segment = mallocSegment) { - assertTrue(segment.hasAccessModes(ALL_ACCESS)); - assertEquals(segment.accessModes(), ALL_ACCESS); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment mallocSegment = addr.asSegment(12, () -> free(addr), scope); + assertFalse(mallocSegment.isReadOnly()); } } @Test public void testDefaultAccessModesEverthing() { - MemorySegment everything = MemorySegment.ofNativeRestricted(); - assertTrue(everything.hasAccessModes(READ | WRITE)); - assertEquals(everything.accessModes(), READ | WRITE); + MemorySegment everything = MemorySegment.globalNativeSegment(); + assertFalse(everything.isReadOnly()); } @Test public void testMallocSegment() { MemoryAddress addr = allocate(12); - MemorySegment mallocSegment = addr.asSegmentRestricted(12, () -> free(addr), null); - assertEquals(mallocSegment.byteSize(), 12); - mallocSegment.close(); //free here - assertTrue(!mallocSegment.isAlive()); + MemorySegment mallocSegment = null; + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + mallocSegment = addr.asSegment(12, () -> free(addr), scope); + assertEquals(mallocSegment.byteSize(), 12); + //free here + } + assertTrue(!mallocSegment.scope().isAlive()); } @Test public void testEverythingSegment() { MemoryAddress addr = allocate(4); - MemorySegment everything = MemorySegment.ofNativeRestricted(); + MemorySegment everything = MemorySegment.globalNativeSegment(); MemoryAccess.setIntAtOffset(everything, addr.toRawLongValue(), 42); assertEquals(MemoryAccess.getIntAtOffset(everything, addr.toRawLongValue()), 42); free(addr); @@ -208,8 +212,9 @@ public class TestNative { @Test(expectedExceptions = IllegalArgumentException.class) public void testBadResize() { - try (MemorySegment segment = MemorySegment.allocateNative(4)) { - segment.address().asSegmentRestricted(0); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(4, 1, scope); + segment.address().asSegment(0, ResourceScope.globalScope()); } } diff --git a/test/jdk/java/foreign/TestNativeScope.java b/test/jdk/java/foreign/TestNativeScope.java deleted file mode 100644 index 78102ea194b..00000000000 --- a/test/jdk/java/foreign/TestNativeScope.java +++ /dev/null @@ -1,536 +0,0 @@ -/* - * 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 - * @run testng/othervm TestNativeScope - */ - -import jdk.incubator.foreign.*; - -import org.testng.annotations.*; - -import java.lang.invoke.VarHandle; -import java.lang.reflect.Array; -import java.lang.reflect.InvocationTargetException; -import java.lang.reflect.Method; -import java.lang.reflect.Modifier; -import java.nio.ByteBuffer; -import java.nio.ByteOrder; -import java.nio.DoubleBuffer; -import java.nio.FloatBuffer; -import java.nio.IntBuffer; -import java.nio.LongBuffer; -import java.nio.ShortBuffer; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.concurrent.atomic.AtomicBoolean; -import java.util.function.Function; -import java.util.function.Predicate; -import java.util.stream.IntStream; -import java.util.stream.LongStream; - -import static jdk.incubator.foreign.MemorySegment.CLOSE; -import static jdk.incubator.foreign.MemorySegment.HANDOFF; -import static org.testng.Assert.*; - -public class TestNativeScope { - - final static int ELEMS = 128; - final static Class ADDRESS_CARRIER = MemoryLayouts.ADDRESS.bitSize() == 64 ? long.class : int.class; - - @Test(dataProvider = "nativeScopes") - public void testAllocation(Z value, ScopeFactory scopeFactory, ValueLayout layout, AllocationFunction allocationFunction, Function handleFactory) { - ValueLayout[] layouts = { - layout, - layout.withBitAlignment(layout.bitAlignment() * 2), - layout.withBitAlignment(layout.bitAlignment() * 4), - layout.withBitAlignment(layout.bitAlignment() * 8) - }; - for (ValueLayout alignedLayout : layouts) { - List addressList = new ArrayList<>(); - int elems = ELEMS / ((int)alignedLayout.byteAlignment() / (int)layout.byteAlignment()); - try (NativeScope scope = scopeFactory.make((int)alignedLayout.byteSize() * ELEMS)) { - for (int i = 0 ; i < elems ; i++) { - MemorySegment address = allocationFunction.allocate(scope, alignedLayout, value); - assertEquals(address.byteSize(), alignedLayout.byteSize()); - addressList.add(address); - VarHandle handle = handleFactory.apply(alignedLayout); - assertEquals(value, handle.get(address)); - try { - address.close(); - fail(); - } catch (UnsupportedOperationException uoe) { - //failure is expected - assertTrue(true); - } - } - boolean isBound = scope.byteSize().isPresent(); - try { - allocationFunction.allocate(scope, alignedLayout, value); //too much, should fail if bound - assertFalse(isBound); - } catch (OutOfMemoryError ex) { - //failure is expected if bound - assertTrue(isBound); - } - } - // addresses should be invalid now - for (MemorySegment address : addressList) { - assertFalse(address.isAlive()); - } - } - } - - static final int SIZE_256M = 1024 * 1024 * 256; - - @Test - public void testBigAllocationInUnboundedScope() { - try (NativeScope scope = NativeScope.unboundedScope()) { - for (int i = 8 ; i < SIZE_256M ; i *= 8) { - MemorySegment address = scope.allocate(i); - //check size - assertEquals(address.byteSize(), i); - //check alignment - assertTrue(address.address().toRawLongValue() % i == 0); - } - } - } - - @Test - public void testAttachClose() { - MemorySegment s1 = MemorySegment.ofArray(new byte[1]); - MemorySegment s2 = MemorySegment.ofArray(new byte[1]); - MemorySegment s3 = MemorySegment.ofArray(new byte[1]); - assertTrue(s1.isAlive()); - assertTrue(s2.isAlive()); - assertTrue(s3.isAlive()); - try (NativeScope scope = NativeScope.boundedScope(10)) { - MemorySegment ss1 = s1.handoff(scope); - assertFalse(s1.isAlive()); - assertTrue(ss1.isAlive()); - s1 = ss1; - MemorySegment ss2 = s2.handoff(scope); - assertFalse(s2.isAlive()); - assertTrue(ss2.isAlive()); - s2 = ss2; - MemorySegment ss3 = s3.handoff(scope); - assertFalse(s3.isAlive()); - assertTrue(ss3.isAlive()); - s3 = ss3; - } - assertFalse(s1.isAlive()); - assertFalse(s2.isAlive()); - assertFalse(s3.isAlive()); - } - - @Test - public void testNoTerminalOps() { - try (NativeScope scope = NativeScope.boundedScope(10)) { - MemorySegment s1 = MemorySegment.ofArray(new byte[1]); - MemorySegment attached = s1.handoff(scope); - int[] terminalOps = {CLOSE, HANDOFF}; - for (int mode : terminalOps) { - if (attached.hasAccessModes(mode)) { - fail(); - } - } - } - } - - @Test(expectedExceptions = UnsupportedOperationException.class) - public void testNoReattach() { - MemorySegment s1 = MemorySegment.ofArray(new byte[1]); - NativeScope scope1 = NativeScope.boundedScope(10); - NativeScope scope2 = NativeScope.boundedScope(10); - s1.handoff(scope1).handoff(scope2); - } - - @Test(expectedExceptions = IllegalStateException.class) - public void testNotAliveClaim() { - MemorySegment segment = MemorySegment.ofArray(new byte[1]); - segment.close(); - segment.handoff(NativeScope.boundedScope(10)); - } - - @Test - public void testRegisterFromUnconfined() { - MemorySegment unconfined = MemorySegment.allocateNative(10).share(); - NativeScope scope = NativeScope.boundedScope(10); - MemorySegment registered = unconfined.handoff(scope); - assertFalse(unconfined.isAlive()); - assertEquals(registered.ownerThread(), scope.ownerThread()); - scope.close(); - assertFalse(registered.isAlive()); - } - - @Test(dataProvider = "arrayScopes") - public void testArray(ScopeFactory scopeFactory, ValueLayout layout, AllocationFunction allocationFunction, ToArrayHelper arrayHelper) { - Z arr = arrayHelper.array(); - try (NativeScope scope = scopeFactory.make(100)) { - MemorySegment address = allocationFunction.allocate(scope, layout, arr); - Z found = arrayHelper.toArray(address, layout); - assertEquals(found, arr); - } - } - - @DataProvider(name = "nativeScopes") - static Object[][] nativeScopes() { - return new Object[][] { - { (byte)42, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_8_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(byte.class) }, - { (short)42, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_16_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(short.class) }, - { (char)42, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_16_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(char.class) }, - { 42, (ScopeFactory) NativeScope::boundedScope, - MemoryLayouts.BITS_32_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(int.class) }, - { 42f, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_32_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(float.class) }, - { 42L, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_64_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(long.class) }, - { 42d, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_64_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(double.class) }, - { MemoryAddress.ofLong(42), (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.ADDRESS.withOrder(ByteOrder.BIG_ENDIAN), - (AllocationFunction) NativeScope::allocate, - (Function)l -> MemoryHandles.asAddressVarHandle(l.varHandle(ADDRESS_CARRIER)) }, - - { (byte)42, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_8_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(byte.class) }, - { (short)42, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_16_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(short.class) }, - { (char)42, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_16_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(char.class) }, - { 42, (ScopeFactory) NativeScope::boundedScope, - MemoryLayouts.BITS_32_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(int.class) }, - { 42f, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_32_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(float.class) }, - { 42L, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_64_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(long.class) }, - { 42d, (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_64_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(double.class) }, - { MemoryAddress.ofLong(42), (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.ADDRESS.withOrder(ByteOrder.LITTLE_ENDIAN), - (AllocationFunction) NativeScope::allocate, - (Function)l -> MemoryHandles.asAddressVarHandle(l.varHandle(ADDRESS_CARRIER)) }, - - { (byte)42, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_8_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(byte.class) }, - { (short)42, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_16_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(short.class) }, - { (char)42, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_16_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(char.class) }, - { 42, (ScopeFactory)size -> NativeScope.unboundedScope(), - MemoryLayouts.BITS_32_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(int.class) }, - { 42f, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_32_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(float.class) }, - { 42L, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_64_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(long.class) }, - { 42d, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_64_BE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(double.class) }, - { MemoryAddress.ofLong(42), (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.ADDRESS.withOrder(ByteOrder.BIG_ENDIAN), - (AllocationFunction) NativeScope::allocate, - (Function)l -> MemoryHandles.asAddressVarHandle(l.varHandle(ADDRESS_CARRIER)) }, - - { (byte)42, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_8_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(byte.class) }, - { (short)42, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_16_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(short.class) }, - { (char)42, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_16_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(char.class) }, - { 42, (ScopeFactory)size -> NativeScope.unboundedScope(), - MemoryLayouts.BITS_32_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(int.class) }, - { 42f, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_32_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(float.class) }, - { 42L, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_64_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(long.class) }, - { 42d, (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_64_LE, - (AllocationFunction) NativeScope::allocate, - (Function)l -> l.varHandle(double.class) }, - { MemoryAddress.ofLong(42), (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.ADDRESS.withOrder(ByteOrder.LITTLE_ENDIAN), - (AllocationFunction) NativeScope::allocate, - (Function)l -> MemoryHandles.asAddressVarHandle(l.varHandle(ADDRESS_CARRIER)) }, - }; - } - - @DataProvider(name = "arrayScopes") - static Object[][] arrayScopes() { - return new Object[][] { - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_8_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toByteArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_16_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toShortArray }, - { (ScopeFactory) NativeScope::boundedScope, - MemoryLayouts.BITS_32_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toIntArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_32_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toFloatArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_64_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toLongArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_64_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toDoubleArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.ADDRESS.withOrder(ByteOrder.LITTLE_ENDIAN), - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toAddressArray }, - - - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_8_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toByteArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_16_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toShortArray }, - { (ScopeFactory) NativeScope::boundedScope, - MemoryLayouts.BITS_32_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toIntArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_32_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toFloatArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_64_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toLongArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.BITS_64_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toDoubleArray }, - { (ScopeFactory) NativeScope::boundedScope, MemoryLayouts.ADDRESS.withOrder(ByteOrder.BIG_ENDIAN), - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toAddressArray }, - - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_8_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toByteArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_16_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toShortArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), - MemoryLayouts.BITS_32_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toIntArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_32_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toFloatArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_64_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toLongArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_64_LE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toDoubleArray }, - { (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.ADDRESS.withOrder(ByteOrder.LITTLE_ENDIAN), - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toAddressArray }, - - - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_8_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toByteArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_16_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toShortArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), - MemoryLayouts.BITS_32_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toIntArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_32_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toFloatArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_64_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toLongArray }, - { (ScopeFactory) size -> NativeScope.unboundedScope(), MemoryLayouts.BITS_64_BE, - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toDoubleArray }, - { (ScopeFactory)size -> NativeScope.unboundedScope(), MemoryLayouts.ADDRESS.withOrder(ByteOrder.BIG_ENDIAN), - (AllocationFunction) NativeScope::allocateArray, - ToArrayHelper.toAddressArray }, - }; - } - - interface AllocationFunction { - MemorySegment allocate(NativeScope scope, ValueLayout layout, X value); - } - - interface ScopeFactory { - NativeScope make(int size); - } - - interface ToArrayHelper { - T array(); - T toArray(MemorySegment segment, ValueLayout layout); - - ToArrayHelper toByteArray = new ToArrayHelper<>() { - @Override - public byte[] array() { - return new byte[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; - } - - @Override - public byte[] toArray(MemorySegment segment, ValueLayout layout) { - ByteBuffer buffer = segment.asByteBuffer().order(layout.order()); - byte[] found = new byte[buffer.limit()]; - buffer.get(found); - return found; - } - }; - - ToArrayHelper toShortArray = new ToArrayHelper<>() { - @Override - public short[] array() { - return new short[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; - } - - @Override - public short[] toArray(MemorySegment segment, ValueLayout layout) { - ShortBuffer buffer = segment.asByteBuffer().order(layout.order()).asShortBuffer(); - short[] found = new short[buffer.limit()]; - buffer.get(found); - return found; - } - }; - - ToArrayHelper toIntArray = new ToArrayHelper<>() { - @Override - public int[] array() { - return new int[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; - } - - @Override - public int[] toArray(MemorySegment segment, ValueLayout layout) { - IntBuffer buffer = segment.asByteBuffer().order(layout.order()).asIntBuffer(); - int[] found = new int[buffer.limit()]; - buffer.get(found); - return found; - } - }; - - ToArrayHelper toFloatArray = new ToArrayHelper<>() { - @Override - public float[] array() { - return new float[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; - } - - @Override - public float[] toArray(MemorySegment segment, ValueLayout layout) { - FloatBuffer buffer = segment.asByteBuffer().order(layout.order()).asFloatBuffer(); - float[] found = new float[buffer.limit()]; - buffer.get(found); - return found; - } - }; - - ToArrayHelper toLongArray = new ToArrayHelper<>() { - @Override - public long[] array() { - return new long[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; - } - - @Override - public long[] toArray(MemorySegment segment, ValueLayout layout) { - LongBuffer buffer = segment.asByteBuffer().order(layout.order()).asLongBuffer(); - long[] found = new long[buffer.limit()]; - buffer.get(found); - return found; - } - }; - - ToArrayHelper toDoubleArray = new ToArrayHelper<>() { - @Override - public double[] array() { - return new double[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; - } - - @Override - public double[] toArray(MemorySegment segment, ValueLayout layout) { - DoubleBuffer buffer = segment.asByteBuffer().order(layout.order()).asDoubleBuffer(); - double[] found = new double[buffer.limit()]; - buffer.get(found); - return found; - } - }; - - ToArrayHelper toAddressArray = new ToArrayHelper<>() { - @Override - public MemoryAddress[] array() { - return switch ((int)MemoryLayouts.ADDRESS.byteSize()) { - case 4 -> wrap(toIntArray.array()); - case 8 -> wrap(toLongArray.array()); - default -> throw new IllegalStateException("Cannot get here"); - }; - } - - @Override - public MemoryAddress[] toArray(MemorySegment segment, ValueLayout layout) { - return switch ((int)layout.byteSize()) { - case 4 -> wrap(toIntArray.toArray(segment, layout)); - case 8 -> wrap(toLongArray.toArray(segment, layout)); - default -> throw new IllegalStateException("Cannot get here"); - }; - } - - private MemoryAddress[] wrap(int[] ints) { - return IntStream.of(ints).mapToObj(MemoryAddress::ofLong).toArray(MemoryAddress[]::new); - } - - private MemoryAddress[] wrap(long[] ints) { - return LongStream.of(ints).mapToObj(MemoryAddress::ofLong).toArray(MemoryAddress[]::new); - } - }; - } -} diff --git a/test/jdk/java/foreign/TestNoForeignUnsafeOverride.java b/test/jdk/java/foreign/TestNoForeignUnsafeOverride.java index c39796ea212..ac47ea3113f 100644 --- a/test/jdk/java/foreign/TestNoForeignUnsafeOverride.java +++ b/test/jdk/java/foreign/TestNoForeignUnsafeOverride.java @@ -21,23 +21,14 @@ * questions. */ -/* - * @test - * @run testng TestNoForeignUnsafeOverride - */ +package org.openjdk.foreigntest; -import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.*; -import jdk.incubator.foreign.MemorySegment; -import org.testng.annotations.Test; - -public class TestNoForeignUnsafeOverride { - static { - System.setProperty("foreign.restricted", "permit"); - } - - @Test(expectedExceptions = IllegalAccessError.class) - public void testUnsafeAccess() { - MemorySegment.ofNativeRestricted(); - } +public class PanamaMain { + public static void main(String[] args) { + System.out.println("Trying to get CLinker"); + CLinker.getInstance(); + System.out.println("Got CLinker"); + } } diff --git a/test/jdk/java/foreign/TestNulls.java b/test/jdk/java/foreign/TestNulls.java index 888bd0b93b0..5f773adf70c 100644 --- a/test/jdk/java/foreign/TestNulls.java +++ b/test/jdk/java/foreign/TestNulls.java @@ -26,7 +26,9 @@ * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" * @modules java.base/jdk.internal.ref * jdk.incubator.foreign - * @run testng/othervm -Dforeign.restricted=permit TestNulls + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * TestNulls */ import jdk.incubator.foreign.*; @@ -53,6 +55,7 @@ import java.nio.file.Path; import java.util.*; import java.util.concurrent.atomic.AtomicReference; import java.util.function.Consumer; +import java.util.function.Supplier; import java.util.function.UnaryOperator; import java.util.stream.Collectors; import java.util.stream.Stream; @@ -82,21 +85,22 @@ public class TestNulls { GroupLayout.class, Addressable.class, MemoryAccess.class, - MappedMemorySegments.class, MemoryLayouts.class, MemoryHandles.class, - NativeScope.class, CLinker.class, CLinker.VaList.class, CLinker.VaList.Builder.class, FunctionDescriptor.class, - LibraryLookup.class + LibraryLookup.class, + SegmentAllocator.class, + ResourceScope.class }; static final Set EXCLUDE_LIST = Set.of( "jdk.incubator.foreign.MemoryLayout/withAttribute(java.lang.String,java.lang.constant.Constable)/1/0", - "jdk.incubator.foreign.MemoryAddress/asSegmentRestricted(long,java.lang.Runnable,java.lang.Object)/1/0", - "jdk.incubator.foreign.MemoryAddress/asSegmentRestricted(long,java.lang.Runnable,java.lang.Object)/2/0", + "jdk.incubator.foreign.MemoryAddress/asSegment(long,java.lang.Runnable,java.lang.Object)/1/0", + "jdk.incubator.foreign.MemoryAddress/asSegment(long,java.lang.Runnable,java.lang.Object)/2/0", + "jdk.incubator.foreign.MemoryAddress/asSegment(long,java.lang.Runnable,jdk.incubator.foreign.ResourceScope)/1/0", "jdk.incubator.foreign.SequenceLayout/withAttribute(java.lang.String,java.lang.constant.Constable)/1/0", "jdk.incubator.foreign.ValueLayout/withAttribute(java.lang.String,java.lang.constant.Constable)/1/0", "jdk.incubator.foreign.GroupLayout/withAttribute(java.lang.String,java.lang.constant.Constable)/1/0", @@ -122,6 +126,7 @@ public class TestNulls { addDefaultMapping(float.class, 0f); addDefaultMapping(long.class, 0L); addDefaultMapping(double.class, 0d); + addDefaultMapping(boolean.class, true); addDefaultMapping(ByteOrder.class, ByteOrder.nativeOrder()); addDefaultMapping(Thread.class, Thread.currentThread()); addDefaultMapping(Cleaner.class, CleanerFactory.cleaner()); @@ -144,15 +149,18 @@ public class TestNulls { addDefaultMapping(Addressable.class, MemoryAddress.NULL); addDefaultMapping(MemoryLayout.class, MemoryLayouts.JAVA_INT); addDefaultMapping(ValueLayout.class, MemoryLayouts.JAVA_INT); - addDefaultMapping(GroupLayout.class, MemoryLayout.ofStruct(MemoryLayouts.JAVA_INT)); - addDefaultMapping(SequenceLayout.class, MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT)); + addDefaultMapping(GroupLayout.class, MemoryLayout.structLayout(MemoryLayouts.JAVA_INT)); + addDefaultMapping(SequenceLayout.class, MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT)); addDefaultMapping(MemorySegment.class, MemorySegment.ofArray(new byte[10])); - addDefaultMapping(NativeScope.class, NativeScope.boundedScope(10)); addDefaultMapping(FunctionDescriptor.class, FunctionDescriptor.ofVoid()); addDefaultMapping(CLinker.class, CLinker.getInstance()); addDefaultMapping(CLinker.VaList.class, VaListHelper.vaList); addDefaultMapping(CLinker.VaList.Builder.class, VaListHelper.vaListBuilder); addDefaultMapping(LibraryLookup.class, LibraryLookup.ofDefault()); + addDefaultMapping(ResourceScope.class, ResourceScope.newImplicitScope()); + addDefaultMapping(SegmentAllocator.class, (size, align) -> null); + addDefaultMapping(Supplier.class, () -> null); + addDefaultMapping(ResourceScope.Handle.class, ResourceScope.globalScope().acquire()); } static class VaListHelper { @@ -164,7 +172,7 @@ public class TestNulls { vaList = CLinker.VaList.make(b -> { builderRef.set(b); b.vargFromLong(CLinker.C_LONG_LONG, 42L); - }); + }, ResourceScope.newImplicitScope()); vaListBuilder = builderRef.get(); } } diff --git a/test/jdk/java/foreign/TestRebase.java b/test/jdk/java/foreign/TestRebase.java index 12ccaec7f5a..0396523c8a5 100644 --- a/test/jdk/java/foreign/TestRebase.java +++ b/test/jdk/java/foreign/TestRebase.java @@ -29,12 +29,11 @@ import jdk.incubator.foreign.MemoryAccess; import jdk.incubator.foreign.MemoryAddress; -import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; -import java.lang.invoke.VarHandle; import java.util.ArrayList; import java.util.List; import java.util.function.IntFunction; @@ -81,7 +80,7 @@ public class TestRebase { static class SegmentSlice { enum Kind { - NATIVE(MemorySegment::allocateNative), + NATIVE(i -> MemorySegment.allocateNative(i, ResourceScope.newImplicitScope())), ARRAY(i -> MemorySegment.ofArray(new byte[i])); final IntFunction segmentFactory; diff --git a/test/jdk/java/foreign/TestReshape.java b/test/jdk/java/foreign/TestReshape.java index 2ca842a93ad..dc53e83ee0f 100644 --- a/test/jdk/java/foreign/TestReshape.java +++ b/test/jdk/java/foreign/TestReshape.java @@ -43,7 +43,7 @@ public class TestReshape { @Test(dataProvider = "shapes") public void testReshape(MemoryLayout layout, long[] expectedShape) { long flattenedSize = LongStream.of(expectedShape).reduce(1L, Math::multiplyExact); - SequenceLayout seq_flattened = MemoryLayout.ofSequence(flattenedSize, layout); + SequenceLayout seq_flattened = MemoryLayout.sequenceLayout(flattenedSize, layout); assertDimensions(seq_flattened, flattenedSize); for (long[] shape : new Shape(expectedShape)) { SequenceLayout seq_shaped = seq_flattened.reshape(shape); @@ -54,43 +54,43 @@ public class TestReshape { @Test(expectedExceptions = IllegalArgumentException.class) public void testInvalidReshape() { - SequenceLayout seq = MemoryLayout.ofSequence(4, MemoryLayouts.JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(4, MemoryLayouts.JAVA_INT); seq.reshape(3, 2); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadReshapeInference() { - SequenceLayout seq = MemoryLayout.ofSequence(4, MemoryLayouts.JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(4, MemoryLayouts.JAVA_INT); seq.reshape(-1, -1); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadReshapeParameterZero() { - SequenceLayout seq = MemoryLayout.ofSequence(4, MemoryLayouts.JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(4, MemoryLayouts.JAVA_INT); seq.reshape(0, 4); } @Test(expectedExceptions = IllegalArgumentException.class) public void testBadReshapeParameterNegative() { - SequenceLayout seq = MemoryLayout.ofSequence(4, MemoryLayouts.JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(4, MemoryLayouts.JAVA_INT); seq.reshape(-2, 2); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testReshapeOnUnboundSequence() { - SequenceLayout seq = MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT); seq.reshape(3, 2); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testFlattenOnUnboundSequence() { - SequenceLayout seq = MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT); + SequenceLayout seq = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT); seq.flatten(); } @Test(expectedExceptions = UnsupportedOperationException.class) public void testFlattenOnUnboundNestedSequence() { - SequenceLayout seq = MemoryLayout.ofSequence(4, MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT)); + SequenceLayout seq = MemoryLayout.sequenceLayout(4, MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT)); seq.flatten(); } @@ -124,7 +124,7 @@ public class TestReshape { } } - static MemoryLayout POINT = MemoryLayout.ofStruct( + static MemoryLayout POINT = MemoryLayout.structLayout( MemoryLayouts.JAVA_INT, MemoryLayouts.JAVA_INT ); diff --git a/test/jdk/java/foreign/TestResourceScope.java b/test/jdk/java/foreign/TestResourceScope.java new file mode 100644 index 00000000000..829859677bd --- /dev/null +++ b/test/jdk/java/foreign/TestResourceScope.java @@ -0,0 +1,306 @@ +/* + * 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 java.base/jdk.internal.ref + * jdk.incubator.foreign/jdk.incubator.foreign + * @run testng/othervm TestResourceScope + */ + +import java.lang.ref.Cleaner; + +import jdk.incubator.foreign.ResourceScope; +import jdk.internal.ref.CleanerFactory; + +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; +import static org.testng.Assert.*; + +import java.util.ArrayList; +import java.util.List; +import java.util.concurrent.atomic.AtomicInteger; +import java.util.concurrent.atomic.AtomicReference; +import java.util.function.Supplier; +import java.util.stream.IntStream; + +public class TestResourceScope { + + final static int N_THREADS = 10000; + + @Test(dataProvider = "cleaners") + public void testConfined(Supplier cleanerSupplier) { + AtomicInteger acc = new AtomicInteger(); + Cleaner cleaner = cleanerSupplier.get(); + ResourceScope scope = cleaner != null ? + ResourceScope.newConfinedScope(cleaner) : + ResourceScope.newConfinedScope(); + for (int i = 0 ; i < N_THREADS ; i++) { + int delta = i; + scope.addCloseAction(() -> acc.addAndGet(delta)); + } + assertEquals(acc.get(), 0); + + if (cleaner == null) { + scope.close(); + assertEquals(acc.get(), IntStream.range(0, N_THREADS).sum()); + } else { + scope = null; + int expected = IntStream.range(0, N_THREADS).sum(); + while (acc.get() != expected) { + kickGC(); + } + } + } + + @Test(dataProvider = "cleaners") + public void testSharedSingleThread(Supplier cleanerSupplier) { + AtomicInteger acc = new AtomicInteger(); + Cleaner cleaner = cleanerSupplier.get(); + ResourceScope scope = cleaner != null ? + ResourceScope.newSharedScope(cleaner) : + ResourceScope.newSharedScope(); + for (int i = 0 ; i < N_THREADS ; i++) { + int delta = i; + scope.addCloseAction(() -> acc.addAndGet(delta)); + } + assertEquals(acc.get(), 0); + + if (cleaner == null) { + scope.close(); + assertEquals(acc.get(), IntStream.range(0, N_THREADS).sum()); + } else { + scope = null; + int expected = IntStream.range(0, N_THREADS).sum(); + while (acc.get() != expected) { + kickGC(); + } + } + } + + @Test(dataProvider = "cleaners") + public void testSharedMultiThread(Supplier cleanerSupplier) { + AtomicInteger acc = new AtomicInteger(); + Cleaner cleaner = cleanerSupplier.get(); + List threads = new ArrayList<>(); + ResourceScope scope = cleaner != null ? + ResourceScope.newSharedScope(cleaner) : + ResourceScope.newSharedScope(); + AtomicReference scopeRef = new AtomicReference<>(scope); + for (int i = 0 ; i < N_THREADS ; i++) { + int delta = i; + Thread thread = new Thread(() -> { + try { + scopeRef.get().addCloseAction(() -> { + acc.addAndGet(delta); + }); + } catch (IllegalStateException ex) { + // already closed - we need to call cleanup manually + acc.addAndGet(delta); + } + }); + threads.add(thread); + } + assertEquals(acc.get(), 0); + threads.forEach(Thread::start); + + // if no cleaner, close - not all segments might have been added to the scope! + // if cleaner, don't unset the scope - after all, the scope is kept alive by threads + if (cleaner == null) { + scope.close(); + } + + threads.forEach(t -> { + try { + t.join(); + } catch (InterruptedException ex) { + fail(); + } + }); + + if (cleaner == null) { + assertEquals(acc.get(), IntStream.range(0, N_THREADS).sum()); + } else { + scope = null; + scopeRef.set(null); + int expected = IntStream.range(0, N_THREADS).sum(); + while (acc.get() != expected) { + kickGC(); + } + } + } + + @Test(dataProvider = "cleaners") + public void testLockSingleThread(Supplier cleanerSupplier) { + Cleaner cleaner = cleanerSupplier.get(); + ResourceScope scope = cleaner != null ? + ResourceScope.newConfinedScope(cleaner) : + ResourceScope.newConfinedScope(); + List handles = new ArrayList<>(); + for (int i = 0 ; i < N_THREADS ; i++) { + handles.add(scope.acquire()); + } + + while (true) { + try { + scope.close(); + assertEquals(handles.size(), 0); + break; + } catch (IllegalStateException ex) { + assertTrue(handles.size() > 0); + ResourceScope.Handle handle = handles.remove(0); + scope.release(handle); + scope.release(handle); // make sure it's idempotent + scope.release(handle); // make sure it's idempotent + } + } + } + + @Test(dataProvider = "cleaners") + public void testLockSharedMultiThread(Supplier cleanerSupplier) { + Cleaner cleaner = cleanerSupplier.get(); + ResourceScope scope = cleaner != null ? + ResourceScope.newSharedScope(cleaner) : + ResourceScope.newSharedScope(); + AtomicInteger lockCount = new AtomicInteger(); + for (int i = 0 ; i < N_THREADS ; i++) { + new Thread(() -> { + lockCount.incrementAndGet(); + try { + ResourceScope.Handle handle = scope.acquire(); + waitSomeTime(); + scope.release(handle); + scope.release(handle); // make sure it's idempotent + scope.release(handle); // make sure it's idempotent + } catch (IllegalStateException ex) { + // might be already closed - do nothing + } finally { + lockCount.decrementAndGet(); + } + }).start(); + } + + while (lockCount.get() > 0) { + try { + scope.close(); + assertEquals(lockCount.get(), 0); + break; + } catch (IllegalStateException ex) { + waitSomeTime(); + } + } + } + + @Test + public void testCloseEmptyConfinedScope() { + ResourceScope.newConfinedScope().close(); + } + + @Test + public void testCloseEmptySharedScope() { + ResourceScope.newSharedScope().close(); + } + + @Test + public void testCloseConfinedLock() { + ResourceScope scope = ResourceScope.newConfinedScope(); + ResourceScope.Handle handle = scope.acquire(); + AtomicReference failure = new AtomicReference<>(); + Thread t = new Thread(() -> { + try { + scope.release(handle); + scope.release(handle); // make sure it's idempotent + scope.release(handle); // make sure it's idempotent + } catch (Throwable ex) { + failure.set(ex); + } + }); + t.start(); + try { + t.join(); + assertNotNull(failure.get()); + assertEquals(failure.get().getClass(), IllegalStateException.class); + } catch (Throwable ex) { + throw new AssertionError(ex); + } + } + + @Test(dataProvider = "scopes") + public void testScopeHandles(Supplier scopeFactory) { + ResourceScope scope = scopeFactory.get(); + acquireRecursive(scope, 5); + if (!scope.isImplicit()) { + scope.close(); + } + } + + private void acquireRecursive(ResourceScope scope, int acquireCount) { + ResourceScope.Handle handle = scope.acquire(); + assertEquals(handle.scope(), scope); + if (acquireCount > 0) { + // recursive acquire + acquireRecursive(scope, acquireCount - 1); + } + if (!scope.isImplicit()) { + assertThrows(IllegalStateException.class, scope::close); + } + scope.release(handle); + scope.release(handle); // make sure it's idempotent + scope.release(handle); // make sure it's idempotent + } + + private void waitSomeTime() { + try { + Thread.sleep(10); + } catch (InterruptedException ex) { + // ignore + } + } + + private void kickGC() { + for (int i = 0 ; i < 100 ; i++) { + byte[] b = new byte[100]; + System.gc(); + Thread.onSpinWait(); + } + } + + @DataProvider + static Object[][] cleaners() { + return new Object[][] { + { (Supplier)() -> null }, + { (Supplier)Cleaner::create }, + { (Supplier)CleanerFactory::cleaner } + }; + } + + @DataProvider + static Object[][] scopes() { + return new Object[][] { + { (Supplier)ResourceScope::newConfinedScope }, + { (Supplier)ResourceScope::newSharedScope }, + { (Supplier)ResourceScope::newImplicitScope }, + { (Supplier)ResourceScope::globalScope } + }; + } +} diff --git a/test/jdk/java/foreign/TestRestricted.java b/test/jdk/java/foreign/TestRestricted.java new file mode 100644 index 00000000000..68d23b6eb23 --- /dev/null +++ b/test/jdk/java/foreign/TestRestricted.java @@ -0,0 +1,74 @@ +/* + * Copyright (c) 2021, 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. + */ + +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import org.testng.annotations.Test; + +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; +import java.lang.reflect.Method; +import java.lang.reflect.InvocationTargetException; + +/* + * @test + * @run testng TestRestricted + */ +public class TestRestricted { + @Test(expectedExceptions = InvocationTargetException.class) + public void testReflection() throws Throwable { + Method method = MemorySegment.class.getDeclaredMethod("globalNativeSegment"); + method.invoke(null); + } + + @Test(expectedExceptions = IllegalCallerException.class) + public void testInvoke() throws Throwable { + var mh = MethodHandles.lookup().findStatic(MemorySegment.class, + "globalNativeSegment", MethodType.methodType(MemorySegment.class)); + var seg = (MemorySegment)mh.invokeExact(); + } + + @Test(expectedExceptions = IllegalCallerException.class) + public void testDirectAccess() throws Throwable { + MemorySegment.globalNativeSegment(); + } + + @Test(expectedExceptions = InvocationTargetException.class) + public void testReflection2() throws Throwable { + Method method = MemoryAddress.class.getDeclaredMethod("asSegment", long.class, ResourceScope.class); + method.invoke(MemoryAddress.NULL, 4000L, ResourceScope.globalScope()); + } + + @Test(expectedExceptions = IllegalCallerException.class) + public void testInvoke2() throws Throwable { + var mh = MethodHandles.lookup().findVirtual(MemoryAddress.class, "asSegment", + MethodType.methodType(MemorySegment.class, long.class, ResourceScope.class)); + var seg = (MemorySegment)mh.invokeExact(MemoryAddress.NULL, 4000L, ResourceScope.globalScope()); + } + + @Test(expectedExceptions = IllegalCallerException.class) + public void testDirectAccess2() throws Throwable { + MemoryAddress.NULL.asSegment(4000L, ResourceScope.globalScope()); + } +} diff --git a/test/jdk/java/foreign/TestScopedOperations.java b/test/jdk/java/foreign/TestScopedOperations.java new file mode 100644 index 00000000000..9254a9e5b3a --- /dev/null +++ b/test/jdk/java/foreign/TestScopedOperations.java @@ -0,0 +1,273 @@ +/* + * Copyright (c) 2021, 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 + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestScopedOperations + */ + +import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemoryLayouts; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; +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; +import java.util.ArrayList; +import java.util.List; +import java.util.concurrent.atomic.AtomicReference; +import java.util.function.Consumer; +import java.util.function.Function; + +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") + public void testOpAfterClose(String name, ScopedOperation scopedOperation) { + ResourceScope scope = ResourceScope.newConfinedScope(); + scope.close(); + try { + scopedOperation.accept(scope); + fail(); + } catch (IllegalStateException ex) { + assertTrue(ex.getMessage().contains("closed")); + } + } + + @Test(dataProvider = "scopedOperations") + public void testOpOutsideConfinement(String name, ScopedOperation scopedOperation) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + AtomicReference failed = new AtomicReference<>(); + Thread t = new Thread(() -> { + try { + scopedOperation.accept(scope); + } catch (Throwable ex) { + failed.set(ex); + } + }); + t.start(); + t.join(); + assertNotNull(failed.get()); + assertEquals(failed.get().getClass(), IllegalStateException.class); + assertTrue(failed.get().getMessage().contains("outside")); + } catch (InterruptedException ex) { + throw new AssertionError(ex); + } + } + + static List scopedOperations = new ArrayList<>(); + + static { + // scope operations + ScopedOperation.ofScope(scope -> scope.addCloseAction(() -> { + }), "ResourceScope::addOnClose"); + ScopedOperation.ofScope(scope -> { + ResourceScope.Handle handle = scope.acquire(); + scope.release(handle); + }, "ResourceScope::lock"); + ScopedOperation.ofScope(scope -> MemorySegment.allocateNative(100, scope), "MemorySegment::allocateNative"); + ScopedOperation.ofScope(scope -> { + try { + MemorySegment.mapFile(tempPath, 0, 10, FileChannel.MapMode.READ_WRITE, scope); + } catch (IOException ex) { + fail(); + } + }, "MemorySegment::mapFromFile"); + ScopedOperation.ofScope(scope -> CLinker.VaList.make(b -> {}, scope), "VaList::make"); + ScopedOperation.ofScope(scope -> CLinker.VaList.ofAddress(MemoryAddress.ofLong(42), scope), "VaList::make"); + ScopedOperation.ofScope(scope -> CLinker.toCString("Hello", scope), "CLinker::toCString"); + ScopedOperation.ofScope(SegmentAllocator::arenaAllocator, "SegmentAllocator::arenaAllocator"); + // segment operations + ScopedOperation.ofSegment(MemorySegment::toByteArray, "MemorySegment::toByteArray"); + ScopedOperation.ofSegment(MemorySegment::toCharArray, "MemorySegment::toCharArray"); + ScopedOperation.ofSegment(MemorySegment::toShortArray, "MemorySegment::toShortArray"); + ScopedOperation.ofSegment(MemorySegment::toIntArray, "MemorySegment::toIntArray"); + ScopedOperation.ofSegment(MemorySegment::toFloatArray, "MemorySegment::toFloatArray"); + ScopedOperation.ofSegment(MemorySegment::toLongArray, "MemorySegment::toLongArray"); + ScopedOperation.ofSegment(MemorySegment::toDoubleArray, "MemorySegment::toDoubleArray"); + ScopedOperation.ofSegment(MemorySegment::address, "MemorySegment::address"); + ScopedOperation.ofSegment(s -> MemoryLayout.sequenceLayout(s.byteSize(), MemoryLayouts.JAVA_BYTE), "MemorySegment::spliterator"); + 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"); + // address operations + ScopedOperation.ofAddress(a -> a.toRawLongValue(), "MemoryAddress::toRawLongValue"); + ScopedOperation.ofAddress(a -> a.asSegment(100, ResourceScope.globalScope()), "MemoryAddress::asSegment"); + // valist operations + ScopedOperation.ofVaList(CLinker.VaList::address, "VaList::address"); + ScopedOperation.ofVaList(CLinker.VaList::copy, "VaList::copy"); + ScopedOperation.ofVaList(list -> list.vargAsAddress(MemoryLayouts.ADDRESS), "VaList::vargAsAddress"); + ScopedOperation.ofVaList(list -> list.vargAsInt(MemoryLayouts.JAVA_INT), "VaList::vargAsInt"); + ScopedOperation.ofVaList(list -> list.vargAsLong(MemoryLayouts.JAVA_LONG), "VaList::vargAsLong"); + ScopedOperation.ofVaList(list -> list.vargAsDouble(MemoryLayouts.JAVA_DOUBLE), "VaList::vargAsDouble"); + ScopedOperation.ofVaList(CLinker.VaList::skip, "VaList::skip"); + ScopedOperation.ofVaList(list -> list.vargAsSegment(MemoryLayout.structLayout(MemoryLayouts.JAVA_INT), ResourceScope.newImplicitScope()), "VaList::vargAsSegment/1"); + // allocator operations + ScopedOperation.ofAllocator(a -> a.allocate(1), "NativeAllocator::allocate/size"); + ScopedOperation.ofAllocator(a -> a.allocate(1, 1), "NativeAllocator::allocate/size/align"); + ScopedOperation.ofAllocator(a -> a.allocate(MemoryLayouts.JAVA_BYTE), "NativeAllocator::allocate/layout"); + ScopedOperation.ofAllocator(a -> a.allocate(MemoryLayouts.JAVA_BYTE, (byte) 0), "NativeAllocator::allocate/byte"); + ScopedOperation.ofAllocator(a -> a.allocate(MemoryLayouts.JAVA_CHAR, (char) 0), "NativeAllocator::allocate/char"); + ScopedOperation.ofAllocator(a -> a.allocate(MemoryLayouts.JAVA_SHORT, (short) 0), "NativeAllocator::allocate/short"); + ScopedOperation.ofAllocator(a -> a.allocate(MemoryLayouts.JAVA_INT, 0), "NativeAllocator::allocate/int"); + ScopedOperation.ofAllocator(a -> a.allocate(MemoryLayouts.JAVA_FLOAT, 0f), "NativeAllocator::allocate/float"); + ScopedOperation.ofAllocator(a -> a.allocate(MemoryLayouts.JAVA_LONG, 0L), "NativeAllocator::allocate/long"); + ScopedOperation.ofAllocator(a -> a.allocate(MemoryLayouts.JAVA_DOUBLE, 0d), "NativeAllocator::allocate/double"); + ScopedOperation.ofAllocator(a -> a.allocateArray(MemoryLayouts.JAVA_BYTE, 1L), "NativeAllocator::allocateArray/size"); + ScopedOperation.ofAllocator(a -> a.allocateArray(MemoryLayouts.JAVA_BYTE, new byte[]{0}), "NativeAllocator::allocateArray/byte"); + ScopedOperation.ofAllocator(a -> a.allocateArray(MemoryLayouts.JAVA_CHAR, new char[]{0}), "NativeAllocator::allocateArray/char"); + ScopedOperation.ofAllocator(a -> a.allocateArray(MemoryLayouts.JAVA_SHORT, new short[]{0}), "NativeAllocator::allocateArray/short"); + ScopedOperation.ofAllocator(a -> a.allocateArray(MemoryLayouts.JAVA_INT, new int[]{0}), "NativeAllocator::allocateArray/int"); + ScopedOperation.ofAllocator(a -> a.allocateArray(MemoryLayouts.JAVA_FLOAT, new float[]{0}), "NativeAllocator::allocateArray/float"); + ScopedOperation.ofAllocator(a -> a.allocateArray(MemoryLayouts.JAVA_LONG, new long[]{0}), "NativeAllocator::allocateArray/long"); + ScopedOperation.ofAllocator(a -> a.allocateArray(MemoryLayouts.JAVA_DOUBLE, new double[]{0}), "NativeAllocator::allocateArray/double"); + }; + + @DataProvider(name = "scopedOperations") + static Object[][] scopedOperations() { + return scopedOperations.stream().map(op -> new Object[] { op.name, op }).toArray(Object[][]::new); + } + + static class ScopedOperation implements Consumer { + + final Consumer scopeConsumer; + final String name; + + private ScopedOperation(Consumer scopeConsumer, String name) { + this.scopeConsumer = scopeConsumer; + this.name = name; + } + + @Override + public void accept(ResourceScope scope) { + scopeConsumer.accept(scope); + } + + static void ofScope(Consumer scopeConsumer, String name) { + scopedOperations.add(new ScopedOperation(scopeConsumer::accept, name)); + } + + static void ofVaList(Consumer vaListConsumer, String name) { + scopedOperations.add(new ScopedOperation(scope -> { + CLinker.VaList vaList = CLinker.VaList.make((builder) -> {}, scope); + vaListConsumer.accept(vaList); + }, name)); + } + + static void ofSegment(Consumer segmentConsumer, String name) { + for (SegmentFactory segmentFactory : SegmentFactory.values()) { + scopedOperations.add(new ScopedOperation(scope -> { + MemorySegment segment = segmentFactory.segmentFactory.apply(scope); + segmentConsumer.accept(segment); + }, segmentFactory.name() + "/" + name)); + } + } + + static void ofAddress(Consumer addressConsumer, String name) { + for (SegmentFactory segmentFactory : SegmentFactory.values()) { + scopedOperations.add(new ScopedOperation(scope -> { + MemoryAddress segment = segmentFactory.segmentFactory.apply(scope).address(); + addressConsumer.accept(segment); + }, segmentFactory.name() + "/" + name)); + } + } + + static void ofAllocator(Consumer allocatorConsumer, String name) { + for (AllocatorFactory allocatorFactory : AllocatorFactory.values()) { + scopedOperations.add(new ScopedOperation(scope -> { + SegmentAllocator allocator = allocatorFactory.allocatorFactory.apply(scope); + allocatorConsumer.accept(allocator); + }, allocatorFactory.name() + "/" + name)); + } + } + + enum SegmentFactory { + + NATIVE(scope -> MemorySegment.allocateNative(10, scope)), + MAPPED(scope -> { + try { + return MemorySegment.mapFile(Path.of("foo.txt"), 0, 10, FileChannel.MapMode.READ_WRITE, scope); + } catch (IOException ex) { + throw new AssertionError(ex); + } + }), + UNSAFE(scope -> MemoryAddress.NULL.asSegment(10, scope)); + + static { + try { + File f = new File("foo.txt"); + f.createNewFile(); + f.deleteOnExit(); + } catch (IOException ex) { + throw new ExceptionInInitializerError(ex); + } + } + + final Function segmentFactory; + + SegmentFactory(Function segmentFactory) { + this.segmentFactory = segmentFactory; + } + } + + enum AllocatorFactory { + ARENA_BOUNDED(scope -> SegmentAllocator.arenaAllocator(1000, scope)), + ARENA_UNBOUNDED(SegmentAllocator::arenaAllocator), + FROM_SEGMENT(scope -> { + MemorySegment segment = MemorySegment.allocateNative(10, scope); + return SegmentAllocator.ofSegment(segment); + }), + FROM_SCOPE(SegmentAllocator::ofScope); + + final Function allocatorFactory; + + AllocatorFactory(Function allocatorFactory) { + this.allocatorFactory = allocatorFactory; + } + } + } +} diff --git a/test/jdk/java/foreign/TestSegmentAllocators.java b/test/jdk/java/foreign/TestSegmentAllocators.java new file mode 100644 index 00000000000..3d2538ad897 --- /dev/null +++ b/test/jdk/java/foreign/TestSegmentAllocators.java @@ -0,0 +1,502 @@ +/* + * 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 + * @run testng/othervm TestSegmentAllocators + */ + +import jdk.incubator.foreign.*; + +import org.testng.annotations.*; + +import java.lang.invoke.VarHandle; +import java.nio.ByteBuffer; +import java.nio.ByteOrder; +import java.nio.DoubleBuffer; +import java.nio.FloatBuffer; +import java.nio.IntBuffer; +import java.nio.LongBuffer; +import java.nio.ShortBuffer; +import java.util.ArrayList; +import java.util.List; +import java.util.function.BiFunction; +import java.util.function.Function; +import java.util.stream.IntStream; +import java.util.stream.LongStream; + +import static org.testng.Assert.*; + +public class TestSegmentAllocators { + + final static int ELEMS = 128; + final static Class ADDRESS_CARRIER = MemoryLayouts.ADDRESS.bitSize() == 64 ? long.class : int.class; + + @Test(dataProvider = "nativeScopes") + public void testAllocation(Z value, AllocationFactory allocationFactory, ValueLayout layout, AllocationFunction allocationFunction, Function handleFactory) { + ValueLayout[] layouts = { + layout, + layout.withBitAlignment(layout.bitAlignment() * 2), + layout.withBitAlignment(layout.bitAlignment() * 4), + layout.withBitAlignment(layout.bitAlignment() * 8) + }; + for (ValueLayout alignedLayout : layouts) { + List addressList = new ArrayList<>(); + int elems = ELEMS / ((int)alignedLayout.byteAlignment() / (int)layout.byteAlignment()); + ResourceScope[] scopes = { + ResourceScope.newConfinedScope(), + ResourceScope.newSharedScope() + }; + for (ResourceScope scope : scopes) { + try (scope) { + SegmentAllocator allocator = allocationFactory.allocator(alignedLayout.byteSize() * ELEMS, scope); + for (int i = 0; i < elems; i++) { + MemorySegment address = allocationFunction.allocate(allocator, alignedLayout, value); + assertEquals(address.byteSize(), alignedLayout.byteSize()); + addressList.add(address); + VarHandle handle = handleFactory.apply(alignedLayout); + assertEquals(value, handle.get(address)); + } + boolean isBound = allocationFactory.isBound(); + try { + allocationFunction.allocate(allocator, alignedLayout, value); //too much, should fail if bound + assertFalse(isBound); + } catch (OutOfMemoryError ex) { + //failure is expected if bound + assertTrue(isBound); + } + } + // addresses should be invalid now + for (MemorySegment address : addressList) { + assertFalse(address.scope().isAlive()); + } + } + } + } + + static final int SIZE_256M = 1024 * 1024 * 256; + + @Test + public void testBigAllocationInUnboundedScope() { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + SegmentAllocator allocator = SegmentAllocator.arenaAllocator(scope); + for (int i = 8 ; i < SIZE_256M ; i *= 8) { + MemorySegment address = allocator.allocate(i, i); + //check size + assertEquals(address.byteSize(), i); + //check alignment + assertEquals(address.address().toRawLongValue() % i, 0); + } + } + } + + @Test(expectedExceptions = OutOfMemoryError.class) + public void testTooBigForBoundedArena() { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + SegmentAllocator allocator = SegmentAllocator.arenaAllocator(10, scope); + allocator.allocate(12); + } + } + + @Test + public void testBiggerThanBlockForBoundedArena() { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + SegmentAllocator allocator = SegmentAllocator.arenaAllocator(4 * 1024 * 2, scope); + allocator.allocate(4 * 1024 + 1); // should be ok + } + } + + @Test(dataProvider = "arrayScopes") + public void testArray(AllocationFactory allocationFactory, ValueLayout layout, AllocationFunction allocationFunction, ToArrayHelper arrayHelper) { + Z arr = arrayHelper.array(); + ResourceScope[] scopes = { + ResourceScope.newConfinedScope(), + ResourceScope.newSharedScope() + }; + for (ResourceScope scope : scopes) { + try (scope) { + SegmentAllocator allocator = allocationFactory.allocator(100, scope); + MemorySegment address = allocationFunction.allocate(allocator, layout, arr); + Z found = arrayHelper.toArray(address, layout); + assertEquals(found, arr); + } + } + } + + @DataProvider(name = "nativeScopes") + static Object[][] nativeScopes() { + return new Object[][] { + { (byte)42, AllocationFactory.BOUNDED, MemoryLayouts.BITS_8_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(byte.class) }, + { (short)42, AllocationFactory.BOUNDED, MemoryLayouts.BITS_16_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(short.class) }, + { (char)42, AllocationFactory.BOUNDED, MemoryLayouts.BITS_16_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(char.class) }, + { 42, AllocationFactory.BOUNDED, + MemoryLayouts.BITS_32_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(int.class) }, + { 42f, AllocationFactory.BOUNDED, MemoryLayouts.BITS_32_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(float.class) }, + { 42L, AllocationFactory.BOUNDED, MemoryLayouts.BITS_64_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(long.class) }, + { 42d, AllocationFactory.BOUNDED, MemoryLayouts.BITS_64_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(double.class) }, + { MemoryAddress.ofLong(42), AllocationFactory.BOUNDED, MemoryLayouts.ADDRESS.withOrder(ByteOrder.BIG_ENDIAN), + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> MemoryHandles.asAddressVarHandle(l.varHandle(ADDRESS_CARRIER)) }, + + { (byte)42, AllocationFactory.BOUNDED, MemoryLayouts.BITS_8_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(byte.class) }, + { (short)42, AllocationFactory.BOUNDED, MemoryLayouts.BITS_16_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(short.class) }, + { (char)42, AllocationFactory.BOUNDED, MemoryLayouts.BITS_16_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(char.class) }, + { 42, AllocationFactory.BOUNDED, + MemoryLayouts.BITS_32_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(int.class) }, + { 42f, AllocationFactory.BOUNDED, MemoryLayouts.BITS_32_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(float.class) }, + { 42L, AllocationFactory.BOUNDED, MemoryLayouts.BITS_64_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(long.class) }, + { 42d, AllocationFactory.BOUNDED, MemoryLayouts.BITS_64_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(double.class) }, + { MemoryAddress.ofLong(42), AllocationFactory.BOUNDED, MemoryLayouts.ADDRESS.withOrder(ByteOrder.LITTLE_ENDIAN), + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> MemoryHandles.asAddressVarHandle(l.varHandle(ADDRESS_CARRIER)) }, + + { (byte)42, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_8_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(byte.class) }, + { (short)42, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_16_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(short.class) }, + { (char)42, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_16_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(char.class) }, + { 42, AllocationFactory.UNBOUNDED, + MemoryLayouts.BITS_32_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(int.class) }, + { 42f, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_32_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(float.class) }, + { 42L, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_64_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(long.class) }, + { 42d, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_64_BE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(double.class) }, + { MemoryAddress.ofLong(42), AllocationFactory.UNBOUNDED, MemoryLayouts.ADDRESS.withOrder(ByteOrder.BIG_ENDIAN), + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> MemoryHandles.asAddressVarHandle(l.varHandle(ADDRESS_CARRIER)) }, + + { (byte)42, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_8_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(byte.class) }, + { (short)42, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_16_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(short.class) }, + { (char)42, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_16_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(char.class) }, + { 42, AllocationFactory.UNBOUNDED, + MemoryLayouts.BITS_32_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(int.class) }, + { 42f, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_32_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(float.class) }, + { 42L, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_64_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(long.class) }, + { 42d, AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_64_LE, + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> l.varHandle(double.class) }, + { MemoryAddress.ofLong(42), AllocationFactory.UNBOUNDED, MemoryLayouts.ADDRESS.withOrder(ByteOrder.LITTLE_ENDIAN), + (AllocationFunction) SegmentAllocator::allocate, + (Function)l -> MemoryHandles.asAddressVarHandle(l.varHandle(ADDRESS_CARRIER)) }, + }; + } + + @DataProvider(name = "arrayScopes") + static Object[][] arrayScopes() { + return new Object[][] { + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_8_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toByteArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_16_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toShortArray }, + { AllocationFactory.BOUNDED, + MemoryLayouts.BITS_32_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toIntArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_32_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toFloatArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_64_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toLongArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_64_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toDoubleArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.ADDRESS.withOrder(ByteOrder.LITTLE_ENDIAN), + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toAddressArray }, + + + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_8_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toByteArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_16_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toShortArray }, + { AllocationFactory.BOUNDED, + MemoryLayouts.BITS_32_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toIntArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_32_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toFloatArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_64_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toLongArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.BITS_64_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toDoubleArray }, + { AllocationFactory.BOUNDED, MemoryLayouts.ADDRESS.withOrder(ByteOrder.BIG_ENDIAN), + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toAddressArray }, + + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_8_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toByteArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_16_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toShortArray }, + { AllocationFactory.UNBOUNDED, + MemoryLayouts.BITS_32_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toIntArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_32_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toFloatArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_64_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toLongArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_64_LE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toDoubleArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.ADDRESS.withOrder(ByteOrder.LITTLE_ENDIAN), + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toAddressArray }, + + + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_8_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toByteArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_16_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toShortArray }, + { AllocationFactory.UNBOUNDED, + MemoryLayouts.BITS_32_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toIntArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_32_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toFloatArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_64_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toLongArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.BITS_64_BE, + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toDoubleArray }, + { AllocationFactory.UNBOUNDED, MemoryLayouts.ADDRESS.withOrder(ByteOrder.BIG_ENDIAN), + (AllocationFunction) SegmentAllocator::allocateArray, + ToArrayHelper.toAddressArray }, + }; + } + + interface AllocationFunction { + MemorySegment allocate(SegmentAllocator allocator, ValueLayout layout, X value); + } + + static class AllocationFactory { + private final boolean isBound; + private final BiFunction factory; + + private AllocationFactory(boolean isBound, BiFunction factory) { + this.isBound = isBound; + this.factory = factory; + } + + SegmentAllocator allocator(long size, ResourceScope scope) { + return factory.apply(size, scope); + } + + public boolean isBound() { + return isBound; + } + + static AllocationFactory BOUNDED = new AllocationFactory(true, SegmentAllocator::arenaAllocator); + static AllocationFactory UNBOUNDED = new AllocationFactory(false, (size, scope) -> SegmentAllocator.arenaAllocator(scope)); + } + + interface ToArrayHelper { + T array(); + T toArray(MemorySegment segment, ValueLayout layout); + + ToArrayHelper toByteArray = new ToArrayHelper<>() { + @Override + public byte[] array() { + return new byte[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; + } + + @Override + public byte[] toArray(MemorySegment segment, ValueLayout layout) { + ByteBuffer buffer = segment.asByteBuffer().order(layout.order()); + byte[] found = new byte[buffer.limit()]; + buffer.get(found); + return found; + } + }; + + ToArrayHelper toShortArray = new ToArrayHelper<>() { + @Override + public short[] array() { + return new short[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; + } + + @Override + public short[] toArray(MemorySegment segment, ValueLayout layout) { + ShortBuffer buffer = segment.asByteBuffer().order(layout.order()).asShortBuffer(); + short[] found = new short[buffer.limit()]; + buffer.get(found); + return found; + } + }; + + ToArrayHelper toIntArray = new ToArrayHelper<>() { + @Override + public int[] array() { + return new int[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; + } + + @Override + public int[] toArray(MemorySegment segment, ValueLayout layout) { + IntBuffer buffer = segment.asByteBuffer().order(layout.order()).asIntBuffer(); + int[] found = new int[buffer.limit()]; + buffer.get(found); + return found; + } + }; + + ToArrayHelper toFloatArray = new ToArrayHelper<>() { + @Override + public float[] array() { + return new float[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; + } + + @Override + public float[] toArray(MemorySegment segment, ValueLayout layout) { + FloatBuffer buffer = segment.asByteBuffer().order(layout.order()).asFloatBuffer(); + float[] found = new float[buffer.limit()]; + buffer.get(found); + return found; + } + }; + + ToArrayHelper toLongArray = new ToArrayHelper<>() { + @Override + public long[] array() { + return new long[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; + } + + @Override + public long[] toArray(MemorySegment segment, ValueLayout layout) { + LongBuffer buffer = segment.asByteBuffer().order(layout.order()).asLongBuffer(); + long[] found = new long[buffer.limit()]; + buffer.get(found); + return found; + } + }; + + ToArrayHelper toDoubleArray = new ToArrayHelper<>() { + @Override + public double[] array() { + return new double[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; + } + + @Override + public double[] toArray(MemorySegment segment, ValueLayout layout) { + DoubleBuffer buffer = segment.asByteBuffer().order(layout.order()).asDoubleBuffer(); + double[] found = new double[buffer.limit()]; + buffer.get(found); + return found; + } + }; + + ToArrayHelper toAddressArray = new ToArrayHelper<>() { + @Override + public MemoryAddress[] array() { + return switch ((int)MemoryLayouts.ADDRESS.byteSize()) { + case 4 -> wrap(toIntArray.array()); + case 8 -> wrap(toLongArray.array()); + default -> throw new IllegalStateException("Cannot get here"); + }; + } + + @Override + public MemoryAddress[] toArray(MemorySegment segment, ValueLayout layout) { + return switch ((int)layout.byteSize()) { + case 4 -> wrap(toIntArray.toArray(segment, layout)); + case 8 -> wrap(toLongArray.toArray(segment, layout)); + default -> throw new IllegalStateException("Cannot get here"); + }; + } + + private MemoryAddress[] wrap(int[] ints) { + return IntStream.of(ints).mapToObj(MemoryAddress::ofLong).toArray(MemoryAddress[]::new); + } + + private MemoryAddress[] wrap(long[] ints) { + return LongStream.of(ints).mapToObj(MemoryAddress::ofLong).toArray(MemoryAddress[]::new); + } + }; + } +} diff --git a/test/jdk/java/foreign/TestSegments.java b/test/jdk/java/foreign/TestSegments.java index 70521378ffb..beadf1618ac 100644 --- a/test/jdk/java/foreign/TestSegments.java +++ b/test/jdk/java/foreign/TestSegments.java @@ -27,99 +27,54 @@ * @run testng/othervm -Xmx4G -XX:MaxDirectMemorySize=1M TestSegments */ -import jdk.incubator.foreign.MappedMemorySegments; +import jdk.incubator.foreign.MemoryAccess; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; -import java.io.File; -import java.io.IOException; import java.lang.invoke.VarHandle; -import java.lang.reflect.InvocationTargetException; -import java.lang.reflect.Method; -import java.lang.reflect.Modifier; import java.nio.ByteBuffer; import java.nio.ByteOrder; -import java.nio.channels.FileChannel; -import java.nio.file.Path; -import java.util.ArrayList; import java.util.List; -import java.util.concurrent.atomic.AtomicBoolean; import java.util.concurrent.atomic.AtomicReference; -import java.util.function.Function; import java.util.function.IntFunction; import java.util.function.LongFunction; import java.util.function.Supplier; -import java.util.stream.Stream; -import static jdk.incubator.foreign.MemorySegment.*; + import static org.testng.Assert.*; public class TestSegments { @Test(dataProvider = "badSizeAndAlignments", expectedExceptions = IllegalArgumentException.class) public void testBadAllocateAlign(long size, long align) { - MemorySegment.allocateNative(size, align); + MemorySegment.allocateNative(size, align, ResourceScope.newImplicitScope()); } @Test(dataProvider = "badLayouts", expectedExceptions = UnsupportedOperationException.class) public void testBadAllocateLayout(MemoryLayout layout) { - MemorySegment.allocateNative(layout); + MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); } @Test(expectedExceptions = { OutOfMemoryError.class, IllegalArgumentException.class }) public void testAllocateTooBig() { - MemorySegment.allocateNative(Long.MAX_VALUE); - } - - @Test(dataProvider = "segmentOperations") - public void testOpOutsideConfinement(SegmentMember member) throws Throwable { - try (MemorySegment segment = MemorySegment.allocateNative(4)) { - AtomicBoolean failed = new AtomicBoolean(false); - Thread t = new Thread(() -> { - try { - Object o = member.method.invoke(segment, member.params); - } catch (ReflectiveOperationException ex) { - throw new IllegalStateException(ex); - } - }); - t.setUncaughtExceptionHandler((thread, ex) -> failed.set(true)); - t.start(); - t.join(); - assertEquals(failed.get(), member.isConfined()); - } - } - - @Test(dataProvider = "segmentOperations") - public void testOpAfterClose(SegmentMember member) throws Throwable { - MemorySegment segment = MemorySegment.allocateNative(4); - segment.close(); - try { - Object o = member.method.invoke(segment, member.params); - assertFalse(member.isConfined()); - } catch (InvocationTargetException ex) { - assertTrue(member.isConfined()); - Throwable target = ex.getTargetException(); - assertTrue(target instanceof NullPointerException || - target instanceof UnsupportedOperationException || - target instanceof IllegalStateException); - } + MemorySegment.allocateNative(Long.MAX_VALUE, ResourceScope.newImplicitScope()); } @Test(expectedExceptions = OutOfMemoryError.class) public void testNativeAllocationTooBig() { - try (MemorySegment segment = MemorySegment.allocateNative(1024 * 1024 * 8 * 2)) { // 2M - // do nothing - } + MemorySegment segment = MemorySegment.allocateNative(1024 * 1024 * 8 * 2, ResourceScope.newImplicitScope()); // 2M } @Test public void testNativeSegmentIsZeroed() { - VarHandle byteHandle = MemoryLayout.ofSequence(MemoryLayouts.JAVA_BYTE) + VarHandle byteHandle = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_BYTE) .varHandle(byte.class, MemoryLayout.PathElement.sequenceElement()); - try (MemorySegment segment = MemorySegment.allocateNative(1000)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(1000, 1, scope); for (long i = 0 ; i < segment.byteSize() ; i++) { assertEquals(0, (byte)byteHandle.get(segment, i)); } @@ -128,9 +83,10 @@ public class TestSegments { @Test public void testSlices() { - VarHandle byteHandle = MemoryLayout.ofSequence(MemoryLayouts.JAVA_BYTE) + VarHandle byteHandle = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_BYTE) .varHandle(byte.class, MemoryLayout.PathElement.sequenceElement()); - try (MemorySegment segment = MemorySegment.allocateNative(10)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(10, 1, scope); //init for (byte i = 0 ; i < segment.byteSize() ; i++) { byteHandle.set(segment, (long)i, i); @@ -147,28 +103,30 @@ public class TestSegments { } } - @Test(dataProvider = "segmentFactories") - public void testAccessModesOfFactories(Supplier memorySegmentSupplier) { - try (MemorySegment segment = memorySegmentSupplier.get()) { - assertTrue(segment.hasAccessModes(ALL_ACCESS)); - assertEquals(segment.accessModes(), ALL_ACCESS); - } + @Test(expectedExceptions = IndexOutOfBoundsException.class) + public void testSmallSegmentMax() { + long offset = (long)Integer.MAX_VALUE + (long)Integer.MAX_VALUE + 2L + 6L; // overflows to 6 when casted to int + MemorySegment memorySegment = MemorySegment.allocateNative(10, ResourceScope.newImplicitScope()); + MemoryAccess.getIntAtOffset(memorySegment, offset); } - @Test(dataProvider = "accessModes") - public void testAccessModes(int accessModes) { - int[] arr = new int[1]; - for (AccessActions action : AccessActions.values()) { - MemorySegment segment = MemorySegment.ofArray(arr); - MemorySegment restrictedSegment = segment.withAccessModes(accessModes); - assertEquals(restrictedSegment.accessModes(), accessModes); - boolean shouldFail = !restrictedSegment.hasAccessModes(action.accessMode); - try { - action.run(restrictedSegment); - assertFalse(shouldFail); - } catch (UnsupportedOperationException ex) { - assertTrue(shouldFail); - } + @Test(expectedExceptions = IndexOutOfBoundsException.class) + public void testSmallSegmentMin() { + long offset = ((long)Integer.MIN_VALUE * 2L) + 6L; // underflows to 6 when casted to int + MemorySegment memorySegment = MemorySegment.allocateNative(10, ResourceScope.newImplicitScope()); + MemoryAccess.getIntAtOffset(memorySegment, offset); + } + + @Test(dataProvider = "segmentFactories") + public void testAccessModesOfFactories(Supplier memorySegmentSupplier) { + MemorySegment segment = memorySegmentSupplier.get(); + assertFalse(segment.isReadOnly()); + tryClose(segment); + } + + static void tryClose(MemorySegment segment) { + if (!segment.scope().isImplicit()) { + segment.scope().close(); } } @@ -182,74 +140,103 @@ public class TestSegments { () -> MemorySegment.ofArray(new int[] { 1, 2, 3, 4 }), () -> MemorySegment.ofArray(new long[] { 1l, 2l, 3l, 4l } ), () -> MemorySegment.ofArray(new short[] { 1, 2, 3, 4 } ), - () -> MemorySegment.allocateNative(4), - () -> MemorySegment.allocateNative(4, 8), - () -> MemorySegment.allocateNative(MemoryLayout.ofValueBits(32, ByteOrder.nativeOrder())) + () -> MemorySegment.allocateNative(4, ResourceScope.newImplicitScope()), + () -> MemorySegment.allocateNative(4, 8, ResourceScope.newImplicitScope()), + () -> MemorySegment.allocateNative(MemoryLayout.valueLayout(32, ByteOrder.nativeOrder()), ResourceScope.newImplicitScope()), + () -> MemorySegment.allocateNative(4, ResourceScope.newConfinedScope()), + () -> MemorySegment.allocateNative(4, 8, ResourceScope.newConfinedScope()), + () -> MemorySegment.allocateNative(MemoryLayout.valueLayout(32, ByteOrder.nativeOrder()), ResourceScope.newConfinedScope()) + ); return l.stream().map(s -> new Object[] { s }).toArray(Object[][]::new); } @Test(dataProvider = "segmentFactories") public void testFill(Supplier memorySegmentSupplier) { - VarHandle byteHandle = MemoryLayout.ofSequence(MemoryLayouts.JAVA_BYTE) + VarHandle byteHandle = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_BYTE) .varHandle(byte.class, MemoryLayout.PathElement.sequenceElement()); for (byte value : new byte[] {(byte) 0xFF, (byte) 0x00, (byte) 0x45}) { - try (MemorySegment segment = memorySegmentSupplier.get()) { - segment.fill(value); - for (long l = 0; l < segment.byteSize(); l++) { - assertEquals((byte) byteHandle.get(segment, l), value); - } - - // fill a slice - var sliceSegment = segment.asSlice(1, segment.byteSize() - 2).fill((byte) ~value); - for (long l = 0; l < sliceSegment.byteSize(); l++) { - assertEquals((byte) byteHandle.get(sliceSegment, l), ~value); - } - // assert enclosing slice - assertEquals((byte) byteHandle.get(segment, 0L), value); - for (long l = 1; l < segment.byteSize() - 2; l++) { - assertEquals((byte) byteHandle.get(segment, l), (byte) ~value); - } - assertEquals((byte) byteHandle.get(segment, segment.byteSize() - 1L), value); + MemorySegment segment = memorySegmentSupplier.get(); + segment.fill(value); + for (long l = 0; l < segment.byteSize(); l++) { + assertEquals((byte) byteHandle.get(segment, l), value); } - } - } - @Test(dataProvider = "segmentFactories", expectedExceptions = IllegalStateException.class) - public void testFillClosed(Supplier memorySegmentSupplier) { - MemorySegment segment = memorySegmentSupplier.get(); - segment.close(); - segment.fill((byte) 0xFF); - } - - @Test(dataProvider = "segmentFactories", expectedExceptions = UnsupportedOperationException.class) - public void testFillIllegalAccessMode(Supplier memorySegmentSupplier) { - try (MemorySegment segment = memorySegmentSupplier.get()) { - segment.withAccessModes(segment.accessModes() & ~WRITE).fill((byte) 0xFF); + // fill a slice + var sliceSegment = segment.asSlice(1, segment.byteSize() - 2).fill((byte) ~value); + for (long l = 0; l < sliceSegment.byteSize(); l++) { + assertEquals((byte) byteHandle.get(sliceSegment, l), ~value); + } + // assert enclosing slice + assertEquals((byte) byteHandle.get(segment, 0L), value); + for (long l = 1; l < segment.byteSize() - 2; l++) { + assertEquals((byte) byteHandle.get(segment, l), (byte) ~value); + } + assertEquals((byte) byteHandle.get(segment, segment.byteSize() - 1L), value); + tryClose(segment); } } @Test(dataProvider = "segmentFactories") - public void testFillThread(Supplier memorySegmentSupplier) throws Exception { - try (MemorySegment segment = memorySegmentSupplier.get()) { - AtomicReference exception = new AtomicReference<>(); - Runnable action = () -> { - try { - segment.fill((byte) 0xBA); - } catch (RuntimeException e) { - exception.set(e); - } - }; - Thread thread = new Thread(action); - thread.start(); - thread.join(); + public void testFillClosed(Supplier memorySegmentSupplier) { + MemorySegment segment = memorySegmentSupplier.get(); + tryClose(segment); + if (!segment.scope().isAlive()) { + try { + segment.fill((byte) 0xFF); + fail(); + } catch (IllegalStateException ex) { + assertTrue(true); + } + } + } + @Test(dataProvider = "segmentFactories") + public void testNativeSegments(Supplier memorySegmentSupplier) throws Exception { + MemorySegment segment = memorySegmentSupplier.get(); + try { + segment.address().toRawLongValue(); + assertTrue(segment.isNative()); + assertTrue(segment.address().isNative()); + } catch (UnsupportedOperationException exception) { + assertFalse(segment.isNative()); + assertFalse(segment.address().isNative()); + } + tryClose(segment); + } + + @Test(dataProvider = "segmentFactories", expectedExceptions = UnsupportedOperationException.class) + public void testFillIllegalAccessMode(Supplier memorySegmentSupplier) { + MemorySegment segment = memorySegmentSupplier.get(); + segment.asReadOnly().fill((byte) 0xFF); + tryClose(segment); + } + + @Test(dataProvider = "segmentFactories") + public void testFillThread(Supplier memorySegmentSupplier) throws Exception { + MemorySegment segment = memorySegmentSupplier.get(); + AtomicReference exception = new AtomicReference<>(); + Runnable action = () -> { + try { + segment.fill((byte) 0xBA); + } catch (RuntimeException e) { + exception.set(e); + } + }; + Thread thread = new Thread(action); + thread.start(); + thread.join(); + + if (segment.scope().ownerThread() != null) { RuntimeException e = exception.get(); if (!(e instanceof IllegalStateException)) { throw e; } + } else { + assertNull(exception.get()); } + tryClose(segment); } @Test @@ -259,27 +246,6 @@ public class TestSegments { MemorySegment.ofByteBuffer(ByteBuffer.allocateDirect(0)).fill((byte) 0xFF); } - @Test(expectedExceptions = IllegalArgumentException.class) - public void testWithAccessModesBadUnsupportedMode() { - int[] arr = new int[1]; - MemorySegment segment = MemorySegment.ofArray(arr); - segment.withAccessModes((1 << AccessActions.values().length) + 1); - } - - @Test(expectedExceptions = IllegalArgumentException.class) - public void testBadWithAccessModesBadStrongerMode() { - int[] arr = new int[1]; - MemorySegment segment = MemorySegment.ofArray(arr).withAccessModes(READ); - segment.withAccessModes(WRITE); - } - - @Test(expectedExceptions = IllegalArgumentException.class) - public void testBadHasAccessModes() { - int[] arr = new int[1]; - MemorySegment segment = MemorySegment.ofArray(arr); - segment.hasAccessModes((1 << AccessActions.values().length) + 1); - } - @Test(dataProvider = "heapFactories") public void testBigHeapSegments(IntFunction heapSegmentFactory, int factor) { int bigSize = (Integer.MAX_VALUE / factor) + 1; @@ -301,16 +267,16 @@ public class TestSegments { SizedLayoutFactory[] layoutFactories = SizedLayoutFactory.values(); Object[][] values = new Object[layoutFactories.length * 2][2]; for (int i = 0; i < layoutFactories.length ; i++) { - values[i * 2] = new Object[] { MemoryLayout.ofStruct(layoutFactories[i].make(7), MemoryLayout.ofPaddingBits(9)) }; // good size, bad align + values[i * 2] = new Object[] { MemoryLayout.structLayout(layoutFactories[i].make(7), MemoryLayout.paddingLayout(9)) }; // good size, bad align values[(i * 2) + 1] = new Object[] { layoutFactories[i].make(15).withBitAlignment(16) }; // bad size, good align } return values; } enum SizedLayoutFactory { - VALUE_BE(size -> MemoryLayout.ofValueBits(size, ByteOrder.BIG_ENDIAN)), - VALUE_LE(size -> MemoryLayout.ofValueBits(size, ByteOrder.LITTLE_ENDIAN)), - PADDING(MemoryLayout::ofPaddingBits); + VALUE_BE(size -> MemoryLayout.valueLayout(size, ByteOrder.BIG_ENDIAN)), + VALUE_LE(size -> MemoryLayout.valueLayout(size, ByteOrder.LITTLE_ENDIAN)), + PADDING(MemoryLayout::paddingLayout); private final LongFunction factory; @@ -323,139 +289,6 @@ public class TestSegments { } } - @DataProvider(name = "segmentOperations") - static Object[][] segmentMembers() { - List members = new ArrayList<>(); - for (Method m : MemorySegment.class.getDeclaredMethods()) { - //skip defaults, statics and method declared in j.l.Object - if (m.isDefault() || - m.getDeclaringClass().equals(Object.class) || - (m.getModifiers() & Modifier.STATIC) != 0) continue; - Object[] args = Stream.of(m.getParameterTypes()) - .map(TestSegments::defaultValue) - .toArray(); - members.add(new SegmentMember(m, args)); - } - return members.stream().map(ms -> new Object[] { ms }).toArray(Object[][]::new); - } - - static class SegmentMember { - final Method method; - final Object[] params; - - final static List CONFINED_NAMES = List.of( - "address", - "close", - "share", - "handoff", - "registerCleaner", - "fill", - "spliterator", - "copyFrom", - "mismatch", - "toByteArray", - "toCharArray", - "toShortArray", - "toIntArray", - "toFloatArray", - "toLongArray", - "toDoubleArray" - ); - - public SegmentMember(Method method, Object[] params) { - this.method = method; - this.params = params; - } - - boolean isConfined() { - return CONFINED_NAMES.contains(method.getName()); - } - - @Override - public String toString() { - return method.getName(); - } - } - - static Object defaultValue(Class c) { - if (c.isPrimitive()) { - if (c == char.class) { - return (char)0; - } else if (c == boolean.class) { - return false; - } else if (c == byte.class) { - return (byte)0; - } else if (c == short.class) { - return (short)0; - } else if (c == int.class) { - return 0; - } else if (c == long.class) { - return 0L; - } else if (c == float.class) { - return 0f; - } else if (c == double.class) { - return 0d; - } else { - throw new IllegalStateException(); - } - } else { - return null; - } - } - - @DataProvider(name = "accessModes") - public Object[][] accessModes() { - int nActions = AccessActions.values().length; - Object[][] results = new Object[1 << nActions][]; - for (int accessModes = 0 ; accessModes < results.length ; accessModes++) { - results[accessModes] = new Object[] { accessModes }; - } - return results; - } - - enum AccessActions { - SHARE(MemorySegment.SHARE) { - @Override - void run(MemorySegment segment) { - segment.share(); - } - }, - CLOSE(MemorySegment.CLOSE) { - @Override - void run(MemorySegment segment) { - segment.close(); - } - }, - READ(MemorySegment.READ) { - @Override - void run(MemorySegment segment) { - INT_HANDLE.get(segment); - } - }, - WRITE(MemorySegment.WRITE) { - @Override - void run(MemorySegment segment) { - INT_HANDLE.set(segment, 42); - } - }, - HANDOFF(MemorySegment.HANDOFF) { - @Override - void run(MemorySegment segment) { - segment.handoff(new Thread()); - } - }; - - final int accessMode; - - static VarHandle INT_HANDLE = MemoryLayouts.JAVA_INT.varHandle(int.class); - - AccessActions(int accessMode) { - this.accessMode = accessMode; - } - - abstract void run(MemorySegment segment); - } - @DataProvider(name = "heapFactories") public Object[][] heapFactories() { return new Object[][] { diff --git a/test/jdk/java/foreign/TestSharedAccess.java b/test/jdk/java/foreign/TestSharedAccess.java index 29e1fbeb219..b5b5f717b46 100644 --- a/test/jdk/java/foreign/TestSharedAccess.java +++ b/test/jdk/java/foreign/TestSharedAccess.java @@ -24,7 +24,7 @@ /* * @test - * @run testng/othervm -Dforeign.restricted=permit TestSharedAccess + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestSharedAccess */ import jdk.incubator.foreign.*; @@ -38,7 +38,6 @@ import java.util.Spliterator; import java.util.concurrent.CompletableFuture; import java.util.concurrent.CountDownLatch; import java.util.concurrent.atomic.AtomicInteger; -import java.util.concurrent.atomic.AtomicReference; import static org.testng.Assert.*; @@ -46,42 +45,17 @@ public class TestSharedAccess { static final VarHandle intHandle = MemoryLayouts.JAVA_INT.varHandle(int.class); - @Test - public void testConfined() throws Throwable { - Thread owner = Thread.currentThread(); - MemorySegment s = MemorySegment.allocateNative(4); - AtomicReference confined = new AtomicReference<>(s); - setInt(s, 42); - assertEquals(getInt(s), 42); - List threads = new ArrayList<>(); - for (int i = 0 ; i < 1000 ; i++) { - threads.add(new Thread(() -> { - assertEquals(getInt(confined.get()), 42); - confined.set(confined.get().handoff(owner)); - })); - } - threads.forEach(t -> { - confined.set(confined.get().handoff(t)); - t.start(); - try { - t.join(); - } catch (Throwable e) { - throw new IllegalStateException(e); - } - }); - confined.get().close(); - } - @Test public void testShared() throws Throwable { - SequenceLayout layout = MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_INT); - try (MemorySegment s = MemorySegment.allocateNative(layout).share()) { + SequenceLayout layout = MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_INT); + try (ResourceScope scope = ResourceScope.newSharedScope()) { + MemorySegment s = MemorySegment.allocateNative(layout, scope); for (int i = 0 ; i < layout.elementCount().getAsLong() ; i++) { setInt(s.asSlice(i * 4), 42); } List threads = new ArrayList<>(); List> spliterators = new ArrayList<>(); - spliterators.add(s.spliterator(layout)); + spliterators.add(s.spliterator(layout.elementLayout())); while (true) { boolean progress = false; List> newSpliterators = new ArrayList<>(); @@ -119,11 +93,12 @@ public class TestSharedAccess { @Test public void testSharedUnsafe() throws Throwable { - try (MemorySegment s = MemorySegment.allocateNative(4)) { + try (ResourceScope scope = ResourceScope.newSharedScope()) { + MemorySegment s = MemorySegment.allocateNative(4, 1, scope); setInt(s, 42); assertEquals(getInt(s), 42); List threads = new ArrayList<>(); - MemorySegment sharedSegment = s.address().asSegmentRestricted(s.byteSize()).share(); + MemorySegment sharedSegment = s.address().asSegment(s.byteSize(), scope); for (int i = 0 ; i < 1000 ; i++) { threads.add(new Thread(() -> { assertEquals(getInt(sharedSegment), 42); @@ -140,40 +115,13 @@ public class TestSharedAccess { } } - @Test - public void testHandoffToSelf() { - MemorySegment s1 = MemorySegment.ofArray(new int[4]); - MemorySegment s2 = s1.handoff(Thread.currentThread()); - assertFalse(s1.isAlive()); - assertTrue(s2.isAlive()); - } - - @Test - public void testShareTwice() { - MemorySegment s1 = MemorySegment.ofArray(new int[4]).share(); - MemorySegment s2 = s1.share(); - assertFalse(s1.isAlive()); - assertTrue(s2.isAlive()); - } - - @Test(expectedExceptions=UnsupportedOperationException.class) - public void testBadHandoffNoAccess() { - MemorySegment.ofArray(new int[4]) - .withAccessModes(MemorySegment.CLOSE).handoff(new Thread()); - } - - @Test(expectedExceptions=UnsupportedOperationException.class) - public void testBadShareNoAccess() { - MemorySegment.ofArray(new int[4]) - .withAccessModes(MemorySegment.CLOSE).share(); - } - @Test public void testOutsideConfinementThread() throws Throwable { CountDownLatch a = new CountDownLatch(1); CountDownLatch b = new CountDownLatch(1); CompletableFuture r; - try (MemorySegment s1 = MemorySegment.allocateNative(MemoryLayout.ofSequence(2, MemoryLayouts.JAVA_INT))) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s1 = MemorySegment.allocateNative(MemoryLayout.sequenceLayout(2, MemoryLayouts.JAVA_INT), scope); r = CompletableFuture.runAsync(() -> { try { ByteBuffer bb = s1.asByteBuffer(); diff --git a/test/jdk/java/foreign/TestSlices.java b/test/jdk/java/foreign/TestSlices.java index 8225bf95a93..da9ad9135b1 100644 --- a/test/jdk/java/foreign/TestSlices.java +++ b/test/jdk/java/foreign/TestSlices.java @@ -22,13 +22,13 @@ * */ -import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; import java.lang.invoke.VarHandle; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.*; import static org.testng.Assert.*; @@ -38,15 +38,16 @@ import static org.testng.Assert.*; */ public class TestSlices { - static MemoryLayout LAYOUT = MemoryLayout.ofSequence(2, - MemoryLayout.ofSequence(5, MemoryLayouts.JAVA_INT)); + static MemoryLayout LAYOUT = MemoryLayout.sequenceLayout(2, + MemoryLayout.sequenceLayout(5, MemoryLayouts.JAVA_INT)); static VarHandle VH_ALL = LAYOUT.varHandle(int.class, MemoryLayout.PathElement.sequenceElement(), MemoryLayout.PathElement.sequenceElement()); @Test(dataProvider = "slices") public void testSlices(VarHandle handle, int lo, int hi, int[] values) { - try (MemorySegment segment = MemorySegment.allocateNative(LAYOUT)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(LAYOUT, scope); //init for (long i = 0 ; i < 2 ; i++) { for (long j = 0 ; j < 5 ; j++) { diff --git a/test/jdk/java/foreign/TestSpliterator.java b/test/jdk/java/foreign/TestSpliterator.java index 233b2ed6a63..d8c3fe24e0a 100644 --- a/test/jdk/java/foreign/TestSpliterator.java +++ b/test/jdk/java/foreign/TestSpliterator.java @@ -29,63 +29,63 @@ import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayouts; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; import java.lang.invoke.VarHandle; import java.util.LinkedList; import java.util.List; -import java.util.Map; import java.util.Spliterator; import java.util.concurrent.CountedCompleter; import java.util.concurrent.RecursiveTask; import java.util.concurrent.atomic.AtomicLong; -import java.util.function.Consumer; -import java.util.function.Supplier; import java.util.stream.LongStream; import java.util.stream.StreamSupport; import org.testng.annotations.*; -import static jdk.incubator.foreign.MemorySegment.*; + import static org.testng.Assert.*; public class TestSpliterator { - static final VarHandle INT_HANDLE = MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT) + static final VarHandle INT_HANDLE = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT) .varHandle(int.class, MemoryLayout.PathElement.sequenceElement()); final static int CARRIER_SIZE = 4; @Test(dataProvider = "splits") public void testSum(int size, int threshold) { - SequenceLayout layout = MemoryLayout.ofSequence(size, MemoryLayouts.JAVA_INT); + SequenceLayout layout = MemoryLayout.sequenceLayout(size, MemoryLayouts.JAVA_INT); //setup - MemorySegment segment = MemorySegment.allocateNative(layout).share(); - for (int i = 0; i < layout.elementCount().getAsLong(); i++) { - INT_HANDLE.set(segment, (long) i, i); + try (ResourceScope scope = ResourceScope.newSharedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); + for (int i = 0; i < layout.elementCount().getAsLong(); i++) { + INT_HANDLE.set(segment, (long) i, i); + } + long expected = LongStream.range(0, layout.elementCount().getAsLong()).sum(); + //serial + long serial = sum(0, segment); + assertEquals(serial, expected); + //parallel counted completer + long parallelCounted = new SumSegmentCounted(null, segment.spliterator(layout.elementLayout()), threshold).invoke(); + assertEquals(parallelCounted, expected); + //parallel recursive action + long parallelRecursive = new SumSegmentRecursive(segment.spliterator(layout.elementLayout()), threshold).invoke(); + assertEquals(parallelRecursive, expected); + //parallel stream + long streamParallel = segment.elements(layout.elementLayout()).parallel() + .reduce(0L, TestSpliterator::sumSingle, Long::sum); + assertEquals(streamParallel, expected); } - long expected = LongStream.range(0, layout.elementCount().getAsLong()).sum(); - //serial - long serial = sum(0, segment); - assertEquals(serial, expected); - //parallel counted completer - long parallelCounted = new SumSegmentCounted(null, segment.spliterator(layout), threshold).invoke(); - assertEquals(parallelCounted, expected); - //parallel recursive action - long parallelRecursive = new SumSegmentRecursive(segment.spliterator(layout), threshold).invoke(); - assertEquals(parallelRecursive, expected); - //parallel stream - long streamParallel = StreamSupport.stream(segment.spliterator(layout), true) - .reduce(0L, TestSpliterator::sumSingle, Long::sum); - assertEquals(streamParallel, expected); - segment.close(); } + @Test public void testSumSameThread() { - SequenceLayout layout = MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_INT); + SequenceLayout layout = MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_INT); //setup - MemorySegment segment = MemorySegment.allocateNative(layout); + MemorySegment segment = MemorySegment.allocateNative(layout, ResourceScope.newImplicitScope()); for (int i = 0; i < layout.elementCount().getAsLong(); i++) { INT_HANDLE.set(segment, (long) i, i); } @@ -93,11 +93,41 @@ public class TestSpliterator { //check that a segment w/o ACQUIRE access mode can still be used from same thread AtomicLong spliteratorSum = new AtomicLong(); - segment.withAccessModes(MemorySegment.READ).spliterator(layout) + segment.spliterator(layout.elementLayout()) .forEachRemaining(s -> spliteratorSum.addAndGet(sumSingle(0L, s))); assertEquals(spliteratorSum.get(), expected); } + @Test(expectedExceptions = IllegalArgumentException.class) + public void testBadSpliteratorElementSizeTooBig() { + MemorySegment.ofArray(new byte[2]).spliterator(MemoryLayouts.JAVA_INT); + } + + @Test(expectedExceptions = IllegalArgumentException.class) + public void testBadStreamElementSizeTooBig() { + MemorySegment.ofArray(new byte[2]).elements(MemoryLayouts.JAVA_INT); + } + + @Test(expectedExceptions = IllegalArgumentException.class) + public void testBadSpliteratorElementSizeNotMultiple() { + MemorySegment.ofArray(new byte[7]).spliterator(MemoryLayouts.JAVA_INT); + } + + @Test(expectedExceptions = IllegalArgumentException.class) + public void testBadStreamElementSizeNotMultiple() { + MemorySegment.ofArray(new byte[7]).elements(MemoryLayouts.JAVA_INT); + } + + @Test(expectedExceptions = IllegalArgumentException.class) + public void testBadSpliteratorElementSizeZero() { + MemorySegment.ofArray(new byte[7]).spliterator(MemoryLayout.sequenceLayout(0, MemoryLayouts.JAVA_INT)); + } + + @Test(expectedExceptions = IllegalArgumentException.class) + public void testBadStreamElementSizeZero() { + MemorySegment.ofArray(new byte[7]).elements(MemoryLayout.sequenceLayout(0, MemoryLayouts.JAVA_INT)); + } + static long sumSingle(long acc, MemorySegment segment) { return acc + (int)INT_HANDLE.get(segment, 0L); } @@ -202,52 +232,4 @@ public class TestSpliterator { { 10000, 10000 }, }; } - - @DataProvider(name = "accessScenarios") - public Object[][] accessScenarios() { - SequenceLayout layout = MemoryLayout.ofSequence(16, MemoryLayouts.JAVA_INT); - var mallocSegment = MemorySegment.allocateNative(layout); - - Map>,Integer> l = Map.of( - () -> mallocSegment.withAccessModes(ALL_ACCESS).spliterator(layout), ALL_ACCESS, - () -> mallocSegment.withAccessModes(0).spliterator(layout), 0, - () -> mallocSegment.withAccessModes(READ).spliterator(layout), READ, - () -> mallocSegment.withAccessModes(CLOSE).spliterator(layout), 0, - () -> mallocSegment.withAccessModes(READ|WRITE).spliterator(layout), READ|WRITE, - () -> mallocSegment.withAccessModes(READ|WRITE| SHARE).spliterator(layout), READ|WRITE| SHARE, - () -> mallocSegment.withAccessModes(READ|WRITE| SHARE |HANDOFF).spliterator(layout), READ|WRITE| SHARE |HANDOFF - - ); - return l.entrySet().stream().map(e -> new Object[] { e.getKey(), e.getValue() }).toArray(Object[][]::new); - } - - static Consumer assertAccessModes(int accessModes) { - return segment -> { - assertTrue(segment.hasAccessModes(accessModes & ~CLOSE)); - assertEquals(segment.accessModes(), accessModes & ~CLOSE); - }; - } - - @Test(dataProvider = "accessScenarios") - public void testAccessModes(Supplier> spliteratorSupplier, - int expectedAccessModes) { - Spliterator spliterator = spliteratorSupplier.get(); - spliterator.forEachRemaining(assertAccessModes(expectedAccessModes)); - - spliterator = spliteratorSupplier.get(); - do { } while (spliterator.tryAdvance(assertAccessModes(expectedAccessModes))); - - splitOrConsume(spliteratorSupplier.get(), assertAccessModes(expectedAccessModes)); - } - - static void splitOrConsume(Spliterator spliterator, - Consumer consumer) { - var s1 = spliterator.trySplit(); - if (s1 != null) { - splitOrConsume(s1, consumer); - splitOrConsume(spliterator, consumer); - } else { - spliterator.forEachRemaining(consumer); - } - } } diff --git a/test/jdk/java/foreign/TestTypeAccess.java b/test/jdk/java/foreign/TestTypeAccess.java index 24c311f6192..9683f58987f 100644 --- a/test/jdk/java/foreign/TestTypeAccess.java +++ b/test/jdk/java/foreign/TestTypeAccess.java @@ -27,15 +27,14 @@ * @run testng TestTypeAccess */ -import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemorySegment; import jdk.incubator.foreign.MemoryLayouts; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.*; import java.lang.invoke.VarHandle; import java.lang.invoke.WrongMethodTypeException; -import java.nio.ByteOrder; public class TestTypeAccess { @@ -45,42 +44,48 @@ public class TestTypeAccess { @Test(expectedExceptions=ClassCastException.class) public void testMemoryAddressCoordinateAsString() { - try (MemorySegment s = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s = MemorySegment.allocateNative(8, 8, scope); int v = (int)INT_HANDLE.get("string"); } } @Test(expectedExceptions=WrongMethodTypeException.class) public void testMemoryCoordinatePrimitive() { - try (MemorySegment s = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s = MemorySegment.allocateNative(8, 8, scope); int v = (int)INT_HANDLE.get(1); } } @Test(expectedExceptions=ClassCastException.class) public void testMemoryAddressValueGetAsString() { - try (MemorySegment s = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s = MemorySegment.allocateNative(8, 8, scope); String address = (String)ADDR_HANDLE.get(s.address()); } } @Test(expectedExceptions=ClassCastException.class) public void testMemoryAddressValueSetAsString() { - try (MemorySegment s = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s = MemorySegment.allocateNative(8, 8, scope); ADDR_HANDLE.set(s.address(), "string"); } } @Test(expectedExceptions=WrongMethodTypeException.class) public void testMemoryAddressValueGetAsPrimitive() { - try (MemorySegment s = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s = MemorySegment.allocateNative(8, 8, scope); int address = (int)ADDR_HANDLE.get(s.address()); } } @Test(expectedExceptions=WrongMethodTypeException.class) public void testMemoryAddressValueSetAsPrimitive() { - try (MemorySegment s = MemorySegment.allocateNative(8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment s = MemorySegment.allocateNative(8, 8, scope); ADDR_HANDLE.set(s.address(), 1); } } diff --git a/test/jdk/java/foreign/TestUnsupportedPlatform.java b/test/jdk/java/foreign/TestUnsupportedPlatform.java index 5a3e16a5527..478cac72e33 100644 --- a/test/jdk/java/foreign/TestUnsupportedPlatform.java +++ b/test/jdk/java/foreign/TestUnsupportedPlatform.java @@ -26,21 +26,12 @@ * @test * @requires !(((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64") * @modules jdk.incubator.foreign/jdk.internal.foreign - * @run testng/othervm -Dforeign.restricted=permit TestUnsupportedPlatform + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestUnsupportedPlatform */ import jdk.incubator.foreign.CLinker; -import jdk.incubator.foreign.MemoryLayout; -import jdk.incubator.foreign.NativeScope; -import org.testng.annotations.DataProvider; import org.testng.annotations.Test; -import java.lang.reflect.Field; -import java.util.ArrayList; -import java.util.List; - -import static jdk.incubator.foreign.MemoryAddress.NULL; -import static jdk.incubator.foreign.MemoryLayouts.JAVA_BYTE; import static org.testng.Assert.assertNull; // tests run on 32-bit platforms, which are currently not supported diff --git a/test/jdk/java/foreign/TestUpcall.java b/test/jdk/java/foreign/TestUpcall.java index 6229d32044b..c6e8835df6e 100644 --- a/test/jdk/java/foreign/TestUpcall.java +++ b/test/jdk/java/foreign/TestUpcall.java @@ -29,24 +29,7 @@ * @build NativeTestHelper CallGeneratorHelper TestUpcall * * @run testng/othervm - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false - * TestUpcall - * @run testng/othervm - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false - * TestUpcall - * @run testng/othervm - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true - * TestUpcall - * @run testng/othervm - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * --enable-native-access=ALL-UNNAMED * TestUpcall */ @@ -56,8 +39,9 @@ import jdk.incubator.foreign.LibraryLookup; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; -import jdk.incubator.foreign.ValueLayout; -import org.testng.annotations.AfterClass; + +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; import org.testng.annotations.BeforeClass; import org.testng.annotations.Test; @@ -86,39 +70,55 @@ public class TestUpcall extends CallGeneratorHelper { static { try { DUMMY = MethodHandles.lookup().findStatic(TestUpcall.class, "dummy", MethodType.methodType(void.class)); - PASS_AND_SAVE = MethodHandles.lookup().findStatic(TestUpcall.class, "passAndSave", MethodType.methodType(Object.class, Object[].class, AtomicReference.class)); + PASS_AND_SAVE = MethodHandles.lookup().findStatic(TestUpcall.class, "passAndSave", + MethodType.methodType(Object.class, Object[].class, AtomicReference.class)); } catch (Throwable ex) { throw new IllegalStateException(ex); } } - static MemorySegment dummyStub; + static MemoryAddress dummyStub; @BeforeClass void setup() { - dummyStub = abi.upcallStub(DUMMY, FunctionDescriptor.ofVoid()); - } - - @AfterClass - void teardown() { - dummyStub.close(); + dummyStub = abi.upcallStub(DUMMY, FunctionDescriptor.ofVoid(), ResourceScope.newImplicitScope()); } @Test(dataProvider="functions", dataProviderClass=CallGeneratorHelper.class) - public void testUpcalls(String fName, Ret ret, List paramTypes, List fields) throws Throwable { - List segments = new ArrayList<>(); + public void testUpcalls(int count, String fName, Ret ret, List paramTypes, List fields) throws Throwable { List> returnChecks = new ArrayList<>(); List> argChecks = new ArrayList<>(); - LibraryLookup.Symbol addr = lib.lookup(fName).get(); - MethodHandle mh = abi.downcallHandle(addr, methodType(ret, paramTypes, fields), function(ret, paramTypes, fields)); - Object[] args = makeArgs(ret, paramTypes, fields, returnChecks, argChecks, segments); - mh = mh.asSpreader(Object[].class, paramTypes.size() + 1); - Object res = mh.invoke(args); + MemoryAddress addr = lib.lookup(fName).get(); + MethodType mtype = methodType(ret, paramTypes, fields); + try (NativeScope scope = new NativeScope()) { + MethodHandle mh = abi.downcallHandle(addr, scope, mtype, function(ret, paramTypes, fields)); + Object[] args = makeArgs(scope.scope(), ret, paramTypes, fields, returnChecks, argChecks); + Object[] callArgs = args; + Object res = mh.invokeWithArguments(callArgs); + argChecks.forEach(c -> c.accept(args)); + if (ret == Ret.NON_VOID) { + returnChecks.forEach(c -> c.accept(res)); + } + } + } + + @Test(dataProvider="functions", dataProviderClass=CallGeneratorHelper.class) + public void testUpcallsNoScope(int count, String fName, Ret ret, List paramTypes, List fields) throws Throwable { + List> returnChecks = new ArrayList<>(); + List> argChecks = new ArrayList<>(); + MemoryAddress addr = lib.lookup(fName).get(); + MethodType mtype = methodType(ret, paramTypes, fields); + MethodHandle mh = abi.downcallHandle(addr, IMPLICIT_ALLOCATOR, mtype, function(ret, paramTypes, fields)); + Object[] args = makeArgs(ResourceScope.newImplicitScope(), ret, paramTypes, fields, returnChecks, argChecks); + Object[] callArgs = args; + if (count % 100 == 0) { + System.gc(); + } + Object res = mh.invokeWithArguments(callArgs); argChecks.forEach(c -> c.accept(args)); if (ret == Ret.NON_VOID) { returnChecks.forEach(c -> c.accept(res)); } - segments.forEach(MemorySegment::close); } static MethodType methodType(Ret ret, List params, List fields) { @@ -140,17 +140,17 @@ public class TestUpcall extends CallGeneratorHelper { FunctionDescriptor.of(layouts[0], layouts); } - static Object[] makeArgs(Ret ret, List params, List fields, List> checks, List> argChecks, List segments) throws ReflectiveOperationException { + static Object[] makeArgs(ResourceScope scope, Ret ret, List params, List fields, List> checks, List> argChecks) throws ReflectiveOperationException { Object[] args = new Object[params.size() + 1]; for (int i = 0 ; i < params.size() ; i++) { - args[i] = makeArg(params.get(i).layout(fields), checks, i == 0, segments); + args[i] = makeArg(params.get(i).layout(fields), checks, i == 0); } - args[params.size()] = makeCallback(ret, params, fields, checks, argChecks, segments); + args[params.size()] = makeCallback(scope, ret, params, fields, checks, argChecks); return args; } @SuppressWarnings("unchecked") - static MemoryAddress makeCallback(Ret ret, List params, List fields, List> checks, List> argChecks, List segments) { + static MemoryAddress makeCallback(ResourceScope scope, Ret ret, List params, List fields, List> checks, List> argChecks) { if (params.isEmpty()) { return dummyStub.address(); } @@ -189,12 +189,18 @@ public class TestUpcall extends CallGeneratorHelper { FunctionDescriptor func = ret != Ret.VOID ? FunctionDescriptor.of(firstlayout, paramLayouts) : FunctionDescriptor.ofVoid(paramLayouts); - MemorySegment stub = abi.upcallStub(mh, func); - segments.add(stub); - return stub.address(); + return abi.upcallStub(mh, func, scope); } static Object passAndSave(Object[] o, AtomicReference ref) { + for (int i = 0; i < o.length; i++) { + if (o[i] instanceof MemorySegment) { + MemorySegment ms = (MemorySegment) o[i]; + MemorySegment copy = MemorySegment.allocateNative(ms.byteSize(), ResourceScope.newImplicitScope()); + copy.copyFrom(ms); + o[i] = copy; + } + } ref.set(o); return o[0]; } diff --git a/test/jdk/java/foreign/TestUpcallHighArity.java b/test/jdk/java/foreign/TestUpcallHighArity.java index fd9c97c6647..9e0eb00a678 100644 --- a/test/jdk/java/foreign/TestUpcallHighArity.java +++ b/test/jdk/java/foreign/TestUpcallHighArity.java @@ -29,24 +29,7 @@ * @build NativeTestHelper CallGeneratorHelper TestUpcallHighArity * * @run testng/othervm/native - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false - * TestUpcallHighArity - * @run testng/othervm/native - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false - * TestUpcallHighArity - * @run testng/othervm/native - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true - * TestUpcallHighArity - * @run testng/othervm/native - * -Dforeign.restricted=permit - * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true - * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true + * --enable-native-access=ALL-UNNAMED * TestUpcallHighArity */ @@ -56,16 +39,15 @@ import jdk.incubator.foreign.LibraryLookup; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; -import java.util.ArrayList; import java.util.List; import java.util.concurrent.atomic.AtomicReference; -import java.util.function.Consumer; import static jdk.incubator.foreign.CLinker.*; import static org.testng.Assert.assertEquals; @@ -76,7 +58,7 @@ public class TestUpcallHighArity extends CallGeneratorHelper { static final CLinker LINKER = CLinker.getInstance(); // struct S_PDI { void* p0; double p1; int p2; }; - static final MemoryLayout S_PDI_LAYOUT = MemoryLayout.ofStruct( + static final MemoryLayout S_PDI_LAYOUT = MemoryLayout.structLayout( C_POINTER.withName("p0"), C_DOUBLE.withName("p1"), C_INT.withName("p2") @@ -106,6 +88,14 @@ public class TestUpcallHighArity extends CallGeneratorHelper { } static void passAndSave(Object[] o, AtomicReference ref) { + for (int i = 0; i < o.length; i++) { + if (o[i] instanceof MemorySegment) { + MemorySegment ms = (MemorySegment) o[i]; + MemorySegment copy = MemorySegment.allocateNative(ms.byteSize(), ResourceScope.newImplicitScope()); + copy.copyFrom(ms); + o[i] = copy; + } + } ref.set(o); } @@ -116,13 +106,13 @@ public class TestUpcallHighArity extends CallGeneratorHelper { MethodHandle target = MethodHandles.insertArguments(MH_passAndSave, 1, capturedArgs) .asCollector(Object[].class, upcallType.parameterCount()) .asType(upcallType); - try (MemorySegment upcallStub = LINKER.upcallStub(target, upcallDescriptor)) { - List segments = new ArrayList<>(); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemoryAddress upcallStub = LINKER.upcallStub(target, upcallDescriptor, scope); Object[] args = new Object[upcallType.parameterCount() + 1]; args[0] = upcallStub.address(); List argLayouts = upcallDescriptor.argumentLayouts(); for (int i = 1; i < args.length; i++) { - args[i] = makeArg(argLayouts.get(i - 1), null, false, segments); + args[i] = makeArg(argLayouts.get(i - 1), null, false); } downcall.invokeWithArguments(args); @@ -135,7 +125,6 @@ public class TestUpcallHighArity extends CallGeneratorHelper { assertEquals(capturedArgsArr[i], args[i + 1]); } } - segments.forEach(MemorySegment::close); } } diff --git a/test/jdk/java/foreign/TestUpcallStructScope.java b/test/jdk/java/foreign/TestUpcallStructScope.java new file mode 100644 index 00000000000..bf6170274d5 --- /dev/null +++ b/test/jdk/java/foreign/TestUpcallStructScope.java @@ -0,0 +1,107 @@ +/* + * 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 + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @modules jdk.incubator.foreign/jdk.internal.foreign + * + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=false + * TestUpcallStructScope + * @run testng/othervm/native + * --enable-native-access=ALL-UNNAMED + * -Djdk.internal.foreign.ProgrammableInvoker.USE_SPEC=true + * TestUpcallStructScope + */ + +import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.FunctionDescriptor; +import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import org.testng.annotations.Test; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; +import java.util.concurrent.atomic.AtomicReference; +import java.util.function.Consumer; + +import static jdk.incubator.foreign.CLinker.C_DOUBLE; +import static jdk.incubator.foreign.CLinker.C_INT; +import static jdk.incubator.foreign.CLinker.C_POINTER; +import static org.testng.Assert.assertFalse; + +public class TestUpcallStructScope { + static final MethodHandle MH_do_upcall; + static final CLinker LINKER = CLinker.getInstance(); + static final MethodHandle MH_Consumer_accept; + + // struct S_PDI { void* p0; double p1; int p2; }; + static final MemoryLayout S_PDI_LAYOUT = MemoryLayout.structLayout( + C_POINTER.withName("p0"), + C_DOUBLE.withName("p1"), + C_INT.withName("p2") + ); + + static { + LibraryLookup lookup = LibraryLookup.ofLibrary("TestUpcallStructScope"); + MH_do_upcall = LINKER.downcallHandle( + lookup.lookup("do_upcall").orElseThrow(), + MethodType.methodType(void.class, MemoryAddress.class, MemorySegment.class), + FunctionDescriptor.ofVoid(C_POINTER, S_PDI_LAYOUT) + ); + + try { + MH_Consumer_accept = MethodHandles.publicLookup().findVirtual(Consumer.class, "accept", + MethodType.methodType(void.class, Object.class)); + } catch (NoSuchMethodException | IllegalAccessException e) { + throw new RuntimeException(e); + } + } + + private static MethodHandle methodHandle (Consumer callback) { + return MH_Consumer_accept.bindTo(callback).asType(MethodType.methodType(void.class, MemorySegment.class)); + } + + @Test + public void testUpcall() throws Throwable { + AtomicReference capturedSegment = new AtomicReference<>(); + MethodHandle target = methodHandle(capturedSegment::set); + FunctionDescriptor upcallDesc = FunctionDescriptor.ofVoid(S_PDI_LAYOUT); + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemoryAddress upcallStub = LINKER.upcallStub(target, upcallDesc, scope); + MemorySegment argSegment = MemorySegment.allocateNative(S_PDI_LAYOUT, scope); + MH_do_upcall.invokeExact(upcallStub.address(), argSegment); + } + + MemorySegment captured = capturedSegment.get(); + assertFalse(captured.scope().isAlive()); + } + +} diff --git a/test/jdk/java/foreign/TestUpcallStubs.java b/test/jdk/java/foreign/TestUpcallStubs.java deleted file mode 100644 index a9c29f00c66..00000000000 --- a/test/jdk/java/foreign/TestUpcallStubs.java +++ /dev/null @@ -1,96 +0,0 @@ -/* - * 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 - * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" - * @run testng/othervm -Dforeign.restricted=permit TestUpcallStubs - */ - -import jdk.incubator.foreign.CLinker; -import jdk.incubator.foreign.FunctionDescriptor; -import jdk.incubator.foreign.MemoryAddress; -import jdk.incubator.foreign.MemorySegment; -import org.testng.annotations.*; - -import java.lang.invoke.MethodHandle; -import java.lang.invoke.MethodHandles; -import java.lang.invoke.MethodType; -import java.lang.invoke.VarHandle; - -import static jdk.incubator.foreign.MemoryLayouts.JAVA_INT; -import static org.testng.Assert.assertFalse; - -public class TestUpcallStubs { - - static final CLinker abi = CLinker.getInstance(); - static final MethodHandle MH_dummy; - - static { - try { - MH_dummy = MethodHandles.lookup() - .findStatic(TestUpcallStubs.class, "dummy", MethodType.methodType(void.class)); - } catch (NoSuchMethodException | IllegalAccessException e) { - throw new BootstrapMethodError(e); - } - } - - private static MemorySegment getStub() { - return abi.upcallStub(MH_dummy, FunctionDescriptor.ofVoid()); - } - - @Test(expectedExceptions = UnsupportedOperationException.class) - public void testNoAccess() { - try (MemorySegment stub = getStub()) { - VarHandle vh = JAVA_INT.varHandle(int.class); - vh.set(stub, 10); - } - } - - @Test - public void testFree() { - MemorySegment stub = getStub(); - stub.close(); - assertFalse(stub.isAlive()); - } - - @Test(expectedExceptions = IllegalStateException.class) - public void testAlreadyFreed() { - MemorySegment stub = getStub(); - stub.close(); - stub.close(); // should fail - } - - @DataProvider - public static Object[][] badAddresses() { - return new Object[][]{ - { MemoryAddress.ofLong(42) /* random address */ }, - { MemorySegment.ofArray(new int []{ 1, 2, 3 }).address() /* heap address */ } - }; - } - - // where - public static void dummy() {} - -} diff --git a/test/jdk/java/foreign/TestVarArgs.java b/test/jdk/java/foreign/TestVarArgs.java index cb4a14feab9..bd31a42c259 100644 --- a/test/jdk/java/foreign/TestVarArgs.java +++ b/test/jdk/java/foreign/TestVarArgs.java @@ -25,7 +25,7 @@ /* * @test * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" - * @run testng/othervm -Dforeign.restricted=permit TestVarArgs + * @run testng/othervm --enable-native-access=ALL-UNNAMED TestVarArgs */ import jdk.incubator.foreign.CLinker; @@ -34,6 +34,7 @@ import jdk.incubator.foreign.LibraryLookup; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.ValueLayout; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; @@ -50,26 +51,27 @@ import static org.testng.Assert.assertEquals; public class TestVarArgs { - static final MemoryLayout ML_CallInfo = MemoryLayout.ofStruct( + static final MemoryLayout ML_CallInfo = MemoryLayout.structLayout( C_POINTER.withName("writeback"), // writeback C_POINTER.withName("argIDs")); // arg ids static final VarHandle VH_CallInfo_writeback = ML_CallInfo.varHandle(long.class, groupElement("writeback")); static final VarHandle VH_CallInfo_argIDs = ML_CallInfo.varHandle(long.class, groupElement("argIDs")); - static final VarHandle VH_IntArray = MemoryLayout.ofSequence(C_INT).varHandle(int.class, sequenceElement()); + static final VarHandle VH_IntArray = MemoryLayout.sequenceLayout(C_INT).varHandle(int.class, sequenceElement()); static final CLinker abi = CLinker.getInstance(); - static final LibraryLookup.Symbol varargsAddr = LibraryLookup.ofLibrary("VarArgs") + static final MemoryAddress varargsAddr = LibraryLookup.ofLibrary("VarArgs") .lookup("varargs").get(); static final int WRITEBACK_BYTES_PER_ARG = 8; @Test(dataProvider = "args") public void testVarArgs(List args) throws Throwable { - try (MemorySegment writeBack = MemorySegment.allocateNative(args.size() * WRITEBACK_BYTES_PER_ARG); - MemorySegment callInfo = MemorySegment.allocateNative(ML_CallInfo); - MemorySegment argIDs = MemorySegment.allocateNative(MemoryLayout.ofSequence(args.size(), C_INT))) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment writeBack = MemorySegment.allocateNative(args.size() * WRITEBACK_BYTES_PER_ARG, WRITEBACK_BYTES_PER_ARG, scope); + MemorySegment callInfo = MemorySegment.allocateNative(ML_CallInfo, scope); + MemorySegment argIDs = MemorySegment.allocateNative(MemoryLayout.sequenceLayout(args.size(), C_INT), scope); MemoryAddress callInfoPtr = callInfo.address(); diff --git a/test/jdk/java/foreign/TestVarHandleCombinators.java b/test/jdk/java/foreign/TestVarHandleCombinators.java index c3840dbad38..2090c31fdd9 100644 --- a/test/jdk/java/foreign/TestVarHandleCombinators.java +++ b/test/jdk/java/foreign/TestVarHandleCombinators.java @@ -28,12 +28,12 @@ */ import jdk.incubator.foreign.MemoryHandles; +import jdk.incubator.foreign.ResourceScope; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; import jdk.incubator.foreign.MemorySegment; -import java.lang.invoke.MethodHandles; import java.lang.invoke.VarHandle; import java.nio.ByteOrder; @@ -74,7 +74,7 @@ public class TestVarHandleCombinators { public void testAlign() { VarHandle vh = MemoryHandles.varHandle(byte.class, 2, ByteOrder.nativeOrder()); - MemorySegment segment = MemorySegment.allocateNative(1, 2); + MemorySegment segment = MemorySegment.allocateNative(1, 2, ResourceScope.newImplicitScope()); vh.set(segment, 0L, (byte) 10); // fine, memory region is aligned assertEquals((byte) vh.get(segment, 0L), (byte) 10); } @@ -108,7 +108,8 @@ public class TestVarHandleCombinators { VarHandle vh = MemoryHandles.varHandle(int.class, ByteOrder.nativeOrder()); int count = 0; - try (MemorySegment segment = MemorySegment.allocateNative(inner_size * outer_size * 8)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(inner_size * outer_size * 8, 4, scope); for (long i = 0; i < outer_size; i++) { for (long j = 0; j < inner_size; j++) { vh.set(segment, i * 40 + j * 8, count); diff --git a/test/jdk/java/foreign/callarranger/TestAarch64CallArranger.java b/test/jdk/java/foreign/callarranger/TestAarch64CallArranger.java index f9de8b71e5e..eef17a4aad1 100644 --- a/test/jdk/java/foreign/callarranger/TestAarch64CallArranger.java +++ b/test/jdk/java/foreign/callarranger/TestAarch64CallArranger.java @@ -144,10 +144,10 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { @DataProvider public static Object[][] structs() { - MemoryLayout struct2 = MemoryLayout.ofStruct(C_INT, C_INT, C_DOUBLE, C_INT); + MemoryLayout struct2 = MemoryLayout.structLayout(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[] { + { MemoryLayout.structLayout(C_INT, C_INT, C_DOUBLE), new Binding[] { dup(), // s.a & s.b bufferLoad(0, long.class), vmStore(r0, long.class), @@ -162,7 +162,7 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { 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[] { + { MemoryLayout.structLayout(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), @@ -170,7 +170,7 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { bufferLoad(8, long.class), vmStore(r1, long.class), }}, // struct s { float a; /* padding */ double b }; - { MemoryLayout.ofStruct(C_FLOAT, MemoryLayout.ofPaddingBits(32), C_DOUBLE), + { MemoryLayout.structLayout(C_FLOAT, MemoryLayout.paddingLayout(32), C_DOUBLE), new Binding[] { dup(), // s.a @@ -183,8 +183,8 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { @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); + MemoryLayout struct1 = MemoryLayout.structLayout(C_INT, C_INT, C_DOUBLE, C_INT); + MemoryLayout struct2 = MemoryLayout.structLayout(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); @@ -216,7 +216,7 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { @Test public void testReturnStruct1() { - MemoryLayout struct = MemoryLayout.ofStruct(C_LONG, C_LONG, C_FLOAT); + MemoryLayout struct = MemoryLayout.structLayout(C_LONG, C_LONG, C_FLOAT); MethodType mt = MethodType.methodType(MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.of(struct); @@ -239,7 +239,7 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { @Test public void testReturnStruct2() { - MemoryLayout struct = MemoryLayout.ofStruct(C_LONG, C_LONG); + MemoryLayout struct = MemoryLayout.structLayout(C_LONG, C_LONG); MethodType mt = MethodType.methodType(MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.of(struct); @@ -265,7 +265,7 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { @Test public void testStructHFA1() { - MemoryLayout hfa = MemoryLayout.ofStruct(C_FLOAT, C_FLOAT); + MemoryLayout hfa = MemoryLayout.structLayout(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); @@ -281,27 +281,27 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { { vmStore(r0, int.class) }, { dup(), - bufferLoad(0, int.class), - vmStore(v1, int.class), - bufferLoad(4, int.class), - vmStore(v2, int.class) + bufferLoad(0, float.class), + vmStore(v1, float.class), + bufferLoad(4, float.class), + vmStore(v2, float.class) } }); checkReturnBindings(callingSequence, new Binding[]{ allocate(hfa), dup(), - vmLoad(v0, int.class), - bufferStore(0, int.class), + vmLoad(v0, float.class), + bufferStore(0, float.class), dup(), - vmLoad(v1, int.class), - bufferStore(4, int.class), + vmLoad(v1, float.class), + bufferStore(4, float.class), }); } @Test public void testStructHFA3() { - MemoryLayout struct = MemoryLayout.ofStruct(C_FLOAT, C_FLOAT, C_FLOAT); + MemoryLayout struct = MemoryLayout.structLayout(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); @@ -315,23 +315,23 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { checkArgumentBindings(callingSequence, new Binding[][]{ { dup(), - bufferLoad(0, int.class), - vmStore(v0, int.class), + bufferLoad(0, float.class), + vmStore(v0, float.class), dup(), - bufferLoad(4, int.class), - vmStore(v1, int.class), - bufferLoad(8, int.class), - vmStore(v2, int.class) + bufferLoad(4, float.class), + vmStore(v1, float.class), + bufferLoad(8, float.class), + vmStore(v2, float.class) }, { dup(), - bufferLoad(0, int.class), - vmStore(v3, int.class), + bufferLoad(0, float.class), + vmStore(v3, float.class), dup(), - bufferLoad(4, int.class), - vmStore(v4, int.class), - bufferLoad(8, int.class), - vmStore(v5, int.class) + bufferLoad(4, float.class), + vmStore(v4, float.class), + bufferLoad(8, float.class), + vmStore(v5, float.class) }, { dup(), @@ -351,7 +351,7 @@ public class TestAarch64CallArranger extends CallArrangerTestBase { // 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); + MemoryLayout struct = MemoryLayout.structLayout(C_INT, C_INT, C_DOUBLE, C_INT); MethodType mt = MethodType.methodType( void.class, MemorySegment.class, MemorySegment.class, int.class, int.class, diff --git a/test/jdk/java/foreign/callarranger/TestSysVCallArranger.java b/test/jdk/java/foreign/callarranger/TestSysVCallArranger.java index 48cef83c867..668132bc1de 100644 --- a/test/jdk/java/foreign/callarranger/TestSysVCallArranger.java +++ b/test/jdk/java/foreign/callarranger/TestSysVCallArranger.java @@ -75,9 +75,9 @@ public class TestSysVCallArranger extends CallArrangerTestBase { @Test public void testNestedStructs() { - MemoryLayout POINT = MemoryLayout.ofStruct( + MemoryLayout POINT = MemoryLayout.structLayout( C_INT, - MemoryLayout.ofStruct( + MemoryLayout.structLayout( C_INT, C_INT ) @@ -104,11 +104,11 @@ public class TestSysVCallArranger extends CallArrangerTestBase { @Test public void testNestedUnion() { - MemoryLayout POINT = MemoryLayout.ofStruct( + MemoryLayout POINT = MemoryLayout.structLayout( C_INT, - MemoryLayout.ofPaddingBits(32), - MemoryLayout.ofUnion( - MemoryLayout.ofStruct(C_INT, C_INT), + MemoryLayout.paddingLayout(32), + MemoryLayout.unionLayout( + MemoryLayout.structLayout(C_INT, C_INT), C_LONG ) ); @@ -134,9 +134,9 @@ public class TestSysVCallArranger extends CallArrangerTestBase { @Test public void testNestedStructsUnaligned() { - MemoryLayout POINT = MemoryLayout.ofStruct( + MemoryLayout POINT = MemoryLayout.structLayout( C_INT, - MemoryLayout.ofStruct( + MemoryLayout.structLayout( C_LONG, C_INT ) @@ -163,10 +163,10 @@ public class TestSysVCallArranger extends CallArrangerTestBase { @Test public void testNestedUnionUnaligned() { - MemoryLayout POINT = MemoryLayout.ofStruct( + MemoryLayout POINT = MemoryLayout.structLayout( C_INT, - MemoryLayout.ofUnion( - MemoryLayout.ofStruct(C_INT, C_INT), + MemoryLayout.unionLayout( + MemoryLayout.structLayout(C_INT, C_INT), C_LONG ) ); @@ -309,7 +309,7 @@ public class TestSysVCallArranger extends CallArrangerTestBase { */ @Test public void testAbiExample() { - MemoryLayout struct = MemoryLayout.ofStruct(C_INT, C_INT, C_DOUBLE); + MemoryLayout struct = MemoryLayout.structLayout(C_INT, C_INT, C_DOUBLE); MethodType mt = MethodType.methodType(void.class, int.class, int.class, MemorySegment.class, int.class, int.class, @@ -329,7 +329,7 @@ public class TestSysVCallArranger extends CallArrangerTestBase { { dup(), bufferLoad(0, long.class), vmStore(rdx, long.class), - bufferLoad(8, long.class), vmStore(xmm0, long.class) + bufferLoad(8, double.class), vmStore(xmm0, double.class) }, { vmStore(rcx, int.class) }, { vmStore(r8, int.class) }, @@ -400,17 +400,17 @@ public class TestSysVCallArranger extends CallArrangerTestBase { @DataProvider public static Object[][] structs() { return new Object[][]{ - { MemoryLayout.ofStruct(C_LONG), new Binding[]{ + { MemoryLayout.structLayout(C_LONG), new Binding[]{ bufferLoad(0, long.class), vmStore(rdi, long.class) } }, - { MemoryLayout.ofStruct(C_LONG, C_LONG), new Binding[]{ + { MemoryLayout.structLayout(C_LONG, C_LONG), new Binding[]{ dup(), bufferLoad(0, long.class), vmStore(rdi, long.class), bufferLoad(8, long.class), vmStore(rsi, long.class) } }, - { MemoryLayout.ofStruct(C_LONG, C_LONG, C_LONG), new Binding[]{ + { MemoryLayout.structLayout(C_LONG, C_LONG, C_LONG), new Binding[]{ dup(), bufferLoad(0, long.class), vmStore(stackStorage(0), long.class), dup(), @@ -418,7 +418,7 @@ public class TestSysVCallArranger extends CallArrangerTestBase { bufferLoad(16, long.class), vmStore(stackStorage(2), long.class) } }, - { MemoryLayout.ofStruct(C_LONG, C_LONG, C_LONG, C_LONG), new Binding[]{ + { MemoryLayout.structLayout(C_LONG, C_LONG, C_LONG, C_LONG), new Binding[]{ dup(), bufferLoad(0, long.class), vmStore(stackStorage(0), long.class), dup(), @@ -433,7 +433,7 @@ public class TestSysVCallArranger extends CallArrangerTestBase { @Test public void testReturnRegisterStruct() { - MemoryLayout struct = MemoryLayout.ofStruct(C_LONG, C_LONG); + MemoryLayout struct = MemoryLayout.structLayout(C_LONG, C_LONG); MethodType mt = MethodType.methodType(MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.of(struct); @@ -463,7 +463,7 @@ public class TestSysVCallArranger extends CallArrangerTestBase { @Test public void testIMR() { - MemoryLayout struct = MemoryLayout.ofStruct(C_LONG, C_LONG, C_LONG); + MemoryLayout struct = MemoryLayout.structLayout(C_LONG, C_LONG, C_LONG); MethodType mt = MethodType.methodType(MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.of(struct); @@ -484,4 +484,28 @@ public class TestSysVCallArranger extends CallArrangerTestBase { assertEquals(bindings.nVectorArgs, 0); } + @Test + public void testFloatStructsUpcall() { + MemoryLayout struct = MemoryLayout.structLayout(C_FLOAT); // should be passed in float regs + + MethodType mt = MethodType.methodType(MemorySegment.class, MemorySegment.class); + FunctionDescriptor fd = FunctionDescriptor.of(struct, struct); + CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, true); + + assertFalse(bindings.isInMemoryReturn); + CallingSequence callingSequence = bindings.callingSequence; + assertEquals(callingSequence.methodType(), mt); + assertEquals(callingSequence.functionDesc(), fd); + + checkArgumentBindings(callingSequence, new Binding[][]{ + { allocate(struct), dup(), vmLoad(xmm0, float.class), bufferStore(0, float.class) }, + }); + + checkReturnBindings(callingSequence, new Binding[] { + bufferLoad(0, float.class), vmStore(xmm0, float.class) + }); + + assertEquals(bindings.nVectorArgs, 1); + } + } diff --git a/test/jdk/java/foreign/callarranger/TestWindowsCallArranger.java b/test/jdk/java/foreign/callarranger/TestWindowsCallArranger.java index 8580fc97b75..cb85c6f55e8 100644 --- a/test/jdk/java/foreign/callarranger/TestWindowsCallArranger.java +++ b/test/jdk/java/foreign/callarranger/TestWindowsCallArranger.java @@ -137,7 +137,7 @@ public class TestWindowsCallArranger extends CallArrangerTestBase { @Test public void testAbiExample() { - MemoryLayout structLayout = MemoryLayout.ofStruct(C_INT, C_INT, C_DOUBLE); + MemoryLayout structLayout = MemoryLayout.structLayout(C_INT, C_INT, C_DOUBLE); MethodType mt = MethodType.methodType(void.class, int.class, int.class, MemorySegment.class, int.class, int.class, double.class, double.class, double.class, int.class, int.class, int.class); @@ -208,7 +208,7 @@ public class TestWindowsCallArranger extends CallArrangerTestBase { */ @Test public void testStructRegister() { - MemoryLayout struct = MemoryLayout.ofStruct(C_LONG_LONG); + MemoryLayout struct = MemoryLayout.structLayout(C_LONG_LONG); MethodType mt = MethodType.methodType(void.class, MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid(struct); @@ -237,7 +237,7 @@ public class TestWindowsCallArranger extends CallArrangerTestBase { */ @Test public void testStructReference() { - MemoryLayout struct = MemoryLayout.ofStruct(C_LONG_LONG, C_LONG_LONG); + MemoryLayout struct = MemoryLayout.structLayout(C_LONG_LONG, C_LONG_LONG); MethodType mt = MethodType.methodType(void.class, MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid(struct); @@ -288,7 +288,7 @@ public class TestWindowsCallArranger extends CallArrangerTestBase { @Test public void testReturnRegisterStruct() { - MemoryLayout struct = MemoryLayout.ofStruct(C_LONG_LONG); + MemoryLayout struct = MemoryLayout.structLayout(C_LONG_LONG); MethodType mt = MethodType.methodType(MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.of(struct); @@ -310,7 +310,7 @@ public class TestWindowsCallArranger extends CallArrangerTestBase { @Test public void testIMR() { - MemoryLayout struct = MemoryLayout.ofStruct(C_LONG_LONG, C_LONG_LONG); + MemoryLayout struct = MemoryLayout.structLayout(C_LONG_LONG, C_LONG_LONG); MethodType mt = MethodType.methodType(MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.of(struct); @@ -330,7 +330,7 @@ public class TestWindowsCallArranger extends CallArrangerTestBase { @Test public void testStackStruct() { - MemoryLayout struct = MemoryLayout.ofStruct(C_POINTER, C_DOUBLE, C_INT); + MemoryLayout struct = MemoryLayout.structLayout(C_POINTER, C_DOUBLE, C_INT); MethodType mt = MethodType.methodType(void.class, MemorySegment.class, int.class, double.class, MemoryAddress.class, diff --git a/test/jdk/java/foreign/channels/AbstractChannelsTest.java b/test/jdk/java/foreign/channels/AbstractChannelsTest.java new file mode 100644 index 00000000000..81c6f0bf193 --- /dev/null +++ b/test/jdk/java/foreign/channels/AbstractChannelsTest.java @@ -0,0 +1,208 @@ +/* + * Copyright (c) 2021, 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. + */ + +import java.io.IOException; +import java.lang.ref.Cleaner; +import java.nio.ByteBuffer; +import java.util.Arrays; +import java.util.Random; +import java.util.concurrent.ExecutionException; +import java.util.function.Supplier; +import java.util.stream.Stream; +import jdk.incubator.foreign.MemoryAccess; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.test.lib.RandomFactory; +import org.testng.annotations.*; +import static org.testng.Assert.*; + +/** + * Not a test, but infra for channel tests. + */ +public class AbstractChannelsTest { + + static final Class IOE = IOException.class; + static final Class EE = ExecutionException.class; + static final Class ISE = IllegalStateException.class; + + @FunctionalInterface + interface ThrowingConsumer { + void accept(T action) throws X; + } + + static ResourceScope closeableScopeOrNull(ResourceScope scope) { + if (scope.isImplicit()) + return null; + return scope; + } + + static long remaining(ByteBuffer[] buffers) { + return Arrays.stream(buffers).mapToLong(ByteBuffer::remaining).sum(); + } + + static ByteBuffer[] flip(ByteBuffer[] buffers) { + Arrays.stream(buffers).forEach(ByteBuffer::flip); + return buffers; + } + + static ByteBuffer[] clear(ByteBuffer[] buffers) { + Arrays.stream(buffers).forEach(ByteBuffer::clear); + return buffers; + } + + static final Random RANDOM = RandomFactory.getRandom(); + + static ByteBuffer segmentBufferOfSize(ResourceScope scope, int size) { + var segment = MemorySegment.allocateNative(size, 1, scope); + for (int i = 0; i < size; i++) { + MemoryAccess.setByteAtOffset(segment, i, ((byte)RANDOM.nextInt())); + } + return segment.asByteBuffer(); + } + + static ByteBuffer[] segmentBuffersOfSize(int len, ResourceScope scope, int size) { + ByteBuffer[] bufs = new ByteBuffer[len]; + for (int i = 0; i < len; i++) + bufs[i] = segmentBufferOfSize(scope, size); + return bufs; + } + + /** + * Returns an array of mixed source byte buffers; both heap and direct, + * where heap can be from the global scope or scope-less, and direct are + * associated with the given scope. + */ + static ByteBuffer[] mixedBuffersOfSize(int len, ResourceScope scope, int size) { + ByteBuffer[] bufs; + boolean atLeastOneScopeBuffer = false; + do { + bufs = new ByteBuffer[len]; + for (int i = 0; i < len; i++) { + bufs[i] = switch (RANDOM.nextInt(3)) { + case 0 -> { byte[] b = new byte[size]; + RANDOM.nextBytes(b); + yield ByteBuffer.wrap(b); } + case 1 -> { byte[] b = new byte[size]; + RANDOM.nextBytes(b); + yield MemorySegment.ofArray(b).asByteBuffer(); } + case 2 -> { atLeastOneScopeBuffer = true; + yield segmentBufferOfSize(scope, size); } + default -> throw new AssertionError("cannot happen"); + }; + } + } while (!atLeastOneScopeBuffer); + return bufs; + } + + static void assertMessage(Exception ex, String msg) { + assertTrue(ex.getMessage().contains(msg), "Expected [%s], in: [%s]".formatted(msg, ex.getMessage())); + } + + static void assertCauses(Throwable ex, Class... exceptions) { + for (var expectedClass : exceptions) { + ex = ex.getCause(); + assertTrue(expectedClass.isInstance(ex), "Expected %s, got: %s".formatted(expectedClass, ex)); + } + } + + @DataProvider(name = "confinedScopes") + public static Object[][] confinedScopes() { + return new Object[][] { + { ScopeSupplier.NEW_CONFINED }, + { ScopeSupplier.NEW_CONFINED_EXPLICIT }, + }; + } + + @DataProvider(name = "sharedScopes") + public static Object[][] sharedScopes() { + return new Object[][] { + { ScopeSupplier.NEW_SHARED }, + { ScopeSupplier.NEW_SHARED_EXPLICIT }, + }; + } + + @DataProvider(name = "closeableScopes") + public static Object[][] closeableScopes() { + return Stream.of(sharedScopes(), confinedScopes()) + .flatMap(Arrays::stream) + .toArray(Object[][]::new); + } + + @DataProvider(name = "implicitScopes") + public static Object[][] implicitScopes() { + return new Object[][] { + { ScopeSupplier.NEW_IMPLICIT }, + { ScopeSupplier.GLOBAL }, + }; + } + + @DataProvider(name = "sharedAndImplicitScopes") + public static Object[][] sharedAndImplicitScopes() { + return Stream.of(sharedScopes(), implicitScopes()) + .flatMap(Arrays::stream) + .toArray(Object[][]::new); + } + + @DataProvider(name = "allScopes") + public static Object[][] allScopes() { + return Stream.of(implicitScopes(), closeableScopes()) + .flatMap(Arrays::stream) + .toArray(Object[][]::new); + } + + @DataProvider(name = "sharedScopesAndTimeouts") + public static Object[][] sharedScopesAndTimeouts() { + return new Object[][] { + { ScopeSupplier.NEW_SHARED , 0 }, + { ScopeSupplier.NEW_SHARED_EXPLICIT , 0 }, + { ScopeSupplier.NEW_SHARED , 30 }, + { ScopeSupplier.NEW_SHARED_EXPLICIT , 30 }, + }; + } + + static class ScopeSupplier implements Supplier { + + static final Supplier NEW_CONFINED = + new ScopeSupplier(() -> ResourceScope.newConfinedScope(), "newConfinedScope()"); + static final Supplier NEW_CONFINED_EXPLICIT = + new ScopeSupplier(() -> ResourceScope.newConfinedScope(Cleaner.create()), "newConfinedScope(Cleaner)"); + static final Supplier NEW_SHARED = + new ScopeSupplier(() -> ResourceScope.newSharedScope(), "newSharedScope()"); + static final Supplier NEW_SHARED_EXPLICIT = + new ScopeSupplier(() -> ResourceScope.newSharedScope(Cleaner.create()), "newSharedScope(Cleaner)"); + static final Supplier NEW_IMPLICIT = + new ScopeSupplier(() -> ResourceScope.newImplicitScope(), "newImplicitScope()"); + static final Supplier GLOBAL = + new ScopeSupplier(() -> ResourceScope.globalScope(), "globalScope()"); + + private final Supplier supplier; + private final String str; + private ScopeSupplier(Supplier supplier, String str) { + this.supplier = supplier; + this.str = str; + } + @Override public String toString() { return str; } + @Override public ResourceScope get() { return supplier.get(); } + } +} + diff --git a/test/jdk/java/foreign/channels/TestAsyncSocketChannels.java b/test/jdk/java/foreign/channels/TestAsyncSocketChannels.java new file mode 100644 index 00000000000..6fb0793c03f --- /dev/null +++ b/test/jdk/java/foreign/channels/TestAsyncSocketChannels.java @@ -0,0 +1,387 @@ +/* + * Copyright (c) 2021, 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 + * @library /test/lib + * @modules java.base/sun.nio.ch + * jdk.incubator.foreign/jdk.internal.foreign + * @key randomness + * @run testng/othervm TestAsyncSocketChannels + * @run testng/othervm -Dsun.nio.ch.disableSynchronousRead=true TestAsyncSocketChannels + * @run testng/othervm -Dsun.nio.ch.disableSynchronousRead=false TestAsyncSocketChannels + */ + +import java.io.IOException; +import java.net.InetAddress; +import java.net.InetSocketAddress; +import java.nio.ByteBuffer; +import java.nio.channels.AsynchronousServerSocketChannel; +import java.nio.channels.AsynchronousSocketChannel; +import java.nio.channels.CompletionHandler; +import java.util.List; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.Future; +import java.util.concurrent.atomic.AtomicBoolean; +import java.util.concurrent.atomic.AtomicInteger; +import java.util.concurrent.atomic.AtomicLong; +import java.util.function.Supplier; +import jdk.incubator.foreign.MemoryAccess; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import org.testng.annotations.*; +import static java.lang.System.out; +import static java.util.concurrent.TimeUnit.SECONDS; +import static org.testng.Assert.*; + +/** + * Tests consisting of buffer views with asynchronous NIO network channels. + */ +public class TestAsyncSocketChannels extends AbstractChannelsTest { + + static final Class IOE = IOException.class; + static final Class EE = ExecutionException.class; + static final Class ISE = IllegalStateException.class; + + /** Tests that confined scopes are not supported. */ + @Test(dataProvider = "confinedScopes") + public void testWithConfined(Supplier scopeSupplier) + throws Throwable + { + try (var channel = AsynchronousSocketChannel.open(); + var server = AsynchronousServerSocketChannel.open(); + var connectedChannel = connectChannels(server, channel); + var scope = scopeSupplier.get()) { + var segment = MemorySegment.allocateNative(10, 1, scope); + var bb = segment.asByteBuffer(); + var bba = new ByteBuffer[] { bb }; + List> ioOps = List.of( + handler -> handler.propagateHandlerFromFuture(channel.write(bb)), + handler -> handler.propagateHandlerFromFuture(channel.read(bb)), + handler -> channel.write(bb, null, handler), + handler -> channel.read( bb, null, handler), + handler -> channel.write(bb , 0L, SECONDS, null, handler), + handler -> channel.read( bb, 0L, SECONDS, null, handler), + handler -> channel.write(bba, 0, bba.length, 0L, SECONDS, null, handler), + handler -> channel.read( bba, 0, bba.length, 0L, SECONDS, null, handler) + ); + for (var ioOp : ioOps) { + out.println("testAsyncWithConfined - op"); + var handler = new TestHandler(); + ioOp.accept(handler); + handler.await() + .assertFailedWith(ISE) + .assertExceptionMessage("Confined scope not supported"); + } + } + } + + /** Tests that I/O with a closed scope throws a suitable exception. */ + @Test(dataProvider = "sharedScopesAndTimeouts") + public void testIOWithClosedSharedScope(Supplier scopeSupplier, int timeout) + throws Exception + { + try (var channel = AsynchronousSocketChannel.open(); + var server = AsynchronousServerSocketChannel.open(); + var connectedChannel = connectChannels(server, channel)) { + ResourceScope scope = scopeSupplier.get(); + ByteBuffer bb = segmentBufferOfSize(scope, 64); + ByteBuffer[] buffers = segmentBuffersOfSize(8, scope, 32); + scope.close(); + { + assertCauses(expectThrows(EE, () -> connectedChannel.read(bb).get()), IOE, ISE); + } + { + var handler = new TestHandler(); + connectedChannel.read(bb, null, handler); + handler.await().assertFailedWith(ISE).assertExceptionMessage("Already closed"); + } + { + var handler = new TestHandler(); + connectedChannel.read(bb, timeout, SECONDS, null, handler); + handler.await().assertFailedWith(ISE).assertExceptionMessage("Already closed"); + } + { + var handler = new TestHandler(); + connectedChannel.read(buffers, 0, buffers.length, timeout, SECONDS, null, handler); + handler.await().assertFailedWith(ISE).assertExceptionMessage("Already closed"); + } + { + assertCauses(expectThrows(EE, () -> connectedChannel.write(bb).get()), IOE, ISE); + } + { + var handler = new TestHandler(); + connectedChannel.write(bb, null, handler); + handler.await().assertFailedWith(ISE).assertExceptionMessage("Already closed"); + } + { + var handler = new TestHandler(); + connectedChannel.write(bb, timeout, SECONDS, null, handler); + handler.await().assertFailedWith(ISE).assertExceptionMessage("Already closed"); + } + { + var handler = new TestHandler(); + connectedChannel.write(buffers, 0, buffers.length, timeout, SECONDS, null, handler); + handler.await().assertFailedWith(ISE).assertExceptionMessage("Already closed"); + } + } + } + + /** Tests basic I/O operations work with views over implicit and shared scopes. */ + @Test(dataProvider = "sharedAndImplicitScopes") + public void testBasicIOWithSupportedScope(Supplier scopeSupplier) + throws Exception + { + ResourceScope scope; + try (var asc1 = AsynchronousSocketChannel.open(); + var assc = AsynchronousServerSocketChannel.open(); + var asc2 = connectChannels(assc, asc1); + var scp = closeableScopeOrNull(scope = scopeSupplier.get())) { + MemorySegment segment1 = MemorySegment.allocateNative(10, 1, scope); + MemorySegment segment2 = MemorySegment.allocateNative(10, 1, scope); + for (int i = 0; i < 10; i++) { + MemoryAccess.setByteAtOffset(segment1, i, (byte) i); + } + { // Future variants + ByteBuffer bb1 = segment1.asByteBuffer(); + ByteBuffer bb2 = segment2.asByteBuffer(); + assertEquals((int)asc1.write(bb1).get(), 10); + assertEquals((int)asc2.read(bb2).get(), 10); + assertEquals(bb2.flip(), ByteBuffer.wrap(new byte[]{0, 1, 2, 3, 4, 5, 6, 7, 8, 9})); + } + { // CompletionHandler variants + ByteBuffer bb1 = segment1.asByteBuffer(); + ByteBuffer bb2 = segment2.asByteBuffer(); + var writeHandler = new TestHandler(); + asc1.write(new ByteBuffer[]{bb1}, 0, 1, 30L, SECONDS, null, writeHandler); + writeHandler.await().assertCompleteWith(10L); + var readHandler = new TestHandler(); + asc2.read(new ByteBuffer[]{bb2}, 0, 1, 30L, SECONDS, null, readHandler); + readHandler.await().assertCompleteWith(10L); + assertEquals(bb2.flip(), ByteBuffer.wrap(new byte[]{0, 1, 2, 3, 4, 5, 6, 7, 8, 9})); + } + { // Gathering/Scattering variants + var writeBuffers = mixedBuffersOfSize(16, scope, 32); + var readBuffers = mixedBuffersOfSize(16, scope, 32); + long expectedCount = remaining(writeBuffers); + var writeHandler = new TestHandler(); + asc1.write(writeBuffers, 0, 16, 30L, SECONDS, null, writeHandler); + writeHandler.await().assertCompleteWith(expectedCount); + var readHandler = new TestHandler(); + asc2.read(readBuffers, 0, 16, 30L, SECONDS, null, readHandler); + readHandler.await().assertCompleteWith(expectedCount); + assertEquals(flip(readBuffers), clear(writeBuffers)); + } + } + } + + /** Tests that a scope is not closeable when there is an outstanding read operation. */ + @Test(dataProvider = "sharedScopesAndTimeouts") + public void testCloseWithOutstandingRead(Supplier scopeSupplier, int timeout) + throws Throwable + { + try (var asc1 = AsynchronousSocketChannel.open(); + var assc = AsynchronousServerSocketChannel.open(); + var asc2 = connectChannels(assc, asc1); + var scope = scopeSupplier.get()) { + var segment = MemorySegment.allocateNative(10, 1, scope); + var bb = segment.asByteBuffer(); + var bba = new ByteBuffer[] { bb }; + List> readOps = List.of( + handler -> handler.propagateHandlerFromFuture(asc1.read(bb)), + handler -> asc1.read(bb, null, handler), + handler -> asc1.read(bb, timeout, SECONDS, null, handler), + handler -> asc1.read(bba, 0, bba.length, timeout, SECONDS, null, handler) + ); + for (var ioOp : readOps) { + out.println("testCloseWithOutstandingRead - op"); + var handler = new TestHandler(); + ioOp.accept(handler); + assertFalse(handler.isDone()); + assertTrue(scope.isAlive()); + assertMessage(expectThrows(ISE, () -> scope.close()), "Scope is acquired by"); + + // write to allow the blocking read complete, which will + // in turn unlock the scope and allow it to be closed. + asc2.write(ByteBuffer.wrap(new byte[] { 0x01 })).get(); + handler.await().assertCompleteWith(1L); + assertTrue(scope.isAlive()); + } + } + } + + /** Tests that a scope is not closeable when there is an outstanding write operation. */ + // Note: limited scenarios are checked, given the 5 sec sleep! + @Test(dataProvider = "sharedScopesAndTimeouts") + public void testCloseWithOutstandingWrite(Supplier scopeSupplier, int timeout) + throws Throwable + { + try (var asc1 = AsynchronousSocketChannel.open(); + var assc = AsynchronousServerSocketChannel.open(); + var asc2 = connectChannels(assc, asc1); + var scope = scopeSupplier.get()) { + + // number of bytes written + final AtomicLong bytesWritten = new AtomicLong(0); + // set to true to signal that no more buffers should be written + final AtomicBoolean continueWriting = new AtomicBoolean(true); + final AtomicInteger outstandingWriteOps = new AtomicInteger(0); + + // write until socket buffer is full so as to create the conditions + // for when a write does not complete immediately + var bba = segmentBuffersOfSize(32, scope, 128); + TestHandler handler; + outstandingWriteOps.getAndIncrement(); + asc1.write(bba, 0, bba.length, timeout, SECONDS, null, + (handler = new TestHandler<>() { + public void completed(Long result, Void att) { + super.completed(result, att); + bytesWritten.addAndGet(result); + if (continueWriting.get()) { + var bba = segmentBuffersOfSize(32, scope, 128); + outstandingWriteOps.getAndIncrement(); + asc1.write(bba, 0, bba.length, timeout, SECONDS, null, this); + } + outstandingWriteOps.getAndDecrement(); + } + })); + // give time for socket buffer to fill up. + Thread.sleep(5*1000); + + assertMessage(expectThrows(ISE, () -> scope.close()), "Scope is acquired by"); + assertTrue(scope.isAlive()); + + // signal handler to stop further writing + continueWriting.set(false); + + // read to allow the outstanding write complete, which will + // in turn unlock the scope and allow it to be closed. + readNBytes(asc2, bytesWritten.get()); + assertTrue(scope.isAlive()); + out.println("outstanding writes: " + outstandingWriteOps.get()); + while (outstandingWriteOps.get() > 0 ) { + out.println("spinning"); + Thread.onSpinWait(); + } + handler.await(); + } + } + + /** Completion handler that exposes conveniences to assert results. */ + static class TestHandler implements CompletionHandler { + volatile V result; + volatile Throwable throwable; + final CountDownLatch latch = new CountDownLatch(1); + + /** Starts a thread that complete the handled with the Future result. */ + TestHandler propagateHandlerFromFuture(Future future) { + Runnable runnable = () -> { + try { + this.completed((V)future.get(), null); + } catch (Throwable t) { + // assert and unwrap exception added by Future + assertTrue(ExecutionException.class.isInstance(t)); + t = t.getCause(); + assertTrue(IOException.class.isInstance(t)); + t = t.getCause(); + this.failed(t, null); + } + }; + Thread t = new Thread(runnable); + t.start(); + return this; + } + + @Override + public void completed(V result, Void att) { + assert result.longValue() >= 0; + this.result = result; + latch.countDown(); + } + @Override + public void failed(Throwable exc, Void att){ + this.throwable = tolerateIOEOnWindows(exc); + latch.countDown(); + } + + TestHandler await() throws InterruptedException{ + latch.await(); + return this; + } + + TestHandler assertCompleteWith(V value) { + assertEquals(result.longValue(), value.longValue()); + assertEquals(throwable, null); + return this; + } + + TestHandler assertFailedWith(Class expectedException) { + assertTrue(expectedException.isInstance(throwable), + "Expected type:%s, got:%s".formatted(expectedException, throwable) ); + assertEquals(result, null, "Unexpected result: " + result); + return this; + } + + TestHandler assertExceptionMessage(String expectedMessage) { + assertEquals(throwable.getMessage(), expectedMessage); + return this; + } + + boolean isDone() { + return latch.getCount() == 0; + } + } + + static AsynchronousSocketChannel connectChannels(AsynchronousServerSocketChannel assc, + AsynchronousSocketChannel asc) + throws Exception + { + assc.bind(new InetSocketAddress(InetAddress.getLoopbackAddress(), 0)); + asc.connect(assc.getLocalAddress()).get(); + return assc.accept().get(); + } + + /** Tolerate the additional level of IOException wrapping of unchecked exceptions + * On Windows, when completing the completion handler with a failure. */ + static Throwable tolerateIOEOnWindows(Throwable t) { + if (System.getProperty("os.name").startsWith("Windows")) { + if (t instanceof IOException) + return t.getCause(); + } + return t; + } + + static void readNBytes(AsynchronousSocketChannel channel, long len) + throws Exception + { + var buf = ByteBuffer.allocateDirect(4096); + long total = 0L; + do { + int n = channel.read(buf).get(); + assertTrue(n > 0, "got:" + n); + buf.clear(); + total += n; + } while (total < len); + } +} diff --git a/test/jdk/java/foreign/channels/TestSocketChannels.java b/test/jdk/java/foreign/channels/TestSocketChannels.java new file mode 100644 index 00000000000..3ca3e2dc11a --- /dev/null +++ b/test/jdk/java/foreign/channels/TestSocketChannels.java @@ -0,0 +1,238 @@ +/* + * Copyright (c) 2021, 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 + * @library /test/lib + * @modules java.base/sun.nio.ch + * jdk.incubator.foreign/jdk.internal.foreign + * @key randomness + * @run testng/othervm TestSocketChannels + */ + +import java.net.InetAddress; +import java.net.InetSocketAddress; +import java.nio.ByteBuffer; +import java.nio.channels.ServerSocketChannel; +import java.nio.channels.SocketChannel; +import java.util.Arrays; +import java.util.List; +import java.util.concurrent.atomic.AtomicReference; +import java.util.function.Supplier; +import java.util.stream.Stream; +import jdk.incubator.foreign.MemoryAccess; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import org.testng.annotations.*; +import static org.testng.Assert.*; + +/** + * Tests consisting of buffer views with synchronous NIO network channels. + */ +public class TestSocketChannels extends AbstractChannelsTest { + + static final Class ISE = IllegalStateException.class; + + @Test(dataProvider = "closeableScopes") + public void testBasicIOWithClosedSegment(Supplier scopeSupplier) + throws Exception + { + try (var channel = SocketChannel.open(); + var server = ServerSocketChannel.open(); + var connectedChannel = connectChannels(server, channel)) { + ResourceScope scope = scopeSupplier.get(); + ByteBuffer bb = segmentBufferOfSize(scope, 16); + scope.close(); + assertMessage(expectThrows(ISE, () -> channel.read(bb)), "Already closed"); + assertMessage(expectThrows(ISE, () -> channel.read(new ByteBuffer[] {bb})), "Already closed"); + assertMessage(expectThrows(ISE, () -> channel.read(new ByteBuffer[] {bb}, 0, 1)), "Already closed"); + assertMessage(expectThrows(ISE, () -> channel.write(bb)), "Already closed"); + assertMessage(expectThrows(ISE, () -> channel.write(new ByteBuffer[] {bb})), "Already closed"); + assertMessage(expectThrows(ISE, () -> channel.write(new ByteBuffer[] {bb}, 0 ,1)), "Already closed"); + } + } + + @Test(dataProvider = "closeableScopes") + public void testScatterGatherWithClosedSegment(Supplier scopeSupplier) + throws Exception + { + try (var channel = SocketChannel.open(); + var server = ServerSocketChannel.open(); + var connectedChannel = connectChannels(server, channel)) { + ResourceScope scope = scopeSupplier.get(); + ByteBuffer[] buffers = segmentBuffersOfSize(8, scope, 16); + scope.close(); + assertMessage(expectThrows(ISE, () -> channel.write(buffers)), "Already closed"); + assertMessage(expectThrows(ISE, () -> channel.read(buffers)), "Already closed"); + assertMessage(expectThrows(ISE, () -> channel.write(buffers, 0 ,8)), "Already closed"); + assertMessage(expectThrows(ISE, () -> channel.read(buffers, 0, 8)), "Already closed"); + } + } + + @Test(dataProvider = "allScopes") + public void testBasicIO(Supplier scopeSupplier) + throws Exception + { + ResourceScope scope; + try (var sc1 = SocketChannel.open(); + var ssc = ServerSocketChannel.open(); + var sc2 = connectChannels(ssc, sc1); + var scp = closeableScopeOrNull(scope = scopeSupplier.get())) { + MemorySegment segment1 = MemorySegment.allocateNative(10, 1, scope); + MemorySegment segment2 = MemorySegment.allocateNative(10, 1, scope); + for (int i = 0; i < 10; i++) { + MemoryAccess.setByteAtOffset(segment1, i, (byte) i); + } + ByteBuffer bb1 = segment1.asByteBuffer(); + ByteBuffer bb2 = segment2.asByteBuffer(); + assertEquals(sc1.write(bb1), 10); + assertEquals(sc2.read(bb2), 10); + assertEquals(bb2.flip(), ByteBuffer.wrap(new byte[]{0, 1, 2, 3, 4, 5, 6, 7, 8, 9})); + } + } + + @Test + public void testBasicHeapIOWithGlobalScope() throws Exception { + try (var sc1 = SocketChannel.open(); + var ssc = ServerSocketChannel.open(); + var sc2 = connectChannels(ssc, sc1)) { + var segment1 = MemorySegment.ofArray(new byte[10]); + var segment2 = MemorySegment.ofArray(new byte[10]); + for (int i = 0; i < 10; i++) { + MemoryAccess.setByteAtOffset(segment1, i, (byte) i); + } + ByteBuffer bb1 = segment1.asByteBuffer(); + ByteBuffer bb2 = segment2.asByteBuffer(); + assertEquals(sc1.write(bb1), 10); + assertEquals(sc2.read(bb2), 10); + assertEquals(bb2.flip(), ByteBuffer.wrap(new byte[]{0, 1, 2, 3, 4, 5, 6, 7, 8, 9})); + } + } + + @Test(dataProvider = "confinedScopes") + public void testIOOnConfinedFromAnotherThread(Supplier scopeSupplier) + throws Exception + { + try (var channel = SocketChannel.open(); + var server = ServerSocketChannel.open(); + var connected = connectChannels(server, channel); + var scope = scopeSupplier.get()) { + var segment = MemorySegment.allocateNative(10, 1, scope); + ByteBuffer bb = segment.asByteBuffer(); + List ioOps = List.of( + () -> channel.write(bb), + () -> channel.read(bb), + () -> channel.write(new ByteBuffer[] {bb}), + () -> channel.read(new ByteBuffer[] {bb}), + () -> channel.write(new ByteBuffer[] {bb}, 0, 1), + () -> channel.read(new ByteBuffer[] {bb}, 0, 1) + ); + for (var ioOp : ioOps) { + AtomicReference exception = new AtomicReference<>(); + Runnable task = () -> exception.set(expectThrows(ISE, ioOp)); + var t = new Thread(task); + t.start(); + t.join(); + assertMessage(exception.get(), "Attempted access outside owning thread"); + } + } + } + + @Test(dataProvider = "allScopes") + public void testScatterGatherIO(Supplier scopeSupplier) + throws Exception + { + ResourceScope scope; + try (var sc1 = SocketChannel.open(); + var ssc = ServerSocketChannel.open(); + var sc2 = connectChannels(ssc, sc1); + var scp = closeableScopeOrNull(scope = scopeSupplier.get())) { + var writeBuffers = mixedBuffersOfSize(32, scope, 64); + var readBuffers = mixedBuffersOfSize(32, scope, 64); + long expectedCount = remaining(writeBuffers); + assertEquals(writeNBytes(sc1, writeBuffers, 0, 32, expectedCount), expectedCount); + assertEquals(readNBytes(sc2, readBuffers, 0, 32, expectedCount), expectedCount); + assertEquals(flip(readBuffers), clear(writeBuffers)); + } + } + + @Test(dataProvider = "closeableScopes") + public void testBasicIOWithDifferentScopes(Supplier scopeSupplier) + throws Exception + { + try (var sc1 = SocketChannel.open(); + var ssc = ServerSocketChannel.open(); + var sc2 = connectChannels(ssc, sc1); + var scope1 = scopeSupplier.get(); + var scope2 = scopeSupplier.get()) { + var writeBuffers = Stream.of(mixedBuffersOfSize(16, scope1, 64), mixedBuffersOfSize(16, scope2, 64)) + .flatMap(Arrays::stream) + .toArray(ByteBuffer[]::new); + var readBuffers = Stream.of(mixedBuffersOfSize(16, scope1, 64), mixedBuffersOfSize(16, scope2, 64)) + .flatMap(Arrays::stream) + .toArray(ByteBuffer[]::new); + + long expectedCount = remaining(writeBuffers); + assertEquals(writeNBytes(sc1, writeBuffers, 0, 32, expectedCount), expectedCount); + assertEquals(readNBytes(sc2, readBuffers, 0, 32, expectedCount), expectedCount); + assertEquals(flip(readBuffers), clear(writeBuffers)); + } + } + + static SocketChannel connectChannels(ServerSocketChannel ssc, SocketChannel sc) + throws Exception + { + ssc.bind(new InetSocketAddress(InetAddress.getLoopbackAddress(), 0)); + sc.connect(ssc.getLocalAddress()); + return ssc.accept(); + } + + static long writeNBytes(SocketChannel channel, + ByteBuffer[] buffers, int offset, int len, + long bytes) + throws Exception + { + long total = 0L; + do { + long n = channel.write(buffers, offset, len); + assertTrue(n > 0, "got:" + n); + total += n; + } while (total < bytes); + return total; + } + + static long readNBytes(SocketChannel channel, + ByteBuffer[] buffers, int offset, int len, + long bytes) + throws Exception + { + long total = 0L; + do { + long n = channel.read(buffers, offset, len); + assertTrue(n > 0, "got:" + n); + total += n; + } while (total < bytes); + return total; + } +} + diff --git a/test/jdk/java/foreign/enablenativeaccess/TestDriver.java b/test/jdk/java/foreign/enablenativeaccess/TestDriver.java new file mode 100644 index 00000000000..382d37e70f6 --- /dev/null +++ b/test/jdk/java/foreign/enablenativeaccess/TestDriver.java @@ -0,0 +1,119 @@ +/* + * Copyright (c) 2021, 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 id=panama_enable_native_access + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm --enable-native-access=panama_module panama_module/org.openjdk.foreigntest.PanamaMain + * @summary with --enable-native-access access to specific module Panama unsafe API succeeds + */ + +/** + * @test id=panama_enable_native_access_reflection + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm --enable-native-access=panama_module panama_module/org.openjdk.foreigntest.PanamaMainReflection + * @summary with --enable-native-access access to specific module Panama unsafe API succeeds + */ + +/** + * @test id=panama_enable_native_access_invoke + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm --enable-native-access=panama_module panama_module/org.openjdk.foreigntest.PanamaMainInvoke + * @summary with --enable-native-access access to specific module Panama unsafe API succeeds + */ + +/** + * @test id=panama_comma_separated_enable + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm --enable-native-access=com.acme,panama_module panama_module/org.openjdk.foreigntest.PanamaMain + * @summary with --enable-native-access access to comma separated list of modules + */ + +/** + * @test id=panama_comma_separated_enable_reflection + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm --enable-native-access=com.acme,panama_module panama_module/org.openjdk.foreigntest.PanamaMainReflection + * @summary with --enable-native-access access to comma separated list of modules + */ + +/** + * @test id=panama_comma_separated_enable_invoke + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm --enable-native-access=com.acme,panama_module panama_module/org.openjdk.foreigntest.PanamaMainInvoke + * @summary with --enable-native-access access to comma separated list of modules + */ + +/** + * @test id=panama_no_enable_native_access_fail + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm/fail panama_module/org.openjdk.foreigntest.PanamaMain + * @summary without --enable-native-access access to Panama unsafe API fails + */ + +/** + * @test id=panama_no_enable_native_access_fail_reflection + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm/fail panama_module/org.openjdk.foreigntest.PanamaMainReflection + * @summary without --enable-native-access access to Panama unsafe API fails + */ + +/** + * @test id=panama_no_enable_native_access_fail_invoke + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm/fail panama_module/org.openjdk.foreigntest.PanamaMainInvoke + * @summary without --enable-native-access access to Panama unsafe API fails + */ + +/** + * @test id=panama_no_all_module_path_blanket_native_access + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build panama_module/* + * @run main/othervm/fail --enable-native-access=ALL-MODULE-PATH panama_module/org.openjdk.foreigntest.PanamaMain + * @summary --enable-native-access does not work with ALL-MODULE-PATH + */ + +/** + * @test id=panama_no_unnamed_module_native_access + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build org.openjdk.foreigntest.PanamaMainUnnamedModule + * @run testng/othervm/fail org.openjdk.foreigntest.PanamaMainUnnamedModule + * @summary --enable-native-access does not work without ALL-UNNAMED + */ + +/** + * @test id=panama_all_unnamed_module_native_access + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build org.openjdk.foreigntest.PanamaMainUnnamedModule + * @run testng/othervm --enable-native-access=ALL-UNNAMED org.openjdk.foreigntest.PanamaMainUnnamedModule + * @summary --enable-native-access ALL-UNNAMED works + */ diff --git a/test/jdk/java/foreign/enablenativeaccess/org/openjdk/foreigntest/PanamaMainUnnamedModule.java b/test/jdk/java/foreign/enablenativeaccess/org/openjdk/foreigntest/PanamaMainUnnamedModule.java new file mode 100644 index 00000000000..bf32954eee3 --- /dev/null +++ b/test/jdk/java/foreign/enablenativeaccess/org/openjdk/foreigntest/PanamaMainUnnamedModule.java @@ -0,0 +1,57 @@ +/* + * Copyright (c) 2021, 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. + */ + +package org.openjdk.foreigntest; + +import jdk.incubator.foreign.*; +import org.testng.annotations.Test; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; +import java.lang.reflect.Method; + +public class PanamaMainUnnamedModule { + @Test + public void testReflection() throws Throwable { + Method method = CLinker.class.getDeclaredMethod("getInstance"); + method.invoke(null); + } + + @Test + public void testSetAccessible() throws Throwable { + Method method = CLinker.class.getDeclaredMethod("getInstance"); + method.setAccessible(true); + method.invoke(null); + } + + @Test + public void testInvoke() throws Throwable { + var mh = MethodHandles.lookup().findStatic(CLinker.class, "getInstance", + MethodType.methodType(CLinker.class)); + var linker = (CLinker)mh.invokeExact(); + } + + @Test + public void testDirectAccess() throws Throwable { + CLinker.getInstance(); + } +} diff --git a/test/jdk/java/foreign/enablenativeaccess/panama_module/module-info.java b/test/jdk/java/foreign/enablenativeaccess/panama_module/module-info.java new file mode 100644 index 00000000000..af14d21a628 --- /dev/null +++ b/test/jdk/java/foreign/enablenativeaccess/panama_module/module-info.java @@ -0,0 +1,27 @@ +/* + * Copyright (c) 2021, 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. + */ + +module panama_module { + requires jdk.incubator.foreign; + exports org.openjdk.foreigntest; +} diff --git a/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMain.java b/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMain.java new file mode 100644 index 00000000000..b4416f87808 --- /dev/null +++ b/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMain.java @@ -0,0 +1,34 @@ +/* + * Copyright (c) 2021, 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. + */ + +package org.openjdk.foreigntest; + +import jdk.incubator.foreign.*; + +public class PanamaMain { + public static void main(String[] args) { + System.out.println("Trying to get CLinker"); + CLinker.getInstance(); + System.out.println("Got CLinker"); + } +} diff --git a/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMainInvoke.java b/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMainInvoke.java new file mode 100644 index 00000000000..6a5a2bd4a92 --- /dev/null +++ b/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMainInvoke.java @@ -0,0 +1,35 @@ +/* + * Copyright (c) 2021, 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. + */ + +package org.openjdk.foreigntest; + +import java.lang.invoke.*; +import jdk.incubator.foreign.*; + +public class PanamaMainInvoke { + public static void main(String[] args) throws Throwable { + var mh = MethodHandles.lookup().findStatic(CLinker.class, "getInstance", + MethodType.methodType(CLinker.class)); + var linker = (CLinker)mh.invokeExact(); + } +} diff --git a/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMainReflection.java b/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMainReflection.java new file mode 100644 index 00000000000..0ddd28b97ee --- /dev/null +++ b/test/jdk/java/foreign/enablenativeaccess/panama_module/org/openjdk/foreigntest/PanamaMainReflection.java @@ -0,0 +1,34 @@ +/* + * Copyright (c) 2021, 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. + */ + +package org.openjdk.foreigntest; + +import jdk.incubator.foreign.*; +import java.lang.reflect.Method; + +public class PanamaMainReflection { + public static void main(String[] args) throws Throwable { + Method method = CLinker.class.getDeclaredMethod("getInstance"); + method.invoke(null); + } +} diff --git a/test/jdk/java/foreign/handles/Driver.java b/test/jdk/java/foreign/handles/Driver.java new file mode 100644 index 00000000000..695cfbca02f --- /dev/null +++ b/test/jdk/java/foreign/handles/Driver.java @@ -0,0 +1,30 @@ +/* + * Copyright (c) 2021, 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 + * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" + * @build invoker_module/* lookup_module/* + * @run testng/othervm --enable-native-access=invoker_module + lookup_module/handle.lookup.MethodHandleLookup + */ diff --git a/test/jdk/java/foreign/handles/invoker_module/handle/invoker/MethodHandleInvoker.java b/test/jdk/java/foreign/handles/invoker_module/handle/invoker/MethodHandleInvoker.java new file mode 100644 index 00000000000..088e5abf37c --- /dev/null +++ b/test/jdk/java/foreign/handles/invoker_module/handle/invoker/MethodHandleInvoker.java @@ -0,0 +1,96 @@ +/* + * Copyright (c) 2021, 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. + */ + +package handle.invoker; + +import jdk.incubator.foreign.Addressable; +import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.FunctionDescriptor; +import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemoryLayouts; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; +import java.nio.charset.Charset; +import java.nio.file.Path; +import java.util.HashMap; +import java.util.Map; + +public class MethodHandleInvoker { + public void call(MethodHandle methodHandle) throws Throwable { + try { + Object[] args = makeArgs(methodHandle.type()); + methodHandle.invokeWithArguments(args); + throw new AssertionError("Call to restricted method did not fail as expected!"); + } catch (IllegalCallerException ex) { + if (!ex.getMessage().contains("lookup_module")) { + throw new AssertionError("Caller module is not lookup_module!"); + } + } catch (Throwable ex) { + throw new AssertionError("Call to restricted method did not fail as expected!"); + } + } + + static final Map, Object> DEFAULT_VALUES = new HashMap<>(); + + static void addDefaultMapping(Class carrier, Z value) { + DEFAULT_VALUES.put(carrier, value); + } + + static { + addDefaultMapping(CLinker.class, CLinker.getInstance()); + addDefaultMapping(long.class, 0L); + addDefaultMapping(Path.class, Path.of("nonExistent")); + addDefaultMapping(String.class, "Hello!"); + addDefaultMapping(Runnable.class, () -> {}); + addDefaultMapping(MethodHandle.class, MethodHandles.identity(int.class)); + addDefaultMapping(Charset.class, Charset.defaultCharset()); + addDefaultMapping(MethodType.class, MethodType.methodType(void.class)); + addDefaultMapping(MemoryAddress.class, MemoryAddress.NULL); + addDefaultMapping(Addressable.class, MemoryAddress.NULL); + addDefaultMapping(MemoryLayout.class, MemoryLayouts.JAVA_INT); + addDefaultMapping(FunctionDescriptor.class, FunctionDescriptor.ofVoid()); + addDefaultMapping(LibraryLookup.class, LibraryLookup.ofDefault()); + addDefaultMapping(ResourceScope.class, ResourceScope.newImplicitScope()); + addDefaultMapping(SegmentAllocator.class, (size, align) -> null); + } + + static Object[] makeArgs(MethodType type) { + return type.parameterList().stream() + .map(MethodHandleInvoker::makeArg) + .toArray(); + } + + static Object makeArg(Class clazz) { + Object value = DEFAULT_VALUES.get(clazz); + if (value == null) { + throw new UnsupportedOperationException(clazz.getName()); + } + return value; + } +} diff --git a/test/jdk/java/foreign/handles/invoker_module/module-info.java b/test/jdk/java/foreign/handles/invoker_module/module-info.java new file mode 100644 index 00000000000..39ddafabd65 --- /dev/null +++ b/test/jdk/java/foreign/handles/invoker_module/module-info.java @@ -0,0 +1,27 @@ +/* + * Copyright (c) 2021, 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. + */ + +module invoker_module { + requires jdk.incubator.foreign; + exports handle.invoker; +} diff --git a/test/jdk/java/foreign/handles/lookup_module/handle/lookup/MethodHandleLookup.java b/test/jdk/java/foreign/handles/lookup_module/handle/lookup/MethodHandleLookup.java new file mode 100644 index 00000000000..9a2322d2a14 --- /dev/null +++ b/test/jdk/java/foreign/handles/lookup_module/handle/lookup/MethodHandleLookup.java @@ -0,0 +1,100 @@ +/* + * Copyright (c) 2021, 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. + */ + +package handle.lookup; + +import jdk.incubator.foreign.Addressable; +import jdk.incubator.foreign.CLinker; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; +import java.nio.charset.Charset; +import java.nio.file.Path; +import java.util.Optional; + +import jdk.incubator.foreign.FunctionDescriptor; +import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; + +import org.testng.annotations.*; + +public class MethodHandleLookup { + + @Test(dataProvider = "restrictedMethods") + public void testRestrictedHandles(MethodHandle handle, String testName) throws Throwable { + new handle.invoker.MethodHandleInvoker().call(handle); + } + + @DataProvider(name = "restrictedMethods") + static Object[][] restrictedMethods() { + try { + return new Object[][]{ + { MethodHandles.lookup().findStatic(CLinker.class, "getInstance", + MethodType.methodType(CLinker.class)), "CLinker::getInstance" }, + { MethodHandles.lookup().findStatic(CLinker.class, "toJavaString", + MethodType.methodType(String.class, MemoryAddress.class)), + "CLinker::toJavaString/1" }, + { MethodHandles.lookup().findStatic(CLinker.class, "toJavaString", + MethodType.methodType(String.class, MemoryAddress.class, Charset.class)), + "CLinker::toJavaString/2" }, + { MethodHandles.lookup().findStatic(CLinker.class, "allocateMemory", + MethodType.methodType(MemoryAddress.class, long.class)), + "CLinker::allocateMemory" }, + { MethodHandles.lookup().findStatic(CLinker.class, "freeMemory", + MethodType.methodType(void.class, MemoryAddress.class)), + "CLinker::freeMemory" }, + { MethodHandles.lookup().findStatic(CLinker.VaList.class, "ofAddress", + MethodType.methodType(CLinker.VaList.class, MemoryAddress.class)), + "VaList::ofAddress/1" }, + { MethodHandles.lookup().findStatic(CLinker.VaList.class, "ofAddress", + MethodType.methodType(CLinker.VaList.class, MemoryAddress.class, ResourceScope.class)), + "VaList::ofAddress/2" }, + { MethodHandles.lookup().findStatic(LibraryLookup.class, "ofPath", + MethodType.methodType(LibraryLookup.class, Path.class)), + "LibraryLookup::ofPath" }, + { MethodHandles.lookup().findStatic(LibraryLookup.class, "ofLibrary", + MethodType.methodType(LibraryLookup.class, String.class)), + "LibraryLookup::ofLibrary" }, + { MethodHandles.lookup().findStatic(LibraryLookup.class, "ofDefault", + MethodType.methodType(LibraryLookup.class)), + "LibraryLookup::ofDefault" }, + { MethodHandles.lookup().findVirtual(MemoryAddress.class, "asSegment", + MethodType.methodType(MemorySegment.class, long.class, ResourceScope.class)), + "MemoryAddress::asSegment/1" }, + { MethodHandles.lookup().findVirtual(MemoryAddress.class, "asSegment", + MethodType.methodType(MemorySegment.class, long.class, Runnable.class, ResourceScope.class)), + "MemoryAddress::asSegment/2" }, + { MethodHandles.lookup().findStatic(MemorySegment.class, "globalNativeSegment", + MethodType.methodType(MemorySegment.class)), + "MemoryAddress::globalNativeSegment" } + }; + } catch (Throwable ex) { + throw new ExceptionInInitializerError((ex)); + } + } +} diff --git a/test/jdk/java/foreign/handles/lookup_module/module-info.java b/test/jdk/java/foreign/handles/lookup_module/module-info.java new file mode 100644 index 00000000000..c9542a5af19 --- /dev/null +++ b/test/jdk/java/foreign/handles/lookup_module/module-info.java @@ -0,0 +1,29 @@ +/* + * Copyright (c) 2021, 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. + */ + +open module lookup_module { + requires testng; + requires jdk.incubator.foreign; + requires invoker_module; + exports handle.lookup; +} diff --git a/test/jdk/java/foreign/libLookupTest.c b/test/jdk/java/foreign/libLookupTest.c index 5eab5048caf..7aa6ca36031 100644 --- a/test/jdk/java/foreign/libLookupTest.c +++ b/test/jdk/java/foreign/libLookupTest.c @@ -29,4 +29,5 @@ #endif EXPORT void f() { } + EXPORT int c = 42; diff --git a/test/jdk/java/foreign/libSafeAccess.c b/test/jdk/java/foreign/libSafeAccess.c new file mode 100644 index 00000000000..a5bcbe0e892 --- /dev/null +++ b/test/jdk/java/foreign/libSafeAccess.c @@ -0,0 +1,38 @@ +/* + * Copyright (c) 2021, 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. + * + */ + +#ifdef _WIN64 +#define EXPORT __declspec(dllexport) +#else +#define EXPORT +#endif + +struct Point { + int x; + int y; +}; + +EXPORT void struct_func(struct Point p) { } + +EXPORT void addr_func(struct Point* p) { } diff --git a/test/jdk/java/foreign/libTestUpcallStructScope.c b/test/jdk/java/foreign/libTestUpcallStructScope.c new file mode 100644 index 00000000000..e27ab0092cc --- /dev/null +++ b/test/jdk/java/foreign/libTestUpcallStructScope.c @@ -0,0 +1,34 @@ +/* + * 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. + */ + +#ifdef _WIN64 +#define EXPORT __declspec(dllexport) +#else +#define EXPORT +#endif + +struct S_PDI { void* p0; double p1; int p2; }; + +EXPORT void do_upcall(void (*cb)(struct S_PDI), struct S_PDI a0) { + cb(a0); +} diff --git a/test/jdk/java/foreign/stackwalk/TestStackWalk.java b/test/jdk/java/foreign/stackwalk/TestStackWalk.java index bbe02440bea..1a5d5a771a0 100644 --- a/test/jdk/java/foreign/stackwalk/TestStackWalk.java +++ b/test/jdk/java/foreign/stackwalk/TestStackWalk.java @@ -33,7 +33,7 @@ * -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=true - * -Dforeign.restricted=permit + * --enable-native-access=ALL-UNNAMED * -Xbatch * TestStackWalk * @@ -42,7 +42,7 @@ * -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI * -Djdk.internal.foreign.ProgrammableInvoker.USE_INTRINSICS=false - * -Dforeign.restricted=permit + * --enable-native-access=ALL-UNNAMED * -Xbatch * TestStackWalk */ @@ -57,6 +57,7 @@ import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodType; import java.lang.ref.Reference; +import jdk.incubator.foreign.ResourceScope; import sun.hotspot.WhiteBox; import static java.lang.invoke.MethodHandles.lookup; @@ -86,7 +87,8 @@ public class TestStackWalk { static boolean armed; public static void main(String[] args) throws Throwable { - try (MemorySegment stub = linker.upcallStub(MH_m, FunctionDescriptor.ofVoid())) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemoryAddress stub = linker.upcallStub(MH_m, FunctionDescriptor.ofVoid(), scope); MemoryAddress stubAddress = stub.address(); armed = false; for (int i = 0; i < 20_000; i++) { @@ -105,7 +107,8 @@ public class TestStackWalk { static void m() { if (armed) { - WB.verifyFrames(true); + WB.verifyFrames(/*log=*/true, /*updateRegisterMap=*/true); + WB.verifyFrames(/*log=*/true, /*updateRegisterMap=*/false); // triggers different code paths } } diff --git a/test/jdk/java/foreign/valist/VaListTest.java b/test/jdk/java/foreign/valist/VaListTest.java index eef6ec6f0f1..614da0bb546 100644 --- a/test/jdk/java/foreign/valist/VaListTest.java +++ b/test/jdk/java/foreign/valist/VaListTest.java @@ -24,18 +24,18 @@ /* * @test + * @library ../ * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" * @modules jdk.incubator.foreign/jdk.internal.foreign * jdk.incubator.foreign/jdk.internal.foreign.abi * jdk.incubator.foreign/jdk.internal.foreign.abi.aarch64 * jdk.incubator.foreign/jdk.internal.foreign.abi.x64.windows * jdk.incubator.foreign/jdk.internal.foreign.abi.x64.sysv - * @run testng/othervm -Dforeign.restricted=permit VaListTest + * @run testng/othervm --enable-native-access=ALL-UNNAMED VaListTest */ import jdk.incubator.foreign.*; import jdk.incubator.foreign.CLinker.VaList; -import jdk.internal.foreign.abi.SharedUtils; import jdk.internal.foreign.abi.aarch64.AArch64Linker; import jdk.internal.foreign.abi.x64.sysv.SysVx64Linker; import jdk.internal.foreign.abi.x64.windows.Windowsx64Linker; @@ -47,9 +47,6 @@ import java.lang.invoke.MethodHandleProxies; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; import java.lang.invoke.VarHandle; -import java.lang.reflect.InvocationTargetException; -import java.lang.reflect.Method; -import java.util.Arrays; import java.util.function.BiFunction; import java.util.function.Consumer; import java.util.function.Function; @@ -67,7 +64,7 @@ import static jdk.incubator.foreign.MemoryLayouts.JAVA_INT; import static jdk.internal.foreign.PlatformLayouts.*; import static org.testng.Assert.*; -public class VaListTest { +public class VaListTest extends NativeTestHelper { private static final CLinker abi = CLinker.getInstance(); private static final LibraryLookup lookup = LibraryLookup.ofLibrary("VaList"); @@ -109,22 +106,22 @@ public class VaListTest { } private static final Function, VaList> winVaListFactory - = actions -> Windowsx64Linker.newVaList(actions, MemorySegment::allocateNative); + = actions -> Windowsx64Linker.newVaList(actions, ResourceScope.newConfinedScope()); private static final Function, VaList> sysvVaListFactory - = actions -> SysVx64Linker.newVaList(actions, MemorySegment::allocateNative); + = actions -> SysVx64Linker.newVaList(actions, ResourceScope.newConfinedScope()); private static final Function, VaList> aarch64VaListFactory - = actions -> AArch64Linker.newVaList(actions, MemorySegment::allocateNative); + = actions -> AArch64Linker.newVaList(actions, ResourceScope.newConfinedScope()); private static final Function, VaList> platformVaListFactory - = VaList::make; + = (builder) -> VaList.make(builder, ResourceScope.newConfinedScope()); private static final BiFunction, NativeScope, VaList> winVaListScopedFactory - = (actions, scope) -> Windowsx64Linker.newVaList(actions, SharedUtils.Allocator.ofScope(scope)); + = (builder, scope) -> Windowsx64Linker.newVaList(builder, scope.scope()); private static final BiFunction, NativeScope, VaList> sysvVaListScopedFactory - = (actions, scope) -> SysVx64Linker.newVaList(actions, SharedUtils.Allocator.ofScope(scope)); + = (builder, scope) -> SysVx64Linker.newVaList(builder, scope.scope()); private static final BiFunction, NativeScope, VaList> aarch64VaListScopedFactory - = (actions, scope) -> AArch64Linker.newVaList(actions, SharedUtils.Allocator.ofScope(scope)); + = (builder, scope) -> AArch64Linker.newVaList(builder, scope.scope()); private static final BiFunction, NativeScope, VaList> platformVaListScopedFactory - = VaList::make; + = (builder, scope) -> VaList.make(builder, scope.scope()); @DataProvider @SuppressWarnings("unchecked") @@ -145,13 +142,13 @@ public class VaListTest { public void testIntSum(Function, VaList> vaListFactory, BiFunction sumInts, ValueLayout intLayout) { - try (VaList vaList = vaListFactory.apply(b -> - b.vargFromInt(intLayout, 10) - .vargFromInt(intLayout, 15) - .vargFromInt(intLayout, 20))) { - int x = sumInts.apply(3, vaList); - assertEquals(x, 45); - } + VaList vaList = vaListFactory.apply(b -> + b.vargFromInt(intLayout, 10) + .vargFromInt(intLayout, 15) + .vargFromInt(intLayout, 20)); + int x = sumInts.apply(3, vaList); + assertEquals(x, 45); + vaList.scope().close(); } @DataProvider @@ -173,13 +170,13 @@ public class VaListTest { public void testDoubleSum(Function, VaList> vaListFactory, BiFunction sumDoubles, ValueLayout doubleLayout) { - try (VaList vaList = vaListFactory.apply(b -> - b.vargFromDouble(doubleLayout, 3.0D) - .vargFromDouble(doubleLayout, 4.0D) - .vargFromDouble(doubleLayout, 5.0D))) { - double x = sumDoubles.apply(3, vaList); - assertEquals(x, 12.0D); - } + VaList vaList = vaListFactory.apply(b -> + b.vargFromDouble(doubleLayout, 3.0D) + .vargFromDouble(doubleLayout, 4.0D) + .vargFromDouble(doubleLayout, 5.0D)); + double x = sumDoubles.apply(3, vaList); + assertEquals(x, 12.0D); + vaList.scope().close(); } @DataProvider @@ -188,7 +185,7 @@ public class VaListTest { Function> getIntJavaFact = layout -> list -> { MemoryAddress ma = list.vargAsAddress(layout); - return MemoryAccess.getIntAtOffset(MemorySegment.ofNativeRestricted(), ma.toRawLongValue()); + return MemoryAccess.getIntAtOffset(MemorySegment.globalNativeSegment(), ma.toRawLongValue()); }; Function getIntNative = MethodHandleProxies.asInterfaceInstance(Function.class, MH_getInt); return new Object[][]{ @@ -203,12 +200,13 @@ public class VaListTest { public void testVaListMemoryAddress(Function, VaList> vaListFactory, Function getFromPointer, ValueLayout pointerLayout) { - try (MemorySegment msInt = MemorySegment.allocateNative(JAVA_INT)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment msInt = MemorySegment.allocateNative(JAVA_INT, scope); MemoryAccess.setInt(msInt, 10); - try (VaList vaList = vaListFactory.apply(b -> b.vargFromAddress(pointerLayout, msInt.address()))) { - int x = getFromPointer.apply(vaList); - assertEquals(x, 10); - } + VaList vaList = vaListFactory.apply(b -> b.vargFromAddress(pointerLayout, msInt.address())); + int x = getFromPointer.apply(vaList); + assertEquals(x, 10); + vaList.scope().close(); } } @@ -222,7 +220,7 @@ public class VaListTest { TriFunction> sumStructJavaFact = (pointLayout, VH_Point_x, VH_Point_y) -> list -> { - MemorySegment struct = list.vargAsSegment(pointLayout); + MemorySegment struct = list.vargAsSegment(pointLayout, ResourceScope.newImplicitScope()); int x = (int) VH_Point_x.get(struct); int y = (int) VH_Point_y.get(struct); return x + y; @@ -235,7 +233,7 @@ public class VaListTest { TriFunction, VaList>, MemoryLayout, TriFunction>, Object[]> argsFact = (vaListFact, intLayout, sumStructFact) -> { - GroupLayout pointLayout = MemoryLayout.ofStruct( + GroupLayout pointLayout = MemoryLayout.structLayout( intLayout.withName("x"), intLayout.withName("y") ); @@ -256,14 +254,15 @@ public class VaListTest { public void testStruct(Function, VaList> vaListFactory, Function sumStruct, GroupLayout Point_LAYOUT, VarHandle VH_Point_x, VarHandle VH_Point_y) { - try (MemorySegment struct = MemorySegment.allocateNative(Point_LAYOUT)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment struct = MemorySegment.allocateNative(Point_LAYOUT, scope); VH_Point_x.set(struct, 5); VH_Point_y.set(struct, 10); - try (VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(Point_LAYOUT, struct))) { - int sum = sumStruct.apply(vaList); - assertEquals(sum, 15); - } + VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(Point_LAYOUT, struct)); + int sum = sumStruct.apply(vaList); + assertEquals(sum, 15); + vaList.scope().close(); } } @@ -273,7 +272,7 @@ public class VaListTest { TriFunction> sumStructJavaFact = (BigPoint_LAYOUT, VH_BigPoint_x, VH_BigPoint_y) -> list -> { - MemorySegment struct = list.vargAsSegment(BigPoint_LAYOUT); + MemorySegment struct = list.vargAsSegment(BigPoint_LAYOUT, ResourceScope.newImplicitScope()); long x = (long) VH_BigPoint_x.get(struct); long y = (long) VH_BigPoint_y.get(struct); return x + y; @@ -286,7 +285,7 @@ public class VaListTest { TriFunction, VaList>, MemoryLayout, TriFunction>, Object[]> argsFact = (vaListFact, longLongLayout, sumBigStructFact) -> { - GroupLayout BigPoint_LAYOUT = MemoryLayout.ofStruct( + GroupLayout BigPoint_LAYOUT = MemoryLayout.structLayout( longLongLayout.withName("x"), longLongLayout.withName("y") ); @@ -307,14 +306,15 @@ public class VaListTest { public void testBigStruct(Function, VaList> vaListFactory, Function sumBigStruct, GroupLayout BigPoint_LAYOUT, VarHandle VH_BigPoint_x, VarHandle VH_BigPoint_y) { - try (MemorySegment struct = MemorySegment.allocateNative(BigPoint_LAYOUT)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment struct = MemorySegment.allocateNative(BigPoint_LAYOUT, scope); VH_BigPoint_x.set(struct, 5); VH_BigPoint_y.set(struct, 10); - try (VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(BigPoint_LAYOUT, struct))) { - long sum = sumBigStruct.apply(vaList); - assertEquals(sum, 15); - } + VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(BigPoint_LAYOUT, struct)); + long sum = sumBigStruct.apply(vaList); + assertEquals(sum, 15); + vaList.scope().close(); } } @@ -324,7 +324,7 @@ public class VaListTest { TriFunction> sumStructJavaFact = (FloatPoint_LAYOUT, VH_FloatPoint_x, VH_FloatPoint_y) -> list -> { - MemorySegment struct = list.vargAsSegment(FloatPoint_LAYOUT); + MemorySegment struct = list.vargAsSegment(FloatPoint_LAYOUT, ResourceScope.newImplicitScope()); float x = (float) VH_FloatPoint_x.get(struct); float y = (float) VH_FloatPoint_y.get(struct); return x + y; @@ -337,7 +337,7 @@ public class VaListTest { TriFunction, VaList>, MemoryLayout, TriFunction>, Object[]> argsFact = (vaListFact, floatLayout, sumFloatStructFact) -> { - GroupLayout FloatPoint_LAYOUT = MemoryLayout.ofStruct( + GroupLayout FloatPoint_LAYOUT = MemoryLayout.structLayout( floatLayout.withName("x"), floatLayout.withName("y") ); @@ -359,14 +359,15 @@ public class VaListTest { Function sumFloatStruct, GroupLayout FloatPoint_LAYOUT, VarHandle VH_FloatPoint_x, VarHandle VH_FloatPoint_y) { - try (MemorySegment struct = MemorySegment.allocateNative(FloatPoint_LAYOUT)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment struct = MemorySegment.allocateNative(FloatPoint_LAYOUT, scope); VH_FloatPoint_x.set(struct, 1.234f); VH_FloatPoint_y.set(struct, 3.142f); - try (VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(FloatPoint_LAYOUT, struct))) { - float sum = sumFloatStruct.apply(vaList); - assertEquals(sum, 4.376f, 0.00001f); - } + VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(FloatPoint_LAYOUT, struct)); + float sum = sumFloatStruct.apply(vaList); + assertEquals(sum, 4.376f, 0.00001f); + vaList.scope().close(); } } @@ -380,7 +381,7 @@ public class VaListTest { QuadFunc> sumStructJavaFact = (HugePoint_LAYOUT, VH_HugePoint_x, VH_HugePoint_y, VH_HugePoint_z) -> list -> { - MemorySegment struct = list.vargAsSegment(HugePoint_LAYOUT); + MemorySegment struct = list.vargAsSegment(HugePoint_LAYOUT, ResourceScope.newImplicitScope()); long x = (long) VH_HugePoint_x.get(struct); long y = (long) VH_HugePoint_y.get(struct); long z = (long) VH_HugePoint_z.get(struct); @@ -394,7 +395,7 @@ public class VaListTest { TriFunction, VaList>, MemoryLayout, QuadFunc>, Object[]> argsFact = (vaListFact, longLongLayout, sumBigStructFact) -> { - GroupLayout HugePoint_LAYOUT = MemoryLayout.ofStruct( + GroupLayout HugePoint_LAYOUT = MemoryLayout.structLayout( longLongLayout.withName("x"), longLongLayout.withName("y"), longLongLayout.withName("z") @@ -421,15 +422,16 @@ public class VaListTest { VarHandle VH_HugePoint_x, VarHandle VH_HugePoint_y, VarHandle VH_HugePoint_z) { // On AArch64 a struct needs to be larger than 16 bytes to be // passed by reference. - try (MemorySegment struct = MemorySegment.allocateNative(HugePoint_LAYOUT)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment struct = MemorySegment.allocateNative(HugePoint_LAYOUT, scope); VH_HugePoint_x.set(struct, 1); VH_HugePoint_y.set(struct, 2); VH_HugePoint_z.set(struct, 3); - try (VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(HugePoint_LAYOUT, struct))) { - long sum = sumHugeStruct.apply(vaList); - assertEquals(sum, 6); - } + VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(HugePoint_LAYOUT, struct)); + long sum = sumHugeStruct.apply(vaList); + assertEquals(sum, 6); + vaList.scope().close(); } } @@ -463,7 +465,7 @@ public class VaListTest { { winVaListFactory, sumStackJavaFact.apply(Win64.C_LONG_LONG, Win64.C_DOUBLE), Win64.C_LONG_LONG, Win64.C_DOUBLE }, { sysvVaListFactory, sumStackJavaFact.apply(SysV.C_LONG_LONG, SysV.C_DOUBLE), SysV.C_LONG_LONG, SysV.C_DOUBLE }, { aarch64VaListFactory, sumStackJavaFact.apply(AArch64.C_LONG_LONG, AArch64.C_DOUBLE), AArch64.C_LONG_LONG, AArch64.C_DOUBLE }, - { platformVaListFactory, sumStackNative, C_LONG_LONG, C_DOUBLE }, + { platformVaListFactory, sumStackNative, C_LONG_LONG, C_DOUBLE }, }; } @@ -472,8 +474,9 @@ public class VaListTest { SumStackFunc sumStack, ValueLayout longLayout, ValueLayout doubleLayout) { - try (MemorySegment longSum = MemorySegment.allocateNative(longLayout); - MemorySegment doubleSum = MemorySegment.allocateNative(doubleLayout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment longSum = MemorySegment.allocateNative(longLayout, scope); + MemorySegment doubleSum = MemorySegment.allocateNative(doubleLayout, scope); MemoryAccess.setLong(longSum, 0L); MemoryAccess.setDouble(doubleSum, 0D); @@ -486,8 +489,10 @@ public class VaListTest { } }); - try (list) { + try { sumStack.invoke(longSum, doubleSum, list); + } finally { + list.scope().close(); } long lSum = MemoryAccess.getLong(longSum); @@ -501,7 +506,8 @@ public class VaListTest { @Test(dataProvider = "upcalls") public void testUpcall(MethodHandle target, MethodHandle callback) throws Throwable { FunctionDescriptor desc = FunctionDescriptor.ofVoid(C_VA_LIST); - try (MemorySegment stub = abi.upcallStub(callback, desc)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemoryAddress stub = abi.upcallStub(callback, desc, scope); target.invokeExact(stub.address()); } } @@ -519,10 +525,10 @@ public class VaListTest { } @Test(expectedExceptions = UnsupportedOperationException.class, - expectedExceptionsMessageRegExp = ".*Empty VaList.*", + expectedExceptionsMessageRegExp = ".*Scope cannot be closed.*", dataProvider = "emptyVaLists") public void testEmptyNotCloseable(VaList emptyList) { - emptyList.close(); + emptyList.scope().close(); } @DataProvider @@ -545,7 +551,7 @@ public class VaListTest { BiFunction sumInts, ValueLayout intLayout) { VaList listLeaked; - try (NativeScope scope = NativeScope.unboundedScope()) { + try (NativeScope scope = new NativeScope()) { VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4) .vargFromInt(intLayout, 8), scope); @@ -553,7 +559,7 @@ public class VaListTest { assertEquals(x, 12); listLeaked = list; } - assertFalse(listLeaked.isAlive()); + assertFalse(listLeaked.scope().isAlive()); } @Test(dataProvider = "structs") @@ -561,20 +567,21 @@ public class VaListTest { Function sumStruct, // ignored GroupLayout Point_LAYOUT, VarHandle VH_Point_x, VarHandle VH_Point_y) { MemorySegment pointOut; - try (NativeScope scope = NativeScope.unboundedScope()) { - try (MemorySegment pointIn = MemorySegment.allocateNative(Point_LAYOUT)) { + try (NativeScope scope = new NativeScope()) { + try (ResourceScope innerScope = ResourceScope.newConfinedScope()) { + MemorySegment pointIn = MemorySegment.allocateNative(Point_LAYOUT, innerScope); VH_Point_x.set(pointIn, 3); VH_Point_y.set(pointIn, 6); - try (VaList list = vaListFactory.apply(b -> b.vargFromSegment(Point_LAYOUT, pointIn))) { - pointOut = list.vargAsSegment(Point_LAYOUT, scope); - assertEquals((int) VH_Point_x.get(pointOut), 3); - assertEquals((int) VH_Point_y.get(pointOut), 6); - } - assertTrue(pointOut.isAlive()); // after VaList freed + VaList list = vaListFactory.apply(b -> b.vargFromSegment(Point_LAYOUT, pointIn)); + pointOut = list.vargAsSegment(Point_LAYOUT, scope); + assertEquals((int) VH_Point_x.get(pointOut), 3); + assertEquals((int) VH_Point_y.get(pointOut), 6); + list.scope().close(); + assertTrue(pointOut.scope().isAlive()); // after VaList freed } - assertTrue(pointOut.isAlive()); // after input MS freed + assertTrue(pointOut.scope().isAlive()); // after inner scope freed } - assertFalse(pointOut.isAlive()); // after scope freed + assertFalse(pointOut.scope().isAlive()); // after outer scope freed } @DataProvider @@ -588,36 +595,22 @@ public class VaListTest { @Test(dataProvider = "copy") public void testCopy(Function, VaList> vaListFactory, ValueLayout intLayout) { - try (VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4) - .vargFromInt(intLayout, 8))) { - VaList copy = list.copy(); - assertEquals(copy.vargAsInt(intLayout), 4); - assertEquals(copy.vargAsInt(intLayout), 8); - copy.close(); + VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4) + .vargFromInt(intLayout, 8)); + VaList copy = list.copy(); + assertEquals(copy.vargAsInt(intLayout), 4); + assertEquals(copy.vargAsInt(intLayout), 8); - assertFalse(copy.isAlive()); +// try { // this logic only works on Windows! +// int x = copy.vargAsInt(intLayout); +// fail(); +// } catch (IndexOutOfBoundsException ex) { +// // ok - we exhausted the list +// } - assertEquals(list.vargAsInt(intLayout), 4); - assertEquals(list.vargAsInt(intLayout), 8); - } - } - - @Test(dataProvider = "copy") - public void testScopedCopy(Function, VaList> vaListFactory, ValueLayout intLayout) { - try (VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4) - .vargFromInt(intLayout, 8))) { - VaList copy; - try (NativeScope scope = NativeScope.unboundedScope()) { - copy = list.copy(scope); - - assertEquals(copy.vargAsInt(intLayout), 4); - assertEquals(copy.vargAsInt(intLayout), 8); - } - assertFalse(copy.isAlive()); - - assertEquals(list.vargAsInt(intLayout), 4); - assertEquals(list.vargAsInt(intLayout), 8); - } + assertEquals(list.vargAsInt(intLayout), 4); + assertEquals(list.vargAsInt(intLayout), 8); + list.scope().close(); } @Test(dataProvider = "copy", @@ -626,48 +619,33 @@ public class VaListTest { ValueLayout intLayout) { VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4) .vargFromInt(intLayout, 8)); - try (VaList copy = list.copy()) { - list.close(); + VaList copy = list.copy(); + list.scope().close(); - copy.vargAsInt(intLayout); // should throw - } - } - - @Test(dataProvider = "copy", - expectedExceptions = IllegalStateException.class) - public void testCopyUnusableAfterOriginalClosedScope(Function, VaList> vaListFactory, - ValueLayout intLayout) { - VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4) - .vargFromInt(intLayout, 8)); - try (NativeScope scope = NativeScope.unboundedScope()) { - VaList copy = list.copy(scope); - list.close(); - - copy.vargAsInt(intLayout); // should throw - } + copy.vargAsInt(intLayout); // should throw } @DataProvider public static Object[][] upcalls() { - GroupLayout BigPoint_LAYOUT = MemoryLayout.ofStruct( + GroupLayout BigPoint_LAYOUT = MemoryLayout.structLayout( C_LONG_LONG.withName("x"), C_LONG_LONG.withName("y") ); VarHandle VH_BigPoint_x = BigPoint_LAYOUT.varHandle(long.class, groupElement("x")); VarHandle VH_BigPoint_y = BigPoint_LAYOUT.varHandle(long.class, groupElement("y")); - GroupLayout Point_LAYOUT = MemoryLayout.ofStruct( + GroupLayout Point_LAYOUT = MemoryLayout.structLayout( C_INT.withName("x"), C_INT.withName("y") ); VarHandle VH_Point_x = Point_LAYOUT.varHandle(int.class, groupElement("x")); VarHandle VH_Point_y = Point_LAYOUT.varHandle(int.class, groupElement("y")); - GroupLayout FloatPoint_LAYOUT = MemoryLayout.ofStruct( + GroupLayout FloatPoint_LAYOUT = MemoryLayout.structLayout( C_FLOAT.withName("x"), C_FLOAT.withName("y") ); VarHandle VH_FloatPoint_x = FloatPoint_LAYOUT.varHandle(float.class, groupElement("x")); VarHandle VH_FloatPoint_y = FloatPoint_LAYOUT.varHandle(float.class, groupElement("y")); - GroupLayout HugePoint_LAYOUT = MemoryLayout.ofStruct( + GroupLayout HugePoint_LAYOUT = MemoryLayout.structLayout( C_LONG_LONG.withName("x"), C_LONG_LONG.withName("y"), C_LONG_LONG.withName("z") @@ -678,49 +656,43 @@ public class VaListTest { return new Object[][]{ { linkVaListCB("upcallBigStruct"), VaListConsumer.mh(vaList -> { - try (MemorySegment struct = vaList.vargAsSegment(BigPoint_LAYOUT)) { - assertEquals((long) VH_BigPoint_x.get(struct), 8); - assertEquals((long) VH_BigPoint_y.get(struct), 16); - } + MemorySegment struct = vaList.vargAsSegment(BigPoint_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((long) VH_BigPoint_x.get(struct), 8); + assertEquals((long) VH_BigPoint_y.get(struct), 16); })}, { linkVaListCB("upcallBigStruct"), VaListConsumer.mh(vaList -> { VaList copy = vaList.copy(); - try (MemorySegment struct = vaList.vargAsSegment(BigPoint_LAYOUT)) { - assertEquals((long) VH_BigPoint_x.get(struct), 8); - assertEquals((long) VH_BigPoint_y.get(struct), 16); + MemorySegment struct = vaList.vargAsSegment(BigPoint_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((long) VH_BigPoint_x.get(struct), 8); + assertEquals((long) VH_BigPoint_y.get(struct), 16); - VH_BigPoint_x.set(struct, 0); - VH_BigPoint_y.set(struct, 0); - } + VH_BigPoint_x.set(struct, 0); + VH_BigPoint_y.set(struct, 0); // should be independent - try (MemorySegment struct = copy.vargAsSegment(BigPoint_LAYOUT)) { - assertEquals((long) VH_BigPoint_x.get(struct), 8); - assertEquals((long) VH_BigPoint_y.get(struct), 16); - } + struct = copy.vargAsSegment(BigPoint_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((long) VH_BigPoint_x.get(struct), 8); + assertEquals((long) VH_BigPoint_y.get(struct), 16); })}, { linkVaListCB("upcallStruct"), VaListConsumer.mh(vaList -> { - try (MemorySegment struct = vaList.vargAsSegment(Point_LAYOUT)) { - assertEquals((int) VH_Point_x.get(struct), 5); - assertEquals((int) VH_Point_y.get(struct), 10); - } + MemorySegment struct = vaList.vargAsSegment(Point_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((int) VH_Point_x.get(struct), 5); + assertEquals((int) VH_Point_y.get(struct), 10); })}, { linkVaListCB("upcallHugeStruct"), VaListConsumer.mh(vaList -> { - try (MemorySegment struct = vaList.vargAsSegment(HugePoint_LAYOUT)) { - assertEquals((long) VH_HugePoint_x.get(struct), 1); - assertEquals((long) VH_HugePoint_y.get(struct), 2); - assertEquals((long) VH_HugePoint_z.get(struct), 3); - } + MemorySegment struct = vaList.vargAsSegment(HugePoint_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((long) VH_HugePoint_x.get(struct), 1); + assertEquals((long) VH_HugePoint_y.get(struct), 2); + assertEquals((long) VH_HugePoint_z.get(struct), 3); })}, { linkVaListCB("upcallFloatStruct"), VaListConsumer.mh(vaList -> { - try (MemorySegment struct = vaList.vargAsSegment(FloatPoint_LAYOUT)) { - assertEquals((float) VH_FloatPoint_x.get(struct), 1.0f); - assertEquals((float) VH_FloatPoint_y.get(struct), 2.0f); - } + MemorySegment struct = vaList.vargAsSegment(FloatPoint_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((float) VH_FloatPoint_x.get(struct), 1.0f); + assertEquals((float) VH_FloatPoint_y.get(struct), 2.0f); })}, { linkVaListCB("upcallMemoryAddress"), VaListConsumer.mh(vaList -> { MemoryAddress intPtr = vaList.vargAsAddress(C_POINTER); - MemorySegment ms = intPtr.asSegmentRestricted(C_INT.byteSize()); + MemorySegment ms = intPtr.asSegment(C_INT.byteSize(), ResourceScope.globalScope()); int x = MemoryAccess.getInt(ms); assertEquals(x, 10); })}, @@ -759,25 +731,22 @@ public class VaListTest { assertEquals((float) vaList.vargAsDouble(C_DOUBLE), 13.0F); assertEquals(vaList.vargAsDouble(C_DOUBLE), 14.0D); - try (MemorySegment point = vaList.vargAsSegment(Point_LAYOUT)) { - assertEquals((int) VH_Point_x.get(point), 5); - assertEquals((int) VH_Point_y.get(point), 10); - } + MemorySegment point = vaList.vargAsSegment(Point_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((int) VH_Point_x.get(point), 5); + assertEquals((int) VH_Point_y.get(point), 10); VaList copy = vaList.copy(); - try (MemorySegment bigPoint = vaList.vargAsSegment(BigPoint_LAYOUT)) { - assertEquals((long) VH_BigPoint_x.get(bigPoint), 15); - assertEquals((long) VH_BigPoint_y.get(bigPoint), 20); + MemorySegment bigPoint = vaList.vargAsSegment(BigPoint_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((long) VH_BigPoint_x.get(bigPoint), 15); + assertEquals((long) VH_BigPoint_y.get(bigPoint), 20); - VH_BigPoint_x.set(bigPoint, 0); - VH_BigPoint_y.set(bigPoint, 0); - } + VH_BigPoint_x.set(bigPoint, 0); + VH_BigPoint_y.set(bigPoint, 0); // should be independent - try (MemorySegment struct = copy.vargAsSegment(BigPoint_LAYOUT)) { - assertEquals((long) VH_BigPoint_x.get(struct), 15); - assertEquals((long) VH_BigPoint_y.get(struct), 20); - } + MemorySegment struct = copy.vargAsSegment(BigPoint_LAYOUT, ResourceScope.newImplicitScope()); + assertEquals((long) VH_BigPoint_x.get(struct), 15); + assertEquals((long) VH_BigPoint_y.get(struct), 20); })}, // test skip { linkVaListCB("upcallStack"), VaListConsumer.mh(vaList -> { diff --git a/test/jdk/java/foreign/virtual/TestVirtualCalls.java b/test/jdk/java/foreign/virtual/TestVirtualCalls.java new file mode 100644 index 00000000000..dd8e667a4b1 --- /dev/null +++ b/test/jdk/java/foreign/virtual/TestVirtualCalls.java @@ -0,0 +1,73 @@ +/* + * Copyright (c) 2021, 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 + * @requires os.arch=="amd64" | os.arch=="x86_64" | os.arch=="aarch64" + * @run testng/othervm + * --enable-native-access=ALL-UNNAMED + * TestVirtualCalls + */ + +import jdk.incubator.foreign.Addressable; +import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.FunctionDescriptor; +import jdk.incubator.foreign.LibraryLookup; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodType; + +import jdk.incubator.foreign.MemoryAddress; +import org.testng.annotations.*; + +import static jdk.incubator.foreign.CLinker.*; +import static org.testng.Assert.assertEquals; + +public class TestVirtualCalls { + + static final CLinker abi = CLinker.getInstance(); + static final LibraryLookup lookup = LibraryLookup.ofLibrary("Virtual"); + + static final MethodHandle func; + static final MemoryAddress funcA; + static final MemoryAddress funcB; + static final MemoryAddress funcC; + + static { + func = abi.downcallHandle( + MethodType.methodType(int.class), + FunctionDescriptor.of(C_INT)); + + funcA = lookup.lookup("funcA").orElseThrow(); + funcB = lookup.lookup("funcB").orElseThrow(); + funcC = lookup.lookup("funcC").orElseThrow(); + } + + @Test + public void testVirtualCalls() throws Throwable { + assertEquals((int) func.invokeExact((Addressable) funcA), 1); + assertEquals((int) func.invokeExact((Addressable) funcB), 2); + assertEquals((int) func.invokeExact((Addressable) funcC), 3); + } + +} diff --git a/test/jdk/java/foreign/virtual/libVirtual.c b/test/jdk/java/foreign/virtual/libVirtual.c new file mode 100644 index 00000000000..86c4c737a94 --- /dev/null +++ b/test/jdk/java/foreign/virtual/libVirtual.c @@ -0,0 +1,41 @@ +/* + * Copyright (c) 2021, 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. + * + */ + +#ifdef _WIN64 +#define EXPORT __declspec(dllexport) +#else +#define EXPORT +#endif + +EXPORT int funcA() { + return 1; +} + +EXPORT int funcB() { + return 2; +} + +EXPORT int funcC() { + return 3; +} diff --git a/test/jdk/java/lang/invoke/VarHandles/VarHandleTestExact.java b/test/jdk/java/lang/invoke/VarHandles/VarHandleTestExact.java index 9d22f8dde86..6c14c885a24 100644 --- a/test/jdk/java/lang/invoke/VarHandles/VarHandleTestExact.java +++ b/test/jdk/java/lang/invoke/VarHandles/VarHandleTestExact.java @@ -48,6 +48,7 @@ import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.internal.access.foreign.MemorySegmentProxy; import org.testng.SkipException; import org.testng.annotations.DataProvider; @@ -170,8 +171,8 @@ public class VarHandleTestExact { @Test(dataProvider = "dataSetMemorySegment") public void testExactSegmentSet(Class carrier, Object testValue, SetSegmentX setter) { VarHandle vh = MemoryHandles.varHandle(carrier, ByteOrder.nativeOrder()); - try (MemorySegment seg = MemorySegment.allocateNative(8)) { - + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment seg = MemorySegment.allocateNative(8, scope); doTest(vh, tvh -> tvh.set(seg, 0L, testValue), tvh -> setter.set(tvh, seg, 0L, testValue), diff --git a/test/jdk/java/util/stream/test/org/openjdk/tests/java/util/stream/SegmentTestDataProvider.java b/test/jdk/java/util/stream/test/org/openjdk/tests/java/util/stream/SegmentTestDataProvider.java index 8c7d8c48921..772e112b23d 100644 --- a/test/jdk/java/util/stream/test/org/openjdk/tests/java/util/stream/SegmentTestDataProvider.java +++ b/test/jdk/java/util/stream/test/org/openjdk/tests/java/util/stream/SegmentTestDataProvider.java @@ -132,13 +132,13 @@ public class SegmentTestDataProvider { } static Object[][] spliteratorTestData = { - { "bytes", MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_BYTE), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsByte }, - { "chars", MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_CHAR), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsChar }, - { "shorts", MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_SHORT), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsShort }, - { "ints", MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_INT), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsInt }, - { "longs", MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_LONG), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsLong }, - { "floats", MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_FLOAT), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsFloat }, - { "doubles", MemoryLayout.ofSequence(1024, MemoryLayouts.JAVA_DOUBLE), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsDouble }, + { "bytes", MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_BYTE), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsByte }, + { "chars", MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_CHAR), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsChar }, + { "shorts", MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_SHORT), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsShort }, + { "ints", MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_INT), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsInt }, + { "longs", MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_LONG), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsLong }, + { "floats", MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_FLOAT), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsFloat }, + { "doubles", MemoryLayout.sequenceLayout(1024, MemoryLayouts.JAVA_DOUBLE), (SpliteratorTestHelper.ContentAsserter)SegmentTestDataProvider::compareSegmentsDouble }, }; // returns an array of (String name, Supplier>, ContentAsserter) diff --git a/test/jdk/java/util/stream/test/org/openjdk/tests/java/util/stream/SpliteratorTest.java b/test/jdk/java/util/stream/test/org/openjdk/tests/java/util/stream/SpliteratorTest.java index 1f9f52ff8e0..37e8aeb03fc 100644 --- a/test/jdk/java/util/stream/test/org/openjdk/tests/java/util/stream/SpliteratorTest.java +++ b/test/jdk/java/util/stream/test/org/openjdk/tests/java/util/stream/SpliteratorTest.java @@ -23,6 +23,7 @@ package org.openjdk.tests.java.util.stream; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; import org.testng.annotations.Test; @@ -64,7 +65,8 @@ public class SpliteratorTest { @Test(dataProvider = "SegmentSpliterator", dataProviderClass = SegmentTestDataProvider.class ) public void testSegmentSpliterator(String name, SequenceLayout layout, SpliteratorTestHelper.ContentAsserter contentAsserter) { - try (MemorySegment segment = MemorySegment.allocateNative(layout)) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(layout, scope); SegmentTestDataProvider.initSegment(segment); SpliteratorTestHelper.testSpliterator(() -> segment.spliterator(layout), SegmentTestDataProvider::segmentCopier, contentAsserter); diff --git a/test/jdk/jdk/nio/zipfs/LargeEntriesTest.java b/test/jdk/jdk/nio/zipfs/LargeEntriesTest.java index 07a403a7359..3b05f3339d7 100644 --- a/test/jdk/jdk/nio/zipfs/LargeEntriesTest.java +++ b/test/jdk/jdk/nio/zipfs/LargeEntriesTest.java @@ -30,6 +30,8 @@ import java.nio.file.FileSystem; import java.nio.file.*; import java.security.SecureRandom; import java.util.Arrays; +import java.util.ArrayList; +import java.util.List; import java.util.Map; import java.util.function.Consumer; import java.util.zip.ZipEntry; diff --git a/test/jdk/tools/jar/modularJar/Basic.java b/test/jdk/tools/jar/modularJar/Basic.java index 4ae99842461..f6f95b63207 100644 --- a/test/jdk/tools/jar/modularJar/Basic.java +++ b/test/jdk/tools/jar/modularJar/Basic.java @@ -170,6 +170,8 @@ public class Basic { conceals = stringToSet(line); } else if (line.contains("VM warning:")) { continue; // ignore server vm warning see#8196748 + } else if (line.contains("WARNING: JNI access from module not specified in --enable-native-access:")) { + continue; } else { throw new AssertionError("Unknown value " + line); } diff --git a/test/lib/sun/hotspot/WhiteBox.java b/test/lib/sun/hotspot/WhiteBox.java index 32c8f8afc50..e7f75e36a94 100644 --- a/test/lib/sun/hotspot/WhiteBox.java +++ b/test/lib/sun/hotspot/WhiteBox.java @@ -634,7 +634,7 @@ public class WhiteBox { public native String getLibcName(); // Walk stack frames of current thread - public native void verifyFrames(boolean log); + public native void verifyFrames(boolean log, boolean updateRegisterMap); public native boolean isJVMTIIncluded(); diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/BulkMismatchAcquire.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/BulkMismatchAcquire.java new file mode 100644 index 00000000000..5d962ee3979 --- /dev/null +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/BulkMismatchAcquire.java @@ -0,0 +1,173 @@ +/* + * Copyright (c) 2021, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ + +package org.openjdk.bench.jdk.incubator.foreign; + +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; +import org.openjdk.jmh.annotations.Benchmark; +import org.openjdk.jmh.annotations.BenchmarkMode; +import org.openjdk.jmh.annotations.Fork; +import org.openjdk.jmh.annotations.Measurement; +import org.openjdk.jmh.annotations.Mode; +import org.openjdk.jmh.annotations.Param; +import org.openjdk.jmh.annotations.OutputTimeUnit; +import org.openjdk.jmh.annotations.Setup; +import org.openjdk.jmh.annotations.State; +import org.openjdk.jmh.annotations.TearDown; +import org.openjdk.jmh.annotations.Warmup; +import sun.misc.Unsafe; + +import jdk.incubator.foreign.MemorySegment; +import java.nio.ByteBuffer; +import java.util.concurrent.TimeUnit; +import java.util.function.Supplier; + +import static jdk.incubator.foreign.MemoryLayouts.JAVA_INT; + +@BenchmarkMode(Mode.AverageTime) +@Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@State(org.openjdk.jmh.annotations.Scope.Thread) +@OutputTimeUnit(TimeUnit.MILLISECONDS) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign" }) +public class BulkMismatchAcquire { + + public enum ScopeKind { + CONFINED(ResourceScope::newConfinedScope), + SHARED(ResourceScope::newSharedScope), + IMPLICIT(ResourceScope::newImplicitScope); + + final Supplier scopeFactory; + + ScopeKind(Supplier scopeFactory) { + this.scopeFactory = scopeFactory; + } + + ResourceScope makeScope() { + return scopeFactory.get(); + } + } + + @Param({"CONFINED", "SHARED", "IMPLICIT"}) + public ScopeKind scopeKind; + + // large(ish) segments/buffers with same content, 0, for mismatch, non-multiple-of-8 sized + static final int SIZE_WITH_TAIL = (1024 * 1024) + 7; + + ResourceScope scope; + MemorySegment mismatchSegmentLarge1; + MemorySegment mismatchSegmentLarge2; + ByteBuffer mismatchBufferLarge1; + ByteBuffer mismatchBufferLarge2; + MemorySegment mismatchSegmentSmall1; + MemorySegment mismatchSegmentSmall2; + ByteBuffer mismatchBufferSmall1; + ByteBuffer mismatchBufferSmall2; + + @Setup + public void setup() { + scope = scopeKind.makeScope(); + mismatchSegmentLarge1 = MemorySegment.allocateNative(SIZE_WITH_TAIL, scope); + mismatchSegmentLarge2 = MemorySegment.allocateNative(SIZE_WITH_TAIL, scope); + mismatchBufferLarge1 = ByteBuffer.allocateDirect(SIZE_WITH_TAIL); + mismatchBufferLarge2 = ByteBuffer.allocateDirect(SIZE_WITH_TAIL); + + // mismatch at first byte + mismatchSegmentSmall1 = MemorySegment.allocateNative(7, scope); + mismatchSegmentSmall2 = MemorySegment.allocateNative(7, scope); + mismatchBufferSmall1 = ByteBuffer.allocateDirect(7); + mismatchBufferSmall2 = ByteBuffer.allocateDirect(7); + { + mismatchSegmentSmall1.fill((byte) 0xFF); + mismatchBufferSmall1.put((byte) 0xFF).clear(); + // verify expected mismatch indices + long si = mismatchSegmentLarge1.mismatch(mismatchSegmentLarge2); + if (si != -1) + throw new AssertionError("Unexpected mismatch index:" + si); + int bi = mismatchBufferLarge1.mismatch(mismatchBufferLarge2); + if (bi != -1) + throw new AssertionError("Unexpected mismatch index:" + bi); + si = mismatchSegmentSmall1.mismatch(mismatchSegmentSmall2); + if (si != 0) + throw new AssertionError("Unexpected mismatch index:" + si); + bi = mismatchBufferSmall1.mismatch(mismatchBufferSmall2); + if (bi != 0) + throw new AssertionError("Unexpected mismatch index:" + bi); + } + } + + @TearDown + public void tearDown() { + if (!scope.isImplicit()) + scope.close(); + } + + @Benchmark + @OutputTimeUnit(TimeUnit.NANOSECONDS) + public long mismatch_large_segment() { + return mismatchSegmentLarge1.mismatch(mismatchSegmentLarge2); + } + + @Benchmark + @OutputTimeUnit(TimeUnit.NANOSECONDS) + public long mismatch_large_segment_acquire() { + var handle = mismatchSegmentLarge1.scope().acquire(); + try { + return mismatchSegmentLarge1.mismatch(mismatchSegmentLarge2); + } finally { + mismatchSegmentLarge1.scope().release(handle); + } + } + + @Benchmark + @OutputTimeUnit(TimeUnit.NANOSECONDS) + public int mismatch_large_bytebuffer() { + return mismatchBufferLarge1.mismatch(mismatchBufferLarge2); + } + + @Benchmark + @OutputTimeUnit(TimeUnit.NANOSECONDS) + public long mismatch_small_segment() { + return mismatchSegmentSmall1.mismatch(mismatchSegmentSmall2); + } + + @Benchmark + @OutputTimeUnit(TimeUnit.NANOSECONDS) + public long mismatch_small_segment_acquire() { + var handle = mismatchSegmentLarge1.scope().acquire(); + try { + return mismatchSegmentSmall1.mismatch(mismatchSegmentSmall2); + } finally { + mismatchSegmentLarge1.scope().release(handle); + } + } + + @Benchmark + @OutputTimeUnit(TimeUnit.NANOSECONDS) + public int mismatch_small_bytebuffer() { + return mismatchBufferSmall1.mismatch(mismatchBufferSmall2); + } +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/BulkOps.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/BulkOps.java index 65a4257fa61..ac92e9429ca 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/BulkOps.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/BulkOps.java @@ -24,6 +24,7 @@ package org.openjdk.bench.jdk.incubator.foreign; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -55,7 +56,7 @@ public class BulkOps { static final int ALLOC_SIZE = ELEM_SIZE * CARRIER_SIZE; static final long unsafe_addr = unsafe.allocateMemory(ALLOC_SIZE); - static final MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE); + static final MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE, ResourceScope.newConfinedScope()); static final int[] bytes = new int[ELEM_SIZE]; static final MemorySegment bytesSegment = MemorySegment.ofArray(bytes); @@ -63,14 +64,14 @@ public class BulkOps { // large(ish) segments/buffers with same content, 0, for mismatch, non-multiple-of-8 sized static final int SIZE_WITH_TAIL = (1024 * 1024) + 7; - static final MemorySegment mismatchSegmentLarge1 = MemorySegment.allocateNative(SIZE_WITH_TAIL); - static final MemorySegment mismatchSegmentLarge2 = MemorySegment.allocateNative(SIZE_WITH_TAIL); + static final MemorySegment mismatchSegmentLarge1 = MemorySegment.allocateNative(SIZE_WITH_TAIL, ResourceScope.newConfinedScope()); + static final MemorySegment mismatchSegmentLarge2 = MemorySegment.allocateNative(SIZE_WITH_TAIL, ResourceScope.newConfinedScope()); static final ByteBuffer mismatchBufferLarge1 = ByteBuffer.allocateDirect(SIZE_WITH_TAIL); static final ByteBuffer mismatchBufferLarge2 = ByteBuffer.allocateDirect(SIZE_WITH_TAIL); // mismatch at first byte - static final MemorySegment mismatchSegmentSmall1 = MemorySegment.allocateNative(7); - static final MemorySegment mismatchSegmentSmall2 = MemorySegment.allocateNative(7); + static final MemorySegment mismatchSegmentSmall1 = MemorySegment.allocateNative(7, ResourceScope.newConfinedScope()); + static final MemorySegment mismatchSegmentSmall2 = MemorySegment.allocateNative(7, ResourceScope.newConfinedScope()); static final ByteBuffer mismatchBufferSmall1 = ByteBuffer.allocateDirect(7); static final ByteBuffer mismatchBufferSmall2 = ByteBuffer.allocateDirect(7); static { diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverhead.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverhead.java deleted file mode 100644 index 5f58e529732..00000000000 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverhead.java +++ /dev/null @@ -1,161 +0,0 @@ -/* - * 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. - */ -package org.openjdk.bench.jdk.incubator.foreign; - -import jdk.incubator.foreign.FunctionDescriptor; -import jdk.incubator.foreign.LibraryLookup; -import jdk.incubator.foreign.MemoryAddress; -import jdk.incubator.foreign.MemoryLayout; -import jdk.incubator.foreign.MemorySegment; -import jdk.incubator.foreign.CLinker; -import org.openjdk.jmh.annotations.Benchmark; -import org.openjdk.jmh.annotations.BenchmarkMode; -import org.openjdk.jmh.annotations.Fork; -import org.openjdk.jmh.annotations.Measurement; -import org.openjdk.jmh.annotations.Mode; -import org.openjdk.jmh.annotations.OutputTimeUnit; -import org.openjdk.jmh.annotations.State; -import org.openjdk.jmh.annotations.Warmup; - -import java.lang.invoke.MethodHandle; -import java.lang.invoke.MethodType; -import java.util.concurrent.TimeUnit; - -import static jdk.incubator.foreign.CLinker.C_DOUBLE; -import static jdk.incubator.foreign.CLinker.C_INT; -import static jdk.incubator.foreign.CLinker.C_LONG_LONG; -import static jdk.incubator.foreign.CLinker.C_POINTER; - -@BenchmarkMode(Mode.AverageTime) -@Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS) -@Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) -@State(org.openjdk.jmh.annotations.Scope.Thread) -@OutputTimeUnit(TimeUnit.NANOSECONDS) -@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "-Dforeign.restricted=permit" }) -public class CallOverhead { - - static final CLinker abi = CLinker.getInstance(); - - static final MethodHandle func; - static final MethodHandle identity; - static final MethodHandle identity_struct; - static final MethodHandle identity_memory_address; - static final MethodHandle args5; - static final MethodHandle args10; - static final MethodHandle func_trivial; - static final MethodHandle identity_trivial; - - static final MemoryLayout POINT_LAYOUT = MemoryLayout.ofStruct( - C_LONG_LONG, C_LONG_LONG - ); - - static final MemorySegment point = MemorySegment.allocateNative(POINT_LAYOUT); - - static { - System.loadLibrary("CallOverheadJNI"); - - LibraryLookup ll = LibraryLookup.ofLibrary("CallOverhead"); - { - LibraryLookup.Symbol addr = ll.lookup("func").get(); - MethodType mt = MethodType.methodType(void.class); - FunctionDescriptor fd = FunctionDescriptor.ofVoid(); - func = abi.downcallHandle(addr, mt, fd); - func_trivial = abi.downcallHandle(addr, mt, fd.withAttribute(FunctionDescriptor.TRIVIAL_ATTRIBUTE_NAME, true)); - } - { - LibraryLookup.Symbol addr = ll.lookup("identity").get(); - MethodType mt = MethodType.methodType(int.class, int.class); - FunctionDescriptor fd = FunctionDescriptor.of(C_INT, C_INT); - identity = abi.downcallHandle(addr, mt, fd); - identity_trivial = abi.downcallHandle(addr, mt, fd.withAttribute(FunctionDescriptor.TRIVIAL_ATTRIBUTE_NAME, true)); - } - identity_struct = abi.downcallHandle(ll.lookup("identity_struct").get(), - MethodType.methodType(MemorySegment.class, MemorySegment.class), - FunctionDescriptor.of(POINT_LAYOUT, POINT_LAYOUT)); - identity_memory_address = abi.downcallHandle(ll.lookup("identity_memory_address").get(), - MethodType.methodType(MemoryAddress.class, MemoryAddress.class), - FunctionDescriptor.of(C_POINTER, C_POINTER)); - args5 = abi.downcallHandle(ll.lookup("args5").get(), - MethodType.methodType(void.class, long.class, double.class, long.class, double.class, long.class), - FunctionDescriptor.ofVoid(C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG)); - args10 = abi.downcallHandle(ll.lookup("args10").get(), - MethodType.methodType(void.class, long.class, double.class, long.class, double.class, long.class, - double.class, long.class, double.class, long.class, double.class), - FunctionDescriptor.ofVoid(C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG, - C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE)); - } - - static native void blank(); - static native int identity(int x); - - @Benchmark - public void jni_blank() throws Throwable { - blank(); - } - - @Benchmark - public void panama_blank() throws Throwable { - func.invokeExact(); - } - - @Benchmark - public void panama_blank_trivial() throws Throwable { - func_trivial.invokeExact(); - } - - @Benchmark - public int jni_identity() throws Throwable { - return identity(10); - } - - @Benchmark - public int panama_identity() throws Throwable { - return (int) identity.invokeExact(10); - } - - @Benchmark - public int panama_identity_trivial() throws Throwable { - return (int) identity_trivial.invokeExact(10); - } - - @Benchmark - public MemorySegment panama_identity_struct() throws Throwable { - return (MemorySegment) identity_struct.invokeExact(point); - } - - @Benchmark - public MemoryAddress panama_identity_memory_address() throws Throwable { - return (MemoryAddress) identity_memory_address.invokeExact(MemoryAddress.NULL); - } - - @Benchmark - public void panama_args5() throws Throwable { - args5.invokeExact(10L, 11D, 12L, 13D, 14L); - } - - @Benchmark - public void panama_args10() throws Throwable { - args10.invokeExact(10L, 11D, 12L, 13D, 14L, - 15D, 16L, 17D, 18L, 19D); - } -} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadConstant.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadConstant.java new file mode 100644 index 00000000000..bd627646770 --- /dev/null +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadConstant.java @@ -0,0 +1,118 @@ +/* + * Copyright (c) 2021, 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. + */ +package org.openjdk.bench.jdk.incubator.foreign; + +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemorySegment; +import org.openjdk.jmh.annotations.Benchmark; +import org.openjdk.jmh.annotations.BenchmarkMode; +import org.openjdk.jmh.annotations.Fork; +import org.openjdk.jmh.annotations.Measurement; +import org.openjdk.jmh.annotations.Mode; +import org.openjdk.jmh.annotations.OutputTimeUnit; +import org.openjdk.jmh.annotations.State; +import org.openjdk.jmh.annotations.Warmup; + +import java.util.concurrent.TimeUnit; + +import static org.openjdk.bench.jdk.incubator.foreign.CallOverheadHelper.*; + +@BenchmarkMode(Mode.AverageTime) +@Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@State(org.openjdk.jmh.annotations.Scope.Thread) +@OutputTimeUnit(TimeUnit.NANOSECONDS) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) +public class CallOverheadConstant { + + @Benchmark + public void jni_blank() throws Throwable { + blank(); + } + + @Benchmark + public void panama_blank() throws Throwable { + func.invokeExact(); + } + + @Benchmark + public void panama_blank_trivial() throws Throwable { + func_trivial.invokeExact(); + } + + @Benchmark + public int jni_identity() throws Throwable { + return identity(10); + } + + @Benchmark + public int panama_identity() throws Throwable { + return (int) identity.invokeExact(10); + } + + @Benchmark + public int panama_identity_trivial() throws Throwable { + return (int) identity_trivial.invokeExact(10); + } + + @Benchmark + public MemorySegment panama_identity_struct() throws Throwable { + return (MemorySegment) identity_struct.invokeExact(recycling_allocator, point); + } + + @Benchmark + public MemoryAddress panama_identity_memory_address() throws Throwable { + return (MemoryAddress) identity_memory_address.invokeExact(MemoryAddress.NULL); + } + + @Benchmark + public void panama_args_01() throws Throwable { + args1.invokeExact(10L); + } + + @Benchmark + public void panama_args_02() throws Throwable { + args2.invokeExact(10L, 11D); + } + + @Benchmark + public void panama_args_03() throws Throwable { + args3.invokeExact(10L, 11D, 12L); + } + + @Benchmark + public void panama_args_04() throws Throwable { + args4.invokeExact(10L, 11D, 12L, 13D); + } + + @Benchmark + public void panama_args_05() throws Throwable { + args5.invokeExact(10L, 11D, 12L, 13D, 14L); + } + + @Benchmark + public void panama_args_10() throws Throwable { + args10.invokeExact(10L, 11D, 12L, 13D, 14L, + 15D, 16L, 17D, 18L, 19D); + } +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadHelper.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadHelper.java new file mode 100644 index 00000000000..453ab59e2aa --- /dev/null +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadHelper.java @@ -0,0 +1,166 @@ +/* + * Copyright (c) 2021, 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. + */ +package org.openjdk.bench.jdk.incubator.foreign; + +import jdk.incubator.foreign.Addressable; +import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.FunctionDescriptor; +import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodType; + +import static java.lang.invoke.MethodHandles.insertArguments; +import static jdk.incubator.foreign.CLinker.C_DOUBLE; +import static jdk.incubator.foreign.CLinker.C_INT; +import static jdk.incubator.foreign.CLinker.C_LONG_LONG; +import static jdk.incubator.foreign.CLinker.C_POINTER; + +public class CallOverheadHelper { + + static final CLinker abi = CLinker.getInstance(); + + static final MethodHandle func; + static final MethodHandle func_v; + static Addressable func_addr; + static final MethodHandle identity; + static final MethodHandle identity_v; + static Addressable identity_addr; + static final MethodHandle identity_struct; + static final MethodHandle identity_struct_v; + static Addressable identity_struct_addr; + static final MethodHandle identity_memory_address; + static final MethodHandle identity_memory_address_v; + static Addressable identity_memory_address_addr; + static final MethodHandle args1; + static final MethodHandle args1_v; + static Addressable args1_addr; + static final MethodHandle args2; + static final MethodHandle args2_v; + static Addressable args2_addr; + static final MethodHandle args3; + static final MethodHandle args3_v; + static Addressable args3_addr; + static final MethodHandle args4; + static final MethodHandle args4_v; + static Addressable args4_addr; + static final MethodHandle args5; + static final MethodHandle args5_v; + static Addressable args5_addr; + static final MethodHandle args10; + static final MethodHandle args10_v; + static Addressable args10_addr; + static final MethodHandle func_trivial; + static final MethodHandle func_trivial_v; + static final MethodHandle identity_trivial; + static final MethodHandle identity_trivial_v; + + static final MemoryLayout POINT_LAYOUT = MemoryLayout.structLayout( + C_LONG_LONG, C_LONG_LONG + ); + + static final MemorySegment point = MemorySegment.allocateNative(POINT_LAYOUT, ResourceScope.newImplicitScope()); + + static final SegmentAllocator recycling_allocator = SegmentAllocator.ofSegment(MemorySegment.allocateNative(POINT_LAYOUT, ResourceScope.newImplicitScope())); + + static { + System.loadLibrary("CallOverheadJNI"); + + LibraryLookup ll = LibraryLookup.ofLibrary("CallOverhead"); + { + func_addr = ll.lookup("func").orElseThrow(); + MethodType mt = MethodType.methodType(void.class); + FunctionDescriptor fd = FunctionDescriptor.ofVoid(); + func_v = abi.downcallHandle(mt, fd); + func = insertArguments(func_v, 0, func_addr); + func_trivial_v = abi.downcallHandle(mt, fd.withAttribute(FunctionDescriptor.TRIVIAL_ATTRIBUTE_NAME, true)); + func_trivial = insertArguments(func_trivial_v, 0, func_addr); + } + { + identity_addr = ll.lookup("identity").orElseThrow(); + MethodType mt = MethodType.methodType(int.class, int.class); + FunctionDescriptor fd = FunctionDescriptor.of(C_INT, C_INT); + identity_v = abi.downcallHandle(mt, fd); + identity = insertArguments(identity_v, 0, identity_addr); + identity_trivial_v = abi.downcallHandle(mt, fd.withAttribute(FunctionDescriptor.TRIVIAL_ATTRIBUTE_NAME, true)); + identity_trivial = insertArguments(identity_trivial_v, 0, identity_addr); + } + identity_struct_addr = ll.lookup("identity_struct").orElseThrow(); + identity_struct_v = abi.downcallHandle( + MethodType.methodType(MemorySegment.class, MemorySegment.class), + FunctionDescriptor.of(POINT_LAYOUT, POINT_LAYOUT)); + identity_struct = insertArguments(identity_struct_v, 0, identity_struct_addr); + + identity_memory_address_addr = ll.lookup("identity_memory_address").orElseThrow(); + identity_memory_address_v = abi.downcallHandle( + MethodType.methodType(MemoryAddress.class, MemoryAddress.class), + FunctionDescriptor.of(C_POINTER, C_POINTER)); + identity_memory_address = insertArguments(identity_memory_address_v, 0, identity_memory_address_addr); + + args1_addr = ll.lookup("args1").orElseThrow(); + args1_v = abi.downcallHandle( + MethodType.methodType(void.class, long.class), + FunctionDescriptor.ofVoid(C_LONG_LONG)); + args1 = insertArguments(args1_v, 0, args1_addr); + + args2_addr = ll.lookup("args2").orElseThrow(); + args2_v = abi.downcallHandle( + MethodType.methodType(void.class, long.class, double.class), + FunctionDescriptor.ofVoid(C_LONG_LONG, C_DOUBLE)); + args2 = insertArguments(args2_v, 0, args2_addr); + + args3_addr = ll.lookup("args3").orElseThrow(); + args3_v = abi.downcallHandle( + MethodType.methodType(void.class, long.class, double.class, long.class), + FunctionDescriptor.ofVoid(C_LONG_LONG, C_DOUBLE, C_LONG_LONG)); + args3 = insertArguments(args3_v, 0, args3_addr); + + args4_addr = ll.lookup("args4").orElseThrow(); + args4_v = abi.downcallHandle( + MethodType.methodType(void.class, long.class, double.class, long.class, double.class), + FunctionDescriptor.ofVoid(C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE)); + args4 = insertArguments(args4_v, 0, args4_addr); + + args5_addr = ll.lookup("args5").orElseThrow(); + args5_v = abi.downcallHandle( + MethodType.methodType(void.class, long.class, double.class, long.class, double.class, long.class), + FunctionDescriptor.ofVoid(C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG)); + args5 = insertArguments(args5_v, 0, args5_addr); + + args10_addr = ll.lookup("args10").orElseThrow(); + args10_v = abi.downcallHandle( + MethodType.methodType(void.class, long.class, double.class, long.class, double.class, long.class, + double.class, long.class, double.class, long.class, double.class), + FunctionDescriptor.ofVoid(C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG, + C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE)); + args10 = insertArguments(args10_v, 0, args10_addr); + } + + static native void blank(); + static native int identity(int x); +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadVirtual.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadVirtual.java new file mode 100644 index 00000000000..e1e557bad23 --- /dev/null +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/CallOverheadVirtual.java @@ -0,0 +1,119 @@ +/* + * Copyright (c) 2021, 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. + */ +package org.openjdk.bench.jdk.incubator.foreign; + +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemorySegment; +import org.openjdk.jmh.annotations.Benchmark; +import org.openjdk.jmh.annotations.BenchmarkMode; +import org.openjdk.jmh.annotations.Fork; +import org.openjdk.jmh.annotations.Measurement; +import org.openjdk.jmh.annotations.Mode; +import org.openjdk.jmh.annotations.OutputTimeUnit; +import org.openjdk.jmh.annotations.State; +import org.openjdk.jmh.annotations.Warmup; + +import java.util.concurrent.TimeUnit; + +import static org.openjdk.bench.jdk.incubator.foreign.CallOverheadHelper.*; + +@BenchmarkMode(Mode.AverageTime) +@Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@State(org.openjdk.jmh.annotations.Scope.Thread) +@OutputTimeUnit(TimeUnit.NANOSECONDS) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) +public class CallOverheadVirtual { + + @Benchmark + public void jni_blank() throws Throwable { + blank(); + } + + @Benchmark + public void panama_blank() throws Throwable { + func_v.invokeExact(func_addr); + } + + @Benchmark + public void panama_blank_trivial() throws Throwable { + func_trivial_v.invokeExact(func_addr); + } + + @Benchmark + public int jni_identity() throws Throwable { + return identity(10); + } + + @Benchmark + public int panama_identity() throws Throwable { + return (int) identity_v.invokeExact(identity_addr, 10); + } + + @Benchmark + public int panama_identity_trivial() throws Throwable { + return (int) identity_trivial_v.invokeExact(identity_addr, 10); + } + + @Benchmark + public MemorySegment panama_identity_struct() throws Throwable { + return (MemorySegment) identity_struct_v.invokeExact(identity_struct_addr, recycling_allocator, point); + } + + @Benchmark + public MemoryAddress panama_identity_memory_address() throws Throwable { + return (MemoryAddress) identity_memory_address_v.invokeExact(identity_memory_address_addr, MemoryAddress.NULL); + } + + @Benchmark + public void panama_args_01() throws Throwable { + args1_v.invokeExact(args1_addr, 10L); + } + + @Benchmark + public void panama_args_02() throws Throwable { + args2_v.invokeExact(args2_addr, 10L, 11D); + } + + @Benchmark + public void panama_args_03() throws Throwable { + args3_v.invokeExact(args3_addr, 10L, 11D, 12L); + } + + @Benchmark + public void panama_args_04() throws Throwable { + args4_v.invokeExact(args4_addr, 10L, 11D, 12L, 13D); + } + + @Benchmark + public void panama_args_05() throws Throwable { + args5_v.invokeExact(args5_addr, 10L, 11D, 12L, 13D, 14L); + } + + @Benchmark + public void panama_args_10() throws Throwable { + args10_v.invokeExact(args10_addr, + 10L, 11D, 12L, 13D, 14L, + 15D, 16L, 17D, 18L, 19D); + } +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverConstant.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverConstant.java index 48be5907796..64cc3697687 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverConstant.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverConstant.java @@ -23,6 +23,7 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryLayout; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.CompilerControl; @@ -36,7 +37,6 @@ import org.openjdk.jmh.annotations.TearDown; import org.openjdk.jmh.annotations.Warmup; import sun.misc.Unsafe; -import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemorySegment; import java.lang.invoke.VarHandle; import java.nio.ByteBuffer; @@ -72,8 +72,8 @@ public class LoopOverConstant { //setup native memory segment - static final MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE); - static final VarHandle VH_int = MemoryLayout.ofSequence(JAVA_INT).varHandle(int.class, sequenceElement()); + static final MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE, ResourceScope.newImplicitScope()); + static final VarHandle VH_int = MemoryLayout.sequenceLayout(JAVA_INT).varHandle(int.class, sequenceElement()); static { for (int i = 0; i < ELEM_SIZE; i++) { diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNew.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNew.java index c2a32beb13e..0d59d83488f 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNew.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNew.java @@ -24,6 +24,8 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.SegmentAllocator; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -57,8 +59,17 @@ public class LoopOverNew { static final int ELEM_SIZE = 1_000_000; static final int CARRIER_SIZE = (int)JAVA_INT.byteSize(); static final int ALLOC_SIZE = ELEM_SIZE * CARRIER_SIZE; + static final MemoryLayout ALLOC_LAYOUT = MemoryLayout.sequenceLayout(ELEM_SIZE, JAVA_INT); - static final VarHandle VH_int = MemoryLayout.ofSequence(JAVA_INT).varHandle(int.class, sequenceElement()); + static final VarHandle VH_int = MemoryLayout.sequenceLayout(JAVA_INT).varHandle(int.class, sequenceElement()); + + final ResourceScope scope = ResourceScope.newConfinedScope(); + final SegmentAllocator recyclingAlloc = SegmentAllocator.ofSegment(MemorySegment.allocateNative(ALLOC_LAYOUT, scope)); + + @TearDown + public void tearDown() throws Throwable { + scope.close(); + } @Benchmark public void unsafe_loop() { @@ -70,21 +81,31 @@ public class LoopOverNew { } @Benchmark - public void segment_loop() { - MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE); - for (int i = 0; i < ELEM_SIZE; i++) { - VH_int.set(segment, (long) i, i); + public void segment_loop_confined() { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE, 4, scope); + for (int i = 0; i < ELEM_SIZE; i++) { + VH_int.set(segment, (long) i, i); + } } - segment.close(); } @Benchmark public void segment_loop_shared() { - MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE).share(); + try (ResourceScope scope = ResourceScope.newSharedScope()) { + MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE, 4, scope); + for (int i = 0; i < ELEM_SIZE; i++) { + VH_int.set(segment, (long) i, i); + } + } + } + + @Benchmark + public void segment_loop_recycle() { + MemorySegment segment = recyclingAlloc.allocate(ALLOC_SIZE, 4); for (int i = 0; i < ELEM_SIZE; i++) { VH_int.set(segment, (long) i, i); } - segment.close(); } @Benchmark @@ -95,4 +116,31 @@ public class LoopOverNew { } unsafe.invokeCleaner(byteBuffer); } + + // hack to even out calls to System::gc, which allows us to compare how the implicit segment deallocation + // fares compared with ByteBuffer; if there's no call to System.gc() we end up comparing how well the two + // act under significant native memory pressure, and here the ByteBuffer API has more juice, since it features + // a complex exponential back off with multiple GC retries (see ByteBuffer::allocateDirect). Of course, we + // don't care about those cases with segments, as if clients need to allocate/free very frequently + // they should just use deterministic deallocation (with confined scope) instead, which delivers much + // better performances anyway. + static byte gcCount = 0; + + @Benchmark + public void buffer_loop_implicit() { + if (gcCount++ == 0) System.gc(); // GC when we overflow + ByteBuffer byteBuffer = ByteBuffer.allocateDirect(ALLOC_SIZE).order(ByteOrder.nativeOrder()); + for (int i = 0; i < ELEM_SIZE; i++) { + byteBuffer.putInt(i * CARRIER_SIZE , i); + } + } + + @Benchmark + public void segment_loop_implicit() { + if (gcCount++ == 0) System.gc(); // GC when we overflow + MemorySegment segment = MemorySegment.allocateNative(ALLOC_SIZE, 4, ResourceScope.newImplicitScope()); + for (int i = 0; i < ELEM_SIZE; i++) { + VH_int.set(segment, (long) i, i); + } + } } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstant.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstant.java index 690a0c5d5b7..c4af846b685 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstant.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstant.java @@ -23,9 +23,9 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryAccess; -import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -60,7 +60,7 @@ public class LoopOverNonConstant { static final int CARRIER_SIZE = (int)JAVA_INT.byteSize(); static final int ALLOC_SIZE = ELEM_SIZE * CARRIER_SIZE; - static final VarHandle VH_int = MemoryLayout.ofSequence(JAVA_INT).varHandle(int.class, sequenceElement()); + static final VarHandle VH_int = MemoryLayout.sequenceLayout(JAVA_INT).varHandle(int.class, sequenceElement()); MemorySegment segment; long unsafe_addr; @@ -72,7 +72,7 @@ public class LoopOverNonConstant { for (int i = 0; i < ELEM_SIZE; i++) { unsafe.putInt(unsafe_addr + (i * CARRIER_SIZE) , i); } - segment = MemorySegment.allocateNative(ALLOC_SIZE); + segment = MemorySegment.allocateNative(ALLOC_SIZE, ResourceScope.newConfinedScope()); for (int i = 0; i < ELEM_SIZE; i++) { VH_int.set(segment, (long) i, i); } @@ -84,7 +84,7 @@ public class LoopOverNonConstant { @TearDown public void tearDown() { - segment.close(); + segment.scope().close(); unsafe.invokeCleaner(byteBuffer); unsafe.freeMemory(unsafe_addr); } @@ -147,7 +147,7 @@ public class LoopOverNonConstant { @Benchmark public int segment_loop_readonly() { int sum = 0; - MemorySegment base = segment.withAccessModes(MemorySegment.READ); + MemorySegment base = segment.asReadOnly(); for (int i = 0; i < ELEM_SIZE; i++) { sum += (int) VH_int.get(base, (long) i); } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantFP.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantFP.java index d4325c7c5fe..17c8cd6b341 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantFP.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantFP.java @@ -23,9 +23,8 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryAccess; -import jdk.incubator.foreign.MemoryAddress; -import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -38,7 +37,6 @@ import org.openjdk.jmh.annotations.TearDown; import org.openjdk.jmh.annotations.Warmup; import sun.misc.Unsafe; -import java.lang.invoke.VarHandle; import java.nio.ByteBuffer; import java.nio.ByteOrder; import java.util.concurrent.TimeUnit; @@ -74,8 +72,8 @@ public class LoopOverNonConstantFP { for (int i = 0; i < ELEM_SIZE; i++) { unsafe.putDouble(unsafe_addrOut + (i * CARRIER_SIZE), i); } - segmentIn = MemorySegment.allocateNative(ALLOC_SIZE); - segmentOut = MemorySegment.allocateNative(ALLOC_SIZE); + segmentIn = MemorySegment.allocateNative(ALLOC_SIZE, ResourceScope.newConfinedScope()); + segmentOut = MemorySegment.allocateNative(ALLOC_SIZE, ResourceScope.newConfinedScope()); for (int i = 0; i < ELEM_SIZE; i++) { MemoryAccess.setDoubleAtIndex(segmentIn, i, i); } @@ -94,8 +92,8 @@ public class LoopOverNonConstantFP { @TearDown public void tearDown() { - segmentIn.close(); - segmentOut.close(); + segmentIn.scope().close(); + segmentOut.scope().close(); unsafe.invokeCleaner(byteBufferIn); unsafe.invokeCleaner(byteBufferOut); unsafe.freeMemory(unsafe_addrIn); diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantHeap.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantHeap.java index 343ea117bdf..2b2be067fdc 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantHeap.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantHeap.java @@ -23,9 +23,9 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryAccess; -import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -40,6 +40,7 @@ import org.openjdk.jmh.annotations.Warmup; import sun.misc.Unsafe; import java.lang.invoke.VarHandle; +import java.lang.ref.Cleaner; import java.nio.ByteBuffer; import java.nio.ByteOrder; import java.util.concurrent.TimeUnit; @@ -62,7 +63,7 @@ public class LoopOverNonConstantHeap { static final int ALLOC_SIZE = ELEM_SIZE * CARRIER_SIZE; static final int UNSAFE_BYTE_BASE = unsafe.arrayBaseOffset(byte[].class); - static final VarHandle VH_int = MemoryLayout.ofSequence(JAVA_INT).varHandle(int.class, sequenceElement()); + static final VarHandle VH_int = MemoryLayout.sequenceLayout(JAVA_INT).varHandle(int.class, sequenceElement()); MemorySegment segment; byte[] base; @@ -78,14 +79,13 @@ public class LoopOverNonConstantHeap { MemorySegment intI = MemorySegment.ofArray(new int[ALLOC_SIZE]); MemorySegment intD = MemorySegment.ofArray(new double[ALLOC_SIZE]); MemorySegment intF = MemorySegment.ofArray(new float[ALLOC_SIZE]); - try (MemorySegment s = MemorySegment.allocateNative(ALLOC_SIZE)) { - for (int i = 0; i < ALLOC_SIZE; i++) { - MemoryAccess.setByteAtOffset(intB, i, (byte)i); - MemoryAccess.setIntAtIndex(intI, i, i); - MemoryAccess.setDoubleAtIndex(intD, i, i); - MemoryAccess.setFloatAtIndex(intF, i, i); - MemoryAccess.setByteAtOffset(s, i, (byte) i); - } + MemorySegment s = MemorySegment.allocateNative(ALLOC_SIZE, 1, ResourceScope.newConfinedScope(Cleaner.create())); + for (int i = 0; i < ALLOC_SIZE; i++) { + MemoryAccess.setByteAtOffset(intB, i, (byte)i); + MemoryAccess.setIntAtIndex(intI, i, i); + MemoryAccess.setDoubleAtIndex(intD, i, i); + MemoryAccess.setFloatAtIndex(intF, i, i); + MemoryAccess.setByteAtOffset(s, i, (byte) i); } } @@ -97,11 +97,6 @@ public class LoopOverNonConstantHeap { byteBuffer = ByteBuffer.wrap(base).order(ByteOrder.nativeOrder()); } - @TearDown - public void tearDown() { - segment.close(); - } - @Benchmark @OutputTimeUnit(TimeUnit.NANOSECONDS) public int unsafe_get() { @@ -160,7 +155,7 @@ public class LoopOverNonConstantHeap { @Benchmark public int segment_loop_readonly() { int sum = 0; - MemorySegment base = segment.withAccessModes(MemorySegment.READ); + MemorySegment base = segment.asReadOnly(); for (int i = 0; i < ELEM_SIZE; i++) { sum += (int) VH_int.get(base, (long) i); } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantMapped.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantMapped.java index 19f2f30d306..11f1c1f061e 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantMapped.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantMapped.java @@ -25,6 +25,7 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryAccess; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -80,7 +81,7 @@ public class LoopOverNonConstantMapped { } } - static final VarHandle VH_int = MemoryLayout.ofSequence(JAVA_INT).varHandle(int.class, sequenceElement()); + static final VarHandle VH_int = MemoryLayout.sequenceLayout(JAVA_INT).varHandle(int.class, sequenceElement()); MemorySegment segment; long unsafe_addr; @@ -95,13 +96,13 @@ public class LoopOverNonConstantMapped { } ((MappedByteBuffer)byteBuffer).force(); } - segment = MemorySegment.mapFile(tempPath, 0L, ALLOC_SIZE, FileChannel.MapMode.READ_WRITE); + segment = MemorySegment.mapFile(tempPath, 0L, ALLOC_SIZE, FileChannel.MapMode.READ_WRITE, ResourceScope.newConfinedScope()); unsafe_addr = segment.address().toRawLongValue(); } @TearDown public void tearDown() { - segment.close(); + segment.scope().close(); unsafe.invokeCleaner(byteBuffer); } @@ -163,7 +164,7 @@ public class LoopOverNonConstantMapped { @Benchmark public int segment_loop_readonly() { int sum = 0; - MemorySegment base = segment.withAccessModes(MemorySegment.READ); + MemorySegment base = segment.asReadOnly(); for (int i = 0; i < ELEM_SIZE; i++) { sum += (int) VH_int.get(base, (long) i); } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantShared.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantShared.java index 0d629a16dc9..06c6310845a 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantShared.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverNonConstantShared.java @@ -25,6 +25,7 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryAccess; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -59,7 +60,7 @@ public class LoopOverNonConstantShared { static final int CARRIER_SIZE = (int)JAVA_INT.byteSize(); static final int ALLOC_SIZE = ELEM_SIZE * CARRIER_SIZE; - static final VarHandle VH_int = MemoryLayout.ofSequence(JAVA_INT).varHandle(int.class, sequenceElement()); + static final VarHandle VH_int = MemoryLayout.sequenceLayout(JAVA_INT).varHandle(int.class, sequenceElement()); MemorySegment segment; long unsafe_addr; @@ -71,7 +72,7 @@ public class LoopOverNonConstantShared { for (int i = 0; i < ELEM_SIZE; i++) { unsafe.putInt(unsafe_addr + (i * CARRIER_SIZE) , i); } - segment = MemorySegment.allocateNative(ALLOC_SIZE).share(); + segment = MemorySegment.allocateNative(ALLOC_SIZE, CARRIER_SIZE, ResourceScope.newSharedScope()); for (int i = 0; i < ELEM_SIZE; i++) { VH_int.set(segment, (long) i, i); } @@ -83,7 +84,7 @@ public class LoopOverNonConstantShared { @TearDown public void tearDown() { - segment.close(); + segment.scope().close(); unsafe.invokeCleaner(byteBuffer); unsafe.freeMemory(unsafe_addr); } @@ -146,7 +147,7 @@ public class LoopOverNonConstantShared { @Benchmark public int segment_loop_readonly() { int sum = 0; - MemorySegment base = segment.withAccessModes(MemorySegment.READ); + MemorySegment base = segment.asReadOnly(); for (int i = 0; i < ELEM_SIZE; i++) { sum += (int) VH_int.get(base, (long) i); } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverPollutedSegments.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverPollutedSegments.java index 463474ee004..65ae252ee32 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverPollutedSegments.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/LoopOverPollutedSegments.java @@ -25,6 +25,7 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryAccess; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -61,7 +62,7 @@ public class LoopOverPollutedSegments { byte[] arr; long addr; - static final VarHandle intHandle = MemoryLayout.ofSequence(JAVA_INT).varHandle(int.class, MemoryLayout.PathElement.sequenceElement()); + static final VarHandle intHandle = MemoryLayout.sequenceLayout(JAVA_INT).varHandle(int.class, MemoryLayout.PathElement.sequenceElement()); @Setup @@ -71,7 +72,7 @@ public class LoopOverPollutedSegments { unsafe.putInt(addr + (i * 4), i); } arr = new byte[ALLOC_SIZE]; - nativeSegment = MemorySegment.allocateNative(ALLOC_SIZE, 4); + nativeSegment = MemorySegment.allocateNative(ALLOC_SIZE, 4, ResourceScope.newConfinedScope()); heapSegmentBytes = MemorySegment.ofArray(new byte[ALLOC_SIZE]); heapSegmentFloats = MemorySegment.ofArray(new float[ELEM_SIZE]); @@ -93,7 +94,7 @@ public class LoopOverPollutedSegments { @TearDown public void tearDown() { - nativeSegment.close(); + nativeSegment.scope().close(); heapSegmentBytes = null; heapSegmentFloats = null; arr = null; diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/ParallelSum.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/ParallelSum.java index b0f6d3080cd..231c606103b 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/ParallelSum.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/ParallelSum.java @@ -25,6 +25,7 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayouts; +import jdk.incubator.foreign.ResourceScope; import jdk.incubator.foreign.SequenceLayout; import sun.misc.Unsafe; import org.openjdk.jmh.annotations.Benchmark; @@ -38,21 +39,17 @@ import org.openjdk.jmh.annotations.State; import org.openjdk.jmh.annotations.TearDown; import org.openjdk.jmh.annotations.Warmup; -import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemorySegment; import java.lang.invoke.VarHandle; import java.util.LinkedList; import java.util.List; import java.util.Optional; -import java.util.OptionalInt; import java.util.Spliterator; import java.util.concurrent.CountedCompleter; import java.util.concurrent.RecursiveTask; import java.util.concurrent.TimeUnit; -import java.util.function.IntFunction; import java.util.function.Predicate; import java.util.function.ToIntFunction; -import java.util.stream.StreamSupport; import static jdk.incubator.foreign.MemoryLayout.PathElement.sequenceElement; import static jdk.incubator.foreign.MemoryLayouts.JAVA_INT; @@ -68,11 +65,11 @@ public class ParallelSum { final static int CARRIER_SIZE = 4; final static int ALLOC_SIZE = CARRIER_SIZE * 1024 * 1024 * 256; final static int ELEM_SIZE = ALLOC_SIZE / CARRIER_SIZE; - static final VarHandle VH_int = MemoryLayout.ofSequence(JAVA_INT).varHandle(int.class, sequenceElement()); + static final VarHandle VH_int = MemoryLayout.sequenceLayout(JAVA_INT).varHandle(int.class, sequenceElement()); - final static SequenceLayout SEQUENCE_LAYOUT = MemoryLayout.ofSequence(ELEM_SIZE, MemoryLayouts.JAVA_INT); + final static MemoryLayout ELEM_LAYOUT = MemoryLayouts.JAVA_INT; final static int BULK_FACTOR = 512; - final static SequenceLayout SEQUENCE_LAYOUT_BULK = SEQUENCE_LAYOUT.reshape(-1, BULK_FACTOR); + final static SequenceLayout ELEM_LAYOUT_BULK = MemoryLayout.sequenceLayout(BULK_FACTOR, ELEM_LAYOUT); static final Unsafe unsafe = Utils.unsafe; @@ -85,7 +82,7 @@ public class ParallelSum { for (int i = 0; i < ELEM_SIZE; i++) { unsafe.putInt(address + (i * CARRIER_SIZE), i); } - segment = MemorySegment.allocateNative(ALLOC_SIZE).share(); + segment = MemorySegment.allocateNative(ALLOC_SIZE, CARRIER_SIZE, ResourceScope.newSharedScope()); for (int i = 0; i < ELEM_SIZE; i++) { VH_int.set(segment, (long) i, i); } @@ -94,7 +91,7 @@ public class ParallelSum { @TearDown public void tearDown() throws Throwable { unsafe.freeMemory(address); - segment.close(); + segment.scope().close(); } @Benchmark @@ -117,24 +114,22 @@ public class ParallelSum { @Benchmark public int segment_parallel() { - return new SumSegment(segment.spliterator(SEQUENCE_LAYOUT), SEGMENT_TO_INT).invoke(); + return new SumSegment(segment.spliterator(ELEM_LAYOUT), SEGMENT_TO_INT).invoke(); } @Benchmark public int segment_parallel_bulk() { - return new SumSegment(segment.spliterator(SEQUENCE_LAYOUT_BULK), SEGMENT_TO_INT_BULK).invoke(); + return new SumSegment(segment.spliterator(ELEM_LAYOUT_BULK), SEGMENT_TO_INT_BULK).invoke(); } @Benchmark public int segment_stream_parallel() { - return StreamSupport.stream(segment.spliterator(SEQUENCE_LAYOUT), true) - .mapToInt(SEGMENT_TO_INT).sum(); + return segment.elements(ELEM_LAYOUT).parallel().mapToInt(SEGMENT_TO_INT).sum(); } @Benchmark public int segment_stream_parallel_bulk() { - return StreamSupport.stream(segment.spliterator(SEQUENCE_LAYOUT_BULK), true) - .mapToInt(SEGMENT_TO_INT_BULK).sum(); + return segment.elements(ELEM_LAYOUT_BULK).parallel().mapToInt(SEGMENT_TO_INT_BULK).sum(); } final static ToIntFunction SEGMENT_TO_INT = slice -> @@ -150,28 +145,28 @@ public class ParallelSum { @Benchmark public Optional segment_stream_findany_serial() { - return StreamSupport.stream(segment.spliterator(SEQUENCE_LAYOUT), false) + return segment.elements(ELEM_LAYOUT) .filter(FIND_SINGLE) .findAny(); } @Benchmark public Optional segment_stream_findany_parallel() { - return StreamSupport.stream(segment.spliterator(SEQUENCE_LAYOUT), true) + return segment.elements(ELEM_LAYOUT).parallel() .filter(FIND_SINGLE) .findAny(); } @Benchmark public Optional segment_stream_findany_serial_bulk() { - return StreamSupport.stream(segment.spliterator(SEQUENCE_LAYOUT_BULK), false) + return segment.elements(ELEM_LAYOUT_BULK) .filter(FIND_BULK) .findAny(); } @Benchmark public Optional segment_stream_findany_parallel_bulk() { - return StreamSupport.stream(segment.spliterator(SEQUENCE_LAYOUT_BULK), true) + return segment.elements(ELEM_LAYOUT_BULK).parallel() .filter(FIND_BULK) .findAny(); } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/ResourceScopeClose.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/ResourceScopeClose.java new file mode 100644 index 00000000000..7fbe56e05fe --- /dev/null +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/ResourceScopeClose.java @@ -0,0 +1,132 @@ +/* + * Copyright (c) 2021, 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. + */ +package org.openjdk.bench.jdk.incubator.foreign; + +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import org.openjdk.jmh.annotations.Benchmark; +import org.openjdk.jmh.annotations.BenchmarkMode; +import org.openjdk.jmh.annotations.Fork; +import org.openjdk.jmh.annotations.Measurement; +import org.openjdk.jmh.annotations.Mode; +import org.openjdk.jmh.annotations.OutputTimeUnit; +import org.openjdk.jmh.annotations.Param; +import org.openjdk.jmh.annotations.Setup; +import org.openjdk.jmh.annotations.State; +import org.openjdk.jmh.annotations.TearDown; +import org.openjdk.jmh.annotations.Warmup; + +import java.util.ArrayList; +import java.util.List; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.TimeUnit; + +@BenchmarkMode(Mode.AverageTime) +@Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@State(org.openjdk.jmh.annotations.Scope.Thread) +@OutputTimeUnit(TimeUnit.MICROSECONDS) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign" }) +public class ResourceScopeClose { + + static final int ALLOC_SIZE = 1024; + + public enum StressMode { + NONE, + MEMORY, + THREADS + } + + @Param({"NONE", "MEMORY", "THREADS"}) + StressMode mode; + + List arrays; + volatile boolean stop = false; + List threads; + + @Setup + public void setup() throws Throwable { + if (mode == StressMode.MEMORY) { + arrays = new ArrayList<>(); + for (int i = 0; i < 100_000_000; i++) { + arrays.add(new byte[2]); + } + } else if (mode == StressMode.THREADS) { + threads = new ArrayList<>(); + for (int i = 0 ; i < 4 ; i++) { + threads.add(new Thread(() -> { + while (true) { + if (stop) break; + // busy wait + } + })); + } + threads.forEach(Thread::start); + } + + } + + @TearDown + public void tearDown() throws Throwable { + arrays = null; + if (threads != null) { + stop = true; + threads.forEach(t -> { + try { + t.join(); + } catch (InterruptedException ex) { + throw new IllegalStateException(ex); + } + }); + } + } + + @Benchmark + public MemorySegment confined_close() { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + return MemorySegment.allocateNative(ALLOC_SIZE, 4, scope); + } + } + + @Benchmark + public MemorySegment shared_close() { + try (ResourceScope scope = ResourceScope.newSharedScope()) { + return MemorySegment.allocateNative(ALLOC_SIZE, 4, scope); + } + } + + @Benchmark + public MemorySegment implicit_close() { + return MemorySegment.allocateNative(ALLOC_SIZE, 4, ResourceScope.newImplicitScope()); + } + + @Benchmark + public MemorySegment implicit_close_systemgc() { + if (gcCount++ == 0) System.gc(); // GC when we overflow + return MemorySegment.allocateNative(ALLOC_SIZE, 4, ResourceScope.newImplicitScope()); + } + + // keep + static byte gcCount = 0; +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/StrLenTest.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/StrLenTest.java new file mode 100644 index 00000000000..53503ebc80a --- /dev/null +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/StrLenTest.java @@ -0,0 +1,180 @@ +/* + * Copyright (c) 2021, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ + +package org.openjdk.bench.jdk.incubator.foreign; + +import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.FunctionDescriptor; +import jdk.incubator.foreign.LibraryLookup; +import jdk.incubator.foreign.MemoryAccess; +import jdk.incubator.foreign.MemoryAddress; +import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; +import jdk.incubator.foreign.SegmentAllocator; +import org.openjdk.jmh.annotations.Benchmark; +import org.openjdk.jmh.annotations.BenchmarkMode; +import org.openjdk.jmh.annotations.Fork; +import org.openjdk.jmh.annotations.Setup; +import org.openjdk.jmh.annotations.Param; +import org.openjdk.jmh.annotations.TearDown; +import org.openjdk.jmh.annotations.Measurement; +import org.openjdk.jmh.annotations.Mode; +import org.openjdk.jmh.annotations.OutputTimeUnit; +import org.openjdk.jmh.annotations.State; +import org.openjdk.jmh.annotations.Warmup; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodType; +import java.util.concurrent.TimeUnit; + +import static jdk.incubator.foreign.CLinker.*; + +@BenchmarkMode(Mode.AverageTime) +@Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) +@State(org.openjdk.jmh.annotations.Scope.Thread) +@OutputTimeUnit(TimeUnit.NANOSECONDS) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) +public class StrLenTest { + + ResourceScope scope = ResourceScope.newConfinedScope(); + + SegmentAllocator segmentAllocator; + SegmentAllocator arenaAllocator = SegmentAllocator.arenaAllocator(scope); + + @Param({"5", "20", "100"}) + public int size; + public String str; + + static { + System.loadLibrary("StrLen"); + } + + static final MethodHandle STRLEN; + static final MethodHandle STRLEN_TRIVIAL; + static final MethodHandle MALLOC_TRIVIAL; + static final MethodHandle FREE_TRIVIAL; + + static { + LibraryLookup lookup = LibraryLookup.ofDefault(); + CLinker abi = CLinker.getInstance(); + STRLEN = abi.downcallHandle(lookup.lookup("strlen").get(), + MethodType.methodType(int.class, MemoryAddress.class), + FunctionDescriptor.of(C_INT, C_POINTER)); + STRLEN_TRIVIAL = abi.downcallHandle(lookup.lookup("strlen").get(), + MethodType.methodType(int.class, MemoryAddress.class), + FunctionDescriptor.of(C_INT, C_POINTER).withAttribute(FunctionDescriptor.TRIVIAL_ATTRIBUTE_NAME, true)); + MALLOC_TRIVIAL = abi.downcallHandle(lookup.lookup("malloc").get(), + MethodType.methodType(MemoryAddress.class, long.class), + FunctionDescriptor.of(C_POINTER, C_LONG_LONG).withAttribute(FunctionDescriptor.TRIVIAL_ATTRIBUTE_NAME, true)); + + FREE_TRIVIAL = abi.downcallHandle(lookup.lookup("free").get(), + MethodType.methodType(void.class, MemoryAddress.class), + FunctionDescriptor.ofVoid(C_POINTER).withAttribute(FunctionDescriptor.TRIVIAL_ATTRIBUTE_NAME, true)); + } + + @Setup + public void setup() { + str = makeString(size); + segmentAllocator = SegmentAllocator.ofSegment(MemorySegment.allocateNative(size + 1, ResourceScope.newImplicitScope())); + } + + @TearDown + public void tearDown() { + scope.close(); + } + + @Benchmark + public int jni_strlen() throws Throwable { + return strlen(str); + } + + @Benchmark + public int panama_strlen() throws Throwable { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + MemorySegment segment = CLinker.toCString(str, scope); + return (int)STRLEN.invokeExact(segment.address()); + } + } + + @Benchmark + public int panama_strlen_arena() throws Throwable { + return (int)STRLEN.invokeExact(CLinker.toCString(str, arenaAllocator).address()); + } + + @Benchmark + public int panama_strlen_prefix() throws Throwable { + return (int)STRLEN.invokeExact(CLinker.toCString(str, segmentAllocator).address()); + } + + @Benchmark + public int panama_strlen_unsafe() throws Throwable { + MemoryAddress address = makeStringUnsafe(str); + int res = (int) STRLEN.invokeExact(address); + CLinker.freeMemory(address); + return res; + } + + @Benchmark + public int panama_strlen_unsafe_trivial() throws Throwable { + MemoryAddress address = makeStringUnsafeTrivial(str); + int res = (int) STRLEN_TRIVIAL.invokeExact(address); + FREE_TRIVIAL.invokeExact(address); + return res; + } + + static MemoryAddress makeStringUnsafe(String s) { + byte[] bytes = s.getBytes(); + int len = bytes.length; + MemoryAddress address = CLinker.allocateMemory(len + 1); + MemorySegment str = address.asSegment(len + 1, ResourceScope.globalScope()); + str.copyFrom(MemorySegment.ofArray(bytes)); + MemoryAccess.setByteAtOffset(str, len, (byte)0); + return address; + } + + static MemoryAddress makeStringUnsafeTrivial(String s) throws Throwable { + byte[] bytes = s.getBytes(); + int len = bytes.length; + MemoryAddress address = (MemoryAddress)MALLOC_TRIVIAL.invokeExact((long)len + 1); + MemorySegment str = address.asSegment(len + 1, ResourceScope.globalScope()); + str.copyFrom(MemorySegment.ofArray(bytes)); + MemoryAccess.setByteAtOffset(str, len, (byte)0); + return address; + } + + static native int strlen(String str); + + static String makeString(int size) { + String lorem = """ + Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et + dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip + ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu + fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt + mollit anim id est laborum. + """; + return lorem.substring(0, size); + } +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/TestAdaptVarHandles.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/TestAdaptVarHandles.java index da44ad762a3..f8f482b3a28 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/TestAdaptVarHandles.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/TestAdaptVarHandles.java @@ -24,7 +24,6 @@ package org.openjdk.bench.jdk.incubator.foreign; -import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemoryLayouts; @@ -86,7 +85,7 @@ public class TestAdaptVarHandles { static final VarHandle VH_box_int = MemoryHandles.filterValue(VH_int, INTBOX_TO_INT, INT_TO_INTBOX); - static final VarHandle VH_addr_int = MemoryLayout.ofSequence(MemoryLayouts.JAVA_INT) + static final VarHandle VH_addr_int = MemoryLayout.sequenceLayout(MemoryLayouts.JAVA_INT) .varHandle(int.class, MemoryLayout.PathElement.sequenceElement()); static final VarHandle VH_addr_box_int = MemoryHandles.filterValue(VH_addr_int, INTBOX_TO_INT, INT_TO_INTBOX); diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/UnrolledAccess.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/UnrolledAccess.java index 8b1374b97a4..614ff42b8f4 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/UnrolledAccess.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/UnrolledAccess.java @@ -32,9 +32,7 @@ import org.openjdk.jmh.runner.options.OptionsBuilder; import sun.misc.Unsafe; import java.util.concurrent.TimeUnit; -import java.lang.invoke.MethodHandles; import java.lang.invoke.VarHandle; -import java.lang.reflect.Field; @BenchmarkMode(Mode.AverageTime) @Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS) @@ -48,7 +46,7 @@ public class UnrolledAccess { final static int SIZE = 1024; - static final VarHandle LONG_HANDLE = MemoryLayout.ofSequence(SIZE, MemoryLayouts.JAVA_LONG) + static final VarHandle LONG_HANDLE = MemoryLayout.sequenceLayout(SIZE, MemoryLayouts.JAVA_LONG) .varHandle(long.class, MemoryLayout.PathElement.sequenceElement()); @State(Scope.Benchmark) @@ -67,8 +65,8 @@ public class UnrolledAccess { this.outputArray = new double[SIZE]; this.inputAddress = U.allocateMemory(8 * SIZE); this.outputAddress = U.allocateMemory(8 * SIZE); - this.inputSegment = MemoryAddress.ofLong(inputAddress).asSegmentRestricted(8*SIZE); - this.outputSegment = MemoryAddress.ofLong(outputAddress).asSegmentRestricted(8*SIZE); + this.inputSegment = MemoryAddress.ofLong(inputAddress).asSegment(8*SIZE, ResourceScope.globalScope()); + this.outputSegment = MemoryAddress.ofLong(outputAddress).asSegment(8*SIZE, ResourceScope.globalScope()); } } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/Upcalls.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/Upcalls.java index 80efd81ffec..af30a0c326c 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/Upcalls.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/Upcalls.java @@ -26,6 +26,7 @@ import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.LibraryLookup; import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -40,7 +41,9 @@ import java.lang.invoke.MethodType; import java.util.concurrent.TimeUnit; import static java.lang.invoke.MethodHandles.lookup; +import static jdk.incubator.foreign.CLinker.C_DOUBLE; import static jdk.incubator.foreign.CLinker.C_INT; +import static jdk.incubator.foreign.CLinker.C_LONG_LONG; import static jdk.incubator.foreign.CLinker.C_POINTER; @BenchmarkMode(Mode.AverageTime) @@ -48,18 +51,24 @@ import static jdk.incubator.foreign.CLinker.C_POINTER; @Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) @State(org.openjdk.jmh.annotations.Scope.Thread) @OutputTimeUnit(TimeUnit.NANOSECONDS) -@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "-Dforeign.restricted=permit" }) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) public class Upcalls { static final CLinker abi = CLinker.getInstance(); static final MethodHandle blank; static final MethodHandle identity; + static final MethodHandle args5; + static final MethodHandle args10; static final MemoryAddress cb_blank; static final MemoryAddress cb_identity; + static final MemoryAddress cb_args5; + static final MemoryAddress cb_args10; static final long cb_blank_jni; static final long cb_identity_jni; + static final long cb_args5_jni; + static final long cb_args10_jni; static { System.loadLibrary("UpcallsJNI"); @@ -67,38 +76,74 @@ public class Upcalls { String className = "org/openjdk/bench/jdk/incubator/foreign/Upcalls"; cb_blank_jni = makeCB(className, "blank", "()V"); cb_identity_jni = makeCB(className, "identity", "(I)I"); + cb_args5_jni = makeCB(className, "args5", "(JDJDJ)V"); + cb_args10_jni = makeCB(className, "args10", "(JDJDJDJDJD)V"); try { LibraryLookup ll = LibraryLookup.ofLibrary("Upcalls"); { - LibraryLookup.Symbol addr = ll.lookup("blank").get(); - MethodType mt = MethodType.methodType(void.class, MemoryAddress.class); - FunctionDescriptor fd = FunctionDescriptor.ofVoid(C_POINTER); - blank = abi.downcallHandle(addr, mt, fd); + String name = "blank"; + MethodType mt = MethodType.methodType(void.class); + FunctionDescriptor fd = FunctionDescriptor.ofVoid(); - cb_blank = abi.upcallStub( - lookup().findStatic(Upcalls.class, "blank", MethodType.methodType(void.class)), - FunctionDescriptor.ofVoid() - ).address(); + blank = linkFunc(ll, name, mt, fd); + cb_blank = makeCB(name, mt, fd); } { - LibraryLookup.Symbol addr = ll.lookup("identity").get(); - MethodType mt = MethodType.methodType(int.class, int.class, MemoryAddress.class); - FunctionDescriptor fd = FunctionDescriptor.of(C_INT, C_INT, C_POINTER); - identity = abi.downcallHandle(addr, mt, fd); + String name = "identity"; + MethodType mt = MethodType.methodType(int.class, int.class); + FunctionDescriptor fd = FunctionDescriptor.of(C_INT, C_INT); - cb_identity = abi.upcallStub( - lookup().findStatic(Upcalls.class, "identity", MethodType.methodType(int.class, int.class)), - FunctionDescriptor.of(C_INT, C_INT) - ).address(); + identity = linkFunc(ll, name, mt, fd); + cb_identity = makeCB(name, mt, fd); + } + { + String name = "args5"; + MethodType mt = MethodType.methodType(void.class, + long.class, double.class, long.class, double.class, long.class); + FunctionDescriptor fd = FunctionDescriptor.ofVoid( + C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG); + + args5 = linkFunc(ll, name, mt, fd); + cb_args5 = makeCB(name, mt, fd); + } + { + String name = "args10"; + MethodType mt = MethodType.methodType(void.class, + long.class, double.class, long.class, double.class, long.class, + double.class, long.class, double.class, long.class, double.class); + FunctionDescriptor fd = FunctionDescriptor.ofVoid( + C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG, + C_DOUBLE, C_LONG_LONG, C_DOUBLE, C_LONG_LONG, C_DOUBLE); + + args10 = linkFunc(ll, name, mt, fd); + cb_args10 = makeCB(name, mt, fd); } } catch (ReflectiveOperationException e) { throw new BootstrapMethodError(e); } } + static MethodHandle linkFunc(LibraryLookup ll, String name, MethodType baseType, FunctionDescriptor baseDesc) { + return abi.downcallHandle( + ll.lookup(name).orElseThrow(), + baseType.insertParameterTypes(baseType.parameterCount(), MemoryAddress.class), + baseDesc.withAppendedArgumentLayouts(C_POINTER) + ); + } + + static MemoryAddress makeCB(String name, MethodType mt, FunctionDescriptor fd) throws ReflectiveOperationException { + return abi.upcallStub( + lookup().findStatic(Upcalls.class, name, mt), + fd, ResourceScope.globalScope() + ).address(); + } + static native void blank(long cb); static native int identity(int x, long cb); + static native void args5(long a0, double a1, long a2, double a3, long a4, long cb); + static native void args10(long a0, double a1, long a2, double a3, long a4, + double a5, long a6, double a7, long a8, double a9, long cb); static native long makeCB(String holder, String name, String signature); @Benchmark @@ -116,11 +161,34 @@ public class Upcalls { return identity(10, cb_identity_jni); } + @Benchmark + public void jni_args5() throws Throwable { + args5(1L, 2D, 3L, 4D, 5L, cb_args5_jni); + } + + @Benchmark + public void jni_args10() throws Throwable { + args10(1L, 2D, 3L, 4D, 5L, 6D, 7L, 8D, 9L, 10D, cb_args10_jni); + } + @Benchmark public int panama_identity() throws Throwable { return (int) identity.invokeExact(10, cb_identity); } + @Benchmark + public void panama_args5() throws Throwable { + args5.invokeExact(1L, 2D, 3L, 4D, 5L, cb_args5); + } + + @Benchmark + public void panama_args10() throws Throwable { + args10.invokeExact(1L, 2D, 3L, 4D, 5L, 6D, 7L, 8D, 9L, 10D, cb_args10); + } + static void blank() {} static int identity(int x) { return x; } + static void args5(long a0, double a1, long a2, double a3, long a4) { } + static void args10(long a0, double a1, long a2, double a3, long a4, + double a5, long a6, double a7, long a8, double a9) { } } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/VaList.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/VaList.java index c1823fef49b..cd02807cdce 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/VaList.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/VaList.java @@ -25,6 +25,7 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.FunctionDescriptor; import jdk.incubator.foreign.LibraryLookup; import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -49,7 +50,7 @@ import static jdk.incubator.foreign.CLinker.asVarArg; @Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) @State(org.openjdk.jmh.annotations.Scope.Thread) @OutputTimeUnit(TimeUnit.NANOSECONDS) -@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "-Dforeign.restricted=permit" }) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) public class VaList { static final CLinker linker = CLinker.getInstance(); @@ -63,7 +64,7 @@ public class VaList { MethodType.methodType(void.class, int.class, int.class, double.class, long.class), FunctionDescriptor.ofVoid(C_INT, asVarArg(C_INT), asVarArg(C_DOUBLE), asVarArg(C_LONG_LONG))); MH_vaList = linker.downcallHandle(lookup.lookup("vaList").get(), - MethodType.methodType(void.class, int.class, VaList.class), + MethodType.methodType(void.class, int.class, CLinker.VaList.class), FunctionDescriptor.ofVoid(C_INT, C_VA_LIST)); } @@ -75,13 +76,13 @@ public class VaList { @Benchmark public void vaList() throws Throwable { - try (CLinker.VaList vaList = CLinker.VaList.make(b -> - b.vargFromInt(C_INT, 1) - .vargFromDouble(C_DOUBLE, 2D) - .vargFromLong(C_LONG_LONG, 3L) - )) { + try (ResourceScope scope = ResourceScope.newConfinedScope()) { + CLinker.VaList vaList = CLinker.VaList.make(b -> + b.vargFromInt(C_INT, 1) + .vargFromDouble(C_DOUBLE, 2D) + .vargFromLong(C_LONG_LONG, 3L), scope); MH_vaList.invokeExact(3, - vaList); + vaList); } } } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/VarHandleExact.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/VarHandleExact.java index 12ae364f148..d560765c48c 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/VarHandleExact.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/VarHandleExact.java @@ -24,6 +24,7 @@ package org.openjdk.bench.jdk.incubator.foreign; import jdk.incubator.foreign.MemoryHandles; import jdk.incubator.foreign.MemorySegment; +import jdk.incubator.foreign.ResourceScope; import org.openjdk.jmh.annotations.Benchmark; import org.openjdk.jmh.annotations.BenchmarkMode; import org.openjdk.jmh.annotations.Fork; @@ -61,12 +62,12 @@ public class VarHandleExact { @Setup public void setup() { - data = MemorySegment.allocateNative(JAVA_INT); + data = MemorySegment.allocateNative(JAVA_INT, ResourceScope.newConfinedScope()); } @TearDown public void tearDown() { - data.close(); + data.scope().close(); } @Benchmark diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libCallOverhead.c b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libCallOverhead.c index 6cd014e8663..f42aec0172c 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libCallOverhead.c +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libCallOverhead.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -46,6 +46,10 @@ EXPORT void* identity_memory_address(void* p) { return p; } +EXPORT void args1(long long a0) {} +EXPORT void args2(long long a0, double a1) {} +EXPORT void args3(long long a0, double a1, long long a2) {} +EXPORT void args4(long long a0, double a1, long long a2, double a3) {} EXPORT void args5(long long a0, double a1, long long a2, double a3, long long a4) {} EXPORT void args10(long long a0, double a1, long long a2, double a3, long long a4, double a5, long long a6, double a7, long long a8, double a9) {} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libCallOverheadJNI.c b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libCallOverheadJNI.c index 1d74e90cbab..e1dd5dd31ee 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libCallOverheadJNI.c +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libCallOverheadJNI.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2020, 2021, 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 @@ -21,19 +21,15 @@ * questions. */ #include +#include +#include "libCallOverhead.c" -void func() {} - -int identity(int x) { - return x; -} - -JNIEXPORT void JNICALL Java_org_openjdk_bench_jdk_incubator_foreign_CallOverhead_blank +JNIEXPORT void JNICALL Java_org_openjdk_bench_jdk_incubator_foreign_CallOverheadHelper_blank (JNIEnv *env, jclass cls) { func(); } -JNIEXPORT jint JNICALL Java_org_openjdk_bench_jdk_incubator_foreign_CallOverhead_identity +JNIEXPORT jint JNICALL Java_org_openjdk_bench_jdk_incubator_foreign_CallOverheadHelper_identity (JNIEnv *env, jclass cls, jint x) { return identity(x); } diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libStrLen.c b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libStrLen.c new file mode 100644 index 00000000000..ebbc2f2caa1 --- /dev/null +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libStrLen.c @@ -0,0 +1,35 @@ +/* + * Copyright (c) 2021, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ + +#include +#include +#include + +JNIEXPORT jint JNICALL Java_org_openjdk_bench_jdk_incubator_foreign_StrLenTest_strlen(JNIEnv *const env, const jclass cls, const jstring text) { + const char *str = (*env)->GetStringUTFChars(env, text, NULL); + int len = (int)strlen(str); + (*env)->ReleaseStringUTFChars(env, text, str); + return len; +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libUpcalls.c b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libUpcalls.c index df0d3b25ee7..220a890a812 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libUpcalls.c +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libUpcalls.c @@ -34,3 +34,15 @@ EXPORT void blank(void (*cb)(void)) { EXPORT int identity(int x, int (*cb)(int)) { return cb(x); } + +EXPORT void args5(long long a0, double a1, long long a2, double a3, long long a4, + void (*cb)(long long, double, long long, double, long long)) { + cb(a0, a1, a2, a3, a4); +} + +EXPORT void args10(long long a0, double a1, long long a2, double a3, long long a4, + double a5, long long a6, double a7, long long a8, double a9, + void (*cb)(long long, double, long long, double, long long, + double, long long, double, long long, double)) { + cb(a0, a1, a2, a3, a4, a5, a6, a7, a8, a9); +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libUpcallsJNI.c b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libUpcallsJNI.c index bf946891578..c44979e3e62 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/libUpcallsJNI.c +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/libUpcallsJNI.c @@ -24,14 +24,6 @@ #include #include "jlong.h" -void blank(void (*cb)(void)) { - cb(); -} - -int identity(int x, int (*cb)(int)) { - return cb(x); -} - typedef struct { jclass holder; jmethodID mid; @@ -82,3 +74,20 @@ JNIEXPORT jint JNICALL Java_org_openjdk_bench_jdk_incubator_foreign_Upcalls_iden JNICB jniCb = jlong_to_ptr(cb); return (*env)->CallStaticIntMethod(env, jniCb->holder, jniCb->mid, x); } + +JNIEXPORT jint JNICALL Java_org_openjdk_bench_jdk_incubator_foreign_Upcalls_args5 + (JNIEnv *env, jclass cls, + jlong a0, jdouble a1, jlong a2, jdouble a3, jlong a4, + jlong cb) { + JNICB jniCb = jlong_to_ptr(cb); + return (*env)->CallStaticIntMethod(env, jniCb->holder, jniCb->mid, a0, a1, a2, a3, a4); +} + +JNIEXPORT jint JNICALL Java_org_openjdk_bench_jdk_incubator_foreign_Upcalls_args10 + (JNIEnv *env, jclass cls, + jlong a0, jdouble a1, jlong a2, jdouble a3, jlong a4, + jdouble a5, jlong a6, jdouble a7, jlong a8, jdouble a9, + jlong cb) { + JNICB jniCb = jlong_to_ptr(cb); + return (*env)->CallStaticIntMethod(env, jniCb->holder, jniCb->mid, a0, a1, a2, a3, a4, a5, a6, a7, a8, a9); +} diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsAccess.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsAccess.java index beb0b5292ed..3eca42c5521 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsAccess.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsAccess.java @@ -43,7 +43,7 @@ import java.util.concurrent.TimeUnit; @Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) @State(org.openjdk.jmh.annotations.Scope.Thread) @OutputTimeUnit(TimeUnit.NANOSECONDS) -@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "-Dforeign.restricted=permit" }) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) public class PointsAccess { BBPoint BBPoint; diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsAlloc.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsAlloc.java index f13648b776a..1c9fd3b69d7 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsAlloc.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsAlloc.java @@ -41,7 +41,7 @@ import java.util.concurrent.TimeUnit; @Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) @State(org.openjdk.jmh.annotations.Scope.Thread) @OutputTimeUnit(TimeUnit.NANOSECONDS) -@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "-Dforeign.restricted=permit" }) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) public class PointsAlloc { @Benchmark diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsDistance.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsDistance.java index 164d74e4236..8c371587665 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsDistance.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsDistance.java @@ -43,7 +43,7 @@ import java.util.concurrent.TimeUnit; @Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) @State(org.openjdk.jmh.annotations.Scope.Thread) @OutputTimeUnit(TimeUnit.NANOSECONDS) -@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "-Dforeign.restricted=permit" }) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) public class PointsDistance { BBPoint jniP1; diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsFree.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsFree.java index dbf76e8b12b..ab34b41666e 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsFree.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/PointsFree.java @@ -42,7 +42,7 @@ import java.util.concurrent.TimeUnit; @Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS) @State(org.openjdk.jmh.annotations.Scope.Thread) @OutputTimeUnit(TimeUnit.NANOSECONDS) -@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "-Dforeign.restricted=permit" }) +@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" }) public class PointsFree { JNIPoint jniPoint; diff --git a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/support/PanamaPoint.java b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/support/PanamaPoint.java index aea10f5853f..e8eee07a63d 100644 --- a/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/support/PanamaPoint.java +++ b/test/micro/org/openjdk/bench/jdk/incubator/foreign/points/support/PanamaPoint.java @@ -28,6 +28,7 @@ import jdk.incubator.foreign.MemoryAddress; import jdk.incubator.foreign.MemoryLayout; import jdk.incubator.foreign.MemorySegment; import jdk.incubator.foreign.CLinker; +import jdk.incubator.foreign.ResourceScope; import java.lang.invoke.MethodHandle; import java.lang.invoke.VarHandle; @@ -38,7 +39,7 @@ import static jdk.incubator.foreign.CLinker.*; public class PanamaPoint implements AutoCloseable { - public static final MemoryLayout LAYOUT = MemoryLayout.ofStruct( + public static final MemoryLayout LAYOUT = MemoryLayout.structLayout( C_INT.withName("x"), C_INT.withName("y") ); @@ -66,7 +67,7 @@ public class PanamaPoint implements AutoCloseable { private final MemorySegment segment; public PanamaPoint(int x, int y) { - this(MemorySegment.allocateNative(LAYOUT), x, y); + this(MemorySegment.allocateNative(LAYOUT, ResourceScope.newConfinedScope()), x, y); } public PanamaPoint(MemorySegment segment, int x, int y) { @@ -113,6 +114,6 @@ public class PanamaPoint implements AutoCloseable { @Override public void close() { - segment.close(); + segment.scope().close(); } }