diff --git a/src/hotspot/cpu/aarch64/aarch64.ad b/src/hotspot/cpu/aarch64/aarch64.ad index d89363099d3..fcd0ce096f9 100644 --- a/src/hotspot/cpu/aarch64/aarch64.ad +++ b/src/hotspot/cpu/aarch64/aarch64.ad @@ -3893,10 +3893,10 @@ encode %{ // Load markWord from object into displaced_header. __ ldr(disp_hdr, Address(oop, oopDesc::mark_offset_in_bytes())); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { __ load_klass(tmp, oop); __ ldrw(tmp, Address(tmp, Klass::access_flags_offset())); - __ tstw(tmp, JVM_ACC_IS_BOX_CLASS); + __ tstw(tmp, JVM_ACC_IS_VALUE_BASED_CLASS); __ br(Assembler::NE, cont); } diff --git a/src/hotspot/cpu/aarch64/c1_MacroAssembler_aarch64.cpp b/src/hotspot/cpu/aarch64/c1_MacroAssembler_aarch64.cpp index 42d1c5ee359..48759ca7122 100644 --- a/src/hotspot/cpu/aarch64/c1_MacroAssembler_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/c1_MacroAssembler_aarch64.cpp @@ -75,10 +75,10 @@ int C1_MacroAssembler::lock_object(Register hdr, Register obj, Register disp_hdr null_check_offset = offset(); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(hdr, obj); ldrw(hdr, Address(hdr, Klass::access_flags_offset())); - tstw(hdr, JVM_ACC_IS_BOX_CLASS); + tstw(hdr, JVM_ACC_IS_VALUE_BASED_CLASS); br(Assembler::NE, slow_case); } diff --git a/src/hotspot/cpu/aarch64/interp_masm_aarch64.cpp b/src/hotspot/cpu/aarch64/interp_masm_aarch64.cpp index 30a315b20a6..94c379a89e8 100644 --- a/src/hotspot/cpu/aarch64/interp_masm_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/interp_masm_aarch64.cpp @@ -746,10 +746,10 @@ void InterpreterMacroAssembler::lock_object(Register lock_reg) // Load object pointer into obj_reg %c_rarg3 ldr(obj_reg, Address(lock_reg, obj_offset)); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(tmp, obj_reg); ldrw(tmp, Address(tmp, Klass::access_flags_offset())); - tstw(tmp, JVM_ACC_IS_BOX_CLASS); + tstw(tmp, JVM_ACC_IS_VALUE_BASED_CLASS); br(Assembler::NE, slow_case); } diff --git a/src/hotspot/cpu/arm/c1_MacroAssembler_arm.cpp b/src/hotspot/cpu/arm/c1_MacroAssembler_arm.cpp index 7304e3ad15f..9d5c82dceb9 100644 --- a/src/hotspot/cpu/arm/c1_MacroAssembler_arm.cpp +++ b/src/hotspot/cpu/arm/c1_MacroAssembler_arm.cpp @@ -204,10 +204,10 @@ int C1_MacroAssembler::lock_object(Register hdr, Register obj, null_check_offset = offset(); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(tmp1, obj); ldr_u32(tmp1, Address(tmp1, Klass::access_flags_offset())); - tst(tmp1, JVM_ACC_IS_BOX_CLASS); + tst(tmp1, JVM_ACC_IS_VALUE_BASED_CLASS); b(slow_case, ne); } diff --git a/src/hotspot/cpu/arm/c2_MacroAssembler_arm.cpp b/src/hotspot/cpu/arm/c2_MacroAssembler_arm.cpp index 0b14d9ebb14..2211d5c5fa3 100644 --- a/src/hotspot/cpu/arm/c2_MacroAssembler_arm.cpp +++ b/src/hotspot/cpu/arm/c2_MacroAssembler_arm.cpp @@ -90,10 +90,10 @@ void C2_MacroAssembler::fast_lock(Register Roop, Register Rbox, Register Rscratc Label fast_lock, done; - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(Rscratch, Roop); ldr_u32(Rscratch, Address(Rscratch, Klass::access_flags_offset())); - tst(Rscratch, JVM_ACC_IS_BOX_CLASS); + tst(Rscratch, JVM_ACC_IS_VALUE_BASED_CLASS); b(done, ne); } diff --git a/src/hotspot/cpu/arm/interp_masm_arm.cpp b/src/hotspot/cpu/arm/interp_masm_arm.cpp index 01ff3a5d39c..f754b36a27f 100644 --- a/src/hotspot/cpu/arm/interp_masm_arm.cpp +++ b/src/hotspot/cpu/arm/interp_masm_arm.cpp @@ -883,10 +883,10 @@ void InterpreterMacroAssembler::lock_object(Register Rlock) { // Load object pointer ldr(Robj, Address(Rlock, obj_offset)); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(R0, Robj); ldr_u32(R0, Address(R0, Klass::access_flags_offset())); - tst(R0, JVM_ACC_IS_BOX_CLASS); + tst(R0, JVM_ACC_IS_VALUE_BASED_CLASS); b(slow_case, ne); } diff --git a/src/hotspot/cpu/ppc/c1_MacroAssembler_ppc.cpp b/src/hotspot/cpu/ppc/c1_MacroAssembler_ppc.cpp index 4d1dc267fac..1dd18861dea 100644 --- a/src/hotspot/cpu/ppc/c1_MacroAssembler_ppc.cpp +++ b/src/hotspot/cpu/ppc/c1_MacroAssembler_ppc.cpp @@ -105,10 +105,10 @@ void C1_MacroAssembler::lock_object(Register Rmark, Register Roop, Register Rbox // Save object being locked into the BasicObjectLock... std(Roop, BasicObjectLock::obj_offset_in_bytes(), Rbox); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(Rscratch, Roop); lwz(Rscratch, in_bytes(Klass::access_flags_offset()), Rscratch); - testbitdi(CCR0, R0, Rscratch, exact_log2(JVM_ACC_IS_BOX_CLASS)); + testbitdi(CCR0, R0, Rscratch, exact_log2(JVM_ACC_IS_VALUE_BASED_CLASS)); bne(CCR0, slow_int); } diff --git a/src/hotspot/cpu/ppc/interp_masm_ppc_64.cpp b/src/hotspot/cpu/ppc/interp_masm_ppc_64.cpp index b59eb97d447..6427392469a 100644 --- a/src/hotspot/cpu/ppc/interp_masm_ppc_64.cpp +++ b/src/hotspot/cpu/ppc/interp_masm_ppc_64.cpp @@ -910,10 +910,10 @@ void InterpreterMacroAssembler::lock_object(Register monitor, Register object) { // Load markWord from object into displaced_header. ld(displaced_header, oopDesc::mark_offset_in_bytes(), object); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(tmp, object); lwz(tmp, in_bytes(Klass::access_flags_offset()), tmp); - testbitdi(CCR0, R0, tmp, exact_log2(JVM_ACC_IS_BOX_CLASS)); + testbitdi(CCR0, R0, tmp, exact_log2(JVM_ACC_IS_VALUE_BASED_CLASS)); bne(CCR0, slow_case); } diff --git a/src/hotspot/cpu/ppc/macroAssembler_ppc.cpp b/src/hotspot/cpu/ppc/macroAssembler_ppc.cpp index 72b10dff4aa..5dfb5d6b471 100644 --- a/src/hotspot/cpu/ppc/macroAssembler_ppc.cpp +++ b/src/hotspot/cpu/ppc/macroAssembler_ppc.cpp @@ -2818,10 +2818,10 @@ void MacroAssembler::compiler_fast_lock_object(ConditionRegister flag, Register // Load markWord from object into displaced_header. ld(displaced_header, oopDesc::mark_offset_in_bytes(), oop); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(temp, oop); lwz(temp, in_bytes(Klass::access_flags_offset()), temp); - testbitdi(flag, R0, temp, exact_log2(JVM_ACC_IS_BOX_CLASS)); + testbitdi(flag, R0, temp, exact_log2(JVM_ACC_IS_VALUE_BASED_CLASS)); bne(flag, cont); } diff --git a/src/hotspot/cpu/s390/c1_MacroAssembler_s390.cpp b/src/hotspot/cpu/s390/c1_MacroAssembler_s390.cpp index 16e7c8997b4..6c040e0bbcb 100644 --- a/src/hotspot/cpu/s390/c1_MacroAssembler_s390.cpp +++ b/src/hotspot/cpu/s390/c1_MacroAssembler_s390.cpp @@ -91,9 +91,9 @@ void C1_MacroAssembler::lock_object(Register hdr, Register obj, Register disp_hd // Save object being locked into the BasicObjectLock... z_stg(obj, Address(disp_hdr, BasicObjectLock::obj_offset_in_bytes())); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(Z_R1_scratch, obj); - testbit(Address(Z_R1_scratch, Klass::access_flags_offset()), exact_log2(JVM_ACC_IS_BOX_CLASS)); + testbit(Address(Z_R1_scratch, Klass::access_flags_offset()), exact_log2(JVM_ACC_IS_VALUE_BASED_CLASS)); z_btrue(slow_case); } diff --git a/src/hotspot/cpu/s390/interp_masm_s390.cpp b/src/hotspot/cpu/s390/interp_masm_s390.cpp index 275d86051bb..5c7e959937c 100644 --- a/src/hotspot/cpu/s390/interp_masm_s390.cpp +++ b/src/hotspot/cpu/s390/interp_masm_s390.cpp @@ -999,9 +999,9 @@ void InterpreterMacroAssembler::lock_object(Register monitor, Register object) { // Load markWord from object into displaced_header. z_lg(displaced_header, oopDesc::mark_offset_in_bytes(), object); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(Z_R1_scratch, object); - testbit(Address(Z_R1_scratch, Klass::access_flags_offset()), exact_log2(JVM_ACC_IS_BOX_CLASS)); + testbit(Address(Z_R1_scratch, Klass::access_flags_offset()), exact_log2(JVM_ACC_IS_VALUE_BASED_CLASS)); z_btrue(slow_case); } diff --git a/src/hotspot/cpu/s390/macroAssembler_s390.cpp b/src/hotspot/cpu/s390/macroAssembler_s390.cpp index d7c95ee96ee..9c06a20ae70 100644 --- a/src/hotspot/cpu/s390/macroAssembler_s390.cpp +++ b/src/hotspot/cpu/s390/macroAssembler_s390.cpp @@ -3326,11 +3326,11 @@ void MacroAssembler::compiler_fast_lock_object(Register oop, Register box, Regis // Load markWord from oop into mark. z_lg(displacedHeader, 0, oop); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(Z_R1_scratch, oop); z_l(Z_R1_scratch, Address(Z_R1_scratch, Klass::access_flags_offset())); - assert((JVM_ACC_IS_BOX_CLASS & 0xFFFF) == 0, "or change following instruction"); - z_nilh(Z_R1_scratch, JVM_ACC_IS_BOX_CLASS >> 16); + assert((JVM_ACC_IS_VALUE_BASED_CLASS & 0xFFFF) == 0, "or change following instruction"); + z_nilh(Z_R1_scratch, JVM_ACC_IS_VALUE_BASED_CLASS >> 16); z_brne(done); } diff --git a/src/hotspot/cpu/x86/c1_MacroAssembler_x86.cpp b/src/hotspot/cpu/x86/c1_MacroAssembler_x86.cpp index 0ca3ac105fc..687a33b253e 100644 --- a/src/hotspot/cpu/x86/c1_MacroAssembler_x86.cpp +++ b/src/hotspot/cpu/x86/c1_MacroAssembler_x86.cpp @@ -54,10 +54,10 @@ int C1_MacroAssembler::lock_object(Register hdr, Register obj, Register disp_hdr null_check_offset = offset(); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(hdr, obj, rklass_decode_tmp); movl(hdr, Address(hdr, Klass::access_flags_offset())); - testl(hdr, JVM_ACC_IS_BOX_CLASS); + testl(hdr, JVM_ACC_IS_VALUE_BASED_CLASS); jcc(Assembler::notZero, slow_case); } diff --git a/src/hotspot/cpu/x86/c2_MacroAssembler_x86.cpp b/src/hotspot/cpu/x86/c2_MacroAssembler_x86.cpp index f5f712ee8e7..c975b4f3b13 100644 --- a/src/hotspot/cpu/x86/c2_MacroAssembler_x86.cpp +++ b/src/hotspot/cpu/x86/c2_MacroAssembler_x86.cpp @@ -485,10 +485,10 @@ void C2_MacroAssembler::fast_lock(Register objReg, Register boxReg, Register tmp Label IsInflated, DONE_LABEL; - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(tmpReg, objReg, cx1Reg); movl(tmpReg, Address(tmpReg, Klass::access_flags_offset())); - testl(tmpReg, JVM_ACC_IS_BOX_CLASS); + testl(tmpReg, JVM_ACC_IS_VALUE_BASED_CLASS); jcc(Assembler::notZero, DONE_LABEL); } diff --git a/src/hotspot/cpu/x86/interp_masm_x86.cpp b/src/hotspot/cpu/x86/interp_masm_x86.cpp index b6d7bf81856..19c7f49b788 100644 --- a/src/hotspot/cpu/x86/interp_masm_x86.cpp +++ b/src/hotspot/cpu/x86/interp_masm_x86.cpp @@ -1219,10 +1219,10 @@ void InterpreterMacroAssembler::lock_object(Register lock_reg) { // Load object pointer into obj_reg movptr(obj_reg, Address(lock_reg, obj_offset)); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { + if (DiagnoseSyncOnValueBasedClasses != 0) { load_klass(tmp_reg, obj_reg, rklass_decode_tmp); movl(tmp_reg, Address(tmp_reg, Klass::access_flags_offset())); - testl(tmp_reg, JVM_ACC_IS_BOX_CLASS); + testl(tmp_reg, JVM_ACC_IS_VALUE_BASED_CLASS); jcc(Assembler::notZero, slow_case); } diff --git a/src/hotspot/share/classfile/classFileParser.cpp b/src/hotspot/share/classfile/classFileParser.cpp index 124cee2ca8a..9130018b3e3 100644 --- a/src/hotspot/share/classfile/classFileParser.cpp +++ b/src/hotspot/share/classfile/classFileParser.cpp @@ -1091,6 +1091,7 @@ public: _jdk_internal_vm_annotation_Contended, _field_Stable, _jdk_internal_vm_annotation_ReservedStackAccess, + _jdk_internal_ValueBased, _annotation_LIMIT }; const Location _location; @@ -2147,6 +2148,11 @@ AnnotationCollector::annotation_index(const ClassLoaderData* loader_data, if (RestrictReservedStack && !privileged) break; // honor privileges return _jdk_internal_vm_annotation_ReservedStackAccess; } + case VM_SYMBOL_ENUM_NAME(jdk_internal_ValueBased_signature): { + if (_location != _in_class) break; // only allow for classes + if (!privileged) break; // only allow in priviledged code + return _jdk_internal_ValueBased; + } default: { break; } @@ -2190,7 +2196,16 @@ void MethodAnnotationCollector::apply_to(const methodHandle& m) { void ClassFileParser::ClassAnnotationCollector::apply_to(InstanceKlass* ik) { assert(ik != NULL, "invariant"); - ik->set_is_contended(is_contended()); + if (has_annotation(_jdk_internal_vm_annotation_Contended)) { + ik->set_is_contended(is_contended()); + } + if (has_annotation(_jdk_internal_ValueBased)) { + ik->set_has_value_based_class_annotation(); + if (DiagnoseSyncOnValueBasedClasses) { + ik->set_is_value_based(); + ik->set_prototype_header(markWord::prototype()); + } + } } #define MAX_ARGS_SIZE 255 diff --git a/src/hotspot/share/classfile/systemDictionary.cpp b/src/hotspot/share/classfile/systemDictionary.cpp index ab346f7af09..9d7f63eb0ea 100644 --- a/src/hotspot/share/classfile/systemDictionary.cpp +++ b/src/hotspot/share/classfile/systemDictionary.cpp @@ -2142,14 +2142,6 @@ void SystemDictionary::resolve_well_known_classes(TRAPS) { //_box_klasses[T_OBJECT] = WK_KLASS(object_klass); //_box_klasses[T_ARRAY] = WK_KLASS(object_klass); - if (DiagnoseSyncOnPrimitiveWrappers != 0) { - for (int i = T_BOOLEAN; i < T_LONG + 1; i++) { - assert(_box_klasses[i] != NULL, "NULL box class"); - _box_klasses[i]->set_is_box(); - _box_klasses[i]->set_prototype_header(markWord::prototype()); - } - } - #ifdef ASSERT if (UseSharedSpaces) { JVMTI_ONLY(assert(JvmtiExport::is_early_phase(), diff --git a/src/hotspot/share/classfile/vmSymbols.hpp b/src/hotspot/share/classfile/vmSymbols.hpp index de399d809cc..495c30cfa60 100644 --- a/src/hotspot/share/classfile/vmSymbols.hpp +++ b/src/hotspot/share/classfile/vmSymbols.hpp @@ -249,6 +249,7 @@ template(java_util_concurrent_atomic_AtomicReferenceFieldUpdater_Impl, "java/util/concurrent/atomic/AtomicReferenceFieldUpdater$AtomicReferenceFieldUpdaterImpl") \ template(jdk_internal_vm_annotation_Contended_signature, "Ljdk/internal/vm/annotation/Contended;") \ template(jdk_internal_vm_annotation_ReservedStackAccess_signature, "Ljdk/internal/vm/annotation/ReservedStackAccess;") \ + template(jdk_internal_ValueBased_signature, "Ljdk/internal/ValueBased;") \ \ /* class symbols needed by intrinsics */ \ VM_INTRINSICS_DO(VM_INTRINSIC_IGNORE, template, VM_SYMBOL_IGNORE, VM_SYMBOL_IGNORE, VM_ALIAS_IGNORE) \ diff --git a/src/hotspot/share/jfr/metadata/metadata.xml b/src/hotspot/share/jfr/metadata/metadata.xml index 43d980b4e8a..d4484f7db36 100644 --- a/src/hotspot/share/jfr/metadata/metadata.xml +++ b/src/hotspot/share/jfr/metadata/metadata.xml @@ -66,8 +66,8 @@ - - + + has_class_mirror_holder(); } - bool is_box() const { return access_flags().is_box_class(); } - void set_is_box() { _access_flags.set_is_box_class(); } + bool is_value_based() { return _access_flags.is_value_based_class(); } + void set_is_value_based() { _access_flags.set_is_value_based_class(); } bool is_cloneable() const; void set_is_cloneable(); diff --git a/src/hotspot/share/runtime/arguments.cpp b/src/hotspot/share/runtime/arguments.cpp index 68e0ef34b2e..732923b51ac 100644 --- a/src/hotspot/share/runtime/arguments.cpp +++ b/src/hotspot/share/runtime/arguments.cpp @@ -4139,9 +4139,9 @@ jint Arguments::apply_ergo() { } #endif // COMPILER2 - if (FLAG_IS_CMDLINE(DiagnoseSyncOnPrimitiveWrappers)) { - if (DiagnoseSyncOnPrimitiveWrappers == ObjectSynchronizer::LOG_WARNING && !log_is_enabled(Info, primitivewrappers)) { - LogConfiguration::configure_stdout(LogLevel::Info, true, LOG_TAGS(primitivewrappers)); + if (FLAG_IS_CMDLINE(DiagnoseSyncOnValueBasedClasses)) { + if (DiagnoseSyncOnValueBasedClasses == ObjectSynchronizer::LOG_WARNING && !log_is_enabled(Info, valuebasedclasses)) { + LogConfiguration::configure_stdout(LogLevel::Info, true, LOG_TAGS(valuebasedclasses)); } } return JNI_OK; diff --git a/src/hotspot/share/runtime/globals.hpp b/src/hotspot/share/runtime/globals.hpp index 28147427680..7efabe34eeb 100644 --- a/src/hotspot/share/runtime/globals.hpp +++ b/src/hotspot/share/runtime/globals.hpp @@ -851,13 +851,13 @@ const intx ObjectAlignmentInBytes = 8; range(500, max_intx) \ constraint(BiasedLockingDecayTimeFunc,AfterErgo) \ \ - product(intx, DiagnoseSyncOnPrimitiveWrappers, 0, DIAGNOSTIC, \ + product(intx, DiagnoseSyncOnValueBasedClasses, 0, DIAGNOSTIC, \ "Detect and take action upon identifying synchronization on " \ - "primitive wrappers. Modes: " \ + "value based classes. Modes: " \ "0: off; " \ "1: exit with fatal error; " \ "2: log message to stdout. Output file can be specified with " \ - " -Xlog:primitivewrappers. If JFR is running it will " \ + " -Xlog:valuebasedclasses. If JFR is running it will " \ " also generate JFR events.") \ range(0, 2) \ \ diff --git a/src/hotspot/share/runtime/synchronizer.cpp b/src/hotspot/share/runtime/synchronizer.cpp index 17a7ba4f117..1fd6930d96c 100644 --- a/src/hotspot/share/runtime/synchronizer.cpp +++ b/src/hotspot/share/runtime/synchronizer.cpp @@ -333,7 +333,7 @@ bool ObjectSynchronizer::quick_enter(oop obj, Thread* self, NoSafepointVerifier nsv; if (obj == NULL) return false; // Need to throw NPE - if (DiagnoseSyncOnPrimitiveWrappers != 0 && obj->klass()->is_box()) { + if (obj->klass()->is_value_based()) { return false; } @@ -387,17 +387,23 @@ bool ObjectSynchronizer::quick_enter(oop obj, Thread* self, return false; // revert to slow-path } -// Handle notifications when synchronizing on primitive wrappers -void ObjectSynchronizer::handle_sync_on_primitive_wrapper(Handle obj, Thread* current) { +// Handle notifications when synchronizing on value based classes +void ObjectSynchronizer::handle_sync_on_value_based_class(Handle obj, Thread* current) { JavaThread* self = current->as_Java_thread(); frame last_frame = self->last_frame(); - if (last_frame.is_interpreted_frame()) { + bool bcp_was_adjusted = false; + // Don't decrement bcp if it points to the frame's first instruction. This happens when + // handle_sync_on_value_based_class() is called because of a synchronized method. There + // is no actual monitorenter instruction in the byte code in this case. + if (last_frame.is_interpreted_frame() && + (last_frame.interpreter_frame_method()->code_base() < last_frame.interpreter_frame_bcp())) { // adjust bcp to point back to monitorenter so that we print the correct line numbers last_frame.interpreter_frame_set_bcp(last_frame.interpreter_frame_bcp() - 1); + bcp_was_adjusted = true; } - if (DiagnoseSyncOnPrimitiveWrappers == FATAL_EXIT) { + if (DiagnoseSyncOnValueBasedClasses == FATAL_EXIT) { ResourceMark rm(self); stringStream ss; self->print_stack_on(&ss); @@ -408,26 +414,26 @@ void ObjectSynchronizer::handle_sync_on_primitive_wrapper(Handle obj, Thread* cu } fatal("Synchronizing on object " INTPTR_FORMAT " of klass %s %s", p2i(obj()), obj->klass()->external_name(), base); } else { - assert(DiagnoseSyncOnPrimitiveWrappers == LOG_WARNING, "invalid value for DiagnoseSyncOnPrimitiveWrappers"); + assert(DiagnoseSyncOnValueBasedClasses == LOG_WARNING, "invalid value for DiagnoseSyncOnValueBasedClasses"); ResourceMark rm(self); - Log(primitivewrappers) pwlog; + Log(valuebasedclasses) vblog; - pwlog.info("Synchronizing on object " INTPTR_FORMAT " of klass %s", p2i(obj()), obj->klass()->external_name()); + vblog.info("Synchronizing on object " INTPTR_FORMAT " of klass %s", p2i(obj()), obj->klass()->external_name()); if (self->has_last_Java_frame()) { - LogStream info_stream(pwlog.info()); + LogStream info_stream(vblog.info()); self->print_stack_on(&info_stream); } else { - pwlog.info("Cannot find the last Java frame"); + vblog.info("Cannot find the last Java frame"); } - EventSyncOnPrimitiveWrapper event; + EventSyncOnValueBasedClass event; if (event.should_commit()) { - event.set_boxClass(obj->klass()); + event.set_valueBasedClass(obj->klass()); event.commit(); } } - if (last_frame.is_interpreted_frame()) { + if (bcp_was_adjusted) { last_frame.interpreter_frame_set_bcp(last_frame.interpreter_frame_bcp() + 1); } } @@ -439,8 +445,8 @@ void ObjectSynchronizer::handle_sync_on_primitive_wrapper(Handle obj, Thread* cu // changed. The implementation is extremely sensitive to race condition. Be careful. void ObjectSynchronizer::enter(Handle obj, BasicLock* lock, TRAPS) { - if (DiagnoseSyncOnPrimitiveWrappers != 0 && obj->klass()->is_box()) { - handle_sync_on_primitive_wrapper(obj, THREAD); + if (obj->klass()->is_value_based()) { + handle_sync_on_value_based_class(obj, THREAD); } if (UseBiasedLocking) { @@ -586,8 +592,8 @@ void ObjectSynchronizer::reenter(Handle obj, intx recursions, TRAPS) { // JNI locks on java objects // NOTE: must use heavy weight monitor to handle jni monitor enter void ObjectSynchronizer::jni_enter(Handle obj, TRAPS) { - if (DiagnoseSyncOnPrimitiveWrappers != 0 && obj->klass()->is_box()) { - handle_sync_on_primitive_wrapper(obj, THREAD); + if (obj->klass()->is_value_based()) { + handle_sync_on_value_based_class(obj, THREAD); } // the current locking is from JNI instead of Java code diff --git a/src/hotspot/share/runtime/synchronizer.hpp b/src/hotspot/share/runtime/synchronizer.hpp index 16657c60e6f..35fe8def72a 100644 --- a/src/hotspot/share/runtime/synchronizer.hpp +++ b/src/hotspot/share/runtime/synchronizer.hpp @@ -161,7 +161,7 @@ class ObjectSynchronizer : AllStatic { static size_t get_gvars_size(); static u_char* get_gvars_stw_random_addr(); - static void handle_sync_on_primitive_wrapper(Handle obj, Thread* current); + static void handle_sync_on_value_based_class(Handle obj, Thread* current); }; // ObjectLocker enforces balanced locking and can never throw an diff --git a/src/hotspot/share/utilities/accessFlags.hpp b/src/hotspot/share/utilities/accessFlags.hpp index 9fe8f7acf3e..ab2f6a640af 100644 --- a/src/hotspot/share/utilities/accessFlags.hpp +++ b/src/hotspot/share/utilities/accessFlags.hpp @@ -67,7 +67,7 @@ enum { JVM_ACC_HAS_FINAL_METHOD = 0x01000000, // True if klass has final method JVM_ACC_IS_SHARED_CLASS = 0x02000000, // True if klass is shared JVM_ACC_IS_HIDDEN_CLASS = 0x04000000, // True if klass is hidden - JVM_ACC_IS_BOX_CLASS = 0x08000000, // True if klass is primitive wrapper + JVM_ACC_IS_VALUE_BASED_CLASS = 0x08000000, // True if klass is marked as a ValueBased class // Klass* and Method* flags JVM_ACC_HAS_LOCAL_VARIABLE_TABLE= 0x00200000, @@ -152,7 +152,7 @@ class AccessFlags { bool is_cloneable_fast () const { return (_flags & JVM_ACC_IS_CLONEABLE_FAST ) != 0; } bool is_shared_class () const { return (_flags & JVM_ACC_IS_SHARED_CLASS ) != 0; } bool is_hidden_class () const { return (_flags & JVM_ACC_IS_HIDDEN_CLASS ) != 0; } - bool is_box_class () const { return (_flags & JVM_ACC_IS_BOX_CLASS ) != 0; } + bool is_value_based_class () const { return (_flags & JVM_ACC_IS_VALUE_BASED_CLASS ) != 0; } // Klass* and Method* flags bool has_localvariable_table () const { return (_flags & JVM_ACC_HAS_LOCAL_VARIABLE_TABLE) != 0; } @@ -226,7 +226,7 @@ class AccessFlags { void set_has_miranda_methods() { atomic_set_bits(JVM_ACC_HAS_MIRANDA_METHODS); } void set_is_shared_class() { atomic_set_bits(JVM_ACC_IS_SHARED_CLASS); } void set_is_hidden_class() { atomic_set_bits(JVM_ACC_IS_HIDDEN_CLASS); } - void set_is_box_class() { atomic_set_bits(JVM_ACC_IS_BOX_CLASS); } + void set_is_value_based_class() { atomic_set_bits(JVM_ACC_IS_VALUE_BASED_CLASS); } public: // field flags diff --git a/src/java.base/share/classes/java/lang/Boolean.java b/src/java.base/share/classes/java/lang/Boolean.java index fc76dfe460f..0397277e68f 100644 --- a/src/java.base/share/classes/java/lang/Boolean.java +++ b/src/java.base/share/classes/java/lang/Boolean.java @@ -41,16 +41,23 @@ import static java.lang.constant.ConstantDescs.CD_Boolean; * {@code boolean} in an object. An object of type * {@code Boolean} contains a single field whose type is * {@code boolean}. - *

- * In addition, this class provides many methods for + * + *

In addition, this class provides many methods for * converting a {@code boolean} to a {@code String} and a * {@code String} to a {@code boolean}, as well as other * constants and methods useful when dealing with a * {@code boolean}. * + *

This is a value-based + * class; 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, synchronization may fail. + * * @author Arthur van Hoff * @since 1.0 */ +@jdk.internal.ValueBased public final class Boolean implements java.io.Serializable, Comparable, Constable { @@ -98,7 +105,7 @@ public final class Boolean implements java.io.Serializable, * Also consider using the final fields {@link #TRUE} and {@link #FALSE} * if possible. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Boolean(boolean value) { this.value = value; } @@ -118,7 +125,7 @@ public final class Boolean implements java.io.Serializable, * {@code boolean} primitive, or use {@link #valueOf(String)} * to convert a string to a {@code Boolean} object. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Boolean(String s) { this(parseBoolean(s)); } diff --git a/src/java.base/share/classes/java/lang/Byte.java b/src/java.base/share/classes/java/lang/Byte.java index 9b004bfae6d..c80a6fc9fe0 100644 --- a/src/java.base/share/classes/java/lang/Byte.java +++ b/src/java.base/share/classes/java/lang/Byte.java @@ -48,11 +48,18 @@ import static java.lang.constant.ConstantDescs.DEFAULT_NAME; * byte}, as well as other constants and methods useful when dealing * with a {@code byte}. * + *

This is a value-based + * class; 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, synchronization may fail. + * * @author Nakul Saraiya * @author Joseph D. Darcy * @see java.lang.Number * @since 1.1 */ +@jdk.internal.ValueBased public final class Byte extends Number implements Comparable, Constable { /** @@ -337,7 +344,7 @@ public final class Byte extends Number implements Comparable, Constable { * {@link #valueOf(byte)} is generally a better choice, as it is * likely to yield significantly better space and time performance. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Byte(byte value) { this.value = value; } @@ -360,7 +367,7 @@ public final class Byte extends Number implements Comparable, Constable { * {@code byte} primitive, or use {@link #valueOf(String)} * to convert a string to a {@code Byte} object. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Byte(String s) throws NumberFormatException { this.value = parseByte(s, 10); } diff --git a/src/java.base/share/classes/java/lang/Character.java b/src/java.base/share/classes/java/lang/Character.java index 96238563354..fb2431d045b 100644 --- a/src/java.base/share/classes/java/lang/Character.java +++ b/src/java.base/share/classes/java/lang/Character.java @@ -122,6 +122,12 @@ import static java.lang.constant.ConstantDescs.DEFAULT_NAME; * encoding. For more information on Unicode terminology, refer to the * Unicode Glossary. * + *

This is a value-based + * class; 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, synchronization may fail. + * * @author Lee Boynton * @author Guy Steele * @author Akira Tanaka @@ -129,6 +135,7 @@ import static java.lang.constant.ConstantDescs.DEFAULT_NAME; * @author Ulf Zibis * @since 1.0 */ +@jdk.internal.ValueBased public final class Character implements java.io.Serializable, Comparable, Constable { /** @@ -8501,7 +8508,7 @@ class Character implements java.io.Serializable, Comparable, Constabl * {@link #valueOf(char)} is generally a better choice, as it is * likely to yield significantly better space and time performance. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Character(char value) { this.value = value; } diff --git a/src/java.base/share/classes/java/lang/Double.java b/src/java.base/share/classes/java/lang/Double.java index 729adcf5b03..36e9086e1c4 100644 --- a/src/java.base/share/classes/java/lang/Double.java +++ b/src/java.base/share/classes/java/lang/Double.java @@ -46,11 +46,18 @@ import jdk.internal.vm.annotation.IntrinsicCandidate; * constants and methods useful when dealing with a * {@code double}. * + *

This is a value-based + * class; 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, synchronization may fail. + * * @author Lee Boynton * @author Arthur van Hoff * @author Joseph D. Darcy * @since 1.0 */ +@jdk.internal.ValueBased public final class Double extends Number implements Comparable, Constable, ConstantDesc { /** @@ -605,7 +612,7 @@ public final class Double extends Number * {@link #valueOf(double)} is generally a better choice, as it is * likely to yield significantly better space and time performance. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Double(double value) { this.value = value; } @@ -626,7 +633,7 @@ public final class Double extends Number * {@code double} primitive, or use {@link #valueOf(String)} * to convert a string to a {@code Double} object. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Double(String s) throws NumberFormatException { value = parseDouble(s); } diff --git a/src/java.base/share/classes/java/lang/Float.java b/src/java.base/share/classes/java/lang/Float.java index 0cebe7848ac..478b5a27a7b 100644 --- a/src/java.base/share/classes/java/lang/Float.java +++ b/src/java.base/share/classes/java/lang/Float.java @@ -45,11 +45,18 @@ import jdk.internal.vm.annotation.IntrinsicCandidate; * constants and methods useful when dealing with a * {@code float}. * + *

This is a value-based + * class; 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, synchronization may fail. + * * @author Lee Boynton * @author Arthur van Hoff * @author Joseph D. Darcy * @since 1.0 */ +@jdk.internal.ValueBased public final class Float extends Number implements Comparable, Constable, ConstantDesc { /** @@ -518,7 +525,7 @@ public final class Float extends Number * {@link #valueOf(float)} is generally a better choice, as it is * likely to yield significantly better space and time performance. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Float(float value) { this.value = value; } @@ -534,7 +541,7 @@ public final class Float extends Number * static factory method {@link #valueOf(float)} method as follows: * {@code Float.valueOf((float)value)}. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Float(double value) { this.value = (float)value; } @@ -555,7 +562,7 @@ public final class Float extends Number * {@code float} primitive, or use {@link #valueOf(String)} * to convert a string to a {@code Float} object. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Float(String s) throws NumberFormatException { value = parseFloat(s); } diff --git a/src/java.base/share/classes/java/lang/Integer.java b/src/java.base/share/classes/java/lang/Integer.java index f75be73b307..815ae10e540 100644 --- a/src/java.base/share/classes/java/lang/Integer.java +++ b/src/java.base/share/classes/java/lang/Integer.java @@ -50,6 +50,12 @@ import static java.lang.String.UTF16; * {@code int}, as well as other constants and methods useful when * dealing with an {@code int}. * + *

This is a value-based + * class; 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, synchronization may fail. + * *

Implementation note: The implementations of the "bit twiddling" * methods (such as {@link #highestOneBit(int) highestOneBit} and * {@link #numberOfTrailingZeros(int) numberOfTrailingZeros}) are @@ -62,6 +68,7 @@ import static java.lang.String.UTF16; * @author Joseph D. Darcy * @since 1.0 */ +@jdk.internal.ValueBased public final class Integer extends Number implements Comparable, Constable, ConstantDesc { /** @@ -1085,7 +1092,7 @@ public final class Integer extends Number * {@link #valueOf(int)} is generally a better choice, as it is * likely to yield significantly better space and time performance. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Integer(int value) { this.value = value; } @@ -1107,7 +1114,7 @@ public final class Integer extends Number * {@code int} primitive, or use {@link #valueOf(String)} * to convert a string to an {@code Integer} object. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Integer(String s) throws NumberFormatException { this.value = parseInt(s, 10); } diff --git a/src/java.base/share/classes/java/lang/Long.java b/src/java.base/share/classes/java/lang/Long.java index 15c9ee1cc31..21323e19de6 100644 --- a/src/java.base/share/classes/java/lang/Long.java +++ b/src/java.base/share/classes/java/lang/Long.java @@ -50,6 +50,12 @@ import static java.lang.String.UTF16; * long}, as well as other constants and methods useful when dealing * with a {@code long}. * + *

This is a value-based + * class; 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, synchronization may fail. + * *

Implementation note: The implementations of the "bit twiddling" * methods (such as {@link #highestOneBit(long) highestOneBit} and * {@link #numberOfTrailingZeros(long) numberOfTrailingZeros}) are @@ -62,6 +68,7 @@ import static java.lang.String.UTF16; * @author Joseph D. Darcy * @since 1.0 */ +@jdk.internal.ValueBased public final class Long extends Number implements Comparable, Constable, ConstantDesc { /** @@ -1316,7 +1323,7 @@ public final class Long extends Number * {@link #valueOf(long)} is generally a better choice, as it is * likely to yield significantly better space and time performance. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Long(long value) { this.value = value; } @@ -1339,7 +1346,7 @@ public final class Long extends Number * {@code long} primitive, or use {@link #valueOf(String)} * to convert a string to a {@code Long} object. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Long(String s) throws NumberFormatException { this.value = parseLong(s, 10); } diff --git a/src/java.base/share/classes/java/lang/ProcessHandle.java b/src/java.base/share/classes/java/lang/ProcessHandle.java index 32b34d13414..229f98d0b6f 100644 --- a/src/java.base/share/classes/java/lang/ProcessHandle.java +++ b/src/java.base/share/classes/java/lang/ProcessHandle.java @@ -81,16 +81,17 @@ import java.util.stream.Stream; *

* The {@code ProcessHandle} static factory methods return instances that are * value-based, - * immutable and thread-safe. - * Use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on these instances of - * {@code ProcessHandle} may have unpredictable results and should be avoided. - * Use {@link #equals(Object) equals} or - * {@link #compareTo(ProcessHandle) compareTo} methods to compare ProcessHandles. + * immutable and thread-safe. 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, synchronization may fail. + * Use the {@code equals} or {@link #compareTo(ProcessHandle) compareTo} methods + * to compare ProcessHandles. * * @see Process * @since 9 */ +@jdk.internal.ValueBased public interface ProcessHandle extends Comparable { /** diff --git a/src/java.base/share/classes/java/lang/ProcessHandleImpl.java b/src/java.base/share/classes/java/lang/ProcessHandleImpl.java index f51720d99f5..92634d9b120 100644 --- a/src/java.base/share/classes/java/lang/ProcessHandleImpl.java +++ b/src/java.base/share/classes/java/lang/ProcessHandleImpl.java @@ -48,6 +48,7 @@ import static java.security.AccessController.doPrivileged; * @see Process * @since 9 */ +@jdk.internal.ValueBased final class ProcessHandleImpl implements ProcessHandle { /** * Default size of stack for reaper processes. diff --git a/src/java.base/share/classes/java/lang/Runtime.java b/src/java.base/share/classes/java/lang/Runtime.java index dae8be95ce6..2519eb88ca6 100644 --- a/src/java.base/share/classes/java/lang/Runtime.java +++ b/src/java.base/share/classes/java/lang/Runtime.java @@ -941,14 +941,15 @@ public class Runtime { * $VNUM(-$PRE)? * * - *

This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code Version} may have unpredictable results and should be avoided. - *

+ *

This is a value-based + * class; 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, synchronization may fail.

* * @since 9 */ + @jdk.internal.ValueBased public static final class Version implements Comparable { diff --git a/src/java.base/share/classes/java/lang/Short.java b/src/java.base/share/classes/java/lang/Short.java index 4f293496579..00d4a91d37b 100644 --- a/src/java.base/share/classes/java/lang/Short.java +++ b/src/java.base/share/classes/java/lang/Short.java @@ -47,11 +47,18 @@ import static java.lang.constant.ConstantDescs.DEFAULT_NAME; * {@code short}, as well as other constants and methods useful when * dealing with a {@code short}. * + *

This is a value-based + * class; 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, synchronization may fail. + * * @author Nakul Saraiya * @author Joseph D. Darcy * @see java.lang.Number * @since 1.1 */ +@jdk.internal.ValueBased public final class Short extends Number implements Comparable, Constable { /** @@ -342,7 +349,7 @@ public final class Short extends Number implements Comparable, Constable * {@link #valueOf(short)} is generally a better choice, as it is * likely to yield significantly better space and time performance. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Short(short value) { this.value = value; } @@ -365,7 +372,7 @@ public final class Short extends Number implements Comparable, Constable * {@code short} primitive, or use {@link #valueOf(String)} * to convert a string to a {@code Short} object. */ - @Deprecated(since="9") + @Deprecated(since="9", forRemoval = true) public Short(String s) throws NumberFormatException { this.value = parseShort(s, 10); } diff --git a/src/java.base/share/classes/java/lang/constant/ConstantDesc.java b/src/java.base/share/classes/java/lang/constant/ConstantDesc.java index 5ffb5b2aded..9d410e6774b 100644 --- a/src/java.base/share/classes/java/lang/constant/ConstantDesc.java +++ b/src/java.base/share/classes/java/lang/constant/ConstantDesc.java @@ -57,8 +57,8 @@ import java.lang.invoke.VarHandle.VarHandleDesc; *

Constants describing various common constants (such as {@link ClassDesc} * instances for platform types) can be found in {@link ConstantDescs}. * - *

Implementations of {@linkplain ConstantDesc} must be - * value-based classes. + *

Implementations of {@linkplain ConstantDesc} should be immutable + * and their behavior should not rely on object identity. * *

Non-platform classes should not implement {@linkplain ConstantDesc} directly. * Instead, they should extend {@link DynamicConstantDesc} (as {@link EnumDesc} diff --git a/src/java.base/share/classes/java/lang/constant/DynamicCallSiteDesc.java b/src/java.base/share/classes/java/lang/constant/DynamicCallSiteDesc.java index f336bc5fb4e..b57f73c0fb9 100644 --- a/src/java.base/share/classes/java/lang/constant/DynamicCallSiteDesc.java +++ b/src/java.base/share/classes/java/lang/constant/DynamicCallSiteDesc.java @@ -41,8 +41,8 @@ import static java.util.stream.Collectors.joining; * A nominal descriptor for an * {@code invokedynamic} call site. * - *

Concrete subtypes of {@linkplain DynamicCallSiteDesc} must be - * value-based. + *

Concrete subtypes of {@linkplain DynamicCallSiteDesc} should be immutable + * and their behavior should not rely on object identity. * * @since 12 */ diff --git a/src/java.base/share/classes/java/lang/constant/DynamicConstantDesc.java b/src/java.base/share/classes/java/lang/constant/DynamicConstantDesc.java index 2ff7a0ebf89..4efd3dde3c3 100644 --- a/src/java.base/share/classes/java/lang/constant/DynamicConstantDesc.java +++ b/src/java.base/share/classes/java/lang/constant/DynamicConstantDesc.java @@ -49,8 +49,8 @@ import static java.util.stream.Collectors.joining; * dynamic constant (one described in the constant pool with * {@code Constant_Dynamic_info}.) * - *

Concrete subtypes of {@linkplain DynamicConstantDesc} must be - * value-based. + *

Concrete subtypes of {@linkplain DynamicConstantDesc} should be immutable + * and their behavior should not rely on object identity. * * @param the type of the dynamic constant * diff --git a/src/java.base/share/classes/java/lang/doc-files/ValueBased.html b/src/java.base/share/classes/java/lang/doc-files/ValueBased.html index fe720a37be7..86d9b359557 100644 --- a/src/java.base/share/classes/java/lang/doc-files/ValueBased.html +++ b/src/java.base/share/classes/java/lang/doc-files/ValueBased.html @@ -31,36 +31,43 @@

{@index "Value-based Classes"}

-Some classes, such as java.util.Optional and -java.time.LocalDateTime, are value-based. Instances of a -value-based class: +Some classes, such as java.lang.Integer and +java.time.LocalDate, are value-based. +A value-based class has the following properties:
    -
  • are final and immutable (though may contain references to mutable - objects);
  • -
  • have implementations of equals, - hashCode, and toString which are computed - solely from the instance's state and not from its identity or the state - of any other object or variable;
  • -
  • make no use of identity-sensitive operations such as reference - equality (==) between instances, identity hash code of - instances, or synchronization on an instances's intrinsic lock;
  • -
  • are considered equal solely based on equals(), not - based on reference equality (==);
  • -
  • do not have accessible constructors, but are instead instantiated - through factory methods which make no commitment as to the identity - of returned instances;
  • -
  • are freely substitutable when equal, meaning that interchanging - any two instances x and y that are equal - according to equals() in any computation or method - invocation should produce no visible change in behavior. -
  • +
  • the class declares only final instance fields (though these may contain references + to mutable objects);
  • +
  • the class's implementations of equals, hashCode, + and toString compute their results solely from the values + of the class's instance fields (and the members of the objects they + reference), not from the instance's identity;
  • +
  • the class's methods treat instances as freely substitutable + when equal, meaning that interchanging any two instances x and + y that are equal according to equals() produces no + visible change in the behavior of the class's methods;
  • +
  • the class performs no synchronization using an instance's monitor;
  • +
  • the class does not declare (or has deprecated any) accessible constructors;
  • +
  • the class does not provide any instance creation mechanism that promises + a unique identity on each method call—in particular, any factory + method's contract must allow for the possibility that if two independently-produced + instances are equal according to equals(), they may also be + equal according to ==;
  • +
  • the class is final, and extends either Object or a hierarchy of + abstract classes that declare no instance fields or instance initializers + and whose constructors are empty.
-

A program may produce unpredictable results if it attempts to distinguish two - references to equal values of a value-based class, whether directly via reference +

When two instances of a value-based class are equal (according to `equals`), a program + should not attempt to distinguish between their identities, whether directly via reference equality or indirectly via an appeal to synchronization, identity hashing, - serialization, or any other identity-sensitive mechanism. Use of such - identity-sensitive operations on instances of value-based classes may have - unpredictable effects and should be avoided.

+ serialization, or any other identity-sensitive mechanism.

+ +

Synchronization on instances of value-based classes is strongly discouraged, + because the programmer cannot guarantee exclusive ownership of the + associated monitor.

+ +

Identity-related behavior of value-based classes may change in a future release. + For example, synchronization may fail.

+ diff --git a/src/java.base/share/classes/java/lang/invoke/MemberName.java b/src/java.base/share/classes/java/lang/invoke/MemberName.java index 019f07a788c..c11e2837c3b 100644 --- a/src/java.base/share/classes/java/lang/invoke/MemberName.java +++ b/src/java.base/share/classes/java/lang/invoke/MemberName.java @@ -773,7 +773,7 @@ final class MemberName implements Member, Cloneable { } @Override - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal"}) public int hashCode() { // Avoid autoboxing getReferenceKind(), since this is used early and will force // early initialization of Byte$ByteCache diff --git a/src/java.base/share/classes/java/time/Duration.java b/src/java.base/share/classes/java/time/Duration.java index 437a63e823c..ea3a382c69d 100644 --- a/src/java.base/share/classes/java/time/Duration.java +++ b/src/java.base/share/classes/java/time/Duration.java @@ -117,12 +117,12 @@ import java.util.regex.Pattern; * This difference only impacts durations measured near a leap-second and should not affect * most applications. * See {@link Instant} for a discussion as to the meaning of the second and time-scales. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code Duration} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -130,6 +130,7 @@ import java.util.regex.Pattern; * * @since 1.8 */ +@jdk.internal.ValueBased public final class Duration implements TemporalAmount, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/Instant.java b/src/java.base/share/classes/java/time/Instant.java index 80b7ea9fb25..b129b9c90d5 100644 --- a/src/java.base/share/classes/java/time/Instant.java +++ b/src/java.base/share/classes/java/time/Instant.java @@ -191,12 +191,12 @@ import java.util.Objects; * The Java time-scale is used for all date-time classes. * This includes {@code Instant}, {@code LocalDate}, {@code LocalTime}, {@code OffsetDateTime}, * {@code ZonedDateTime} and {@code Duration}. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code Instant} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -204,6 +204,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class Instant implements Temporal, TemporalAdjuster, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/LocalDate.java b/src/java.base/share/classes/java/time/LocalDate.java index be7d47d65e2..2ed2f98544c 100644 --- a/src/java.base/share/classes/java/time/LocalDate.java +++ b/src/java.base/share/classes/java/time/LocalDate.java @@ -123,12 +123,12 @@ import java.util.stream.Stream; * For most applications written today, the ISO-8601 rules are entirely suitable. * However, any application that makes use of historical dates, and requires them * to be accurate will find the ISO-8601 approach unsuitable. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code LocalDate} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -136,6 +136,7 @@ import java.util.stream.Stream; * * @since 1.8 */ +@jdk.internal.ValueBased public final class LocalDate implements Temporal, TemporalAdjuster, ChronoLocalDate, Serializable { diff --git a/src/java.base/share/classes/java/time/LocalDateTime.java b/src/java.base/share/classes/java/time/LocalDateTime.java index 5838f45e7bc..0f6022bb659 100644 --- a/src/java.base/share/classes/java/time/LocalDateTime.java +++ b/src/java.base/share/classes/java/time/LocalDateTime.java @@ -119,12 +119,12 @@ import java.util.Objects; * For most applications written today, the ISO-8601 rules are entirely suitable. * However, any application that makes use of historical dates, and requires them * to be accurate will find the ISO-8601 approach unsuitable. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code LocalDateTime} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -132,6 +132,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class LocalDateTime implements Temporal, TemporalAdjuster, ChronoLocalDateTime, Serializable { diff --git a/src/java.base/share/classes/java/time/LocalTime.java b/src/java.base/share/classes/java/time/LocalTime.java index 4fef814b0ca..a8602b681ca 100644 --- a/src/java.base/share/classes/java/time/LocalTime.java +++ b/src/java.base/share/classes/java/time/LocalTime.java @@ -109,12 +109,12 @@ import java.util.Objects; * The ISO-8601 calendar system is the modern civil calendar system used today * in most of the world. This API assumes that all calendar systems use the same * representation, this class, for time-of-day. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code LocalTime} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -122,6 +122,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class LocalTime implements Temporal, TemporalAdjuster, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/MonthDay.java b/src/java.base/share/classes/java/time/MonthDay.java index 1fe61401013..e126f2dea75 100644 --- a/src/java.base/share/classes/java/time/MonthDay.java +++ b/src/java.base/share/classes/java/time/MonthDay.java @@ -111,12 +111,12 @@ import java.util.Objects; * For most applications written today, the ISO-8601 rules are entirely suitable. * However, any application that makes use of historical dates, and requires them * to be accurate will find the ISO-8601 approach unsuitable. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code MonthDay} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -124,6 +124,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class MonthDay implements TemporalAccessor, TemporalAdjuster, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/OffsetDateTime.java b/src/java.base/share/classes/java/time/OffsetDateTime.java index 1d6f2c48984..16d47f70de2 100644 --- a/src/java.base/share/classes/java/time/OffsetDateTime.java +++ b/src/java.base/share/classes/java/time/OffsetDateTime.java @@ -112,12 +112,12 @@ import java.util.Objects; * It is intended that {@code ZonedDateTime} or {@code Instant} is used to model data * in simpler applications. This class may be used when modeling date-time concepts in * more detail, or when communicating to a database or in a network protocol. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code OffsetDateTime} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -125,6 +125,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class OffsetDateTime implements Temporal, TemporalAdjuster, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/OffsetTime.java b/src/java.base/share/classes/java/time/OffsetTime.java index 290a124366d..89ea67e4442 100644 --- a/src/java.base/share/classes/java/time/OffsetTime.java +++ b/src/java.base/share/classes/java/time/OffsetTime.java @@ -102,12 +102,12 @@ import java.util.Objects; * as well as a zone offset. * For example, the value "13:45:30.123456789+02:00" can be stored * in an {@code OffsetTime}. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code OffsetTime} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -115,6 +115,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class OffsetTime implements Temporal, TemporalAdjuster, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/Period.java b/src/java.base/share/classes/java/time/Period.java index b5e1ce5e571..ba68c5ac9c1 100644 --- a/src/java.base/share/classes/java/time/Period.java +++ b/src/java.base/share/classes/java/time/Period.java @@ -117,12 +117,12 @@ import java.util.regex.Pattern; *

* The period is modeled as a directed amount of time, meaning that individual parts of the * period may be negative. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code Period} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -130,6 +130,7 @@ import java.util.regex.Pattern; * * @since 1.8 */ +@jdk.internal.ValueBased public final class Period implements ChronoPeriod, Serializable { diff --git a/src/java.base/share/classes/java/time/Year.java b/src/java.base/share/classes/java/time/Year.java index 37fd428e7cc..88096a39605 100644 --- a/src/java.base/share/classes/java/time/Year.java +++ b/src/java.base/share/classes/java/time/Year.java @@ -119,12 +119,12 @@ import java.util.Objects; * For most applications written today, the ISO-8601 rules are entirely suitable. * However, any application that makes use of historical dates, and requires them * to be accurate will find the ISO-8601 approach unsuitable. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code Year} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -132,6 +132,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class Year implements Temporal, TemporalAdjuster, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/YearMonth.java b/src/java.base/share/classes/java/time/YearMonth.java index f65c9d9b9af..d4874bd90a1 100644 --- a/src/java.base/share/classes/java/time/YearMonth.java +++ b/src/java.base/share/classes/java/time/YearMonth.java @@ -115,12 +115,12 @@ import java.util.Objects; * For most applications written today, the ISO-8601 rules are entirely suitable. * However, any application that makes use of historical dates, and requires them * to be accurate will find the ISO-8601 approach unsuitable. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code YearMonth} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -128,6 +128,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class YearMonth implements Temporal, TemporalAdjuster, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/ZoneId.java b/src/java.base/share/classes/java/time/ZoneId.java index aaec5cdc1a7..cfb3e570cb7 100644 --- a/src/java.base/share/classes/java/time/ZoneId.java +++ b/src/java.base/share/classes/java/time/ZoneId.java @@ -160,12 +160,12 @@ import static java.util.Map.entry; * However, any call to {@code getRules} will fail with {@code ZoneRulesException}. * This approach is designed to allow a {@link ZonedDateTime} to be loaded and * queried, but not modified, on a Java Runtime with incomplete time-zone information. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code ZoneId} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -175,6 +175,7 @@ import static java.util.Map.entry; * * @since 1.8 */ +@jdk.internal.ValueBased public abstract class ZoneId implements Serializable { /** diff --git a/src/java.base/share/classes/java/time/ZoneOffset.java b/src/java.base/share/classes/java/time/ZoneOffset.java index 172ca768dda..f061040b9e2 100644 --- a/src/java.base/share/classes/java/time/ZoneOffset.java +++ b/src/java.base/share/classes/java/time/ZoneOffset.java @@ -114,12 +114,12 @@ import java.util.concurrent.ConcurrentMap; * Instances of {@code ZoneOffset} must be compared using {@link #equals}. * Implementations may choose to cache certain common offsets, however * applications must not rely on such caching. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code ZoneOffset} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -127,6 +127,7 @@ import java.util.concurrent.ConcurrentMap; * * @since 1.8 */ +@jdk.internal.ValueBased public final class ZoneOffset extends ZoneId implements TemporalAccessor, TemporalAdjuster, Comparable, Serializable { diff --git a/src/java.base/share/classes/java/time/ZonedDateTime.java b/src/java.base/share/classes/java/time/ZonedDateTime.java index a6a33c6ef69..e6bc1d256f0 100644 --- a/src/java.base/share/classes/java/time/ZonedDateTime.java +++ b/src/java.base/share/classes/java/time/ZonedDateTime.java @@ -142,12 +142,12 @@ import java.util.Objects; * of a {@code LocalDateTime} and a {@code ZoneId}. The {@code ZoneOffset} is * a vital, but secondary, piece of information, used to ensure that the class * represents an instant, especially during a daylight savings overlap. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code ZonedDateTime} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -161,6 +161,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class ZonedDateTime implements Temporal, ChronoZonedDateTime, Serializable { diff --git a/src/java.base/share/classes/java/time/chrono/HijrahDate.java b/src/java.base/share/classes/java/time/chrono/HijrahDate.java index 97f47a2606f..9f0b6d7a573 100644 --- a/src/java.base/share/classes/java/time/chrono/HijrahDate.java +++ b/src/java.base/share/classes/java/time/chrono/HijrahDate.java @@ -103,12 +103,12 @@ import java.time.temporal.ValueRange; * to create new HijrahDate instances. * Alternatively, the {@link #withVariant} method can be used to convert * to a new HijrahChronology. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code HijrahDate} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -116,6 +116,7 @@ import java.time.temporal.ValueRange; * * @since 1.8 */ +@jdk.internal.ValueBased public final class HijrahDate extends ChronoLocalDateImpl implements ChronoLocalDate, Serializable { diff --git a/src/java.base/share/classes/java/time/chrono/JapaneseDate.java b/src/java.base/share/classes/java/time/chrono/JapaneseDate.java index 096bdaed393..8fd05d93aa9 100644 --- a/src/java.base/share/classes/java/time/chrono/JapaneseDate.java +++ b/src/java.base/share/classes/java/time/chrono/JapaneseDate.java @@ -110,12 +110,12 @@ import sun.util.calendar.LocalGregorianCalendar; * Calling {@code japaneseDate.get(YEAR)} will return 2012.
* Calling {@code japaneseDate.get(ERA)} will return 2, corresponding to * {@code JapaneseChronology.ERA_HEISEI}.
- * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code JapaneseDate} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -123,6 +123,7 @@ import sun.util.calendar.LocalGregorianCalendar; * * @since 1.8 */ +@jdk.internal.ValueBased public final class JapaneseDate extends ChronoLocalDateImpl implements ChronoLocalDate, Serializable { @@ -140,11 +141,11 @@ public final class JapaneseDate /** * The JapaneseEra of this date. */ - private transient JapaneseEra era; + private final transient JapaneseEra era; /** * The Japanese imperial calendar year of this date. */ - private transient int yearOfEra; + private final transient int yearOfEra; /** * The first day supported by the JapaneseChronology is Meiji 6, January 1st. diff --git a/src/java.base/share/classes/java/time/chrono/MinguoDate.java b/src/java.base/share/classes/java/time/chrono/MinguoDate.java index 2b957230172..6f3170177a5 100644 --- a/src/java.base/share/classes/java/time/chrono/MinguoDate.java +++ b/src/java.base/share/classes/java/time/chrono/MinguoDate.java @@ -90,12 +90,12 @@ import java.util.Objects; * This date operates using the {@linkplain MinguoChronology Minguo calendar}. * This calendar system is primarily used in the Republic of China, often known as Taiwan. * Dates are aligned such that {@code 0001-01-01 (Minguo)} is {@code 1912-01-01 (ISO)}. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code MinguoDate} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -103,6 +103,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class MinguoDate extends ChronoLocalDateImpl implements ChronoLocalDate, Serializable { diff --git a/src/java.base/share/classes/java/time/chrono/ThaiBuddhistChronology.java b/src/java.base/share/classes/java/time/chrono/ThaiBuddhistChronology.java index ea9218f6cd7..28e3282c4ea 100644 --- a/src/java.base/share/classes/java/time/chrono/ThaiBuddhistChronology.java +++ b/src/java.base/share/classes/java/time/chrono/ThaiBuddhistChronology.java @@ -72,7 +72,6 @@ import java.time.temporal.ChronoField; import java.time.temporal.TemporalAccessor; import java.time.temporal.TemporalField; import java.time.temporal.ValueRange; -import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.Locale; diff --git a/src/java.base/share/classes/java/time/chrono/ThaiBuddhistDate.java b/src/java.base/share/classes/java/time/chrono/ThaiBuddhistDate.java index 58894d76346..865d2731abf 100644 --- a/src/java.base/share/classes/java/time/chrono/ThaiBuddhistDate.java +++ b/src/java.base/share/classes/java/time/chrono/ThaiBuddhistDate.java @@ -90,12 +90,12 @@ import java.util.Objects; * This date operates using the {@linkplain ThaiBuddhistChronology Thai Buddhist calendar}. * This calendar system is primarily used in Thailand. * Dates are aligned such that {@code 2484-01-01 (Buddhist)} is {@code 1941-01-01 (ISO)}. - * *

* This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code ThaiBuddhistDate} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * * @implSpec @@ -103,6 +103,7 @@ import java.util.Objects; * * @since 1.8 */ +@jdk.internal.ValueBased public final class ThaiBuddhistDate extends ChronoLocalDateImpl implements ChronoLocalDate, Serializable { diff --git a/src/java.base/share/classes/java/time/format/TextStyle.java b/src/java.base/share/classes/java/time/format/TextStyle.java index 27a6f91ad6b..ec49da271ad 100644 --- a/src/java.base/share/classes/java/time/format/TextStyle.java +++ b/src/java.base/share/classes/java/time/format/TextStyle.java @@ -81,7 +81,7 @@ import java.util.Calendar; * to the word used for month in association with a day and year in a date. * * @implSpec - * This is immutable and thread-safe enum. + * This is an immutable and thread-safe enum. * * @since 1.8 */ diff --git a/src/java.base/share/classes/java/util/ImmutableCollections.java b/src/java.base/share/classes/java/util/ImmutableCollections.java index 4c651250f2e..52a83552f14 100644 --- a/src/java.base/share/classes/java/util/ImmutableCollections.java +++ b/src/java.base/share/classes/java/util/ImmutableCollections.java @@ -141,6 +141,7 @@ class ImmutableCollections { static UnsupportedOperationException uoe() { return new UnsupportedOperationException(); } + @jdk.internal.ValueBased static abstract class AbstractImmutableCollection extends AbstractCollection { // all mutating methods throw UnsupportedOperationException @Override public boolean add(E e) { throw uoe(); } @@ -247,6 +248,7 @@ class ImmutableCollections { // ---------- List Implementations ---------- + @jdk.internal.ValueBased static abstract class AbstractImmutableList extends AbstractImmutableCollection implements List, RandomAccess { @@ -540,6 +542,7 @@ class ImmutableCollections { } } + @jdk.internal.ValueBased static final class List12 extends AbstractImmutableList implements Serializable { @@ -646,6 +649,7 @@ class ImmutableCollections { } } + @jdk.internal.ValueBased static final class ListN extends AbstractImmutableList implements Serializable { @@ -737,6 +741,7 @@ class ImmutableCollections { // ---------- Set Implementations ---------- + @jdk.internal.ValueBased static abstract class AbstractImmutableSet extends AbstractImmutableCollection implements Set { @@ -764,6 +769,7 @@ class ImmutableCollections { public abstract int hashCode(); } + @jdk.internal.ValueBased static final class Set12 extends AbstractImmutableSet implements Serializable { @@ -889,6 +895,7 @@ class ImmutableCollections { * least one null is always present. * @param the element type */ + @jdk.internal.ValueBased static final class SetN extends AbstractImmutableSet implements Serializable { @@ -1055,6 +1062,7 @@ class ImmutableCollections { // ---------- Map Implementations ---------- + @jdk.internal.ValueBased abstract static class AbstractImmutableMap extends AbstractMap implements Serializable { @Override public void clear() { throw uoe(); } @Override public V compute(K key, BiFunction rf) { throw uoe(); } @@ -1085,6 +1093,7 @@ class ImmutableCollections { } } + @jdk.internal.ValueBased static final class Map1 extends AbstractImmutableMap { @Stable private final K k0; @@ -1151,6 +1160,7 @@ class ImmutableCollections { * @param the key type * @param the value type */ + @jdk.internal.ValueBased static final class MapN extends AbstractImmutableMap { @Stable diff --git a/src/java.base/share/classes/java/util/KeyValueHolder.java b/src/java.base/share/classes/java/util/KeyValueHolder.java index 3b7250ee6aa..2333257f59b 100644 --- a/src/java.base/share/classes/java/util/KeyValueHolder.java +++ b/src/java.base/share/classes/java/util/KeyValueHolder.java @@ -31,10 +31,11 @@ import jdk.internal.vm.annotation.Stable; * An immutable container for a key and a value, suitable for use * in creating and populating {@code Map} instances. * - *

This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code KeyValueHolder} may have unpredictable results and should be avoided. + *

This is a value-based + * class; 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, synchronization may fail. * * @apiNote * This class is not public. Instances can be created using the @@ -49,6 +50,7 @@ import jdk.internal.vm.annotation.Stable; * @see Map#ofEntries Map.ofEntries() * @since 9 */ +@jdk.internal.ValueBased final class KeyValueHolder implements Map.Entry { @Stable final K key; diff --git a/src/java.base/share/classes/java/util/List.java b/src/java.base/share/classes/java/util/List.java index 314fb6af6c1..285aa8b536d 100644 --- a/src/java.base/share/classes/java/util/List.java +++ b/src/java.base/share/classes/java/util/List.java @@ -107,10 +107,12 @@ import java.util.function.UnaryOperator; *

  • The lists and their {@link #subList(int, int) subList} views implement the * {@link RandomAccess} interface. *
  • They are value-based. - * Callers should make no assumptions about the identity of the returned instances. - * Factories are free to create new instances or reuse existing ones. Therefore, - * identity-sensitive operations on these instances (reference equality ({@code ==}), - * identity hash code, and synchronization) are unreliable and should be avoided. + * Programmers should treat instances that are {@linkplain #equals(Object) equal} + * as interchangeable and should not use them for synchronization, or + * unpredictable behavior may occur. For example, in a future release, + * synchronization may fail. Callers should make no assumptions about the + * identity of the returned instances. Factories are free to + * create new instances or reuse existing ones. *
  • They are serialized as specified on the * Serialized Form * page. diff --git a/src/java.base/share/classes/java/util/Map.java b/src/java.base/share/classes/java/util/Map.java index b1de34b42a5..e1ec0e4282f 100644 --- a/src/java.base/share/classes/java/util/Map.java +++ b/src/java.base/share/classes/java/util/Map.java @@ -131,10 +131,12 @@ import java.io.Serializable; * passed to a static factory method result in {@code IllegalArgumentException}. *
  • The iteration order of mappings is unspecified and is subject to change. *
  • They are value-based. - * Callers should make no assumptions about the identity of the returned instances. - * Factories are free to create new instances or reuse existing ones. Therefore, - * identity-sensitive operations on these instances (reference equality ({@code ==}), - * identity hash code, and synchronization) are unreliable and should be avoided. + * Programmers should treat instances that are {@linkplain #equals(Object) equal} + * as interchangeable and should not use them for synchronization, or + * unpredictable behavior may occur. For example, in a future release, + * synchronization may fail. Callers should make no assumptions + * about the identity of the returned instances. Factories are free to + * create new instances or reuse existing ones. *
  • They are serialized as specified on the * Serialized Form * page. @@ -1636,10 +1638,12 @@ public interface Map { * on a returned {@code Entry} result in {@code UnsupportedOperationException}. *
  • They are not serializable. *
  • They are value-based. - * Callers should make no assumptions about the identity of the returned instances. - * This method is free to create new instances or reuse existing ones. Therefore, - * identity-sensitive operations on these instances (reference equality ({@code ==}), - * identity hash code, and synchronization) are unreliable and should be avoided. + * Programmers should treat instances that are {@linkplain #equals(Object) equal} + * as interchangeable and should not use them for synchronization, or + * unpredictable behavior may occur. For example, in a future release, + * synchronization may fail. Callers should make no assumptions + * about the identity of the returned instances. This method is free to + * create new instances or reuse existing ones. * * * @apiNote diff --git a/src/java.base/share/classes/java/util/Optional.java b/src/java.base/share/classes/java/util/Optional.java index af8ea6d1f98..efb5bac4dce 100644 --- a/src/java.base/share/classes/java/util/Optional.java +++ b/src/java.base/share/classes/java/util/Optional.java @@ -42,10 +42,11 @@ import java.util.stream.Stream; * {@link #ifPresent(Consumer) ifPresent()} (performs an * action if a value is present). * - *

    This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code Optional} may have unpredictable results and should be avoided. + *

    This is a value-based + * class; 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, synchronization may fail. * * @apiNote * {@code Optional} is primarily intended for use as a method return type where @@ -57,6 +58,7 @@ import java.util.stream.Stream; * @param the type of value * @since 1.8 */ +@jdk.internal.ValueBased public final class Optional { /** * Common instance for {@code empty()}. diff --git a/src/java.base/share/classes/java/util/OptionalDouble.java b/src/java.base/share/classes/java/util/OptionalDouble.java index 7c8ed649cbe..4e645c1aa79 100644 --- a/src/java.base/share/classes/java/util/OptionalDouble.java +++ b/src/java.base/share/classes/java/util/OptionalDouble.java @@ -41,10 +41,11 @@ import java.util.stream.DoubleStream; * {@link #ifPresent(DoubleConsumer) ifPresent()} (performs * an action if a value is present). * - *

    This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code OptionalDouble} may have unpredictable results and should be avoided. + *

    This is a value-based + * class; 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, synchronization may fail. * * @apiNote * {@code OptionalDouble} is primarily intended for use as a method return type where @@ -54,6 +55,7 @@ import java.util.stream.DoubleStream; * * @since 1.8 */ +@jdk.internal.ValueBased public final class OptionalDouble { /** * Common instance for {@code empty()}. diff --git a/src/java.base/share/classes/java/util/OptionalInt.java b/src/java.base/share/classes/java/util/OptionalInt.java index aa8a669d010..457273ca4ec 100644 --- a/src/java.base/share/classes/java/util/OptionalInt.java +++ b/src/java.base/share/classes/java/util/OptionalInt.java @@ -41,10 +41,11 @@ import java.util.stream.IntStream; * {@link #ifPresent(IntConsumer) ifPresent()} (performs an * action if a value is present). * - *

    This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code OptionalInt} may have unpredictable results and should be avoided. + *

    This is a value-based + * class; 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, synchronization may fail. * * @apiNote * {@code OptionalInt} is primarily intended for use as a method return type where @@ -54,6 +55,7 @@ import java.util.stream.IntStream; * * @since 1.8 */ +@jdk.internal.ValueBased public final class OptionalInt { /** * Common instance for {@code empty()}. diff --git a/src/java.base/share/classes/java/util/OptionalLong.java b/src/java.base/share/classes/java/util/OptionalLong.java index c18457358fe..817b4b6f058 100644 --- a/src/java.base/share/classes/java/util/OptionalLong.java +++ b/src/java.base/share/classes/java/util/OptionalLong.java @@ -41,10 +41,11 @@ import java.util.stream.LongStream; * {@link #ifPresent(LongConsumer) ifPresent()} (performs an * action if a value is present). * - *

    This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code OptionalLong} may have unpredictable results and should be avoided. + *

    This is a value-based + * class; 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, synchronization may fail. * * @apiNote * {@code OptionalLong} is primarily intended for use as a method return type where @@ -54,6 +55,7 @@ import java.util.stream.LongStream; * * @since 1.8 */ +@jdk.internal.ValueBased public final class OptionalLong { /** * Common instance for {@code empty()}. diff --git a/src/java.base/share/classes/java/util/Set.java b/src/java.base/share/classes/java/util/Set.java index b8109d0b4fb..4c703008a87 100644 --- a/src/java.base/share/classes/java/util/Set.java +++ b/src/java.base/share/classes/java/util/Set.java @@ -82,10 +82,12 @@ package java.util; * passed to a static factory method result in {@code IllegalArgumentException}. *

  • The iteration order of set elements is unspecified and is subject to change. *
  • They are value-based. - * Callers should make no assumptions about the identity of the returned instances. - * Factories are free to create new instances or reuse existing ones. Therefore, - * identity-sensitive operations on these instances (reference equality ({@code ==}), - * identity hash code, and synchronization) are unreliable and should be avoided. + * Programmers should treat instances that are {@linkplain #equals(Object) equal} + * as interchangeable and should not use them for synchronization, or + * unpredictable behavior may occur. For example, in a future release, + * synchronization may fail. Callers should make no assumptions + * about the identity of the returned instances. Factories are free to + * create new instances or reuse existing ones. *
  • They are serialized as specified on the * Serialized Form * page. diff --git a/src/java.base/share/classes/jdk/internal/ValueBased.java b/src/java.base/share/classes/jdk/internal/ValueBased.java new file mode 100644 index 00000000000..587f159854b --- /dev/null +++ b/src/java.base/share/classes/jdk/internal/ValueBased.java @@ -0,0 +1,44 @@ +/* + * 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; + +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +import static java.lang.annotation.ElementType.TYPE; + +/** + * Indicates the API declaration in question is associated with a Value Based class. + * References to value-based classes + * should produce warnings about behavior that is inconsistent with value based semantics. + * + * @since 16 + */ +@Retention(RetentionPolicy.RUNTIME) +@Target(value={TYPE}) +public @interface ValueBased { +} + diff --git a/src/java.management.rmi/share/classes/javax/management/remote/rmi/RMIConnectionImpl_Stub.java b/src/java.management.rmi/share/classes/javax/management/remote/rmi/RMIConnectionImpl_Stub.java index 05e19235e03..8cee1e5c5f1 100644 --- a/src/java.management.rmi/share/classes/javax/management/remote/rmi/RMIConnectionImpl_Stub.java +++ b/src/java.management.rmi/share/classes/javax/management/remote/rmi/RMIConnectionImpl_Stub.java @@ -258,7 +258,10 @@ public final class RMIConnectionImpl_Stub public javax.management.remote.NotificationResult fetchNotifications(long $param_long_1, int $param_int_2, long $param_long_3) throws java.io.IOException { try { - Object $result = ref.invoke(this, $method_fetchNotifications_7, new java.lang.Object[]{new java.lang.Long($param_long_1), new java.lang.Integer($param_int_2), new java.lang.Long($param_long_3)}, -5037523307973544478L); + Object $result = ref.invoke(this, $method_fetchNotifications_7, + new java.lang.Object[]{$param_long_1, + $param_int_2, + $param_long_3}, -5037523307973544478L); return ((javax.management.remote.NotificationResult) $result); } catch (java.lang.RuntimeException e) { throw e; diff --git a/src/java.xml/share/classes/com/sun/org/apache/xerces/internal/impl/xs/traversers/XSDHandler.java b/src/java.xml/share/classes/com/sun/org/apache/xerces/internal/impl/xs/traversers/XSDHandler.java index 542c2d10417..2faf06ea2da 100644 --- a/src/java.xml/share/classes/com/sun/org/apache/xerces/internal/impl/xs/traversers/XSDHandler.java +++ b/src/java.xml/share/classes/com/sun/org/apache/xerces/internal/impl/xs/traversers/XSDHandler.java @@ -4013,7 +4013,8 @@ public class XSDHandler { ","+oldName:currSchema.fTargetNamespace+","+oldName; int attGroupRefsCount = changeRedefineGroup(processedBaseName, componentType, newName, child, currSchema); if (attGroupRefsCount > 1) { - reportSchemaError("src-redefine.7.1", new Object []{new Integer(attGroupRefsCount)}, child); + reportSchemaError("src-redefine.7.1", + new Object []{attGroupRefsCount}, child); } else if (attGroupRefsCount == 1) { // return true; @@ -4029,7 +4030,7 @@ public class XSDHandler { ","+oldName:currSchema.fTargetNamespace+","+oldName; int groupRefsCount = changeRedefineGroup(processedBaseName, componentType, newName, child, currSchema); if (groupRefsCount > 1) { - reportSchemaError("src-redefine.6.1.1", new Object []{new Integer(groupRefsCount)}, child); + reportSchemaError("src-redefine.6.1.1", new Object []{groupRefsCount}, child); } else if (groupRefsCount == 1) { // return true; diff --git a/src/java.xml/share/classes/com/sun/org/apache/xml/internal/serialize/HTMLdtd.java b/src/java.xml/share/classes/com/sun/org/apache/xml/internal/serialize/HTMLdtd.java index eb1f99ba8d0..21328ef00cd 100644 --- a/src/java.xml/share/classes/com/sun/org/apache/xml/internal/serialize/HTMLdtd.java +++ b/src/java.xml/share/classes/com/sun/org/apache/xml/internal/serialize/HTMLdtd.java @@ -438,8 +438,8 @@ public final class HTMLdtd private static void defineEntity( String name, char value ) { if ( _byName.get( name ) == null ) { - _byName.put( name, new Integer( value ) ); - _byChar.put( new Integer( value ), name ); + _byName.put( name, (int) value); + _byChar.put( (int) value , name ); } } diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Lint.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Lint.java index 2d61cf5fb0b..6526104b67b 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Lint.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Lint.java @@ -125,6 +125,7 @@ public class Lint if (!options.isSet(Option.PREVIEW)) { values.add(LintCategory.PREVIEW); } + values.add(LintCategory.SYNCHRONIZATION); } // Look for specific overrides @@ -282,6 +283,11 @@ public class Lint */ STATIC("static"), + /** + * Warn about synchronization attempts on instances of @ValueBased classes. + */ + SYNCHRONIZATION("synchronization"), + /** * Warn about issues relating to use of text blocks */ diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symtab.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symtab.java index 5a5d78be3c8..d5eb6aa45ea 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symtab.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Symtab.java @@ -220,6 +220,7 @@ public class Symtab { public final Type previewFeatureInternalType; public final Type typeDescriptorType; public final Type recordType; + public final Type valueBasedType; /** The symbol representing the length field of an array. */ @@ -584,6 +585,7 @@ public class Symtab { previewFeatureInternalType = enterSyntheticAnnotation("jdk.internal.PreviewFeature+Annotation"); typeDescriptorType = enterClass("java.lang.invoke.TypeDescriptor"); recordType = enterClass("java.lang.Record"); + valueBasedType = enterClass("jdk.internal.ValueBased"); synthesizeEmptyInterfaceIfMissing(autoCloseableType); synthesizeEmptyInterfaceIfMissing(cloneableType); diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java index 4f314bfa7c7..84fdf02c053 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java @@ -1740,9 +1740,24 @@ public class Attr extends JCTree.Visitor { public void visitSynchronized(JCSynchronized tree) { chk.checkRefType(tree.pos(), attribExpr(tree.lock, env)); + if (env.info.lint.isEnabled(LintCategory.SYNCHRONIZATION) && isValueBased(tree.lock.type)) { + log.warning(LintCategory.SYNCHRONIZATION, tree.pos(), Warnings.AttemptToSynchronizeOnInstanceOfValueBasedClass); + } attribStat(tree.body, env); result = null; } + // where + private boolean isValueBased(Type t) { + if (t != null && t.tsym != null) { + for (Attribute.Compound a: t.tsym.getDeclarationAttributes()) { + if (a.type.tsym == syms.valueBasedType.tsym) { + return true; + } + } + } + return false; + } + public void visitTry(JCTry tree) { // Create a new local environment with a local diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties b/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties index 77339ec8a41..27909bae1bc 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties @@ -3740,3 +3740,7 @@ compiler.err.preview.not.latest=\ compiler.err.preview.without.source.or.release=\ --enable-preview must be used with either -source or --release + +compiler.warn.attempt.to.synchronize.on.instance.of.value.based.class=\ + attempt to synchronize on an instance of a value-based class + diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/javac.properties b/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/javac.properties index bbca9cae250..4bb02bd8a5d 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/javac.properties +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/javac.properties @@ -256,10 +256,13 @@ javac.opt.Xlint.desc.unchecked=\ Warn about unchecked operations. javac.opt.Xlint.desc.varargs=\ - Warn about potentially unsafe vararg methods + Warn about potentially unsafe vararg methods. javac.opt.Xlint.desc.preview=\ - Warn about use of preview language features + Warn about use of preview language features. + +javac.opt.Xlint.desc.synchronization=\ + Warn about synchronization attempts on instances of value-based classes. javac.opt.Xdoclint=\ Enable recommended checks for problems in javadoc comments 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 ed13670cbb6..6e09a2fc56f 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 @@ -44,12 +44,12 @@ import java.util.stream.Collectors; * 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...)}). - * *

    * This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code GroupLayout} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} 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 47ff9e29bce..3348506f8ce 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 @@ -40,9 +40,9 @@ import java.lang.ref.Cleaner; * when performing memory dereference operations using a memory access var handle (see {@link MemoryHandles}). *

    * All implementations of this interface must be value-based; - * use of identity-sensitive operations (including reference equality ({@code ==}), identity hash code, or synchronization) on - * instances of {@code MemoryAddress} may have unpredictable results and should be avoided. The {@code equals} method should - * be used for comparisons. + * 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, + * synchronization may fail. The {@code equals} method should be used for comparisons. *

    * Non-platform classes should not implement {@linkplain MemoryAddress} directly. * 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 6666b9c8bab..9c9e5ffc679 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 @@ -78,9 +78,9 @@ SequenceLayout taggedValues = MemoryLayout.ofSequence(5, * } *

    * All implementations of this interface must be value-based; - * use of identity-sensitive operations (including reference equality ({@code ==}), identity hash code, or synchronization) on - * instances of {@code MemoryLayout} may have unpredictable results and should be avoided. The {@code equals} method should - * be used for comparisons. + * 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, + * synchronization may fail. The {@code equals} method should be used for comparisons. *

    * Non-platform classes should not implement {@linkplain MemoryLayout} directly. * 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 13cef07afa7..d5360101dd3 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 @@ -50,9 +50,9 @@ import java.util.Spliterator; * operations on a segment cannot occur after a memory segment has been closed (see {@link MemorySegment#close()}). *

    * All implementations of this interface must be value-based; - * use of identity-sensitive operations (including reference equality ({@code ==}), identity hash code, or synchronization) on - * instances of {@code MemorySegment} may have unpredictable results and should be avoided. The {@code equals} method should - * be used for comparisons. + * 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, + * synchronization may fail. The {@code equals} method should be used for comparisons. *

    * Non-platform classes should not implement {@linkplain MemorySegment} directly. * diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/PaddingLayout.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/PaddingLayout.java index 6f16f206bff..db65c48669b 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/PaddingLayout.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/PaddingLayout.java @@ -38,9 +38,10 @@ import java.util.OptionalLong; * and is typically used for aligning member layouts around word boundaries. *

    * This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code PaddingLayout} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} 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 1df597c4e40..584c34e9335 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 @@ -55,9 +55,10 @@ MemoryLayout.ofStruct( * *

    * This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code SequenceLayout} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} diff --git a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/ValueLayout.java b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/ValueLayout.java index feb58bca780..26271da07da 100644 --- a/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/ValueLayout.java +++ b/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/ValueLayout.java @@ -40,9 +40,10 @@ import java.util.OptionalLong; * *

    * This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances of - * {@code ValueLayout} may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * *

    Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null} diff --git a/src/jdk.internal.vm.compiler/share/classes/jdk.internal.vm.compiler.collections.test/src/jdk/internal/vm/compiler/collections/test/EconomicMapImplTest.java b/src/jdk.internal.vm.compiler/share/classes/jdk.internal.vm.compiler.collections.test/src/jdk/internal/vm/compiler/collections/test/EconomicMapImplTest.java index 4beea31edd8..206294bcb65 100644 --- a/src/jdk.internal.vm.compiler/share/classes/jdk.internal.vm.compiler.collections.test/src/jdk/internal/vm/compiler/collections/test/EconomicMapImplTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/jdk.internal.vm.compiler.collections.test/src/jdk/internal/vm/compiler/collections/test/EconomicMapImplTest.java @@ -133,7 +133,7 @@ public class EconomicMapImplTest { Assert.assertTrue(set.add(newInteger(0))); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) private static Integer newInteger(int value) { return new Integer(value); } diff --git a/src/jdk.internal.vm.compiler/share/classes/jdk.internal.vm.compiler.collections.test/src/jdk/internal/vm/compiler/collections/test/EconomicSetTest.java b/src/jdk.internal.vm.compiler/share/classes/jdk.internal.vm.compiler.collections.test/src/jdk/internal/vm/compiler/collections/test/EconomicSetTest.java index e6cc8a80532..80fe74517b9 100644 --- a/src/jdk.internal.vm.compiler/share/classes/jdk.internal.vm.compiler.collections.test/src/jdk/internal/vm/compiler/collections/test/EconomicSetTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/jdk.internal.vm.compiler.collections.test/src/jdk/internal/vm/compiler/collections/test/EconomicSetTest.java @@ -161,7 +161,7 @@ public class EconomicSetTest { Assert.assertEquals(newInteger(9), finalList.get(0)); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) private static Integer newInteger(int value) { return new Integer(value); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/EscapeAnalysisTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/EscapeAnalysisTest.java index 6c64ecc8dfb..938f9f20c49 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/EscapeAnalysisTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/EscapeAnalysisTest.java @@ -58,7 +58,7 @@ public class EscapeAnalysisTest extends EATestBase { testEscapeAnalysis("test1Snippet", JavaConstant.forInt(101), false); } - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal"}) public static int test1Snippet() { Integer x = new Integer(101); return x.intValue(); @@ -89,7 +89,7 @@ public class EscapeAnalysisTest extends EATestBase { testEscapeAnalysis("testMonitorSnippet", JavaConstant.forInt(0), false); } - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal", "synchronization"}) public static int testMonitorSnippet() { Integer x = new Integer(0); Double y = new Double(0); @@ -113,7 +113,7 @@ public class EscapeAnalysisTest extends EATestBase { * This test case differs from the last one in that it requires inlining within a synchronized * region. */ - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal", "synchronization"}) public static int testMonitor2Snippet() { Integer x = new Integer(0); Double y = new Double(0); @@ -335,7 +335,7 @@ public class EscapeAnalysisTest extends EATestBase { public volatile Object field; - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal"}) public int testChangeHandlingSnippet(int a) { Object obj; Integer one = 1; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/PEAAssertionsTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/PEAAssertionsTest.java index 7a9ff22ed5a..68cd79247a1 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/PEAAssertionsTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/PEAAssertionsTest.java @@ -42,7 +42,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { public static Object field; - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippet1(int i) { Integer object = new Integer(i); GraalDirectives.ensureVirtualized(object); @@ -53,7 +53,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippet1", 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippet2(int i) { Integer object = new Integer(i); GraalDirectives.ensureVirtualized(object); @@ -65,7 +65,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippet2", 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippet3(int i) { Integer object = new Integer(i); field = object; @@ -77,7 +77,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippet3", 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetHere1(int i) { Integer object = new Integer(i); GraalDirectives.ensureVirtualizedHere(object); @@ -88,7 +88,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippetHere1", 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetHere2(int i) { Integer object = new Integer(i); GraalDirectives.ensureVirtualizedHere(object); @@ -100,7 +100,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippetHere2", 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetHere3(int i) { Integer object = new Integer(i); field = object; @@ -133,7 +133,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippetBoxing2", 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetControlFlow1(boolean b, int i) { Integer object = new Integer(i); if (b) { @@ -148,7 +148,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippetControlFlow1", true, 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetControlFlow2(boolean b, int i) { Integer object = new Integer(i); if (b) { @@ -165,7 +165,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippetControlFlow2", true, 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetControlFlow3(boolean b, int i) { Integer object = new Integer(i); GraalDirectives.ensureVirtualized(object); @@ -183,7 +183,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippetControlFlow3", true, 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetControlFlow4(boolean b, int i) { Integer object = new Integer(i); if (b) { @@ -199,7 +199,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippetControlFlow4", true, 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetControlFlow5(boolean b, int i) { Integer object = new Integer(i); if (b) { @@ -220,7 +220,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { Object b; } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetIndirect1(boolean b, int i) { Integer object = new Integer(i); TestClass t = new TestClass(); @@ -239,7 +239,7 @@ public class PEAAssertionsTest extends GraalCompilerTest { test("snippetIndirect1", true, 1); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static void snippetIndirect2(boolean b, int i) { Integer object = new Integer(i); TestClass t = new TestClass(); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/PartialEscapeAnalysisTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/PartialEscapeAnalysisTest.java index 5093c980519..ee02d0d7c4f 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/PartialEscapeAnalysisTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/PartialEscapeAnalysisTest.java @@ -115,7 +115,7 @@ public class PartialEscapeAnalysisTest extends EATestBase { testPartialEscapeAnalysis("test3Snippet", 0.5, 1, StoreFieldNode.class, LoadFieldNode.class); } - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal"}) public static Object test3Snippet(int a) { if (a < 0) { TestObject obj = new TestObject(1, 2); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/tutorial/StaticAnalysisTests.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/tutorial/StaticAnalysisTests.java index b2c1dc57999..bf436251e73 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/tutorial/StaticAnalysisTests.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/tutorial/StaticAnalysisTests.java @@ -114,7 +114,7 @@ public class StaticAnalysisTests { assertEquals(m(sa, B.class, "foo").getFormalReturn(), t(Data.class)); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) static void test03Entry() { Data data = new Data(); data.f = new Integer(42); @@ -142,7 +142,7 @@ public class StaticAnalysisTests { assertEquals(m(sa, B.class, "foo").getFormalReturn(), t(Data.class), t(Integer.class)); } - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) static void test04Entry() { Data data = null; for (int i = 0; i < 2; i++) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/GraalHotSpotVMConfigAccess.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/GraalHotSpotVMConfigAccess.java index a9fb5ff7de9..b10df1df359 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/GraalHotSpotVMConfigAccess.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/GraalHotSpotVMConfigAccess.java @@ -373,7 +373,7 @@ public class GraalHotSpotVMConfigAccess { /** * @see HotSpotVMConfigAccess#getFlag(String, Class, Object) */ - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal"}) public T getFlag(String name, Class type, T notPresent, boolean expectPresent) { if (expectPresent) { return getFlag(name, type); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/hotpath/HP_allocate02.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/hotpath/HP_allocate02.java index 41e89b8dc1f..e51314a5ea4 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/hotpath/HP_allocate02.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/hotpath/HP_allocate02.java @@ -32,7 +32,7 @@ import org.graalvm.compiler.jtt.JTTTest; */ public class HP_allocate02 extends JTTTest { - @SuppressWarnings({"deprecation", "unused"}) + @SuppressWarnings({"deprecation", "removal", "unused"}) public static int test(int count) { int sum = 0; for (int i = 0; i < count; i++) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.test/src/org/graalvm/compiler/replacements/test/MonitorTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.test/src/org/graalvm/compiler/replacements/test/MonitorTest.java index fefd6b330dd..1fb8c51a472 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.test/src/org/graalvm/compiler/replacements/test/MonitorTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.test/src/org/graalvm/compiler/replacements/test/MonitorTest.java @@ -212,6 +212,7 @@ public class MonitorTest extends GraalCompilerTest { return new String(dst); } + @SuppressWarnings("synchronization") public static String lockBoxedLong(long value) { Long lock = value; synchronized (lock) { diff --git a/src/jdk.jfr/share/conf/jfr/default.jfc b/src/jdk.jfr/share/conf/jfr/default.jfc index a512e76c547..53151be43f0 100644 --- a/src/jdk.jfr/share/conf/jfr/default.jfc +++ b/src/jdk.jfr/share/conf/jfr/default.jfc @@ -91,7 +91,7 @@ 20 ms - + true true diff --git a/src/jdk.jfr/share/conf/jfr/profile.jfc b/src/jdk.jfr/share/conf/jfr/profile.jfc index fa4522ed023..7958f6a1fbe 100644 --- a/src/jdk.jfr/share/conf/jfr/profile.jfc +++ b/src/jdk.jfr/share/conf/jfr/profile.jfc @@ -91,7 +91,7 @@ 10 ms - + true true diff --git a/test/hotspot/jtreg/runtime/Monitor/SyncOnPrimitiveWrapperTest.java b/test/hotspot/jtreg/runtime/Monitor/SyncOnValueBasedClassTest.java similarity index 91% rename from test/hotspot/jtreg/runtime/Monitor/SyncOnPrimitiveWrapperTest.java rename to test/hotspot/jtreg/runtime/Monitor/SyncOnValueBasedClassTest.java index 667ebd8d44a..ad2adf3bcf8 100644 --- a/test/hotspot/jtreg/runtime/Monitor/SyncOnPrimitiveWrapperTest.java +++ b/test/hotspot/jtreg/runtime/Monitor/SyncOnValueBasedClassTest.java @@ -29,12 +29,12 @@ import java.util.stream.*; /* * @test * @bug 8242263 - * @summary Exercise DiagnoseSyncOnPrimitiveWrappers diagnostic flag + * @summary Exercise DiagnoseSyncOnValueBasedClasses diagnostic flag * @library /test/lib - * @run driver/timeout=180000 SyncOnPrimitiveWrapperTest + * @run driver/timeout=180000 SyncOnValueBasedClassTest */ -public class SyncOnPrimitiveWrapperTest { +public class SyncOnValueBasedClassTest { static final int LOOP_COUNT = 3000; static final int THREAD_COUNT = 2; static String[] fatalTests[]; @@ -64,20 +64,20 @@ public class SyncOnPrimitiveWrapperTest { private static void generateTests() { initTestObjects(); - String[] commonFatalTestsFlags = {"-XX:+UnlockDiagnosticVMOptions", "-XX:-CreateCoredumpOnCrash", "-XX:DiagnoseSyncOnPrimitiveWrappers=1"}; + String[] commonFatalTestsFlags = {"-XX:+UnlockDiagnosticVMOptions", "-XX:-CreateCoredumpOnCrash", "-XX:DiagnoseSyncOnValueBasedClasses=1"}; fatalTests = new String[specificFlags.length * testObjects.size()][]; for (int i = 0; i < specificFlags.length; i++) { for (int j = 0; j < testObjects.size(); j++) { int index = i * testObjects.size() + j; - fatalTests[index] = Stream.of(commonFatalTestsFlags, specificFlags[i], new String[] {"SyncOnPrimitiveWrapperTest$FatalTest", Integer.toString(j)}) + fatalTests[index] = Stream.of(commonFatalTestsFlags, specificFlags[i], new String[] {"SyncOnValueBasedClassTest$FatalTest", Integer.toString(j)}) .flatMap(Stream::of) .toArray(String[]::new); } } - String[] commonLogTestsFlags = {"-XX:+UnlockDiagnosticVMOptions", "-XX:DiagnoseSyncOnPrimitiveWrappers=2"}; + String[] commonLogTestsFlags = {"-XX:+UnlockDiagnosticVMOptions", "-XX:DiagnoseSyncOnValueBasedClasses=2"}; logTests = new String[specificFlags.length][]; for (int i = 0; i < specificFlags.length; i++) { - logTests[i] = Stream.of(commonLogTestsFlags, specificFlags[i], new String[] {"SyncOnPrimitiveWrapperTest$LogTest"}) + logTests[i] = Stream.of(commonLogTestsFlags, specificFlags[i], new String[] {"SyncOnValueBasedClassTest$LogTest"}) .flatMap(Stream::of) .toArray(String[]::new); } @@ -89,7 +89,7 @@ public class SyncOnPrimitiveWrapperTest { ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(fatalTests[i]); OutputAnalyzer output = ProcessTools.executeProcess(pb); output.shouldContain("fatal error: Synchronizing on object"); - output.shouldNotContain("synchronization on primitive wrapper did not fail"); + output.shouldNotContain("synchronization on value based class did not fail"); output.shouldNotHaveExitValue(0); } for (int i = 0; i < logTests.length; i++) { @@ -127,7 +127,7 @@ public class SyncOnPrimitiveWrapperTest { public static void main(String[] args) throws Exception { initTestObjects(); synchronized (testObjects.get(Integer.valueOf(args[0]))) { - throw new RuntimeException("synchronization on primitive wrapper did not fail"); + throw new RuntimeException("synchronization on value based class did not fail"); } } } @@ -164,4 +164,4 @@ public class SyncOnPrimitiveWrapperTest { } } } -} \ No newline at end of file +} diff --git a/test/jdk/jdk/jfr/event/metadata/TestLookForUntestedEvents.java b/test/jdk/jdk/jfr/event/metadata/TestLookForUntestedEvents.java index 8ba88792281..1a5e893a916 100644 --- a/test/jdk/jdk/jfr/event/metadata/TestLookForUntestedEvents.java +++ b/test/jdk/jdk/jfr/event/metadata/TestLookForUntestedEvents.java @@ -82,7 +82,7 @@ public class TestLookForUntestedEvents { // Experimental events private static final Set experimentalEvents = new HashSet<>( Arrays.asList( - "Flush", "SyncOnPrimitiveWrapper") + "Flush", "SyncOnValueBasedClass") ); diff --git a/test/jdk/jdk/jfr/event/runtime/TestSyncOnPrimitiveWrapperEvent.java b/test/jdk/jdk/jfr/event/runtime/TestSyncOnValueBasedClassEvent.java similarity index 86% rename from test/jdk/jdk/jfr/event/runtime/TestSyncOnPrimitiveWrapperEvent.java rename to test/jdk/jdk/jfr/event/runtime/TestSyncOnValueBasedClassEvent.java index bf8f31699e4..85b01e4198c 100644 --- a/test/jdk/jdk/jfr/event/runtime/TestSyncOnPrimitiveWrapperEvent.java +++ b/test/jdk/jdk/jfr/event/runtime/TestSyncOnValueBasedClassEvent.java @@ -38,10 +38,10 @@ import jdk.test.lib.jfr.Events; * @requires vm.hasJFR * @key jfr * @library /test/lib - * @run main/othervm -XX:+UnlockDiagnosticVMOptions -XX:DiagnoseSyncOnPrimitiveWrappers=2 jdk.jfr.event.runtime.TestSyncOnPrimitiveWrapperEvent + * @run main/othervm -XX:+UnlockDiagnosticVMOptions -XX:DiagnoseSyncOnValueBasedClasses=2 jdk.jfr.event.runtime.TestSyncOnValueBasedClassEvent */ -public class TestSyncOnPrimitiveWrapperEvent { - static final String EVENT_NAME = EventNames.SyncOnPrimitiveWrapper; +public class TestSyncOnValueBasedClassEvent { + static final String EVENT_NAME = EventNames.SyncOnValueBasedClass; static String[] classesWanted = {"java/lang/Character", "java/lang/Boolean", "java/lang/Byte", "java/lang/Short", "java/lang/Integer", "java/lang/Long", "java/lang/Float", "java/lang/Double"}; static List testObjects = new ArrayList(); @@ -74,7 +74,7 @@ public class TestSyncOnPrimitiveWrapperEvent { List events = Events.fromRecording(recording); Events.hasEvents(events); for (RecordedEvent event : Events.fromRecording(recording)) { - String className = Events.assertField(event, "boxClass.name").notEmpty().getValue(); + String className = Events.assertField(event, "valueBasedClass.name").notEmpty().getValue(); RecordedThread jt = event.getThread(); if (Thread.currentThread().getName().equals(jt.getJavaName())) { classesFound.add(className); @@ -82,11 +82,11 @@ public class TestSyncOnPrimitiveWrapperEvent { } for (String classWanted : classesWanted) { if (!classesFound.contains(classWanted)) { - throw new AssertionError("No matching event SyncOnPrimitiveWrapper with \"boxClass=" + classWanted + "\" and current thread as caller"); + throw new AssertionError("No matching event SyncOnValueBasedClass with \"valueBasedClass=" + classWanted + "\" and current thread as caller"); } } if (classesFound.size() != classesWanted.length) { - throw new AssertionError("Invalid number of SyncOnPrimitiveWrapper events for current thread"); + throw new AssertionError("Invalid number of SyncOnValueBasedClass events for current thread"); } } } diff --git a/test/langtools/jdk/jshell/ImportTest.java b/test/langtools/jdk/jshell/ImportTest.java index d5b03c47f41..e09490052cf 100644 --- a/test/langtools/jdk/jshell/ImportTest.java +++ b/test/langtools/jdk/jshell/ImportTest.java @@ -68,7 +68,7 @@ public class ImportTest extends KullaTesting { public void testImportStatic() { assertImportKeyMatch("import static java.lang.Math.PI;", "PI", SINGLE_STATIC_IMPORT_SUBKIND, added(VALID)); - assertEval("new Double(PI).toString().substring(0, 16).equals(\"3.14159265358979\");", "true"); + assertEval("Double.valueOf(PI).toString().substring(0, 16).equals(\"3.14159265358979\");", "true"); } public void testImportStaticOnDemand() { diff --git a/test/langtools/tools/javac/diags/examples/AttemptToSynchronizeOnInstanceOfVbc.java b/test/langtools/tools/javac/diags/examples/AttemptToSynchronizeOnInstanceOfVbc.java new file mode 100644 index 00000000000..9d63a2af0a4 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/AttemptToSynchronizeOnInstanceOfVbc.java @@ -0,0 +1,32 @@ +/* + * 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. + */ + +// key: compiler.warn.attempt.to.synchronize.on.instance.of.value.based.class +// options: -Xlint:synchronization + +class AttemptToSynchronizeOnInstanceOfVbc { + void foo(Integer i) { + synchronized(i) { + } + } +} diff --git a/test/langtools/tools/javac/lambda/8074381/T8074381a.java b/test/langtools/tools/javac/lambda/8074381/T8074381a.java index d11944d3926..0b5b4695bcb 100644 --- a/test/langtools/tools/javac/lambda/8074381/T8074381a.java +++ b/test/langtools/tools/javac/lambda/8074381/T8074381a.java @@ -13,7 +13,7 @@ class T8074381a { boolean m(String s); } - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal"}) void testRaw() { Sub s1 = c -> true; Sub s2 = Boolean::new; @@ -23,7 +23,7 @@ class T8074381a { }; } - @SuppressWarnings("deprecation") + @SuppressWarnings({"deprecation", "removal"}) void testNonRaw() { Sub s1 = c -> true; Sub s2 = Boolean::new; diff --git a/test/langtools/tools/javac/lint/ExternalAbuseOfVbc.java b/test/langtools/tools/javac/lint/ExternalAbuseOfVbc.java new file mode 100644 index 00000000000..6de1138f527 --- /dev/null +++ b/test/langtools/tools/javac/lint/ExternalAbuseOfVbc.java @@ -0,0 +1,23 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8254274 + * @summary lint should warn when an instance of a value based class is synchronized upon + * @compile/fail/ref=ExternalAbuseOfVbc.out -XDrawDiagnostics -Werror -Xlint ExternalAbuseOfVbc.java + * @compile/fail/ref=ExternalAbuseOfVbc.out -XDrawDiagnostics -Werror -Xlint:all ExternalAbuseOfVbc.java + * @compile/fail/ref=ExternalAbuseOfVbc.out -XDrawDiagnostics -Werror -Xlint:synchronization ExternalAbuseOfVbc.java + * @compile/ref=LintModeOffAbuseOfVbc.out -XDrawDiagnostics -Werror -Xlint:-synchronization ExternalAbuseOfVbc.java + */ + +public final class ExternalAbuseOfVbc { + + final Integer val = Integer.valueOf(42); + final String ref = "String"; + + void abuseVbc() { + synchronized(ref) { // OK + synchronized (val) { // WARN + } + } + } +} + diff --git a/test/langtools/tools/javac/lint/ExternalAbuseOfVbc.out b/test/langtools/tools/javac/lint/ExternalAbuseOfVbc.out new file mode 100644 index 00000000000..b4292879c15 --- /dev/null +++ b/test/langtools/tools/javac/lint/ExternalAbuseOfVbc.out @@ -0,0 +1,4 @@ +ExternalAbuseOfVbc.java:18:13: compiler.warn.attempt.to.synchronize.on.instance.of.value.based.class +- compiler.err.warnings.and.werror +1 error +1 warning diff --git a/test/langtools/tools/javac/lint/JdkInternalAbuseOfVbc.java b/test/langtools/tools/javac/lint/JdkInternalAbuseOfVbc.java new file mode 100644 index 00000000000..2d384dd324d --- /dev/null +++ b/test/langtools/tools/javac/lint/JdkInternalAbuseOfVbc.java @@ -0,0 +1,22 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8254274 + * @summary lint should warn when an instance of a value based class is synchronized upon + * @compile/fail/ref=JdkInternalAbuseOfVbc.out --patch-module java.base=${test.src} -XDrawDiagnostics -Werror -Xlint SomeVbc.java JdkInternalAbuseOfVbc.java + */ + +package java.lang; + +public final class JdkInternalAbuseOfVbc { + + public JdkInternalAbuseOfVbc() {} + + void abuseVbc(SomeVbc vbc) { + + synchronized(this) { // OK + synchronized (vbc) { // WARN + } + } + } +} + diff --git a/test/langtools/tools/javac/lint/JdkInternalAbuseOfVbc.out b/test/langtools/tools/javac/lint/JdkInternalAbuseOfVbc.out new file mode 100644 index 00000000000..cb66f0bcfa6 --- /dev/null +++ b/test/langtools/tools/javac/lint/JdkInternalAbuseOfVbc.out @@ -0,0 +1,6 @@ +SomeVbc.java:38:13: compiler.warn.attempt.to.synchronize.on.instance.of.value.based.class +- compiler.err.warnings.and.werror +SomeVbc.java:49:13: compiler.warn.attempt.to.synchronize.on.instance.of.value.based.class +JdkInternalAbuseOfVbc.java:17:13: compiler.warn.attempt.to.synchronize.on.instance.of.value.based.class +1 error +3 warnings diff --git a/test/langtools/tools/javac/lint/LintModeOffAbuseOfVbc.out b/test/langtools/tools/javac/lint/LintModeOffAbuseOfVbc.out new file mode 100644 index 00000000000..e69de29bb2d diff --git a/test/langtools/tools/javac/lint/SomeVbc.java b/test/langtools/tools/javac/lint/SomeVbc.java new file mode 100644 index 00000000000..b68e4599c81 --- /dev/null +++ b/test/langtools/tools/javac/lint/SomeVbc.java @@ -0,0 +1,54 @@ +/* + * 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 java.lang; + +@jdk.internal.ValueBased +public final class SomeVbc { + + public SomeVbc() {} + + final String ref = "String"; + + void abuseVbc() { + + synchronized(ref) { // OK + synchronized (this) { // WARN + } + } + } +} + +final class AuxilliaryAbuseOfVbc { + + void abuseVbc(SomeVbc vbc) { + + synchronized(this) { // OK + synchronized (vbc) { // WARN + } + } + } +} + diff --git a/test/lib/jdk/test/lib/hexdump/HexPrinter.java b/test/lib/jdk/test/lib/hexdump/HexPrinter.java index 12f0cfd45b0..901d5859f92 100644 --- a/test/lib/jdk/test/lib/hexdump/HexPrinter.java +++ b/test/lib/jdk/test/lib/hexdump/HexPrinter.java @@ -143,9 +143,10 @@ import java.util.Objects; * *

    * This is a value-based - * class; use of identity-sensitive operations (including reference equality - * ({@code ==}), identity hash code, or synchronization) on instances - * may have unpredictable results and should be avoided. + * class; 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, synchronization may fail. * The {@code equals} method should be used for comparisons. * *

    diff --git a/test/lib/jdk/test/lib/jfr/EventNames.java b/test/lib/jdk/test/lib/jfr/EventNames.java index 19fb9f23524..482de84d32a 100644 --- a/test/lib/jdk/test/lib/jfr/EventNames.java +++ b/test/lib/jdk/test/lib/jfr/EventNames.java @@ -61,7 +61,7 @@ public class EventNames { public final static String JavaMonitorEnter = PREFIX + "JavaMonitorEnter"; public final static String JavaMonitorWait = PREFIX + "JavaMonitorWait"; public final static String JavaMonitorInflate = PREFIX + "JavaMonitorInflate"; - public final static String SyncOnPrimitiveWrapper = PREFIX + "SyncOnPrimitiveWrapper"; + public final static String SyncOnValueBasedClass = PREFIX + "SyncOnValueBasedClass"; public final static String ClassLoad = PREFIX + "ClassLoad"; public final static String ClassDefine = PREFIX + "ClassDefine"; public final static String ClassUnload = PREFIX + "ClassUnload";