From 4101ba9a134c1a3969206f5c2af8f7027aa7317d Mon Sep 17 00:00:00 2001 From: Sangheon Kim Date: Thu, 26 Oct 2017 21:30:48 -0700 Subject: [PATCH 01/61] 8185278: TestGreyReclaimedHumongousObjects.java fails guarantee(index != trim_index(_head_index + 1)) failed: should not go past head Add STS to avoid MMU concurrency problem between VM Thread and Concurrent Mark Thread Reviewed-by: tschatzl, ehelin --- .../share/gc/g1/concurrentMarkThread.cpp | 39 +++++++++++++------ .../share/gc/g1/concurrentMarkThread.hpp | 2 + src/hotspot/share/gc/g1/g1MMUTracker.cpp | 3 +- src/hotspot/share/runtime/mutexLocker.cpp | 2 - src/hotspot/share/runtime/mutexLocker.hpp | 2 - .../g1/TestGreyReclaimedHumongousObjects.java | 4 +- 6 files changed, 33 insertions(+), 19 deletions(-) diff --git a/src/hotspot/share/gc/g1/concurrentMarkThread.cpp b/src/hotspot/share/gc/g1/concurrentMarkThread.cpp index feb287a0135..84e3000520b 100644 --- a/src/hotspot/share/gc/g1/concurrentMarkThread.cpp +++ b/src/hotspot/share/gc/g1/concurrentMarkThread.cpp @@ -111,16 +111,31 @@ public: } }; -// Marking pauses can be scheduled flexibly, so we might delay marking to meet MMU. -void ConcurrentMarkThread::delay_to_keep_mmu(G1Policy* g1_policy, bool remark) { +double ConcurrentMarkThread::mmu_sleep_time(G1Policy* g1_policy, bool remark) { + // There are 3 reasons to use SuspendibleThreadSetJoiner. + // 1. To avoid concurrency problem. + // - G1MMUTracker::add_pause(), when_sec() and its variation(when_ms() etc..) can be called + // concurrently from ConcurrentMarkThread and VMThread. + // 2. If currently a gc is running, but it has not yet updated the MMU, + // we will not forget to consider that pause in the MMU calculation. + // 3. If currently a gc is running, ConcurrentMarkThread will wait it to be finished. + // And then sleep for predicted amount of time by delay_to_keep_mmu(). + SuspendibleThreadSetJoiner sts_join; + const G1Analytics* analytics = g1_policy->analytics(); + double now = os::elapsedTime(); + double prediction_ms = remark ? analytics->predict_remark_time_ms() + : analytics->predict_cleanup_time_ms(); + G1MMUTracker *mmu_tracker = g1_policy->mmu_tracker(); + return mmu_tracker->when_ms(now, prediction_ms); +} + +void ConcurrentMarkThread::delay_to_keep_mmu(G1Policy* g1_policy, bool remark) { if (g1_policy->adaptive_young_list_length()) { - double now = os::elapsedTime(); - double prediction_ms = remark ? analytics->predict_remark_time_ms() - : analytics->predict_cleanup_time_ms(); - G1MMUTracker *mmu_tracker = g1_policy->mmu_tracker(); - jlong sleep_time_ms = mmu_tracker->when_ms(now, prediction_ms); - os::sleep(this, sleep_time_ms, false); + jlong sleep_time_ms = mmu_sleep_time(g1_policy, remark); + if (!cm()->has_aborted() && sleep_time_ms > 0) { + os::sleep(this, sleep_time_ms, false); + } } } @@ -349,9 +364,11 @@ void ConcurrentMarkThread::run_service() { if (!cm()->has_aborted()) { delay_to_keep_mmu(g1_policy, false /* cleanup */); - CMCleanUp cl_cl(_cm); - VM_CGC_Operation op(&cl_cl, "Pause Cleanup"); - VMThread::execute(&op); + if (!cm()->has_aborted()) { + CMCleanUp cl_cl(_cm); + VM_CGC_Operation op(&cl_cl, "Pause Cleanup"); + VMThread::execute(&op); + } } else { // We don't want to update the marking status if a GC pause // is already underway. diff --git a/src/hotspot/share/gc/g1/concurrentMarkThread.hpp b/src/hotspot/share/gc/g1/concurrentMarkThread.hpp index 26723581939..dae78bac60b 100644 --- a/src/hotspot/share/gc/g1/concurrentMarkThread.hpp +++ b/src/hotspot/share/gc/g1/concurrentMarkThread.hpp @@ -55,7 +55,9 @@ class ConcurrentMarkThread: public ConcurrentGCThread { ConcurrentGCPhaseManager::Stack _phase_manager_stack; void sleepBeforeNextCycle(); + // Delay marking to meet MMU. void delay_to_keep_mmu(G1Policy* g1_policy, bool remark); + double mmu_sleep_time(G1Policy* g1_policy, bool remark); void run_service(); void stop_service(); diff --git a/src/hotspot/share/gc/g1/g1MMUTracker.cpp b/src/hotspot/share/gc/g1/g1MMUTracker.cpp index e3e522179e9..837eb1576e7 100644 --- a/src/hotspot/share/gc/g1/g1MMUTracker.cpp +++ b/src/hotspot/share/gc/g1/g1MMUTracker.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2017, 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 @@ -117,7 +117,6 @@ void G1MMUTrackerQueue::add_pause(double start, double end) { // of other places (debugging) double G1MMUTrackerQueue::when_sec(double current_time, double pause_time) { - MutexLockerEx x(MMUTracker_lock, Mutex::_no_safepoint_check_flag); remove_expired_entries(current_time); return when_internal(current_time, pause_time); diff --git a/src/hotspot/share/runtime/mutexLocker.cpp b/src/hotspot/share/runtime/mutexLocker.cpp index a14cdbe35be..5927b527f73 100644 --- a/src/hotspot/share/runtime/mutexLocker.cpp +++ b/src/hotspot/share/runtime/mutexLocker.cpp @@ -116,7 +116,6 @@ Mutex* FreeList_lock = NULL; Monitor* SecondaryFreeList_lock = NULL; Mutex* OldSets_lock = NULL; Monitor* RootRegionScan_lock = NULL; -Mutex* MMUTracker_lock = NULL; Monitor* GCTaskManager_lock = NULL; @@ -193,7 +192,6 @@ void mutex_init() { def(SecondaryFreeList_lock , PaddedMonitor, leaf , true, Monitor::_safepoint_check_never); def(OldSets_lock , PaddedMutex , leaf , true, Monitor::_safepoint_check_never); def(RootRegionScan_lock , PaddedMonitor, leaf , true, Monitor::_safepoint_check_never); - def(MMUTracker_lock , PaddedMutex , leaf , true, Monitor::_safepoint_check_never); def(StringDedupQueue_lock , PaddedMonitor, leaf, true, Monitor::_safepoint_check_never); def(StringDedupTable_lock , PaddedMutex , leaf, true, Monitor::_safepoint_check_never); diff --git a/src/hotspot/share/runtime/mutexLocker.hpp b/src/hotspot/share/runtime/mutexLocker.hpp index d7bf9b12fbf..7612059eff5 100644 --- a/src/hotspot/share/runtime/mutexLocker.hpp +++ b/src/hotspot/share/runtime/mutexLocker.hpp @@ -117,8 +117,6 @@ extern Mutex* FreeList_lock; // protects the free region lis extern Monitor* SecondaryFreeList_lock; // protects the secondary free region list extern Mutex* OldSets_lock; // protects the old region sets extern Monitor* RootRegionScan_lock; // used to notify that the CM threads have finished scanning the IM snapshot regions -extern Mutex* MMUTracker_lock; // protects the MMU - // tracker data structures extern Mutex* Management_lock; // a lock used to serialize JVM management extern Monitor* Service_lock; // a lock used for service thread operation diff --git a/test/hotspot/jtreg/gc/g1/TestGreyReclaimedHumongousObjects.java b/test/hotspot/jtreg/gc/g1/TestGreyReclaimedHumongousObjects.java index 3d5cea2457c..f5c389091c8 100644 --- a/test/hotspot/jtreg/gc/g1/TestGreyReclaimedHumongousObjects.java +++ b/test/hotspot/jtreg/gc/g1/TestGreyReclaimedHumongousObjects.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2017, 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 @@ -23,7 +23,7 @@ /* * @test TestGreyReclaimedHumongousObjects.java - * @bug 8069367 + * @bug 8069367 8185278 * @requires vm.gc.G1 * @summary Test handling of marked but unscanned reclaimed humongous objects. * @key gc From 55541e80b537f529a98f44c681be3241c074bed3 Mon Sep 17 00:00:00 2001 From: Jamsheed Mohammed C M Date: Thu, 26 Oct 2017 22:18:15 -0700 Subject: [PATCH 02/61] 8185989: overview.html files should be deleted? Jdk.vm.ci.code/overview.html jdk.vm.ci.meta/overview.html deleted Reviewed-by: kvn, thartmann --- .../classes/jdk.vm.ci.code/overview.html | 37 ------------------ .../classes/jdk.vm.ci.meta/overview.html | 38 ------------------- 2 files changed, 75 deletions(-) delete mode 100644 src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.code/overview.html delete mode 100644 src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/overview.html diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.code/overview.html b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.code/overview.html deleted file mode 100644 index 6f12f4bc25a..00000000000 --- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.code/overview.html +++ /dev/null @@ -1,37 +0,0 @@ - - - - - - - - -The jdk.vm.ci.code project provides an API to the runtime's native code cache. -It allows installation and execution of native code. - - - diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/overview.html b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/overview.html deleted file mode 100644 index 3da323679a1..00000000000 --- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.meta/overview.html +++ /dev/null @@ -1,38 +0,0 @@ - - - - - - - - -The jdk.vm.ci.meta project provides an API to the runtime data structures -for various Java elements. Unlike standard Java reflection, it can model elements that are not yet loaded. -It can also expose profiling information collected by the runtime system. - - - From 0bf8a0fc348e8fd0f5830705098faa68bd4c8244 Mon Sep 17 00:00:00 2001 From: Vladimir Kozlov Date: Fri, 27 Oct 2017 01:09:57 -0700 Subject: [PATCH 03/61] 8189064: Crash with compiler/codegen/*Vect.java on Solaris-sparc Unconditionally call cl->mark_loop_vectorized() when vectors are generated Reviewed-by: thartmann --- src/hotspot/share/opto/superword.cpp | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/src/hotspot/share/opto/superword.cpp b/src/hotspot/share/opto/superword.cpp index b35f9b1da67..b0afe8e5c74 100644 --- a/src/hotspot/share/opto/superword.cpp +++ b/src/hotspot/share/opto/superword.cpp @@ -2168,10 +2168,12 @@ void SuperWord::output() { CountedLoopNode *cl = lpt()->_head->as_CountedLoop(); Compile* C = _phase->C; if (_packset.length() == 0) { - // Instigate more unrolling for optimization when vectorization fails. - C->set_major_progress(); - cl->set_notpassed_slp(); - cl->mark_do_unroll_only(); + if (cl->is_main_loop()) { + // Instigate more unrolling for optimization when vectorization fails. + C->set_major_progress(); + cl->set_notpassed_slp(); + cl->mark_do_unroll_only(); + } return; } @@ -2417,6 +2419,9 @@ void SuperWord::output() { }//for (int i = 0; i < _block.length(); i++) C->set_max_vector_size(max_vlen_in_bytes); + if (max_vlen_in_bytes > 0) { + cl->mark_loop_vectorized(); + } if (SuperWordLoopUnrollAnalysis) { if (cl->has_passed_slp()) { @@ -2439,7 +2444,6 @@ void SuperWord::output() { } if (do_reserve_copy()) { - cl->mark_loop_vectorized(); if (can_process_post_loop) { // Now create the difference of trip and limit and use it as our mask index. // Note: We limited the unroll of the vectorized loop so that From 97d1978c0f0d41e7631ef4e4790b904255caf6d3 Mon Sep 17 00:00:00 2001 From: Leo Korinth Date: Fri, 27 Oct 2017 11:44:56 +0200 Subject: [PATCH 04/61] 8177959: G1CollectedHeap::print_on prints incorrect capacity Reviewed-by: sjohanss, ehelin, pliden --- src/hotspot/share/gc/g1/g1CollectedHeap.cpp | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp index 006c09fa490..3d81bc00965 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp @@ -2390,9 +2390,8 @@ void G1CollectedHeap::print_on(outputStream* st) const { st->print(" %-20s", "garbage-first heap"); st->print(" total " SIZE_FORMAT "K, used " SIZE_FORMAT "K", capacity()/K, used_unlocked()/K); - st->print(" [" PTR_FORMAT ", " PTR_FORMAT ", " PTR_FORMAT ")", + st->print(" [" PTR_FORMAT ", " PTR_FORMAT ")", p2i(_hrm.reserved().start()), - p2i(_hrm.reserved().start() + _hrm.length() + HeapRegion::GrainWords), p2i(_hrm.reserved().end())); st->cr(); st->print(" region size " SIZE_FORMAT "K, ", HeapRegion::GrainBytes / K); From a42299429638023352d6fde81aa151e7a0519de7 Mon Sep 17 00:00:00 2001 From: Ioi Lam Date: Sun, 29 Oct 2017 18:13:18 -0700 Subject: [PATCH 05/61] 8190191: Subclasses of jdk.jfr.Event loaded from CDS breaks -XX:FlightRecorderOptions=retransform=false Do not archive these classes Reviewed-by: jiangli, sspitsyn --- src/hotspot/share/classfile/dictionary.cpp | 26 ++++++++++++++++--- src/hotspot/share/classfile/dictionary.hpp | 2 +- .../share/classfile/systemDictionary.cpp | 2 ++ .../share/classfile/systemDictionary.hpp | 2 +- 4 files changed, 26 insertions(+), 6 deletions(-) diff --git a/src/hotspot/share/classfile/dictionary.cpp b/src/hotspot/share/classfile/dictionary.cpp index 8ffdefa6d4b..c500c47734e 100644 --- a/src/hotspot/share/classfile/dictionary.cpp +++ b/src/hotspot/share/classfile/dictionary.cpp @@ -357,6 +357,16 @@ bool Dictionary::is_valid_protection_domain(int index, unsigned int hash, return entry->is_valid_protection_domain(protection_domain); } +#if INCLUDE_CDS +static bool is_jfr_event_class(Klass *k) { + while (k) { + if (k->name()->equals("jdk/jfr/Event")) { + return true; + } + k = k->super(); + } + return false; +} void Dictionary::reorder_dictionary_for_sharing() { @@ -368,14 +378,22 @@ void Dictionary::reorder_dictionary_for_sharing() { while (p != NULL) { DictionaryEntry* next = p->next(); InstanceKlass*ik = p->instance_klass(); - // we cannot include signed classes in the archive because the certificates - // used during dump time may be different than those used during - // runtime (due to expiration, etc). if (ik->signers() != NULL) { + // We cannot include signed classes in the archive because the certificates + // used during dump time may be different than those used during + // runtime (due to expiration, etc). ResourceMark rm; tty->print_cr("Preload Warning: Skipping %s from signed JAR", ik->name()->as_C_string()); free_entry(p); + } else if (is_jfr_event_class(ik)) { + // We cannot include JFR event classes because they need runtime-specific + // instrumentation in order to work with -XX:FlightRecorderOptions=retransform=false. + // There are only a small number of these classes, so it's not worthwhile to + // support them and make CDS more complicated. + ResourceMark rm; + tty->print_cr("Skipping JFR event class %s", ik->name()->as_C_string()); + free_entry(p); } else { p->set_next(master_list); master_list = p; @@ -400,7 +418,7 @@ void Dictionary::reorder_dictionary_for_sharing() { set_entry(index, p); } } - +#endif SymbolPropertyTable::SymbolPropertyTable(int table_size) : Hashtable(table_size, sizeof(SymbolPropertyEntry)) diff --git a/src/hotspot/share/classfile/dictionary.hpp b/src/hotspot/share/classfile/dictionary.hpp index b2662a1b4c5..d4f56df6787 100644 --- a/src/hotspot/share/classfile/dictionary.hpp +++ b/src/hotspot/share/classfile/dictionary.hpp @@ -88,7 +88,7 @@ public: Handle protection_domain, TRAPS); // Sharing support - void reorder_dictionary_for_sharing(); + void reorder_dictionary_for_sharing() NOT_CDS_RETURN; void print_on(outputStream* st) const; void verify(); diff --git a/src/hotspot/share/classfile/systemDictionary.cpp b/src/hotspot/share/classfile/systemDictionary.cpp index 22bf5b037e7..b72200a6c30 100644 --- a/src/hotspot/share/classfile/systemDictionary.cpp +++ b/src/hotspot/share/classfile/systemDictionary.cpp @@ -2855,9 +2855,11 @@ ProtectionDomainCacheEntry* SystemDictionary::cache_get(Handle protection_domain return _pd_cache_table->get(protection_domain); } +#if INCLUDE_CDS void SystemDictionary::reorder_dictionary_for_sharing() { ClassLoaderData::the_null_class_loader_data()->dictionary()->reorder_dictionary_for_sharing(); } +#endif size_t SystemDictionary::count_bytes_for_buckets() { return ClassLoaderData::the_null_class_loader_data()->dictionary()->count_bytes_for_buckets(); diff --git a/src/hotspot/share/classfile/systemDictionary.hpp b/src/hotspot/share/classfile/systemDictionary.hpp index 17da32a2859..c890afaf321 100644 --- a/src/hotspot/share/classfile/systemDictionary.hpp +++ b/src/hotspot/share/classfile/systemDictionary.hpp @@ -384,7 +384,7 @@ public: public: // Sharing support. - static void reorder_dictionary_for_sharing(); + static void reorder_dictionary_for_sharing() NOT_CDS_RETURN; static void combine_shared_dictionaries(); static size_t count_bytes_for_buckets(); static size_t count_bytes_for_table(); From 43ac55ed1c9864988b50f200e6fc2798998cfd9a Mon Sep 17 00:00:00 2001 From: Martin Doerr Date: Mon, 30 Oct 2017 17:14:39 +0100 Subject: [PATCH 06/61] 8190285: s390: Some java boolean checks are not correct Reviewed-by: lucy, coleenp --- src/hotspot/cpu/s390/c1_LIRGenerator_s390.cpp | 13 +++++--- src/hotspot/cpu/s390/interp_masm_s390.cpp | 32 +++++++++++++++++++ src/hotspot/cpu/s390/interp_masm_s390.hpp | 2 ++ .../templateInterpreterGenerator_s390.cpp | 6 ++++ src/hotspot/cpu/s390/templateTable_s390.cpp | 21 +++++++++++- 5 files changed, 68 insertions(+), 6 deletions(-) diff --git a/src/hotspot/cpu/s390/c1_LIRGenerator_s390.cpp b/src/hotspot/cpu/s390/c1_LIRGenerator_s390.cpp index 8ebecadd85c..e7b7ac76cc0 100644 --- a/src/hotspot/cpu/s390/c1_LIRGenerator_s390.cpp +++ b/src/hotspot/cpu/s390/c1_LIRGenerator_s390.cpp @@ -277,7 +277,7 @@ void LIRGenerator::do_StoreIndexed(StoreIndexed* x) { length.set_instruction(x->length()); length.load_item(); } - if (needs_store_check) { + if (needs_store_check || x->check_boolean()) { value.load_item(); } else { value.load_for_store(x->elt_type()); @@ -327,11 +327,14 @@ void LIRGenerator::do_StoreIndexed(StoreIndexed* x) { // Needs GC write barriers. pre_barrier(LIR_OprFact::address(array_addr), LIR_OprFact::illegalOpr /* pre_val */, true /* do_load */, false /* patch */, NULL); - __ move(value.result(), array_addr, null_check_info); - // Seems to be a precise. + } + + LIR_Opr result = maybe_mask_boolean(x, array.result(), value.result(), null_check_info); + __ move(result, array_addr, null_check_info); + + if (obj_store) { + // Precise card mark post_barrier(LIR_OprFact::address(array_addr), value.result()); - } else { - __ move(value.result(), array_addr, null_check_info); } } diff --git a/src/hotspot/cpu/s390/interp_masm_s390.cpp b/src/hotspot/cpu/s390/interp_masm_s390.cpp index edcbc8f4b1d..382c9113e4c 100644 --- a/src/hotspot/cpu/s390/interp_masm_s390.cpp +++ b/src/hotspot/cpu/s390/interp_masm_s390.cpp @@ -843,6 +843,38 @@ void InterpreterMacroAssembler::unlock_if_synchronized_method(TosState state, verify_oop(Z_tos, state); } +void InterpreterMacroAssembler::narrow(Register result, Register ret_type) { + get_method(ret_type); + z_lg(ret_type, Address(ret_type, in_bytes(Method::const_offset()))); + z_lb(ret_type, Address(ret_type, in_bytes(ConstMethod::result_type_offset()))); + + Label notBool, notByte, notChar, done; + + // common case first + compareU32_and_branch(ret_type, T_INT, bcondEqual, done); + + compareU32_and_branch(ret_type, T_BOOLEAN, bcondNotEqual, notBool); + z_nilf(result, 0x1); + z_bru(done); + + bind(notBool); + compareU32_and_branch(ret_type, T_BYTE, bcondNotEqual, notByte); + z_lbr(result, result); + z_bru(done); + + bind(notByte); + compareU32_and_branch(ret_type, T_CHAR, bcondNotEqual, notChar); + z_nilf(result, 0xffff); + z_bru(done); + + bind(notChar); + // compareU32_and_branch(ret_type, T_SHORT, bcondNotEqual, notShort); + z_lhr(result, result); + + // Nothing to do for T_INT + bind(done); +} + // remove activation // // Unlock the receiver if this is a synchronized method. diff --git a/src/hotspot/cpu/s390/interp_masm_s390.hpp b/src/hotspot/cpu/s390/interp_masm_s390.hpp index bebbb4a7445..eff595648a0 100644 --- a/src/hotspot/cpu/s390/interp_masm_s390.hpp +++ b/src/hotspot/cpu/s390/interp_masm_s390.hpp @@ -86,6 +86,8 @@ class InterpreterMacroAssembler: public MacroAssembler { void dispatch_next_noverify_oop(TosState state, int step = 0); void dispatch_via(TosState state, address* table); + void narrow(Register result, Register ret_type); + // Jump to an invoked target. void prepare_to_jump_from_interpreted(Register method); void jump_from_interpreted(Register method, Register temp); diff --git a/src/hotspot/cpu/s390/templateInterpreterGenerator_s390.cpp b/src/hotspot/cpu/s390/templateInterpreterGenerator_s390.cpp index 5d94689b383..8c95be19dd2 100644 --- a/src/hotspot/cpu/s390/templateInterpreterGenerator_s390.cpp +++ b/src/hotspot/cpu/s390/templateInterpreterGenerator_s390.cpp @@ -2377,6 +2377,12 @@ address TemplateInterpreterGenerator::generate_earlyret_entry_for (TosState stat __ store_const(Address(RjvmtiState, JvmtiThreadState::earlyret_state_offset()), JvmtiThreadState::earlyret_inactive, 4, 4, Z_R0_scratch); + if (state == itos) { + // Narrow result if state is itos but result type is smaller. + // Need to narrow in the return bytecode rather than in generate_return_entry + // since compiled code callers expect the result to already be narrowed. + __ narrow(Z_tos, Z_tmp_1); /* fall through */ + } __ remove_activation(state, Z_tmp_1, // retaddr false, // throw_monitor_exception diff --git a/src/hotspot/cpu/s390/templateTable_s390.cpp b/src/hotspot/cpu/s390/templateTable_s390.cpp index 50477e6e556..7c1e458b9d6 100644 --- a/src/hotspot/cpu/s390/templateTable_s390.cpp +++ b/src/hotspot/cpu/s390/templateTable_s390.cpp @@ -1174,8 +1174,20 @@ void TemplateTable::bastore() { __ pop_i(Z_ARG3); __ pop_ptr(Z_tmp_2); // Z_tos : value - // Z_ARG3 : index + // Z_ARG3 : index // Z_tmp_2 : array + + // Need to check whether array is boolean or byte + // since both types share the bastore bytecode. + __ load_klass(Z_tmp_1, Z_tmp_2); + __ z_llgf(Z_tmp_1, Address(Z_tmp_1, Klass::layout_helper_offset())); + __ z_tmll(Z_tmp_1, Klass::layout_helper_boolean_diffbit()); + Label L_skip; + __ z_bfalse(L_skip); + // if it is a T_BOOLEAN array, mask the stored value to 0/1 + __ z_nilf(Z_tos, 0x1); + __ bind(L_skip); + // No index shift necessary - pass 0. index_check(Z_tmp_2, Z_ARG3, 0); // Prefer index in Z_ARG3. __ z_stc(Z_tos, @@ -2321,6 +2333,13 @@ void TemplateTable::_return(TosState state) { __ bind(skip_register_finalizer); } + if (state == itos) { + // Narrow result if state is itos but result type is smaller. + // Need to narrow in the return bytecode rather than in generate_return_entry + // since compiled code callers expect the result to already be narrowed. + __ narrow(Z_tos, Z_tmp_1); /* fall through */ + } + __ remove_activation(state, Z_R14); __ z_br(Z_R14); } From 832d68597ec6af94bc870a22410d1be95eb2bfa5 Mon Sep 17 00:00:00 2001 From: Tobias Hartmann Date: Tue, 31 Oct 2017 09:33:47 +0100 Subject: [PATCH 07/61] 8190351: InitialAndMaxUsageTest does not free allocated blob Refactored test to add blob to free list and changed the allocation loop to start with blobs of size ~36 Mb. Reviewed-by: kvn --- .../codecache/jmx/InitialAndMaxUsageTest.java | 28 +++++++++---------- 1 file changed, 13 insertions(+), 15 deletions(-) diff --git a/test/hotspot/jtreg/compiler/codecache/jmx/InitialAndMaxUsageTest.java b/test/hotspot/jtreg/compiler/codecache/jmx/InitialAndMaxUsageTest.java index 49b1fa53fe8..f56f2f70857 100644 --- a/test/hotspot/jtreg/compiler/codecache/jmx/InitialAndMaxUsageTest.java +++ b/test/hotspot/jtreg/compiler/codecache/jmx/InitialAndMaxUsageTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2014, 2017, 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 @@ -75,18 +75,12 @@ public class InitialAndMaxUsageTest { } } - private void fillWithSize(long size, List blobs, MemoryPoolMXBean bean) { - long blob; - /* Don't fill too much to have space for adapters. So, stop after crossing 95% and - don't allocate in case we'll cross 97% on next allocation. We can hit situation - like 94% -> (1 allocation) -> 100% otherwise. So, check if - is less than 97%, then allocate in case it is, then, stop - further allocations with given size in case more than 95% */ - while (((double) bean.getUsage().getUsed() + size <= (CACHE_USAGE_COEF + 0.02d) * maxSize) - && (blob = CodeCacheUtils.WB.allocateCodeBlob(size, btype.id)) != 0L - && ((double) bean.getUsage().getUsed() <= CACHE_USAGE_COEF * maxSize)) { - blobs.add(blob); - } + private boolean canAllocate(double size, long maxSize, MemoryPoolMXBean bean) { + // Don't fill too much to have space for adapters. So, stop after crossing 95% and + // don't allocate in case we'll cross 97% on next allocation. + double used = bean.getUsage().getUsed(); + return (used <= CACHE_USAGE_COEF * maxSize) && + (used + size <= (CACHE_USAGE_COEF + 0.02d) * maxSize); } protected void runTest() { @@ -106,8 +100,12 @@ public class InitialAndMaxUsageTest { lots of small allocations takes too much time, so, just a small optimization */ try { - for (int coef = 1000000; coef > 0; coef /= 10) { - fillWithSize(coef * minAllocationUnit, blobs, bean); + for (long size = 100_000 * minAllocationUnit; size > 0; size /= 10) { + long blob = 0; + while (canAllocate(size, maxSize, bean) && + (blob = CodeCacheUtils.WB.allocateCodeBlob(size, btype.id)) != 0) { + blobs.add(blob); + } } Asserts.assertGT((double) bean.getUsage().getUsed(), CACHE_USAGE_COEF * maxSize, String.format("Unable to fill " From b36ba5ff60a7b8c7babee34ba122994cfe873fe8 Mon Sep 17 00:00:00 2001 From: Robin Westberg Date: Mon, 30 Oct 2017 16:37:53 +0100 Subject: [PATCH 08/61] 8189368: Add information on current bias holder for BiasedLockRevocation event Reviewed-by: mgronlun, dholmes, egahlin --- src/hotspot/share/runtime/biasedLocking.cpp | 39 +++++++++++++++------ src/hotspot/share/trace/traceevents.xml | 1 + 2 files changed, 30 insertions(+), 10 deletions(-) diff --git a/src/hotspot/share/runtime/biasedLocking.cpp b/src/hotspot/share/runtime/biasedLocking.cpp index 4440b842ffd..de93a4b370a 100644 --- a/src/hotspot/share/runtime/biasedLocking.cpp +++ b/src/hotspot/share/runtime/biasedLocking.cpp @@ -145,7 +145,9 @@ static GrowableArray* get_or_compute_monitor_info(JavaThread* thre return info; } -static BiasedLocking::Condition revoke_bias(oop obj, bool allow_rebias, bool is_bulk, JavaThread* requesting_thread) { +// After the call, *biased_locker will be set to obj->mark()->biased_locker() if biased_locker != NULL, +// AND it is a living thread. Otherwise it will not be updated, (i.e. the caller is responsible for initialization). +static BiasedLocking::Condition revoke_bias(oop obj, bool allow_rebias, bool is_bulk, JavaThread* requesting_thread, JavaThread** biased_locker) { markOop mark = obj->mark(); if (!mark->has_bias_pattern()) { if (log_is_enabled(Info, biasedlocking)) { @@ -298,6 +300,11 @@ static BiasedLocking::Condition revoke_bias(oop obj, bool allow_rebias, bool is_ } } + // If requested, return information on which thread held the bias + if (biased_locker != NULL) { + *biased_locker = biased_thread; + } + return BiasedLocking::BIAS_REVOKED; } @@ -418,7 +425,7 @@ static BiasedLocking::Condition bulk_revoke_or_rebias_at_safepoint(oop o, // At this point we're done. All we have to do is potentially // adjust the header of the given object to revoke its bias. - revoke_bias(o, attempt_rebias_of_object && klass->prototype_header()->has_bias_pattern(), true, requesting_thread); + revoke_bias(o, attempt_rebias_of_object && klass->prototype_header()->has_bias_pattern(), true, requesting_thread, NULL); } else { if (log_is_enabled(Info, biasedlocking)) { ResourceMark rm; @@ -440,14 +447,14 @@ static BiasedLocking::Condition bulk_revoke_or_rebias_at_safepoint(oop o, oop owner = mon_info->owner(); markOop mark = owner->mark(); if ((owner->klass() == k_o) && mark->has_bias_pattern()) { - revoke_bias(owner, false, true, requesting_thread); + revoke_bias(owner, false, true, requesting_thread, NULL); } } } // Must force the bias of the passed object to be forcibly revoked // as well to ensure guarantees to callers - revoke_bias(o, false, true, requesting_thread); + revoke_bias(o, false, true, requesting_thread, NULL); } log_info(biasedlocking)("* Ending bulk revocation"); @@ -486,19 +493,22 @@ protected: GrowableArray* _objs; JavaThread* _requesting_thread; BiasedLocking::Condition _status_code; + traceid _biased_locker_id; public: VM_RevokeBias(Handle* obj, JavaThread* requesting_thread) : _obj(obj) , _objs(NULL) , _requesting_thread(requesting_thread) - , _status_code(BiasedLocking::NOT_BIASED) {} + , _status_code(BiasedLocking::NOT_BIASED) + , _biased_locker_id(0) {} VM_RevokeBias(GrowableArray* objs, JavaThread* requesting_thread) : _obj(NULL) , _objs(objs) , _requesting_thread(requesting_thread) - , _status_code(BiasedLocking::NOT_BIASED) {} + , _status_code(BiasedLocking::NOT_BIASED) + , _biased_locker_id(0) {} virtual VMOp_Type type() const { return VMOp_RevokeBias; } @@ -525,7 +535,11 @@ public: virtual void doit() { if (_obj != NULL) { log_info(biasedlocking)("Revoking bias with potentially per-thread safepoint:"); - _status_code = revoke_bias((*_obj)(), false, false, _requesting_thread); + JavaThread* biased_locker = NULL; + _status_code = revoke_bias((*_obj)(), false, false, _requesting_thread, &biased_locker); + if (biased_locker != NULL) { + _biased_locker_id = THREAD_TRACE_ID(biased_locker); + } clean_up_cached_monitor_info(); return; } else { @@ -537,6 +551,10 @@ public: BiasedLocking::Condition status_code() const { return _status_code; } + + traceid biased_locker() const { + return _biased_locker_id; + } }; @@ -645,7 +663,7 @@ BiasedLocking::Condition BiasedLocking::revoke_and_rebias(Handle obj, bool attem ResourceMark rm; log_info(biasedlocking)("Revoking bias by walking my own stack:"); EventBiasedLockSelfRevocation event; - BiasedLocking::Condition cond = revoke_bias(obj(), false, false, (JavaThread*) THREAD); + BiasedLocking::Condition cond = revoke_bias(obj(), false, false, (JavaThread*) THREAD, NULL); ((JavaThread*) THREAD)->set_cached_monitor_info(NULL); assert(cond == BIAS_REVOKED, "why not?"); if (event.should_commit()) { @@ -661,6 +679,7 @@ BiasedLocking::Condition BiasedLocking::revoke_and_rebias(Handle obj, bool attem event.set_lockClass(k); // Subtract 1 to match the id of events committed inside the safepoint event.set_safepointId(SafepointSynchronize::safepoint_counter() - 1); + event.set_previousOwner(revoke.biased_locker()); event.commit(); } return revoke.status_code(); @@ -700,7 +719,7 @@ void BiasedLocking::revoke_at_safepoint(Handle h_obj) { oop obj = h_obj(); HeuristicsResult heuristics = update_heuristics(obj, false); if (heuristics == HR_SINGLE_REVOKE) { - revoke_bias(obj, false, false, NULL); + revoke_bias(obj, false, false, NULL, NULL); } else if ((heuristics == HR_BULK_REBIAS) || (heuristics == HR_BULK_REVOKE)) { bulk_revoke_or_rebias_at_safepoint(obj, (heuristics == HR_BULK_REBIAS), false, NULL); @@ -716,7 +735,7 @@ void BiasedLocking::revoke_at_safepoint(GrowableArray* objs) { oop obj = (objs->at(i))(); HeuristicsResult heuristics = update_heuristics(obj, false); if (heuristics == HR_SINGLE_REVOKE) { - revoke_bias(obj, false, false, NULL); + revoke_bias(obj, false, false, NULL, NULL); } else if ((heuristics == HR_BULK_REBIAS) || (heuristics == HR_BULK_REVOKE)) { bulk_revoke_or_rebias_at_safepoint(obj, (heuristics == HR_BULK_REBIAS), false, NULL); diff --git a/src/hotspot/share/trace/traceevents.xml b/src/hotspot/share/trace/traceevents.xml index dbfc5760105..6b09e569571 100644 --- a/src/hotspot/share/trace/traceevents.xml +++ b/src/hotspot/share/trace/traceevents.xml @@ -108,6 +108,7 @@ Declares a structure type that can be used in other events. description="Revoked bias of object" has_thread="true" has_stacktrace="true" is_instant="false"> + Date: Tue, 31 Oct 2017 11:55:09 -0400 Subject: [PATCH 09/61] 8189610: Reconcile jvm.h and all jvm_md.h between java.base and hotspot Removed hotspot version of jvm*h and jni*h files. Reviewed-by: ihse, mchung, dholmes --- make/copy/Copy-java.base.gmk | 2 +- make/hotspot/lib/CompileJvm.gmk | 2 +- make/hotspot/lib/CompileLibjsig.gmk | 2 +- src/hotspot/cpu/aarch64/jni_aarch64.h | 60 - .../cpu/aarch64/macroAssembler_aarch64.cpp | 2 +- src/hotspot/cpu/arm/interp_masm_arm.cpp | 1 + src/hotspot/cpu/arm/jni_arm.h | 52 - src/hotspot/cpu/arm/methodHandles_arm.cpp | 2 +- src/hotspot/cpu/arm/vm_version_arm_32.cpp | 2 +- src/hotspot/cpu/arm/vm_version_arm_64.cpp | 2 +- src/hotspot/cpu/ppc/jni_ppc.h | 55 - src/hotspot/cpu/ppc/methodHandles_ppc.cpp | 2 +- src/hotspot/cpu/ppc/vm_version_ppc.cpp | 2 +- src/hotspot/cpu/s390/jni_s390.h | 45 - src/hotspot/cpu/s390/methodHandles_s390.cpp | 2 +- src/hotspot/cpu/s390/vm_version_s390.cpp | 2 +- src/hotspot/cpu/sparc/jni_sparc.h | 44 - .../cpu/sparc/macroAssembler_sparc.cpp | 2 +- src/hotspot/cpu/sparc/methodHandles_sparc.cpp | 2 +- src/hotspot/cpu/sparc/vm_version_sparc.cpp | 2 +- src/hotspot/cpu/x86/jni_x86.h | 63 - src/hotspot/cpu/x86/macroAssembler_x86.cpp | 2 +- src/hotspot/cpu/x86/methodHandles_x86.cpp | 2 +- src/hotspot/cpu/x86/vm_version_x86.cpp | 2 +- src/hotspot/cpu/zero/jni_zero.h | 50 - src/hotspot/os/aix/jvm_aix.cpp | 2 +- src/hotspot/os/aix/jvm_aix.h | 106 -- src/hotspot/os/aix/os_aix.cpp | 5 +- src/hotspot/os/bsd/decoder_machO.cpp | 2 +- src/hotspot/os/bsd/jvm_bsd.cpp | 2 +- src/hotspot/os/bsd/jvm_bsd.h | 115 -- src/hotspot/os/bsd/os_bsd.cpp | 5 +- src/hotspot/os/linux/decoder_linux.cpp | 2 +- src/hotspot/os/linux/jvm_linux.cpp | 2 +- src/hotspot/os/linux/jvm_linux.h | 97 -- src/hotspot/os/linux/os_linux.cpp | 5 +- src/hotspot/os/posix/os_posix.cpp | 2 +- src/hotspot/os/solaris/jvm_solaris.cpp | 2 +- src/hotspot/os/solaris/jvm_solaris.h | 98 -- src/hotspot/os/solaris/os_solaris.cpp | 3 +- src/hotspot/os/windows/jvm_windows.cpp | 2 +- src/hotspot/os/windows/jvm_windows.h | 118 -- src/hotspot/os/windows/os_windows.cpp | 10 +- .../os/windows/threadCritical_windows.cpp | 2 +- src/hotspot/os_cpu/aix_ppc/os_aix_ppc.cpp | 3 +- src/hotspot/os_cpu/bsd_x86/os_bsd_x86.cpp | 3 +- src/hotspot/os_cpu/bsd_zero/os_bsd_zero.cpp | 3 +- .../os_cpu/linux_aarch64/os_linux_aarch64.cpp | 3 +- src/hotspot/os_cpu/linux_arm/os_linux_arm.cpp | 3 +- src/hotspot/os_cpu/linux_ppc/os_linux_ppc.cpp | 3 +- .../os_cpu/linux_s390/os_linux_s390.cpp | 3 +- .../os_cpu/linux_sparc/os_linux_sparc.cpp | 3 +- src/hotspot/os_cpu/linux_x86/os_linux_x86.cpp | 3 +- .../os_cpu/linux_zero/os_linux_zero.cpp | 3 +- .../os_cpu/solaris_sparc/os_solaris_sparc.cpp | 3 +- .../os_cpu/solaris_x86/os_solaris_x86.cpp | 3 +- .../os_cpu/windows_x86/os_windows_x86.cpp | 3 +- src/hotspot/share/aot/aotLoader.cpp | 2 +- src/hotspot/share/c1/c1_CFGPrinter.cpp | 2 +- src/hotspot/share/c1/c1_Canonicalizer.hpp | 4 + src/hotspot/share/c1/c1_LinearScan.cpp | 2 +- src/hotspot/share/ci/ciEnv.cpp | 2 +- src/hotspot/share/ci/ciFlags.hpp | 2 +- src/hotspot/share/ci/ciReplay.cpp | 4 +- src/hotspot/share/classfile/altHashing.cpp | 22 +- src/hotspot/share/classfile/altHashing.hpp | 4 +- .../share/classfile/classFileParser.cpp | 2 +- src/hotspot/share/classfile/classLoader.cpp | 2 +- .../share/classfile/compactHashtable.cpp | 2 +- src/hotspot/share/classfile/modules.cpp | 2 +- .../share/classfile/systemDictionary.cpp | 2 +- src/hotspot/share/classfile/verifier.cpp | 2 +- src/hotspot/share/classfile/vmSymbols.cpp | 2 +- src/hotspot/share/code/codeBlob.cpp | 2 +- src/hotspot/share/code/nmethod.cpp | 2 +- src/hotspot/share/code/relocInfo.cpp | 2 +- src/hotspot/share/compiler/compileBroker.cpp | 4 +- src/hotspot/share/compiler/compileBroker.hpp | 2 +- src/hotspot/share/compiler/compileLog.cpp | 2 +- src/hotspot/share/compiler/compilerOracle.cpp | 2 +- src/hotspot/share/gc/g1/g1CollectedHeap.cpp | 8 +- src/hotspot/share/gc/g1/satbMarkQueue.cpp | 2 +- src/hotspot/share/gc/shared/ageTable.cpp | 2 +- src/hotspot/share/gc/shared/gcId.cpp | 2 +- src/hotspot/share/gc/shared/gcTimer.hpp | 1 - .../share/gc/shared/gcTraceTime.inline.hpp | 1 - .../share/interpreter/linkResolver.cpp | 2 +- .../share/jvmci/jvmciCodeInstaller.cpp | 2 +- src/hotspot/share/jvmci/jvmciCompiler.cpp | 2 +- src/hotspot/share/jvmci/jvmciCompilerToVM.cpp | 2 +- src/hotspot/share/jvmci/jvmciRuntime.cpp | 4 +- src/hotspot/share/jvmci/jvmci_globals.cpp | 2 +- .../share/logging/logConfiguration.cpp | 2 +- src/hotspot/share/logging/logDecorations.cpp | 2 +- src/hotspot/share/logging/logFileOutput.cpp | 2 +- .../share/logging/logFileStreamOutput.cpp | 2 +- src/hotspot/share/logging/logOutput.cpp | 2 +- src/hotspot/share/logging/logTagSet.cpp | 2 +- src/hotspot/share/memory/filemap.cpp | 2 +- src/hotspot/share/memory/metaspaceShared.cpp | 2 +- src/hotspot/share/oops/constantPool.cpp | 2 +- src/hotspot/share/oops/generateOopMap.cpp | 2 +- src/hotspot/share/oops/instanceKlass.cpp | 2 +- src/hotspot/share/oops/klassVtable.cpp | 2 +- src/hotspot/share/opto/castnode.cpp | 4 +- src/hotspot/share/opto/loopTransform.cpp | 2 +- src/hotspot/share/opto/mulnode.cpp | 8 +- src/hotspot/share/opto/output.cpp | 8 +- src/hotspot/share/opto/parse3.cpp | 8 +- src/hotspot/share/opto/phaseX.cpp | 2 +- src/hotspot/share/opto/type.cpp | 12 +- src/hotspot/share/precompiled/precompiled.hpp | 2 +- src/hotspot/share/prims/jni.cpp | 26 +- src/hotspot/share/prims/jniCheck.cpp | 2 +- src/hotspot/share/prims/jni_md.h | 39 - src/hotspot/share/prims/jvm.cpp | 6 +- src/hotspot/share/prims/jvm.h | 1373 ----------------- src/hotspot/share/prims/jvmtiEnvBase.cpp | 2 +- src/hotspot/share/prims/jvmtiImpl.cpp | 4 +- src/hotspot/share/prims/jvmtiTagMap.cpp | 2 +- src/hotspot/share/prims/methodHandles.hpp | 1 - src/hotspot/share/prims/perf.cpp | 2 +- src/hotspot/share/prims/unsafe.cpp | 2 +- src/hotspot/share/runtime/arguments.cpp | 2 +- .../runtime/commandLineFlagRangeList.cpp | 2 +- src/hotspot/share/runtime/deoptimization.cpp | 2 +- src/hotspot/share/runtime/globals.cpp | 2 +- .../share/runtime/interfaceSupport.cpp | 2 +- src/hotspot/share/runtime/java.cpp | 2 +- src/hotspot/share/runtime/os.cpp | 2 +- src/hotspot/share/runtime/os.hpp | 2 +- src/hotspot/share/runtime/perfData.cpp | 6 +- src/hotspot/share/runtime/perfMemory.cpp | 4 +- src/hotspot/share/runtime/perfMemory.hpp | 2 +- src/hotspot/share/runtime/reflection.cpp | 2 +- src/hotspot/share/runtime/sharedRuntime.cpp | 2 +- src/hotspot/share/runtime/thread.cpp | 4 +- src/hotspot/share/runtime/thread.hpp | 2 +- src/hotspot/share/runtime/vmStructs.cpp | 4 +- .../share/services/diagnosticArgument.cpp | 2 +- .../share/services/diagnosticCommand.cpp | 2 +- .../share/services/diagnosticFramework.cpp | 2 +- src/hotspot/share/services/heapDumper.cpp | 2 +- .../share/services/mallocSiteTable.cpp | 4 +- src/hotspot/share/services/memTracker.cpp | 2 +- src/hotspot/share/utilities/accessFlags.hpp | 2 +- src/hotspot/share/utilities/constantTag.hpp | 2 +- src/hotspot/share/utilities/debug.cpp | 4 +- src/hotspot/share/utilities/decoder.cpp | 2 +- .../share/utilities/elfStringTable.cpp | 2 +- src/hotspot/share/utilities/formatBuffer.cpp | 2 +- src/hotspot/share/utilities/formatBuffer.hpp | 2 +- .../share/utilities/globalDefinitions_xlc.hpp | 1 + src/hotspot/share/utilities/macros.hpp | 2 + src/hotspot/share/utilities/ostream.cpp | 2 +- src/hotspot/share/utilities/vmError.cpp | 2 +- src/java.base/macosx/native/include/jni_md.h | 42 - src/java.base/macosx/native/include/jvm_md.h | 80 - .../native/include/classfile_constants.h | 27 +- src/java.base/share/native/include/jvm.h | 21 +- src/java.base/solaris/native/libjsig/jsig.c | 2 +- src/java.base/unix/native/include/jni_md.h | 11 +- src/java.base/unix/native/include/jvm_md.h | 33 +- .../unix/native/libjava/UnixFileSystem_md.c | 8 +- src/java.base/windows/native/include/jni_md.h | 1 + src/java.base/windows/native/include/jvm_md.h | 37 +- .../gtest/classfile/test_AltHashing.cpp | 2 +- test/hotspot/gtest/logging/logTestFixture.cpp | 2 +- test/hotspot/gtest/logging/test_log.cpp | 2 +- .../gtest/logging/test_logConfiguration.cpp | 2 +- .../gtest/logging/test_logDecorators.cpp | 2 +- .../gtest/logging/test_logFileOutput.cpp | 2 +- .../gtest/logging/test_logMessageTest.cpp | 2 +- .../logging/test_logTagLevelExpression.cpp | 2 +- .../logging/test_logTagSetDescriptions.cpp | 2 +- test/hotspot/gtest/runtime/test_arguments.cpp | 4 +- test/hotspot/gtest/utilities/test_json.cpp | 2 +- 177 files changed, 308 insertions(+), 2713 deletions(-) delete mode 100644 src/hotspot/cpu/aarch64/jni_aarch64.h delete mode 100644 src/hotspot/cpu/arm/jni_arm.h delete mode 100644 src/hotspot/cpu/ppc/jni_ppc.h delete mode 100644 src/hotspot/cpu/s390/jni_s390.h delete mode 100644 src/hotspot/cpu/sparc/jni_sparc.h delete mode 100644 src/hotspot/cpu/x86/jni_x86.h delete mode 100644 src/hotspot/cpu/zero/jni_zero.h delete mode 100644 src/hotspot/os/aix/jvm_aix.h delete mode 100644 src/hotspot/os/bsd/jvm_bsd.h delete mode 100644 src/hotspot/os/linux/jvm_linux.h delete mode 100644 src/hotspot/os/solaris/jvm_solaris.h delete mode 100644 src/hotspot/os/windows/jvm_windows.h delete mode 100644 src/hotspot/share/prims/jni_md.h delete mode 100644 src/hotspot/share/prims/jvm.h delete mode 100644 src/java.base/macosx/native/include/jni_md.h delete mode 100644 src/java.base/macosx/native/include/jvm_md.h diff --git a/make/copy/Copy-java.base.gmk b/make/copy/Copy-java.base.gmk index 89d7d791c06..cc489211b24 100644 --- a/make/copy/Copy-java.base.gmk +++ b/make/copy/Copy-java.base.gmk @@ -42,7 +42,7 @@ $(INCLUDE_DST_DIR)/%.h: $(TOPDIR)/src/java.base/share/native/include/%.h $(call install-file) $(INCLUDE_DST_OS_DIR)/%.h: \ - $(TOPDIR)/src/java.base/$(OPENJDK_TARGET_OS_EXPORT_DIR)/native/include/%.h + $(TOPDIR)/src/java.base/$(OPENJDK_TARGET_OS_TYPE)/native/include/%.h $(call install-file) ################################################################################ diff --git a/make/hotspot/lib/CompileJvm.gmk b/make/hotspot/lib/CompileJvm.gmk index 5c7d7177420..016dedfca32 100644 --- a/make/hotspot/lib/CompileJvm.gmk +++ b/make/hotspot/lib/CompileJvm.gmk @@ -57,8 +57,8 @@ JVM_CFLAGS_INCLUDES += \ $(patsubst %,-I%,$(filter-out $(JVM_VARIANT_OUTPUTDIR)/gensrc/%, $(JVM_SRC_DIRS))) \ -I$(JVM_VARIANT_OUTPUTDIR)/gensrc \ -I$(TOPDIR)/src/hotspot/share/precompiled \ - -I$(TOPDIR)/src/hotspot/share/prims \ -I$(TOPDIR)/src/java.base/share/native/include \ + -I$(TOPDIR)/src/java.base/$(OPENJDK_TARGET_OS_TYPE)/native/include \ # # INCLUDE_SUFFIX_* is only meant for including the proper diff --git a/make/hotspot/lib/CompileLibjsig.gmk b/make/hotspot/lib/CompileLibjsig.gmk index 9c634296e96..b32157aebec 100644 --- a/make/hotspot/lib/CompileLibjsig.gmk +++ b/make/hotspot/lib/CompileLibjsig.gmk @@ -57,7 +57,7 @@ ifneq ($(OPENJDK_TARGET_OS), windows) endif else ifeq ($(OPENJDK_TARGET_OS), solaris) - LIBJSIG_CFLAGS := -m64 -KPIC -mt -I $(TOPDIR)/src/hotspot/os/solaris + LIBJSIG_CFLAGS := -m64 -KPIC -mt -I $(TOPDIR)/src/java.base/unix/native/include LIBJSIG_LDFLAGS := -m64 -mt -xnolib LIBJSIG_LIBS := $(LIBDL) diff --git a/src/hotspot/cpu/aarch64/jni_aarch64.h b/src/hotspot/cpu/aarch64/jni_aarch64.h deleted file mode 100644 index d530e9e1755..00000000000 --- a/src/hotspot/cpu/aarch64/jni_aarch64.h +++ /dev/null @@ -1,60 +0,0 @@ -/* - * Copyright (c) 1997, 2013, 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. - */ - -#ifndef _JAVASOFT_JNI_MD_H_ -#define _JAVASOFT_JNI_MD_H_ - -#if defined(SOLARIS) || defined(LINUX) || defined(_ALLBSD_SOURCE) - - -// Note: please do not change these without also changing jni_md.h in the JDK -// repository -#ifndef __has_attribute - #define __has_attribute(x) 0 -#endif -#if (defined(__GNUC__) && ((__GNUC__ > 4) || (__GNUC__ == 4) && (__GNUC_MINOR__ > 2))) || __has_attribute(visibility) - #define JNIEXPORT __attribute__((visibility("default"))) - #define JNIIMPORT __attribute__((visibility("default"))) -#else - #define JNIEXPORT - #define JNIIMPORT -#endif - - #define JNICALL - typedef int jint; - typedef long jlong; - -#else - #define JNIEXPORT __declspec(dllexport) - #define JNIIMPORT __declspec(dllimport) - #define JNICALL __stdcall - - typedef int jint; - typedef __int64 jlong; -#endif - -typedef signed char jbyte; - -#endif /* !_JAVASOFT_JNI_MD_H_ */ diff --git a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp index a7fb3df4ee7..b4dfe38ed93 100644 --- a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp @@ -26,6 +26,7 @@ #include #include "precompiled.hpp" +#include "jvm.h" #include "asm/assembler.hpp" #include "asm/assembler.inline.hpp" #include "interpreter/interpreter.hpp" @@ -38,7 +39,6 @@ #include "opto/compile.hpp" #include "opto/intrinsicnode.hpp" #include "opto/node.hpp" -#include "prims/jvm.h" #include "runtime/biasedLocking.hpp" #include "runtime/icache.hpp" #include "runtime/interfaceSupport.hpp" diff --git a/src/hotspot/cpu/arm/interp_masm_arm.cpp b/src/hotspot/cpu/arm/interp_masm_arm.cpp index 2a8b8ef853f..d3b6feb11d7 100644 --- a/src/hotspot/cpu/arm/interp_masm_arm.cpp +++ b/src/hotspot/cpu/arm/interp_masm_arm.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "gc/shared/barrierSet.inline.hpp" #include "gc/shared/cardTableModRefBS.inline.hpp" #include "gc/shared/collectedHeap.hpp" diff --git a/src/hotspot/cpu/arm/jni_arm.h b/src/hotspot/cpu/arm/jni_arm.h deleted file mode 100644 index 2a1d4f717b7..00000000000 --- a/src/hotspot/cpu/arm/jni_arm.h +++ /dev/null @@ -1,52 +0,0 @@ -/* - * Copyright (c) 2008, 2013, 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. - */ - -#ifndef _JAVASOFT_JNI_MD_H_ -#define _JAVASOFT_JNI_MD_H_ - -// Note: please do not change these without also changing jni_md.h in the JDK -// repository -#ifndef __has_attribute - #define __has_attribute(x) 0 -#endif -#if (defined(__GNUC__) && ((__GNUC__ > 4) || (__GNUC__ == 4) && (__GNUC_MINOR__ > 2))) || __has_attribute(visibility) - #define JNIEXPORT __attribute__((externally_visible,visibility("default"))) - #define JNIIMPORT __attribute__((externally_visible,visibility("default"))) -#else - #define JNIEXPORT - #define JNIIMPORT -#endif - -#define JNICALL - -typedef int jint; -#if defined(_LP64) - typedef long jlong; -#else - typedef long long jlong; -#endif -typedef signed char jbyte; - -#endif /* !_JAVASOFT_JNI_MD_H_ */ diff --git a/src/hotspot/cpu/arm/methodHandles_arm.cpp b/src/hotspot/cpu/arm/methodHandles_arm.cpp index 83188f92cb6..efc3394ef55 100644 --- a/src/hotspot/cpu/arm/methodHandles_arm.cpp +++ b/src/hotspot/cpu/arm/methodHandles_arm.cpp @@ -27,12 +27,12 @@ // Last synchronization: changeset f8c9417e3571 #include "precompiled.hpp" +#include "jvm.h" #include "classfile/javaClasses.inline.hpp" #include "interpreter/interpreter.hpp" #include "interpreter/interpreterRuntime.hpp" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "prims/methodHandles.hpp" #define __ _masm-> diff --git a/src/hotspot/cpu/arm/vm_version_arm_32.cpp b/src/hotspot/cpu/arm/vm_version_arm_32.cpp index 2d1d4949860..e1bdf1017a2 100644 --- a/src/hotspot/cpu/arm/vm_version_arm_32.cpp +++ b/src/hotspot/cpu/arm/vm_version_arm_32.cpp @@ -23,9 +23,9 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/java.hpp" #include "runtime/os.inline.hpp" #include "runtime/stubCodeGenerator.hpp" diff --git a/src/hotspot/cpu/arm/vm_version_arm_64.cpp b/src/hotspot/cpu/arm/vm_version_arm_64.cpp index 55091befc78..43da23b7fab 100644 --- a/src/hotspot/cpu/arm/vm_version_arm_64.cpp +++ b/src/hotspot/cpu/arm/vm_version_arm_64.cpp @@ -23,9 +23,9 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/java.hpp" #include "runtime/os.inline.hpp" #include "runtime/stubCodeGenerator.hpp" diff --git a/src/hotspot/cpu/ppc/jni_ppc.h b/src/hotspot/cpu/ppc/jni_ppc.h deleted file mode 100644 index 64e4b9bf061..00000000000 --- a/src/hotspot/cpu/ppc/jni_ppc.h +++ /dev/null @@ -1,55 +0,0 @@ -/* - * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright (c) 2012, 2013 SAP SE. 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. - */ - -#ifndef CPU_PPC_VM_JNI_PPC_H -#define CPU_PPC_VM_JNI_PPC_H - -// Note: please do not change these without also changing jni_md.h in the JDK -// repository -#ifndef __has_attribute - #define __has_attribute(x) 0 -#endif -#if (defined(__GNUC__) && ((__GNUC__ > 4) || (__GNUC__ == 4) && (__GNUC_MINOR__ > 2))) || __has_attribute(visibility) - #define JNIEXPORT __attribute__((visibility("default"))) - #define JNIIMPORT __attribute__((visibility("default"))) -#else - #define JNIEXPORT - #define JNIIMPORT -#endif - -#define JNICALL - -typedef int jint; - -#if defined(_LP64) - typedef long jlong; -#else - typedef long long jlong; -#endif - -typedef signed char jbyte; - -#endif // CPU_PPC_VM_JNI_PPC_H diff --git a/src/hotspot/cpu/ppc/methodHandles_ppc.cpp b/src/hotspot/cpu/ppc/methodHandles_ppc.cpp index 683934a1fb7..65b5c93945f 100644 --- a/src/hotspot/cpu/ppc/methodHandles_ppc.cpp +++ b/src/hotspot/cpu/ppc/methodHandles_ppc.cpp @@ -24,12 +24,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.inline.hpp" #include "classfile/javaClasses.inline.hpp" #include "interpreter/interpreter.hpp" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "prims/methodHandles.hpp" #define __ _masm-> diff --git a/src/hotspot/cpu/ppc/vm_version_ppc.cpp b/src/hotspot/cpu/ppc/vm_version_ppc.cpp index a145a52c48a..a27fb75b395 100644 --- a/src/hotspot/cpu/ppc/vm_version_ppc.cpp +++ b/src/hotspot/cpu/ppc/vm_version_ppc.cpp @@ -24,11 +24,11 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/assembler.inline.hpp" #include "asm/macroAssembler.inline.hpp" #include "compiler/disassembler.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/java.hpp" #include "runtime/os.hpp" #include "runtime/stubCodeGenerator.hpp" diff --git a/src/hotspot/cpu/s390/jni_s390.h b/src/hotspot/cpu/s390/jni_s390.h deleted file mode 100644 index fac3f8fe157..00000000000 --- a/src/hotspot/cpu/s390/jni_s390.h +++ /dev/null @@ -1,45 +0,0 @@ -/* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. - * Copyright (c) 2016 SAP SE. 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. - * - */ - -#ifndef _JAVASOFT_JNI_MD_H_ -#define _JAVASOFT_JNI_MD_H_ - -#if defined(__GNUC__) && (__GNUC__ >= 4) - #define JNIEXPORT __attribute__((visibility("default"))) - #define JNIIMPORT __attribute__((visibility("default"))) -#else - #define JNIEXPORT - #define JNIIMPORT -#endif - -#define JNICALL - -typedef int jint; - -typedef long int jlong; - -typedef signed char jbyte; - -#endif // _JAVASOFT_JNI_MD_H_ diff --git a/src/hotspot/cpu/s390/methodHandles_s390.cpp b/src/hotspot/cpu/s390/methodHandles_s390.cpp index d5f9c957fe7..8f7f8c769ea 100644 --- a/src/hotspot/cpu/s390/methodHandles_s390.cpp +++ b/src/hotspot/cpu/s390/methodHandles_s390.cpp @@ -24,12 +24,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.inline.hpp" #include "classfile/javaClasses.inline.hpp" #include "interpreter/interpreter.hpp" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "prims/methodHandles.hpp" #ifdef PRODUCT diff --git a/src/hotspot/cpu/s390/vm_version_s390.cpp b/src/hotspot/cpu/s390/vm_version_s390.cpp index 51383d6db75..72d30b176d9 100644 --- a/src/hotspot/cpu/s390/vm_version_s390.cpp +++ b/src/hotspot/cpu/s390/vm_version_s390.cpp @@ -24,11 +24,11 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/assembler.inline.hpp" #include "compiler/disassembler.hpp" #include "code/compiledIC.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/java.hpp" #include "runtime/stubCodeGenerator.hpp" #include "vm_version_s390.hpp" diff --git a/src/hotspot/cpu/sparc/jni_sparc.h b/src/hotspot/cpu/sparc/jni_sparc.h deleted file mode 100644 index 3402f67c992..00000000000 --- a/src/hotspot/cpu/sparc/jni_sparc.h +++ /dev/null @@ -1,44 +0,0 @@ -/* - * Copyright (c) 1997, 2017, 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. - */ - -// Note: please do not change these without also changing jni_md.h in the JDK -// repository -#ifndef __has_attribute - #define __has_attribute(x) 0 -#endif -#if (defined(__GNUC__) && ((__GNUC__ > 4) || (__GNUC__ == 4) && (__GNUC_MINOR__ > 2))) || __has_attribute(visibility) - #define JNIEXPORT __attribute__((visibility("default"))) - #define JNIIMPORT __attribute__((visibility("default"))) -#else - #define JNIEXPORT - #define JNIIMPORT -#endif -#define JNICALL - -typedef int jint; - -typedef long jlong; - -typedef signed char jbyte; diff --git a/src/hotspot/cpu/sparc/macroAssembler_sparc.cpp b/src/hotspot/cpu/sparc/macroAssembler_sparc.cpp index 70a65b07298..ebd6baa63bb 100644 --- a/src/hotspot/cpu/sparc/macroAssembler_sparc.cpp +++ b/src/hotspot/cpu/sparc/macroAssembler_sparc.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.inline.hpp" #include "compiler/disassembler.hpp" #include "gc/shared/cardTableModRefBS.hpp" @@ -31,7 +32,6 @@ #include "memory/resourceArea.hpp" #include "memory/universe.hpp" #include "oops/klass.inline.hpp" -#include "prims/jvm.h" #include "prims/methodHandles.hpp" #include "runtime/biasedLocking.hpp" #include "runtime/interfaceSupport.hpp" diff --git a/src/hotspot/cpu/sparc/methodHandles_sparc.cpp b/src/hotspot/cpu/sparc/methodHandles_sparc.cpp index be800aafc08..09fdfbeaaa6 100644 --- a/src/hotspot/cpu/sparc/methodHandles_sparc.cpp +++ b/src/hotspot/cpu/sparc/methodHandles_sparc.cpp @@ -23,13 +23,13 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "classfile/javaClasses.inline.hpp" #include "interpreter/interpreter.hpp" #include "interpreter/interp_masm.hpp" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "prims/methodHandles.hpp" #define __ _masm-> diff --git a/src/hotspot/cpu/sparc/vm_version_sparc.cpp b/src/hotspot/cpu/sparc/vm_version_sparc.cpp index b5ef619c35e..584b1b1609e 100644 --- a/src/hotspot/cpu/sparc/vm_version_sparc.cpp +++ b/src/hotspot/cpu/sparc/vm_version_sparc.cpp @@ -23,11 +23,11 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.inline.hpp" #include "logging/log.hpp" #include "logging/logStream.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/java.hpp" #include "runtime/os.hpp" #include "runtime/stubCodeGenerator.hpp" diff --git a/src/hotspot/cpu/x86/jni_x86.h b/src/hotspot/cpu/x86/jni_x86.h deleted file mode 100644 index 0c5030de79e..00000000000 --- a/src/hotspot/cpu/x86/jni_x86.h +++ /dev/null @@ -1,63 +0,0 @@ -/* - * Copyright (c) 1997, 2013, 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. - */ - -#ifndef _JAVASOFT_JNI_MD_H_ -#define _JAVASOFT_JNI_MD_H_ - -#if defined(_WIN32) - #define JNIEXPORT __declspec(dllexport) - #define JNIIMPORT __declspec(dllimport) - #define JNICALL __stdcall - - typedef int jint; - typedef __int64 jlong; -#else - -// Note: please do not change these without also changing jni_md.h in the JDK -// repository -#ifndef __has_attribute - #define __has_attribute(x) 0 -#endif -#if (defined(__GNUC__) && ((__GNUC__ > 4) || (__GNUC__ == 4) && (__GNUC_MINOR__ > 2))) || __has_attribute(visibility) - #define JNIEXPORT __attribute__((visibility("default"))) - #define JNIIMPORT __attribute__((visibility("default"))) -#else - #define JNIEXPORT - #define JNIIMPORT -#endif - - #define JNICALL - typedef int jint; -#if defined(_LP64) - typedef long jlong; -#else - typedef long long jlong; -#endif - -#endif - -typedef signed char jbyte; - -#endif /* !_JAVASOFT_JNI_MD_H_ */ diff --git a/src/hotspot/cpu/x86/macroAssembler_x86.cpp b/src/hotspot/cpu/x86/macroAssembler_x86.cpp index e2be5a3a6a4..8b6b4232850 100644 --- a/src/hotspot/cpu/x86/macroAssembler_x86.cpp +++ b/src/hotspot/cpu/x86/macroAssembler_x86.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/assembler.hpp" #include "asm/assembler.inline.hpp" #include "compiler/disassembler.hpp" @@ -32,7 +33,6 @@ #include "memory/resourceArea.hpp" #include "memory/universe.hpp" #include "oops/klass.inline.hpp" -#include "prims/jvm.h" #include "prims/methodHandles.hpp" #include "runtime/biasedLocking.hpp" #include "runtime/interfaceSupport.hpp" diff --git a/src/hotspot/cpu/x86/methodHandles_x86.cpp b/src/hotspot/cpu/x86/methodHandles_x86.cpp index 81d57ed3422..a329247a70a 100644 --- a/src/hotspot/cpu/x86/methodHandles_x86.cpp +++ b/src/hotspot/cpu/x86/methodHandles_x86.cpp @@ -23,13 +23,13 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "classfile/javaClasses.inline.hpp" #include "interpreter/interpreter.hpp" #include "interpreter/interpreterRuntime.hpp" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "prims/methodHandles.hpp" #define __ _masm-> diff --git a/src/hotspot/cpu/x86/vm_version_x86.cpp b/src/hotspot/cpu/x86/vm_version_x86.cpp index 6e3acc22315..22e7b653f22 100644 --- a/src/hotspot/cpu/x86/vm_version_x86.cpp +++ b/src/hotspot/cpu/x86/vm_version_x86.cpp @@ -23,12 +23,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "asm/macroAssembler.inline.hpp" #include "logging/log.hpp" #include "logging/logStream.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/java.hpp" #include "runtime/os.hpp" #include "runtime/stubCodeGenerator.hpp" diff --git a/src/hotspot/cpu/zero/jni_zero.h b/src/hotspot/cpu/zero/jni_zero.h deleted file mode 100644 index ecf743b65e1..00000000000 --- a/src/hotspot/cpu/zero/jni_zero.h +++ /dev/null @@ -1,50 +0,0 @@ -/* - * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2009 Red Hat, Inc. - * 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. - */ - - - -// Note: please do not change these without also changing jni_md.h in the JDK -// repository -#ifndef __has_attribute - #define __has_attribute(x) 0 -#endif -#if (defined(__GNUC__) && ((__GNUC__ > 4) || (__GNUC__ == 4) && (__GNUC_MINOR__ > 2))) || __has_attribute(visibility) - #define JNIEXPORT __attribute__((visibility("default"))) - #define JNIIMPORT __attribute__((visibility("default"))) -#else - #define JNIEXPORT - #define JNIIMPORT -#endif -#define JNICALL - -typedef int jint; -typedef signed char jbyte; - -#ifdef _LP64 -typedef long jlong; -#else -typedef long long jlong; -#endif diff --git a/src/hotspot/os/aix/jvm_aix.cpp b/src/hotspot/os/aix/jvm_aix.cpp index d6e9038c4c7..50a56d061dc 100644 --- a/src/hotspot/os/aix/jvm_aix.cpp +++ b/src/hotspot/os/aix/jvm_aix.cpp @@ -24,7 +24,7 @@ */ #include "precompiled.hpp" -#include "prims/jvm.h" +#include "jvm.h" #include "runtime/interfaceSupport.hpp" #include "runtime/osThread.hpp" diff --git a/src/hotspot/os/aix/jvm_aix.h b/src/hotspot/os/aix/jvm_aix.h deleted file mode 100644 index 2808e469d4f..00000000000 --- a/src/hotspot/os/aix/jvm_aix.h +++ /dev/null @@ -1,106 +0,0 @@ -/* - * Copyright (c) 1999, 2017, Oracle and/or its affiliates. All rights reserved. - * Copyright (c) 2012, 2013 SAP SE. 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. - * - */ - -#ifndef OS_AIX_VM_JVM_AIX_H -#define OS_AIX_VM_JVM_AIX_H - -// HotSpot integration note: -// -// This is derived from the JDK classic file: -// "$JDK/src/solaris/javavm/export/jvm_md.h":15 (ver. 1.10 98/04/22) -// All local includes have been commented out. - -#ifndef JVM_MD_H -#define JVM_MD_H - -/* - * This file is currently collecting system-specific dregs for the - * JNI conversion, which should be sorted out later. - */ - -#include /* For DIR */ - -// Must redefine NULL because the macro gets redefined to int 0 -// by dirent.h. This redefinition is included later then the standard definition in -// globalDefinitions_.hpp and leads to assertions in the VM initialization. -// We definitely need NULL to have the same lengh as an address pointer. -#ifdef _LP64 -#undef NULL -#define NULL 0L -#else -#ifndef NULL -#define NULL 0 -#endif -#endif - -#include /* For MAXPATHLEN */ -#include /* For socklen_t */ -#include /* For F_OK, R_OK, W_OK */ - -#define JNI_ONLOAD_SYMBOLS {"JNI_OnLoad"} -#define JNI_ONUNLOAD_SYMBOLS {"JNI_OnUnload"} -#define JVM_ONLOAD_SYMBOLS {"JVM_OnLoad"} -#define AGENT_ONLOAD_SYMBOLS {"Agent_OnLoad"} -#define AGENT_ONUNLOAD_SYMBOLS {"Agent_OnUnload"} -#define AGENT_ONATTACH_SYMBOLS {"Agent_OnAttach"} - -#define JNI_LIB_PREFIX "lib" -#define JNI_LIB_SUFFIX ".so" - -#define JVM_MAXPATHLEN MAXPATHLEN - -#define JVM_R_OK R_OK -#define JVM_W_OK W_OK -#define JVM_X_OK X_OK -#define JVM_F_OK F_OK - -/* - * File I/O - */ - -#include -#include -#include -#include - -/* O Flags */ - -#define JVM_O_RDONLY O_RDONLY -#define JVM_O_WRONLY O_WRONLY -#define JVM_O_RDWR O_RDWR -#define JVM_O_O_APPEND O_APPEND -#define JVM_O_EXCL O_EXCL -#define JVM_O_CREAT O_CREAT - -/* Signal definitions */ - -#define BREAK_SIGNAL SIGQUIT /* Thread dumping support. */ -#define SHUTDOWN1_SIGNAL SIGHUP /* Shutdown Hooks support. */ -#define SHUTDOWN2_SIGNAL SIGINT -#define SHUTDOWN3_SIGNAL SIGTERM - -#endif /* JVM_MD_H */ - -#endif // OS_AIX_VM_JVM_AIX_H diff --git a/src/hotspot/os/aix/os_aix.cpp b/src/hotspot/os/aix/os_aix.cpp index eaf7a4f2ae9..c97044fc6c2 100644 --- a/src/hotspot/os/aix/os_aix.cpp +++ b/src/hotspot/os/aix/os_aix.cpp @@ -28,6 +28,7 @@ #pragma alloca // no precompiled headers +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -35,7 +36,6 @@ #include "code/vtableStubs.hpp" #include "compiler/compileBroker.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_aix.h" #include "logging/log.hpp" #include "libo4.hpp" #include "libperfstat_aix.hpp" @@ -49,7 +49,6 @@ #include "os_share_aix.hpp" #include "porting_aix.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/atomic.hpp" @@ -3249,7 +3248,7 @@ void os::run_periodic_checks() { } // ReduceSignalUsage allows the user to override these handlers - // see comments at the very top and jvm_solaris.h + // see comments at the very top and jvm_md.h if (!ReduceSignalUsage) { DO_SIGNAL_CHECK(SHUTDOWN1_SIGNAL); DO_SIGNAL_CHECK(SHUTDOWN2_SIGNAL); diff --git a/src/hotspot/os/bsd/decoder_machO.cpp b/src/hotspot/os/bsd/decoder_machO.cpp index a51400b4e6e..a523e6fc9b4 100644 --- a/src/hotspot/os/bsd/decoder_machO.cpp +++ b/src/hotspot/os/bsd/decoder_machO.cpp @@ -25,7 +25,7 @@ #include "precompiled.hpp" #ifdef __APPLE__ -#include "prims/jvm.h" +#include "jvm.h" #include "decoder_machO.hpp" #include diff --git a/src/hotspot/os/bsd/jvm_bsd.cpp b/src/hotspot/os/bsd/jvm_bsd.cpp index f891f7c1b91..cb28ee043d4 100644 --- a/src/hotspot/os/bsd/jvm_bsd.cpp +++ b/src/hotspot/os/bsd/jvm_bsd.cpp @@ -23,7 +23,7 @@ */ #include "precompiled.hpp" -#include "prims/jvm.h" +#include "jvm.h" #include "runtime/interfaceSupport.hpp" #include "runtime/osThread.hpp" diff --git a/src/hotspot/os/bsd/jvm_bsd.h b/src/hotspot/os/bsd/jvm_bsd.h deleted file mode 100644 index dff2e872173..00000000000 --- a/src/hotspot/os/bsd/jvm_bsd.h +++ /dev/null @@ -1,115 +0,0 @@ -/* - * Copyright (c) 1999, 2017, 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. - * - */ - -#ifndef OS_BSD_VM_JVM_BSD_H -#define OS_BSD_VM_JVM_BSD_H - -/* -// HotSpot integration note: -// -// This is derived from the JDK classic file: -// "$JDK/src/solaris/javavm/export/jvm_md.h":15 (ver. 1.10 98/04/22) -// All local includes have been commented out. -*/ - -#ifndef JVM_MD_H -#define JVM_MD_H - -/* - * This file is currently collecting system-specific dregs for the - * JNI conversion, which should be sorted out later. - */ - -#include /* For DIR */ -#include /* For MAXPATHLEN */ -#include /* For socklen_t */ -#include /* For F_OK, R_OK, W_OK */ - -#define JNI_ONLOAD_SYMBOLS {"JNI_OnLoad"} -#define JNI_ONUNLOAD_SYMBOLS {"JNI_OnUnload"} -#define JVM_ONLOAD_SYMBOLS {"JVM_OnLoad"} -#define AGENT_ONLOAD_SYMBOLS {"Agent_OnLoad"} -#define AGENT_ONUNLOAD_SYMBOLS {"Agent_OnUnload"} -#define AGENT_ONATTACH_SYMBOLS {"Agent_OnAttach"} - -#define JNI_LIB_PREFIX "lib" -#ifdef __APPLE__ -#define JNI_LIB_SUFFIX ".dylib" -#else -#define JNI_LIB_SUFFIX ".so" -#endif - -// Hack: MAXPATHLEN is 4095 on some Bsd and 4096 on others. This may -// cause problems if JVM and the rest of JDK are built on different -// Bsd releases. Here we define JVM_MAXPATHLEN to be MAXPATHLEN + 1, -// so buffers declared in VM are always >= 4096. -#define JVM_MAXPATHLEN MAXPATHLEN + 1 - -#define JVM_R_OK R_OK -#define JVM_W_OK W_OK -#define JVM_X_OK X_OK -#define JVM_F_OK F_OK - -/* - * File I/O - */ - -#include -#include -#include -#include - -/* O Flags */ - -#define JVM_O_RDONLY O_RDONLY -#define JVM_O_WRONLY O_WRONLY -#define JVM_O_RDWR O_RDWR -#define JVM_O_O_APPEND O_APPEND -#define JVM_O_EXCL O_EXCL -#define JVM_O_CREAT O_CREAT - -/* Signal definitions */ - -#define BREAK_SIGNAL SIGQUIT /* Thread dumping support. */ -#define SHUTDOWN1_SIGNAL SIGHUP /* Shutdown Hooks support. */ -#define SHUTDOWN2_SIGNAL SIGINT -#define SHUTDOWN3_SIGNAL SIGTERM - -#ifndef SIGRTMIN -#ifdef __OpenBSD__ -#define SIGRTMIN 1 -#else -#define SIGRTMIN 33 -#endif -#endif -#ifndef SIGRTMAX -#ifdef __OpenBSD__ -#define SIGRTMAX 31 -#else -#define SIGRTMAX 63 -#endif -#endif -#endif /* JVM_MD_H */ - -#endif // OS_BSD_VM_JVM_BSD_H diff --git a/src/hotspot/os/bsd/os_bsd.cpp b/src/hotspot/os/bsd/os_bsd.cpp index bfa7811f8da..0f6dfaa8d80 100644 --- a/src/hotspot/os/bsd/os_bsd.cpp +++ b/src/hotspot/os/bsd/os_bsd.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -31,7 +32,6 @@ #include "compiler/compileBroker.hpp" #include "compiler/disassembler.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_bsd.h" #include "logging/log.hpp" #include "memory/allocation.inline.hpp" #include "memory/filemap.hpp" @@ -39,7 +39,6 @@ #include "os_bsd.inline.hpp" #include "os_share_bsd.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/atomic.hpp" @@ -3240,7 +3239,7 @@ void os::run_periodic_checks() { // ReduceSignalUsage allows the user to override these handlers - // see comments at the very top and jvm_solaris.h + // see comments at the very top and jvm_md.h if (!ReduceSignalUsage) { DO_SIGNAL_CHECK(SHUTDOWN1_SIGNAL); DO_SIGNAL_CHECK(SHUTDOWN2_SIGNAL); diff --git a/src/hotspot/os/linux/decoder_linux.cpp b/src/hotspot/os/linux/decoder_linux.cpp index 0a351969d0d..d972a0cd042 100644 --- a/src/hotspot/os/linux/decoder_linux.cpp +++ b/src/hotspot/os/linux/decoder_linux.cpp @@ -22,7 +22,7 @@ * */ -#include "prims/jvm.h" +#include "jvm.h" #include "utilities/decoder_elf.hpp" #include diff --git a/src/hotspot/os/linux/jvm_linux.cpp b/src/hotspot/os/linux/jvm_linux.cpp index 25e58c01ef4..f7c1c172de5 100644 --- a/src/hotspot/os/linux/jvm_linux.cpp +++ b/src/hotspot/os/linux/jvm_linux.cpp @@ -23,7 +23,7 @@ */ #include "precompiled.hpp" -#include "prims/jvm.h" +#include "jvm.h" #include "runtime/interfaceSupport.hpp" #include "runtime/osThread.hpp" diff --git a/src/hotspot/os/linux/jvm_linux.h b/src/hotspot/os/linux/jvm_linux.h deleted file mode 100644 index 950319be265..00000000000 --- a/src/hotspot/os/linux/jvm_linux.h +++ /dev/null @@ -1,97 +0,0 @@ -/* - * Copyright (c) 1999, 2015, 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. - * - */ - -#ifndef OS_LINUX_VM_JVM_LINUX_H -#define OS_LINUX_VM_JVM_LINUX_H - -/* -// HotSpot integration note: -// -// This is derived from the JDK classic file: -// "$JDK/src/solaris/javavm/export/jvm_md.h":15 (ver. 1.10 98/04/22) -// All local includes have been commented out. -*/ - -#ifndef JVM_MD_H -#define JVM_MD_H - -/* - * This file is currently collecting system-specific dregs for the - * JNI conversion, which should be sorted out later. - */ - -#include /* For DIR */ -#include /* For MAXPATHLEN */ -#include /* For socklen_t */ -#include /* For F_OK, R_OK, W_OK */ - -#define JNI_ONLOAD_SYMBOLS {"JNI_OnLoad"} -#define JNI_ONUNLOAD_SYMBOLS {"JNI_OnUnload"} -#define JVM_ONLOAD_SYMBOLS {"JVM_OnLoad"} -#define AGENT_ONLOAD_SYMBOLS {"Agent_OnLoad"} -#define AGENT_ONUNLOAD_SYMBOLS {"Agent_OnUnload"} -#define AGENT_ONATTACH_SYMBOLS {"Agent_OnAttach"} - -#define JNI_LIB_PREFIX "lib" -#define JNI_LIB_SUFFIX ".so" - -// Hack: MAXPATHLEN is 4095 on some Linux and 4096 on others. This may -// cause problems if JVM and the rest of JDK are built on different -// Linux releases. Here we define JVM_MAXPATHLEN to be MAXPATHLEN + 1, -// so buffers declared in VM are always >= 4096. -#define JVM_MAXPATHLEN MAXPATHLEN + 1 - -#define JVM_R_OK R_OK -#define JVM_W_OK W_OK -#define JVM_X_OK X_OK -#define JVM_F_OK F_OK - -/* - * File I/O - */ - -#include -#include -#include -#include - -/* O Flags */ - -#define JVM_O_RDONLY O_RDONLY -#define JVM_O_WRONLY O_WRONLY -#define JVM_O_RDWR O_RDWR -#define JVM_O_O_APPEND O_APPEND -#define JVM_O_EXCL O_EXCL -#define JVM_O_CREAT O_CREAT - -/* Signal definitions */ - -#define BREAK_SIGNAL SIGQUIT /* Thread dumping support. */ -#define SHUTDOWN1_SIGNAL SIGHUP /* Shutdown Hooks support. */ -#define SHUTDOWN2_SIGNAL SIGINT -#define SHUTDOWN3_SIGNAL SIGTERM - -#endif /* JVM_MD_H */ - -#endif // OS_LINUX_VM_JVM_LINUX_H diff --git a/src/hotspot/os/linux/os_linux.cpp b/src/hotspot/os/linux/os_linux.cpp index 5eabc870287..ca30ac3d19f 100644 --- a/src/hotspot/os/linux/os_linux.cpp +++ b/src/hotspot/os/linux/os_linux.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -31,7 +32,6 @@ #include "compiler/compileBroker.hpp" #include "compiler/disassembler.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_linux.h" #include "logging/log.hpp" #include "memory/allocation.inline.hpp" #include "memory/filemap.hpp" @@ -39,7 +39,6 @@ #include "os_linux.inline.hpp" #include "os_share_linux.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/atomic.hpp" @@ -4664,7 +4663,7 @@ void os::run_periodic_checks() { #endif // ReduceSignalUsage allows the user to override these handlers - // see comments at the very top and jvm_solaris.h + // see comments at the very top and jvm_md.h if (!ReduceSignalUsage) { DO_SIGNAL_CHECK(SHUTDOWN1_SIGNAL); DO_SIGNAL_CHECK(SHUTDOWN2_SIGNAL); diff --git a/src/hotspot/os/posix/os_posix.cpp b/src/hotspot/os/posix/os_posix.cpp index 422f7fcac3c..fed874d30eb 100644 --- a/src/hotspot/os/posix/os_posix.cpp +++ b/src/hotspot/os/posix/os_posix.cpp @@ -22,8 +22,8 @@ * */ +#include "jvm.h" #include "utilities/globalDefinitions.hpp" -#include "prims/jvm.h" #include "runtime/frame.inline.hpp" #include "runtime/interfaceSupport.hpp" #include "runtime/os.hpp" diff --git a/src/hotspot/os/solaris/jvm_solaris.cpp b/src/hotspot/os/solaris/jvm_solaris.cpp index 3dfa84e0f9e..1a2ab0e7026 100644 --- a/src/hotspot/os/solaris/jvm_solaris.cpp +++ b/src/hotspot/os/solaris/jvm_solaris.cpp @@ -23,7 +23,7 @@ */ #include "precompiled.hpp" -#include "prims/jvm.h" +#include "jvm.h" #include "runtime/interfaceSupport.hpp" #include "runtime/osThread.hpp" diff --git a/src/hotspot/os/solaris/jvm_solaris.h b/src/hotspot/os/solaris/jvm_solaris.h deleted file mode 100644 index 92c551c6cb2..00000000000 --- a/src/hotspot/os/solaris/jvm_solaris.h +++ /dev/null @@ -1,98 +0,0 @@ -/* - * Copyright (c) 1998, 2017, 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. - * - */ - -#ifndef OS_SOLARIS_VM_JVM_SOLARIS_H -#define OS_SOLARIS_VM_JVM_SOLARIS_H - -/* -// HotSpot integration note: -// -// This is derived from the JDK classic file: -// "$JDK/src/solaris/javavm/export/jvm_md.h":15 (ver. 1.10 98/04/22) -// All local includes have been commented out. -*/ - -#ifndef JVM_MD_H -#define JVM_MD_H - -/* - * This file is currently collecting system-specific dregs for the - * JNI conversion, which should be sorted out later. - */ - -#include /* For DIR */ -#include /* For MAXPATHLEN */ -#include /* For socklen_t */ -#include /* For F_OK, R_OK, W_OK */ -#include /* for intptr_t types (64 Bit cleanliness) */ - -#define JNI_ONLOAD_SYMBOLS {"JNI_OnLoad"} -#define JNI_ONUNLOAD_SYMBOLS {"JNI_OnUnload"} -#define JVM_ONLOAD_SYMBOLS {"JVM_OnLoad"} -#define AGENT_ONLOAD_SYMBOLS {"Agent_OnLoad"} -#define AGENT_ONUNLOAD_SYMBOLS {"Agent_OnUnload"} -#define AGENT_ONATTACH_SYMBOLS {"Agent_OnAttach"} - -#define JNI_LIB_PREFIX "lib" -#define JNI_LIB_SUFFIX ".so" - -#define JVM_MAXPATHLEN MAXPATHLEN - -#define JVM_R_OK R_OK -#define JVM_W_OK W_OK -#define JVM_X_OK X_OK -#define JVM_F_OK F_OK - -/* - * File I/O - */ - -#include -#include -#include -#include - -/* O Flags */ - -#define JVM_O_RDONLY O_RDONLY -#define JVM_O_WRONLY O_WRONLY -#define JVM_O_RDWR O_RDWR -#define JVM_O_O_APPEND O_APPEND -#define JVM_O_EXCL O_EXCL -#define JVM_O_CREAT O_CREAT - -/* Signal definitions */ - -#define BREAK_SIGNAL SIGQUIT /* Thread dumping support. */ -#define ASYNC_SIGNAL SIGJVM2 /* Event-based suspend/resume support */ -#define SHUTDOWN1_SIGNAL SIGHUP /* Shutdown Hooks support. */ -#define SHUTDOWN2_SIGNAL SIGINT -#define SHUTDOWN3_SIGNAL SIGTERM - -/* With 1.4.1 libjsig added versioning: used in os_solaris.cpp and jsig.c */ -#define JSIG_VERSION_1_4_1 0x30140100 - -#endif /* JVM_MD_H */ - -#endif // OS_SOLARIS_VM_JVM_SOLARIS_H diff --git a/src/hotspot/os/solaris/os_solaris.cpp b/src/hotspot/os/solaris/os_solaris.cpp index 814589d375d..187e2bd2a24 100644 --- a/src/hotspot/os/solaris/os_solaris.cpp +++ b/src/hotspot/os/solaris/os_solaris.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -31,7 +32,6 @@ #include "compiler/compileBroker.hpp" #include "compiler/disassembler.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_solaris.h" #include "logging/log.hpp" #include "memory/allocation.inline.hpp" #include "memory/filemap.hpp" @@ -39,7 +39,6 @@ #include "os_share_solaris.hpp" #include "os_solaris.inline.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/atomic.hpp" diff --git a/src/hotspot/os/windows/jvm_windows.cpp b/src/hotspot/os/windows/jvm_windows.cpp index 1e9a3fe5c49..c81d95175a4 100644 --- a/src/hotspot/os/windows/jvm_windows.cpp +++ b/src/hotspot/os/windows/jvm_windows.cpp @@ -23,7 +23,7 @@ */ #include "precompiled.hpp" -#include "prims/jvm.h" +#include "jvm.h" #include "runtime/interfaceSupport.hpp" #include "runtime/osThread.hpp" diff --git a/src/hotspot/os/windows/jvm_windows.h b/src/hotspot/os/windows/jvm_windows.h deleted file mode 100644 index e5ec82dca90..00000000000 --- a/src/hotspot/os/windows/jvm_windows.h +++ /dev/null @@ -1,118 +0,0 @@ -/* - * Copyright (c) 1998, 2014, 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. - * - */ - -#ifndef OS_WINDOWS_VM_JVM_WINDOWS_H -#define OS_WINDOWS_VM_JVM_WINDOWS_H - -#ifndef _JAVASOFT_JVM_MD_H_ -#define _JAVASOFT_JVM_MD_H_ - -/* - * This file is currently collecting system-specific dregs for the - * JNI conversion, which should be sorted out later. - */ - -// JDK7 requires VS2010 -#if _MSC_VER >= 1600 -// JDK7 minimum platform requirement: Windows XP -#if _WIN32_WINNT < 0x0501 -#undef _WIN32_WINNT -#define _WIN32_WINNT 0x0501 -#endif -#endif - -#include - -#include -#include - -typedef int socklen_t; - -#define JNI_ONLOAD_SYMBOLS {"_JNI_OnLoad@8", "JNI_OnLoad"} -#define JNI_ONUNLOAD_SYMBOLS {"_JNI_OnUnload@8", "JNI_OnUnload"} -#define JVM_ONLOAD_SYMBOLS {"_JVM_OnLoad@12", "JVM_OnLoad"} -#define AGENT_ONLOAD_SYMBOLS {"_Agent_OnLoad@12", "Agent_OnLoad"} -#define AGENT_ONUNLOAD_SYMBOLS {"_Agent_OnUnload@4", "Agent_OnUnload"} -#define AGENT_ONATTACH_SYMBOLS {"_Agent_OnAttach@12", "Agent_OnAttach"} - -#define JNI_LIB_PREFIX "" -#define JNI_LIB_SUFFIX ".dll" - -struct dirent { - char d_name[MAX_PATH]; -}; - -typedef struct { - struct dirent dirent; - char *path; - HANDLE handle; - WIN32_FIND_DATA find_data; -} DIR; - -#include - -#define JVM_MAXPATHLEN _MAX_PATH - -#define JVM_R_OK 4 -#define JVM_W_OK 2 -#define JVM_X_OK 1 -#define JVM_F_OK 0 - -#ifdef __cplusplus -extern "C" { -#endif - -JNIEXPORT void * JNICALL -JVM_GetThreadInterruptEvent(); - -#ifdef __cplusplus -} /* extern "C" */ -#endif /* __cplusplus */ - -/* - * File I/O - */ - -#include - -/* O Flags */ - -#define JVM_O_RDONLY O_RDONLY -#define JVM_O_WRONLY O_WRONLY -#define JVM_O_RDWR O_RDWR -#define JVM_O_O_APPEND O_APPEND -#define JVM_O_EXCL O_EXCL -#define JVM_O_CREAT O_CREAT - -/* Signals */ - -#define JVM_SIGINT SIGINT -#define JVM_SIGTERM SIGTERM - -#define SHUTDOWN1_SIGNAL SIGINT /* Shutdown Hooks support. */ -#define SHUTDOWN2_SIGNAL SIGTERM - -#endif /* !_JAVASOFT_JVM_MD_H_ */ - -#endif // OS_WINDOWS_VM_JVM_WINDOWS_H diff --git a/src/hotspot/os/windows/os_windows.cpp b/src/hotspot/os/windows/os_windows.cpp index dedbc3f2e9b..fa90f1f2595 100644 --- a/src/hotspot/os/windows/os_windows.cpp +++ b/src/hotspot/os/windows/os_windows.cpp @@ -26,6 +26,7 @@ #define _WIN32_WINNT 0x0600 // no precompiled headers +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -34,7 +35,6 @@ #include "compiler/compileBroker.hpp" #include "compiler/disassembler.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_windows.h" #include "logging/log.hpp" #include "memory/allocation.inline.hpp" #include "memory/filemap.hpp" @@ -42,7 +42,6 @@ #include "os_share_windows.hpp" #include "os_windows.inline.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/atomic.hpp" @@ -100,6 +99,7 @@ #include // For os::dll_address_to_function_name // for enumerating dll libraries #include +#include // for timer info max values which include all bits #define ALL_64_BITS CONST64(-1) @@ -3656,7 +3656,7 @@ int os::win32::exit_process_or_thread(Ept what, int exit_code) { static INIT_ONCE init_once_crit_sect = INIT_ONCE_STATIC_INIT; static CRITICAL_SECTION crit_sect; - static volatile jint process_exiting = 0; + static volatile DWORD process_exiting = 0; int i, j; DWORD res; HANDLE hproc, hthr; @@ -3675,7 +3675,7 @@ int os::win32::exit_process_or_thread(Ept what, int exit_code) { if (what != EPT_THREAD) { // Atomically set process_exiting before the critical section // to increase the visibility between racing threads. - Atomic::cmpxchg((jint)GetCurrentThreadId(), &process_exiting, 0); + Atomic::cmpxchg(GetCurrentThreadId(), &process_exiting, (DWORD)0); } EnterCriticalSection(&crit_sect); @@ -3793,7 +3793,7 @@ int os::win32::exit_process_or_thread(Ept what, int exit_code) { if (!registered && OrderAccess::load_acquire(&process_exiting) != 0 && - process_exiting != (jint)GetCurrentThreadId()) { + process_exiting != GetCurrentThreadId()) { // Some other thread is about to call exit(), so we don't let // the current unregistered thread proceed to exit() or _endthreadex() while (true) { diff --git a/src/hotspot/os/windows/threadCritical_windows.cpp b/src/hotspot/os/windows/threadCritical_windows.cpp index 66f19e91fc0..1d0053bc61f 100644 --- a/src/hotspot/os/windows/threadCritical_windows.cpp +++ b/src/hotspot/os/windows/threadCritical_windows.cpp @@ -36,7 +36,7 @@ // static bool initialized = false; -static volatile jint lock_count = -1; +static volatile int lock_count = -1; static HANDLE lock_event; static DWORD lock_owner = -1; diff --git a/src/hotspot/os_cpu/aix_ppc/os_aix_ppc.cpp b/src/hotspot/os_cpu/aix_ppc/os_aix_ppc.cpp index 721d99668b4..992285069c3 100644 --- a/src/hotspot/os_cpu/aix_ppc/os_aix_ppc.cpp +++ b/src/hotspot/os_cpu/aix_ppc/os_aix_ppc.cpp @@ -24,6 +24,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/assembler.inline.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -32,12 +33,10 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_aix.h" #include "memory/allocation.inline.hpp" #include "nativeInst_ppc.hpp" #include "os_share_aix.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "porting_aix.hpp" #include "runtime/arguments.hpp" diff --git a/src/hotspot/os_cpu/bsd_x86/os_bsd_x86.cpp b/src/hotspot/os_cpu/bsd_x86/os_bsd_x86.cpp index 115b48f38dd..36b5872bb03 100644 --- a/src/hotspot/os_cpu/bsd_x86/os_bsd_x86.cpp +++ b/src/hotspot/os_cpu/bsd_x86/os_bsd_x86.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -31,11 +32,9 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_bsd.h" #include "memory/allocation.inline.hpp" #include "os_share_bsd.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/bsd_zero/os_bsd_zero.cpp b/src/hotspot/os_cpu/bsd_zero/os_bsd_zero.cpp index 15a5b7d046b..78a17355062 100644 --- a/src/hotspot/os_cpu/bsd_zero/os_bsd_zero.cpp +++ b/src/hotspot/os_cpu/bsd_zero/os_bsd_zero.cpp @@ -29,6 +29,7 @@ #endif // no precompiled headers +#include "jvm.h" #include "assembler_zero.inline.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -36,12 +37,10 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_bsd.h" #include "memory/allocation.inline.hpp" #include "nativeInst_zero.hpp" #include "os_share_bsd.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp b/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp index 9ef37614ea7..939d782f6ee 100644 --- a/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp +++ b/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp @@ -24,6 +24,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -33,11 +34,9 @@ #include "code/vtableStubs.hpp" #include "code/nativeInst.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_linux.h" #include "memory/allocation.inline.hpp" #include "os_share_linux.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/linux_arm/os_linux_arm.cpp b/src/hotspot/os_cpu/linux_arm/os_linux_arm.cpp index 147eab71daf..4f90cd10bee 100644 --- a/src/hotspot/os_cpu/linux_arm/os_linux_arm.cpp +++ b/src/hotspot/os_cpu/linux_arm/os_linux_arm.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "assembler_arm.inline.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -30,12 +31,10 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_linux.h" #include "memory/allocation.inline.hpp" #include "nativeInst_arm.hpp" #include "os_share_linux.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/linux_ppc/os_linux_ppc.cpp b/src/hotspot/os_cpu/linux_ppc/os_linux_ppc.cpp index 6be73f78e08..80ed5bd8e1f 100644 --- a/src/hotspot/os_cpu/linux_ppc/os_linux_ppc.cpp +++ b/src/hotspot/os_cpu/linux_ppc/os_linux_ppc.cpp @@ -24,6 +24,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/assembler.inline.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -32,12 +33,10 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_linux.h" #include "memory/allocation.inline.hpp" #include "nativeInst_ppc.hpp" #include "os_share_linux.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/linux_s390/os_linux_s390.cpp b/src/hotspot/os_cpu/linux_s390/os_linux_s390.cpp index 272c3e0f46b..01cb7e1a517 100644 --- a/src/hotspot/os_cpu/linux_s390/os_linux_s390.cpp +++ b/src/hotspot/os_cpu/linux_s390/os_linux_s390.cpp @@ -26,6 +26,7 @@ // This file is organized as os_linux_x86.cpp. // no precompiled headers +#include "jvm.h" #include "asm/assembler.inline.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -35,12 +36,10 @@ #include "code/vtableStubs.hpp" #include "compiler/disassembler.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_linux.h" #include "memory/allocation.inline.hpp" #include "nativeInst_s390.hpp" #include "os_share_linux.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/linux_sparc/os_linux_sparc.cpp b/src/hotspot/os_cpu/linux_sparc/os_linux_sparc.cpp index f6656a5d72a..ef4ae04c119 100644 --- a/src/hotspot/os_cpu/linux_sparc/os_linux_sparc.cpp +++ b/src/hotspot/os_cpu/linux_sparc/os_linux_sparc.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -31,12 +32,10 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_linux.h" #include "memory/allocation.inline.hpp" #include "nativeInst_sparc.hpp" #include "os_share_linux.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/linux_x86/os_linux_x86.cpp b/src/hotspot/os_cpu/linux_x86/os_linux_x86.cpp index 67796840c08..eaa0926fcdf 100644 --- a/src/hotspot/os_cpu/linux_x86/os_linux_x86.cpp +++ b/src/hotspot/os_cpu/linux_x86/os_linux_x86.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -31,11 +32,9 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_linux.h" #include "memory/allocation.inline.hpp" #include "os_share_linux.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/linux_zero/os_linux_zero.cpp b/src/hotspot/os_cpu/linux_zero/os_linux_zero.cpp index e62ed20b0c3..b00672edb18 100644 --- a/src/hotspot/os_cpu/linux_zero/os_linux_zero.cpp +++ b/src/hotspot/os_cpu/linux_zero/os_linux_zero.cpp @@ -24,6 +24,7 @@ */ // no precompiled headers +#include "jvm.h" #include "assembler_zero.inline.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -31,12 +32,10 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_linux.h" #include "memory/allocation.inline.hpp" #include "nativeInst_zero.hpp" #include "os_share_linux.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/solaris_sparc/os_solaris_sparc.cpp b/src/hotspot/os_cpu/solaris_sparc/os_solaris_sparc.cpp index 9497d2d604a..60c1237b5b5 100644 --- a/src/hotspot/os_cpu/solaris_sparc/os_solaris_sparc.cpp +++ b/src/hotspot/os_cpu/solaris_sparc/os_solaris_sparc.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "macroAssembler_sparc.hpp" #include "classfile/classLoader.hpp" @@ -32,12 +33,10 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_solaris.h" #include "memory/allocation.inline.hpp" #include "nativeInst_sparc.hpp" #include "os_share_solaris.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/os_cpu/solaris_x86/os_solaris_x86.cpp b/src/hotspot/os_cpu/solaris_x86/os_solaris_x86.cpp index 4934ec39ca3..570ce8c4788 100644 --- a/src/hotspot/os_cpu/solaris_x86/os_solaris_x86.cpp +++ b/src/hotspot/os_cpu/solaris_x86/os_solaris_x86.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -31,11 +32,9 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_solaris.h" #include "memory/allocation.inline.hpp" #include "os_share_solaris.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/atomic.hpp" diff --git a/src/hotspot/os_cpu/windows_x86/os_windows_x86.cpp b/src/hotspot/os_cpu/windows_x86/os_windows_x86.cpp index 3dc70699bfc..80d62854dbb 100644 --- a/src/hotspot/os_cpu/windows_x86/os_windows_x86.cpp +++ b/src/hotspot/os_cpu/windows_x86/os_windows_x86.cpp @@ -23,6 +23,7 @@ */ // no precompiled headers +#include "jvm.h" #include "asm/macroAssembler.hpp" #include "classfile/classLoader.hpp" #include "classfile/systemDictionary.hpp" @@ -30,13 +31,11 @@ #include "code/icBuffer.hpp" #include "code/vtableStubs.hpp" #include "interpreter/interpreter.hpp" -#include "jvm_windows.h" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" #include "nativeInst_x86.hpp" #include "os_share_windows.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/extendedPC.hpp" diff --git a/src/hotspot/share/aot/aotLoader.cpp b/src/hotspot/share/aot/aotLoader.cpp index 531a67f6e9e..c91c04b9b1b 100644 --- a/src/hotspot/share/aot/aotLoader.cpp +++ b/src/hotspot/share/aot/aotLoader.cpp @@ -22,12 +22,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "aot/aotCodeHeap.hpp" #include "aot/aotLoader.inline.hpp" #include "jvmci/jvmciRuntime.hpp" #include "oops/method.hpp" -#include "prims/jvm.h" #include "runtime/os.hpp" #include "runtime/timerTrace.hpp" diff --git a/src/hotspot/share/c1/c1_CFGPrinter.cpp b/src/hotspot/share/c1/c1_CFGPrinter.cpp index 7144a5c976b..86afce18de7 100644 --- a/src/hotspot/share/c1/c1_CFGPrinter.cpp +++ b/src/hotspot/share/c1/c1_CFGPrinter.cpp @@ -23,13 +23,13 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "c1/c1_CFGPrinter.hpp" #include "c1/c1_IR.hpp" #include "c1/c1_InstructionPrinter.hpp" #include "c1/c1_LIR.hpp" #include "c1/c1_LinearScan.hpp" #include "c1/c1_ValueStack.hpp" -#include "prims/jvm.h" #ifndef PRODUCT diff --git a/src/hotspot/share/c1/c1_Canonicalizer.hpp b/src/hotspot/share/c1/c1_Canonicalizer.hpp index e9ba74c1a48..c0d5488d970 100644 --- a/src/hotspot/share/c1/c1_Canonicalizer.hpp +++ b/src/hotspot/share/c1/c1_Canonicalizer.hpp @@ -40,6 +40,10 @@ class Canonicalizer: InstructionVisitor { void set_constant(jlong x) { set_canonical(new Constant(new LongConstant(x))); } void set_constant(jfloat x) { set_canonical(new Constant(new FloatConstant(x))); } void set_constant(jdouble x) { set_canonical(new Constant(new DoubleConstant(x))); } +#ifdef _WINDOWS + // jint is defined as long in jni_md.h, so convert from int to jint + void set_constant(int x) { set_constant((jint)x); } +#endif void move_const_to_right(Op2* x); void do_Op2(Op2* x); void do_UnsafeRawOp(UnsafeRawOp* x); diff --git a/src/hotspot/share/c1/c1_LinearScan.cpp b/src/hotspot/share/c1/c1_LinearScan.cpp index 0e63df153d5..859e6ab9204 100644 --- a/src/hotspot/share/c1/c1_LinearScan.cpp +++ b/src/hotspot/share/c1/c1_LinearScan.cpp @@ -2507,7 +2507,7 @@ void LinearScan::compute_oop_map(IntervalWalker* iw, const LIR_OpVisitState &vis // use). ConstantOopWriteValue* LinearScan::_oop_null_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantOopWriteValue(NULL); ConstantIntValue* LinearScan::_int_m1_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue(-1); -ConstantIntValue* LinearScan::_int_0_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue(0); +ConstantIntValue* LinearScan::_int_0_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue((jint)0); ConstantIntValue* LinearScan::_int_1_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue(1); ConstantIntValue* LinearScan::_int_2_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue(2); LocationValue* _illegal_value = new (ResourceObj::C_HEAP, mtCompiler) LocationValue(Location()); diff --git a/src/hotspot/share/ci/ciEnv.cpp b/src/hotspot/share/ci/ciEnv.cpp index bfde1da61b8..69e338fe269 100644 --- a/src/hotspot/share/ci/ciEnv.cpp +++ b/src/hotspot/share/ci/ciEnv.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "ci/ciConstant.hpp" #include "ci/ciEnv.hpp" #include "ci/ciField.hpp" @@ -49,7 +50,6 @@ #include "oops/objArrayKlass.hpp" #include "oops/objArrayOop.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/jvmtiExport.hpp" #include "runtime/init.hpp" #include "runtime/reflection.hpp" diff --git a/src/hotspot/share/ci/ciFlags.hpp b/src/hotspot/share/ci/ciFlags.hpp index 52b2c6a0322..4c791eff9bc 100644 --- a/src/hotspot/share/ci/ciFlags.hpp +++ b/src/hotspot/share/ci/ciFlags.hpp @@ -25,9 +25,9 @@ #ifndef SHARE_VM_CI_CIFLAGS_HPP #define SHARE_VM_CI_CIFLAGS_HPP +#include "jvm.h" #include "ci/ciClassList.hpp" #include "memory/allocation.hpp" -#include "prims/jvm.h" #include "utilities/accessFlags.hpp" #include "utilities/ostream.hpp" diff --git a/src/hotspot/share/ci/ciReplay.cpp b/src/hotspot/share/ci/ciReplay.cpp index d3306b4a9ac..f11365f26fa 100644 --- a/src/hotspot/share/ci/ciReplay.cpp +++ b/src/hotspot/share/ci/ciReplay.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "ci/ciMethodData.hpp" #include "ci/ciReplay.hpp" #include "ci/ciSymbol.hpp" @@ -33,7 +34,6 @@ #include "memory/oopFactory.hpp" #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "utilities/copy.hpp" #include "utilities/macros.hpp" @@ -790,7 +790,7 @@ class CompileReplay : public StackObj { while (field_signature[rank] == '[') { rank++; } - int* dims = NEW_RESOURCE_ARRAY(int, rank); + jint* dims = NEW_RESOURCE_ARRAY(jint, rank); dims[0] = length; for (int i = 1; i < rank; i++) { dims[i] = 1; // These aren't relevant to the compiler diff --git a/src/hotspot/share/classfile/altHashing.cpp b/src/hotspot/share/classfile/altHashing.cpp index 73601723551..1c73e4e4c60 100644 --- a/src/hotspot/share/classfile/altHashing.cpp +++ b/src/hotspot/share/classfile/altHashing.cpp @@ -42,15 +42,15 @@ static intptr_t object_hash(Klass* k) { juint AltHashing::compute_seed() { jlong nanos = os::javaTimeNanos(); jlong now = os::javaTimeMillis(); - int SEED_MATERIAL[8] = { - (int) object_hash(SystemDictionary::String_klass()), - (int) object_hash(SystemDictionary::System_klass()), - os::random(), // current thread isn't a java thread - (int) (((julong)nanos) >> 32), - (int) nanos, - (int) (((julong)now) >> 32), - (int) now, - (int) (os::javaTimeNanos() >> 2) + jint SEED_MATERIAL[8] = { + (jint) object_hash(SystemDictionary::String_klass()), + (jint) object_hash(SystemDictionary::System_klass()), + (jint) os::random(), // current thread isn't a java thread + (jint) (((julong)nanos) >> 32), + (jint) nanos, + (jint) (((julong)now) >> 32), + (jint) now, + (jint) (os::javaTimeNanos() >> 2) }; return murmur3_32(SEED_MATERIAL, 8); @@ -167,7 +167,7 @@ juint AltHashing::murmur3_32(juint seed, const jchar* data, int len) { } // Hash used for the seed. -juint AltHashing::murmur3_32(juint seed, const int* data, int len) { +juint AltHashing::murmur3_32(juint seed, const jint* data, int len) { juint h1 = seed; int off = 0; @@ -202,6 +202,6 @@ juint AltHashing::murmur3_32(juint seed, const int* data, int len) { return h1; } -juint AltHashing::murmur3_32(const int* data, int len) { +juint AltHashing::murmur3_32(const jint* data, int len) { return murmur3_32(0, data, len); } diff --git a/src/hotspot/share/classfile/altHashing.hpp b/src/hotspot/share/classfile/altHashing.hpp index 15c6a34b25a..3bcfef3888a 100644 --- a/src/hotspot/share/classfile/altHashing.hpp +++ b/src/hotspot/share/classfile/altHashing.hpp @@ -43,8 +43,8 @@ class AltHashing : AllStatic { static juint Integer_rotateLeft(juint i, int distance) { return (i << distance) | (i >> (32 - distance)); } - static juint murmur3_32(const int* data, int len); - static juint murmur3_32(juint seed, const int* data, int len); + static juint murmur3_32(const jint* data, int len); + static juint murmur3_32(juint seed, const jint* data, int len); public: static juint compute_seed(); diff --git a/src/hotspot/share/classfile/classFileParser.cpp b/src/hotspot/share/classfile/classFileParser.cpp index 73ab562145e..5074ba4ca83 100644 --- a/src/hotspot/share/classfile/classFileParser.cpp +++ b/src/hotspot/share/classfile/classFileParser.cpp @@ -22,6 +22,7 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "aot/aotLoader.hpp" #include "classfile/classFileParser.hpp" #include "classfile/classFileStream.hpp" @@ -54,7 +55,6 @@ #include "oops/method.hpp" #include "oops/oop.inline.hpp" #include "oops/symbol.hpp" -#include "prims/jvm.h" #include "prims/jvmtiExport.hpp" #include "prims/jvmtiThreadState.hpp" #include "runtime/javaCalls.hpp" diff --git a/src/hotspot/share/classfile/classLoader.cpp b/src/hotspot/share/classfile/classLoader.cpp index 56bfed2b268..2066da6a5b7 100644 --- a/src/hotspot/share/classfile/classLoader.cpp +++ b/src/hotspot/share/classfile/classLoader.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classFileStream.hpp" #include "classfile/classLoader.hpp" #include "classfile/classLoaderData.inline.hpp" @@ -53,7 +54,6 @@ #include "oops/objArrayOop.inline.hpp" #include "oops/oop.inline.hpp" #include "oops/symbol.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/arguments.hpp" #include "runtime/compilationPolicy.hpp" diff --git a/src/hotspot/share/classfile/compactHashtable.cpp b/src/hotspot/share/classfile/compactHashtable.cpp index b5d04f9e49c..aecf2b4e43e 100644 --- a/src/hotspot/share/classfile/compactHashtable.cpp +++ b/src/hotspot/share/classfile/compactHashtable.cpp @@ -23,12 +23,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/compactHashtable.inline.hpp" #include "classfile/javaClasses.hpp" #include "logging/logMessage.hpp" #include "memory/metadataFactory.hpp" #include "memory/metaspaceShared.hpp" -#include "prims/jvm.h" #include "runtime/vmThread.hpp" #include "utilities/numberSeq.hpp" #include diff --git a/src/hotspot/share/classfile/modules.cpp b/src/hotspot/share/classfile/modules.cpp index 47da2f71c90..efa770e4f33 100644 --- a/src/hotspot/share/classfile/modules.cpp +++ b/src/hotspot/share/classfile/modules.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classFileParser.hpp" #include "classfile/classLoader.hpp" #include "classfile/classLoaderData.inline.hpp" @@ -40,7 +41,6 @@ #include "logging/logStream.hpp" #include "memory/resourceArea.hpp" #include "oops/instanceKlass.hpp" -#include "prims/jvm.h" #include "runtime/arguments.hpp" #include "runtime/handles.inline.hpp" #include "runtime/javaCalls.hpp" diff --git a/src/hotspot/share/classfile/systemDictionary.cpp b/src/hotspot/share/classfile/systemDictionary.cpp index b72200a6c30..9eabbf518d1 100644 --- a/src/hotspot/share/classfile/systemDictionary.cpp +++ b/src/hotspot/share/classfile/systemDictionary.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "aot/aotLoader.hpp" #include "classfile/classFileParser.hpp" #include "classfile/classFileStream.hpp" @@ -61,7 +62,6 @@ #include "oops/oop.inline.hpp" #include "oops/symbol.hpp" #include "oops/typeArrayKlass.hpp" -#include "prims/jvm.h" #include "prims/jvmtiEnvBase.hpp" #include "prims/resolvedMethodTable.hpp" #include "prims/methodHandles.hpp" diff --git a/src/hotspot/share/classfile/verifier.cpp b/src/hotspot/share/classfile/verifier.cpp index ef4a749ff58..e26ce4026dd 100644 --- a/src/hotspot/share/classfile/verifier.cpp +++ b/src/hotspot/share/classfile/verifier.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classFileStream.hpp" #include "classfile/javaClasses.hpp" #include "classfile/stackMapTable.hpp" @@ -40,7 +41,6 @@ #include "oops/instanceKlass.hpp" #include "oops/oop.inline.hpp" #include "oops/typeArrayOop.hpp" -#include "prims/jvm.h" #include "runtime/fieldDescriptor.hpp" #include "runtime/handles.inline.hpp" #include "runtime/interfaceSupport.hpp" diff --git a/src/hotspot/share/classfile/vmSymbols.cpp b/src/hotspot/share/classfile/vmSymbols.cpp index 36a478a7453..77b4b2f181b 100644 --- a/src/hotspot/share/classfile/vmSymbols.cpp +++ b/src/hotspot/share/classfile/vmSymbols.cpp @@ -23,12 +23,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/vmSymbols.hpp" #include "compiler/compilerDirectives.hpp" #include "memory/oopFactory.hpp" #include "memory/metaspaceClosure.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/handles.inline.hpp" #include "utilities/xmlstream.hpp" diff --git a/src/hotspot/share/code/codeBlob.cpp b/src/hotspot/share/code/codeBlob.cpp index a68cb5d5642..bc2fa71b891 100644 --- a/src/hotspot/share/code/codeBlob.cpp +++ b/src/hotspot/share/code/codeBlob.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "code/codeBlob.hpp" #include "code/codeCache.hpp" #include "code/relocInfo.hpp" @@ -33,7 +34,6 @@ #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" #include "prims/forte.hpp" -#include "prims/jvm.h" #include "runtime/handles.inline.hpp" #include "runtime/interfaceSupport.hpp" #include "runtime/mutexLocker.hpp" diff --git a/src/hotspot/share/code/nmethod.cpp b/src/hotspot/share/code/nmethod.cpp index ba0c418aa2b..9a1559f53df 100644 --- a/src/hotspot/share/code/nmethod.cpp +++ b/src/hotspot/share/code/nmethod.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "code/codeCache.hpp" #include "code/compiledIC.hpp" #include "code/dependencies.hpp" @@ -41,7 +42,6 @@ #include "memory/resourceArea.hpp" #include "oops/methodData.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/jvmtiImpl.hpp" #include "runtime/atomic.hpp" #include "runtime/orderAccess.inline.hpp" diff --git a/src/hotspot/share/code/relocInfo.cpp b/src/hotspot/share/code/relocInfo.cpp index 0a4c9121539..e21e92b8004 100644 --- a/src/hotspot/share/code/relocInfo.cpp +++ b/src/hotspot/share/code/relocInfo.cpp @@ -489,7 +489,7 @@ void external_word_Relocation::unpack_data() { #ifndef _LP64 _target = (address) (intptr_t)unpack_1_int(); #else - int32_t lo, hi; + jint lo, hi; unpack_2_ints(lo, hi); jlong t = jlong_from(hi, lo);; _target = (address) t; diff --git a/src/hotspot/share/compiler/compileBroker.cpp b/src/hotspot/share/compiler/compileBroker.cpp index 1169842d3d2..a1354fe08f4 100644 --- a/src/hotspot/share/compiler/compileBroker.cpp +++ b/src/hotspot/share/compiler/compileBroker.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/symbolTable.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -40,7 +41,6 @@ #include "oops/methodData.hpp" #include "oops/method.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/nativeLookup.hpp" #include "prims/whitebox.hpp" #include "runtime/arguments.hpp" @@ -108,7 +108,7 @@ bool CompileBroker::_initialized = false; volatile bool CompileBroker::_should_block = false; -volatile jint CompileBroker::_print_compilation_warning = 0; +volatile int CompileBroker::_print_compilation_warning = 0; volatile jint CompileBroker::_should_compile_new_jobs = run_compilation; // The installed compiler(s) diff --git a/src/hotspot/share/compiler/compileBroker.hpp b/src/hotspot/share/compiler/compileBroker.hpp index ced793a0c4f..0008faec9c1 100644 --- a/src/hotspot/share/compiler/compileBroker.hpp +++ b/src/hotspot/share/compiler/compileBroker.hpp @@ -217,7 +217,7 @@ class CompileBroker: AllStatic { static int _sum_nmethod_code_size; static long _peak_compilation_time; - static volatile jint _print_compilation_warning; + static volatile int _print_compilation_warning; static JavaThread* make_thread(const char* name, CompileQueue* queue, CompilerCounters* counters, AbstractCompiler* comp, bool compiler_thread, TRAPS); static void init_compiler_sweeper_threads(int c1_compiler_count, int c2_compiler_count); diff --git a/src/hotspot/share/compiler/compileLog.cpp b/src/hotspot/share/compiler/compileLog.cpp index eac06f6f548..de8cbd1b080 100644 --- a/src/hotspot/share/compiler/compileLog.cpp +++ b/src/hotspot/share/compiler/compileLog.cpp @@ -23,12 +23,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "ci/ciMethod.hpp" #include "code/codeCache.hpp" #include "compiler/compileLog.hpp" #include "memory/allocation.inline.hpp" #include "oops/method.hpp" -#include "prims/jvm.h" #include "runtime/mutexLocker.hpp" #include "runtime/os.hpp" diff --git a/src/hotspot/share/compiler/compilerOracle.cpp b/src/hotspot/share/compiler/compilerOracle.cpp index 837a2ddf1f5..e4525d4ca06 100644 --- a/src/hotspot/share/compiler/compilerOracle.cpp +++ b/src/hotspot/share/compiler/compilerOracle.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "compiler/compilerOracle.hpp" #include "compiler/methodMatcher.hpp" #include "memory/allocation.inline.hpp" @@ -31,7 +32,6 @@ #include "oops/klass.hpp" #include "oops/method.hpp" #include "oops/symbol.hpp" -#include "prims/jvm.h" #include "runtime/handles.inline.hpp" #include "runtime/jniHandles.hpp" #include "runtime/os.hpp" diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp index 3d81bc00965..e6b8e91190e 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp @@ -3621,7 +3621,7 @@ Monitor* G1CodeCacheUnloadingTask::_lock = new Monitor(Mutex::leaf, "Code Cache class G1KlassCleaningTask : public StackObj { BoolObjectClosure* _is_alive; - volatile jint _clean_klass_tree_claimed; + volatile int _clean_klass_tree_claimed; ClassLoaderDataGraphKlassIteratorAtomic _klass_iterator; public: @@ -3637,7 +3637,7 @@ class G1KlassCleaningTask : public StackObj { return false; } - return Atomic::cmpxchg(1, (jint*)&_clean_klass_tree_claimed, 0) == 0; + return Atomic::cmpxchg(1, &_clean_klass_tree_claimed, 0) == 0; } InstanceKlass* claim_next_klass() { @@ -3674,7 +3674,7 @@ public: class G1ResolvedMethodCleaningTask : public StackObj { BoolObjectClosure* _is_alive; - volatile jint _resolved_method_task_claimed; + volatile int _resolved_method_task_claimed; public: G1ResolvedMethodCleaningTask(BoolObjectClosure* is_alive) : _is_alive(is_alive), _resolved_method_task_claimed(0) {} @@ -3683,7 +3683,7 @@ public: if (_resolved_method_task_claimed) { return false; } - return Atomic::cmpxchg(1, (jint*)&_resolved_method_task_claimed, 0) == 0; + return Atomic::cmpxchg(1, &_resolved_method_task_claimed, 0) == 0; } // These aren't big, one thread can do it all. diff --git a/src/hotspot/share/gc/g1/satbMarkQueue.cpp b/src/hotspot/share/gc/g1/satbMarkQueue.cpp index ac9b0a3d064..ddca44b038c 100644 --- a/src/hotspot/share/gc/g1/satbMarkQueue.cpp +++ b/src/hotspot/share/gc/g1/satbMarkQueue.cpp @@ -23,12 +23,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "gc/g1/g1CollectedHeap.inline.hpp" #include "gc/g1/satbMarkQueue.hpp" #include "gc/shared/collectedHeap.hpp" #include "memory/allocation.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/mutexLocker.hpp" #include "runtime/safepoint.hpp" #include "runtime/thread.hpp" diff --git a/src/hotspot/share/gc/shared/ageTable.cpp b/src/hotspot/share/gc/shared/ageTable.cpp index daa85b5012d..14cac1b4bb5 100644 --- a/src/hotspot/share/gc/shared/ageTable.cpp +++ b/src/hotspot/share/gc/shared/ageTable.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "gc/shared/ageTable.inline.hpp" #include "gc/shared/ageTableTracer.hpp" #include "gc/shared/collectedHeap.hpp" @@ -30,7 +31,6 @@ #include "memory/resourceArea.hpp" #include "logging/log.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "utilities/copy.hpp" /* Copyright (c) 1992, 2016, Oracle and/or its affiliates, and Stanford University. diff --git a/src/hotspot/share/gc/shared/gcId.cpp b/src/hotspot/share/gc/shared/gcId.cpp index 5e622ba2165..cc1d83bf587 100644 --- a/src/hotspot/share/gc/shared/gcId.cpp +++ b/src/hotspot/share/gc/shared/gcId.cpp @@ -23,8 +23,8 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "gc/shared/gcId.hpp" -#include "prims/jvm.h" #include "runtime/safepoint.hpp" #include "runtime/thread.inline.hpp" diff --git a/src/hotspot/share/gc/shared/gcTimer.hpp b/src/hotspot/share/gc/shared/gcTimer.hpp index b58d709362e..14f397dd408 100644 --- a/src/hotspot/share/gc/shared/gcTimer.hpp +++ b/src/hotspot/share/gc/shared/gcTimer.hpp @@ -26,7 +26,6 @@ #define SHARE_VM_GC_SHARED_GCTIMER_HPP #include "memory/allocation.hpp" -#include "prims/jni_md.h" #include "utilities/macros.hpp" #include "utilities/ticks.hpp" diff --git a/src/hotspot/share/gc/shared/gcTraceTime.inline.hpp b/src/hotspot/share/gc/shared/gcTraceTime.inline.hpp index 35fc1961268..06dc268cfd8 100644 --- a/src/hotspot/share/gc/shared/gcTraceTime.inline.hpp +++ b/src/hotspot/share/gc/shared/gcTraceTime.inline.hpp @@ -32,7 +32,6 @@ #include "logging/log.hpp" #include "logging/logStream.hpp" #include "memory/universe.hpp" -#include "prims/jni_md.h" #include "utilities/ticks.hpp" #define LOG_STOP_HEAP_FORMAT SIZE_FORMAT "M->" SIZE_FORMAT "M(" SIZE_FORMAT "M)" diff --git a/src/hotspot/share/interpreter/linkResolver.cpp b/src/hotspot/share/interpreter/linkResolver.cpp index 11bc7f1dd0f..c47069ec0ed 100644 --- a/src/hotspot/share/interpreter/linkResolver.cpp +++ b/src/hotspot/share/interpreter/linkResolver.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/defaultMethods.hpp" #include "classfile/javaClasses.hpp" #include "classfile/resolutionErrors.hpp" @@ -42,7 +43,6 @@ #include "oops/method.hpp" #include "oops/objArrayOop.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/methodHandles.hpp" #include "prims/nativeLookup.hpp" #include "runtime/compilationPolicy.hpp" diff --git a/src/hotspot/share/jvmci/jvmciCodeInstaller.cpp b/src/hotspot/share/jvmci/jvmciCodeInstaller.cpp index 977ff966735..19a5bb6a3f6 100644 --- a/src/hotspot/share/jvmci/jvmciCodeInstaller.cpp +++ b/src/hotspot/share/jvmci/jvmciCodeInstaller.cpp @@ -45,7 +45,7 @@ // use). ConstantOopWriteValue* CodeInstaller::_oop_null_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantOopWriteValue(NULL); ConstantIntValue* CodeInstaller::_int_m1_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue(-1); -ConstantIntValue* CodeInstaller::_int_0_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue(0); +ConstantIntValue* CodeInstaller::_int_0_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue((jint)0); ConstantIntValue* CodeInstaller::_int_1_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue(1); ConstantIntValue* CodeInstaller::_int_2_scope_value = new (ResourceObj::C_HEAP, mtCompiler) ConstantIntValue(2); LocationValue* CodeInstaller::_illegal_value = new (ResourceObj::C_HEAP, mtCompiler) LocationValue(Location()); diff --git a/src/hotspot/share/jvmci/jvmciCompiler.cpp b/src/hotspot/share/jvmci/jvmciCompiler.cpp index 94680ecbe42..e76a600ba25 100644 --- a/src/hotspot/share/jvmci/jvmciCompiler.cpp +++ b/src/hotspot/share/jvmci/jvmciCompiler.cpp @@ -22,10 +22,10 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "memory/oopFactory.hpp" #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/javaCalls.hpp" #include "runtime/handles.hpp" #include "jvmci/jvmciJavaClasses.hpp" diff --git a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp index ca65c1d6414..bccb7fdcff8 100644 --- a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp +++ b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp @@ -1697,7 +1697,7 @@ C2V_VMENTRY(void, writeDebugOutput, (JNIEnv*, jobject, jbyteArray bytes, jint of } while (length > 0) { jbyte* start = array->byte_at_addr(offset); - tty->write((char*) start, MIN2(length, O_BUFLEN)); + tty->write((char*) start, MIN2(length, (jint)O_BUFLEN)); length -= O_BUFLEN; offset += O_BUFLEN; } diff --git a/src/hotspot/share/jvmci/jvmciRuntime.cpp b/src/hotspot/share/jvmci/jvmciRuntime.cpp index 2cc7efe0d26..e64551e08a7 100644 --- a/src/hotspot/share/jvmci/jvmciRuntime.cpp +++ b/src/hotspot/share/jvmci/jvmciRuntime.cpp @@ -22,6 +22,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "asm/codeBuffer.hpp" #include "classfile/javaClasses.inline.hpp" #include "code/codeCache.hpp" @@ -37,7 +38,6 @@ #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" #include "oops/objArrayOop.inline.hpp" -#include "prims/jvm.h" #include "runtime/biasedLocking.hpp" #include "runtime/interfaceSupport.hpp" #include "runtime/reflection.hpp" @@ -611,7 +611,7 @@ JRT_ENTRY(jboolean, JVMCIRuntime::thread_is_interrupted(JavaThread* thread, oopD } JRT_END -JRT_ENTRY(jint, JVMCIRuntime::test_deoptimize_call_int(JavaThread* thread, int value)) +JRT_ENTRY(int, JVMCIRuntime::test_deoptimize_call_int(JavaThread* thread, int value)) deopt_caller(); return value; JRT_END diff --git a/src/hotspot/share/jvmci/jvmci_globals.cpp b/src/hotspot/share/jvmci/jvmci_globals.cpp index 3739a9ea161..f9dba2af188 100644 --- a/src/hotspot/share/jvmci/jvmci_globals.cpp +++ b/src/hotspot/share/jvmci/jvmci_globals.cpp @@ -23,8 +23,8 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "jvmci/jvmci_globals.hpp" -#include "prims/jvm.h" #include "utilities/defaultStream.hpp" #include "runtime/globals_extension.hpp" diff --git a/src/hotspot/share/logging/logConfiguration.cpp b/src/hotspot/share/logging/logConfiguration.cpp index 7320a17c1c8..d1916740f72 100644 --- a/src/hotspot/share/logging/logConfiguration.cpp +++ b/src/hotspot/share/logging/logConfiguration.cpp @@ -22,6 +22,7 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/log.hpp" #include "logging/logConfiguration.hpp" #include "logging/logDecorations.hpp" @@ -34,7 +35,6 @@ #include "logging/logTagSet.hpp" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/os.inline.hpp" #include "runtime/semaphore.hpp" #include "utilities/globalDefinitions.hpp" diff --git a/src/hotspot/share/logging/logDecorations.cpp b/src/hotspot/share/logging/logDecorations.cpp index 45947398d0f..6fdf29f0319 100644 --- a/src/hotspot/share/logging/logDecorations.cpp +++ b/src/hotspot/share/logging/logDecorations.cpp @@ -22,9 +22,9 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/logConfiguration.hpp" #include "logging/logDecorations.hpp" -#include "prims/jvm.h" #include "runtime/os.inline.hpp" #include "runtime/thread.inline.hpp" #include "services/management.hpp" diff --git a/src/hotspot/share/logging/logFileOutput.cpp b/src/hotspot/share/logging/logFileOutput.cpp index 2b4d0e8f851..7b782d7a10c 100644 --- a/src/hotspot/share/logging/logFileOutput.cpp +++ b/src/hotspot/share/logging/logFileOutput.cpp @@ -22,11 +22,11 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/log.hpp" #include "logging/logConfiguration.hpp" #include "logging/logFileOutput.hpp" #include "memory/allocation.inline.hpp" -#include "prims/jvm.h" #include "runtime/arguments.hpp" #include "runtime/os.inline.hpp" #include "utilities/globalDefinitions.hpp" diff --git a/src/hotspot/share/logging/logFileStreamOutput.cpp b/src/hotspot/share/logging/logFileStreamOutput.cpp index 47afc69652c..12a68d03798 100644 --- a/src/hotspot/share/logging/logFileStreamOutput.cpp +++ b/src/hotspot/share/logging/logFileStreamOutput.cpp @@ -22,12 +22,12 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/logDecorators.hpp" #include "logging/logDecorations.hpp" #include "logging/logFileStreamOutput.hpp" #include "logging/logMessageBuffer.hpp" #include "memory/allocation.inline.hpp" -#include "prims/jvm.h" static bool initialized; static union { diff --git a/src/hotspot/share/logging/logOutput.cpp b/src/hotspot/share/logging/logOutput.cpp index 68eb895c653..a9039ff8c7f 100644 --- a/src/hotspot/share/logging/logOutput.cpp +++ b/src/hotspot/share/logging/logOutput.cpp @@ -22,11 +22,11 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/logFileStreamOutput.hpp" #include "logging/logOutput.hpp" #include "logging/logTagSet.hpp" #include "memory/allocation.inline.hpp" -#include "prims/jvm.h" #include "runtime/mutexLocker.hpp" #include "runtime/os.inline.hpp" diff --git a/src/hotspot/share/logging/logTagSet.cpp b/src/hotspot/share/logging/logTagSet.cpp index ec1a5d09524..1be30d36e4a 100644 --- a/src/hotspot/share/logging/logTagSet.cpp +++ b/src/hotspot/share/logging/logTagSet.cpp @@ -22,6 +22,7 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/logDecorations.hpp" #include "logging/logFileStreamOutput.hpp" #include "logging/logLevel.hpp" @@ -31,7 +32,6 @@ #include "logging/logTagSet.hpp" #include "logging/logTagSetDescriptions.hpp" #include "memory/allocation.inline.hpp" -#include "prims/jvm.h" #include "utilities/ostream.hpp" LogTagSet* LogTagSet::_list = NULL; diff --git a/src/hotspot/share/memory/filemap.cpp b/src/hotspot/share/memory/filemap.cpp index e4127d05b4e..75d04c2db7e 100644 --- a/src/hotspot/share/memory/filemap.cpp +++ b/src/hotspot/share/memory/filemap.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/compactHashtable.inline.hpp" #include "classfile/sharedClassUtil.hpp" @@ -42,7 +43,6 @@ #include "memory/metaspaceShared.hpp" #include "memory/oopFactory.hpp" #include "oops/objArrayOop.hpp" -#include "prims/jvm.h" #include "prims/jvmtiExport.hpp" #include "runtime/arguments.hpp" #include "runtime/java.hpp" diff --git a/src/hotspot/share/memory/metaspaceShared.cpp b/src/hotspot/share/memory/metaspaceShared.cpp index ea92ff79d92..7d90abd4aae 100644 --- a/src/hotspot/share/memory/metaspaceShared.cpp +++ b/src/hotspot/share/memory/metaspaceShared.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classListParser.hpp" #include "classfile/classLoaderExt.hpp" #include "classfile/dictionary.hpp" @@ -55,7 +56,6 @@ #include "oops/objArrayOop.hpp" #include "oops/oop.inline.hpp" #include "oops/typeArrayKlass.hpp" -#include "prims/jvm.h" #include "prims/jvmtiRedefineClasses.hpp" #include "runtime/timerTrace.hpp" #include "runtime/os.hpp" diff --git a/src/hotspot/share/oops/constantPool.cpp b/src/hotspot/share/oops/constantPool.cpp index 4b8c191b16f..f1aec2bff2e 100644 --- a/src/hotspot/share/oops/constantPool.cpp +++ b/src/hotspot/share/oops/constantPool.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classLoaderData.hpp" #include "classfile/javaClasses.inline.hpp" #include "classfile/metadataOnStackMark.hpp" @@ -41,7 +42,6 @@ #include "oops/objArrayKlass.hpp" #include "oops/objArrayOop.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/fieldType.hpp" #include "runtime/init.hpp" #include "runtime/javaCalls.hpp" diff --git a/src/hotspot/share/oops/generateOopMap.cpp b/src/hotspot/share/oops/generateOopMap.cpp index 6e0da2b5c61..0cab539d4d5 100644 --- a/src/hotspot/share/oops/generateOopMap.cpp +++ b/src/hotspot/share/oops/generateOopMap.cpp @@ -23,13 +23,13 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "interpreter/bytecodeStream.hpp" #include "logging/log.hpp" #include "logging/logStream.hpp" #include "oops/generateOopMap.hpp" #include "oops/oop.inline.hpp" #include "oops/symbol.hpp" -#include "prims/jvm.h" #include "runtime/handles.inline.hpp" #include "runtime/java.hpp" #include "runtime/relocator.hpp" diff --git a/src/hotspot/share/oops/instanceKlass.cpp b/src/hotspot/share/oops/instanceKlass.cpp index 8da123a5277..4888d180e13 100644 --- a/src/hotspot/share/oops/instanceKlass.cpp +++ b/src/hotspot/share/oops/instanceKlass.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "aot/aotLoader.hpp" #include "classfile/classFileParser.hpp" #include "classfile/classFileStream.hpp" @@ -59,7 +60,6 @@ #include "oops/method.hpp" #include "oops/oop.inline.hpp" #include "oops/symbol.hpp" -#include "prims/jvm.h" #include "prims/jvmtiExport.hpp" #include "prims/jvmtiRedefineClasses.hpp" #include "prims/jvmtiThreadState.hpp" diff --git a/src/hotspot/share/oops/klassVtable.cpp b/src/hotspot/share/oops/klassVtable.cpp index 6ebf6988b43..1789f6c6264 100644 --- a/src/hotspot/share/oops/klassVtable.cpp +++ b/src/hotspot/share/oops/klassVtable.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" #include "gc/shared/gcLocker.hpp" @@ -36,7 +37,6 @@ #include "oops/method.hpp" #include "oops/objArrayOop.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/arguments.hpp" #include "runtime/handles.inline.hpp" #include "utilities/copy.hpp" diff --git a/src/hotspot/share/opto/castnode.cpp b/src/hotspot/share/opto/castnode.cpp index 4113b10be58..ebd7d33112e 100644 --- a/src/hotspot/share/opto/castnode.cpp +++ b/src/hotspot/share/opto/castnode.cpp @@ -237,8 +237,8 @@ Node *CastIINode::Ideal(PhaseGVN *phase, bool can_reshape) { if (in_type != NULL && this_type != NULL && (in_type->_lo != this_type->_lo || in_type->_hi != this_type->_hi)) { - int lo1 = this_type->_lo; - int hi1 = this_type->_hi; + jint lo1 = this_type->_lo; + jint hi1 = this_type->_hi; int w1 = this_type->_widen; if (lo1 >= 0) { diff --git a/src/hotspot/share/opto/loopTransform.cpp b/src/hotspot/share/opto/loopTransform.cpp index 4737a8ffe9e..2bad280129a 100644 --- a/src/hotspot/share/opto/loopTransform.cpp +++ b/src/hotspot/share/opto/loopTransform.cpp @@ -2129,7 +2129,7 @@ int PhaseIdealLoop::do_range_check( IdealLoopTree *loop, Node_List &old_new ) { // Look for trip_counter + offset vs limit Node *rc_exp = cmp->in(1); Node *limit = cmp->in(2); - jint scale_con= 1; // Assume trip counter not scaled + int scale_con= 1; // Assume trip counter not scaled Node *limit_c = get_ctrl(limit); if( loop->is_member(get_loop(limit_c) ) ) { diff --git a/src/hotspot/share/opto/mulnode.cpp b/src/hotspot/share/opto/mulnode.cpp index e0bf418a0bd..b1be31e35cf 100644 --- a/src/hotspot/share/opto/mulnode.cpp +++ b/src/hotspot/share/opto/mulnode.cpp @@ -235,13 +235,13 @@ const Type *MulINode::mul_ring(const Type *t0, const Type *t1) const { const TypeInt *r1 = t1->is_int(); // Fetch endpoints of all ranges - int32_t lo0 = r0->_lo; + jint lo0 = r0->_lo; double a = (double)lo0; - int32_t hi0 = r0->_hi; + jint hi0 = r0->_hi; double b = (double)hi0; - int32_t lo1 = r1->_lo; + jint lo1 = r1->_lo; double c = (double)lo1; - int32_t hi1 = r1->_hi; + jint hi1 = r1->_hi; double d = (double)hi1; // Compute all endpoints & check for overflow diff --git a/src/hotspot/share/opto/output.cpp b/src/hotspot/share/opto/output.cpp index c4a76dc6dd9..db712c80dfc 100644 --- a/src/hotspot/share/opto/output.cpp +++ b/src/hotspot/share/opto/output.cpp @@ -577,10 +577,10 @@ void Compile::FillLocArray( int idx, MachSafePointNode* sfpt, Node *local, // arbitrary, and are not tied to any machine-level encodings.) #ifdef _LP64 if( t->base() == Type::DoubleBot || t->base() == Type::DoubleCon ) { - array->append(new ConstantIntValue(0)); + array->append(new ConstantIntValue((jint)0)); array->append(new_loc_value( _regalloc, regnum, Location::dbl )); } else if ( t->base() == Type::Long ) { - array->append(new ConstantIntValue(0)); + array->append(new ConstantIntValue((jint)0)); array->append(new_loc_value( _regalloc, regnum, Location::lng )); } else if ( t->base() == Type::RawPtr ) { // jsr/ret return address which must be restored into a the full @@ -663,7 +663,7 @@ void Compile::FillLocArray( int idx, MachSafePointNode* sfpt, Node *local, case Type::DoubleCon: { jdouble d = t->is_double_constant()->getd(); #ifdef _LP64 - array->append(new ConstantIntValue(0)); + array->append(new ConstantIntValue((jint)0)); array->append(new ConstantDoubleValue(d)); #else // Repack the double as two jints. @@ -683,7 +683,7 @@ void Compile::FillLocArray( int idx, MachSafePointNode* sfpt, Node *local, case Type::Long: { jlong d = t->is_long()->get_con(); #ifdef _LP64 - array->append(new ConstantIntValue(0)); + array->append(new ConstantIntValue((jint)0)); array->append(new ConstantLongValue(d)); #else // Repack the long as two jints. diff --git a/src/hotspot/share/opto/parse3.cpp b/src/hotspot/share/opto/parse3.cpp index ee0997e6913..282c3072959 100644 --- a/src/hotspot/share/opto/parse3.cpp +++ b/src/hotspot/share/opto/parse3.cpp @@ -412,11 +412,11 @@ void Parse::do_multianewarray() { // The original expression was of this form: new T[length0][length1]... // It is often the case that the lengths are small (except the last). // If that happens, use the fast 1-d creator a constant number of times. - const jint expand_limit = MIN2((jint)MultiArrayExpandLimit, 100); - jint expand_count = 1; // count of allocations in the expansion - jint expand_fanout = 1; // running total fanout + const int expand_limit = MIN2((int)MultiArrayExpandLimit, 100); + int expand_count = 1; // count of allocations in the expansion + int expand_fanout = 1; // running total fanout for (j = 0; j < ndimensions-1; j++) { - jint dim_con = find_int_con(length[j], -1); + int dim_con = find_int_con(length[j], -1); expand_fanout *= dim_con; expand_count += expand_fanout; // count the level-J sub-arrays if (dim_con <= 0 diff --git a/src/hotspot/share/opto/phaseX.cpp b/src/hotspot/share/opto/phaseX.cpp index d7bedfcb396..e10f866b35c 100644 --- a/src/hotspot/share/opto/phaseX.cpp +++ b/src/hotspot/share/opto/phaseX.cpp @@ -723,7 +723,7 @@ ConNode* PhaseValues::uncached_makecon(const Type *t) { //------------------------------intcon----------------------------------------- // Fast integer constant. Same as "transform(new ConINode(TypeInt::make(i)))" -ConINode* PhaseTransform::intcon(int i) { +ConINode* PhaseTransform::intcon(jint i) { // Small integer? Check cache! Check that cached node is not dead if (i >= _icon_min && i <= _icon_max) { ConINode* icon = _icons[i-_icon_min]; diff --git a/src/hotspot/share/opto/type.cpp b/src/hotspot/share/opto/type.cpp index 08b56cd3d14..e4ffa7f41f1 100644 --- a/src/hotspot/share/opto/type.cpp +++ b/src/hotspot/share/opto/type.cpp @@ -1503,7 +1503,7 @@ bool TypeInt::eq( const Type *t ) const { //------------------------------hash------------------------------------------- // Type-specific hashing function. int TypeInt::hash(void) const { - return java_add(java_add(_lo, _hi), java_add(_widen, (int)Type::Int)); + return java_add(java_add(_lo, _hi), java_add((jint)_widen, (jint)Type::Int)); } //------------------------------is_finite-------------------------------------- @@ -2505,7 +2505,7 @@ bool TypePtr::eq( const Type *t ) const { //------------------------------hash------------------------------------------- // Type-specific hashing function. int TypePtr::hash(void) const { - return java_add(java_add(_ptr, _offset), java_add( hash_speculative(), _inline_depth)); + return java_add(java_add((jint)_ptr, (jint)_offset), java_add((jint)hash_speculative(), (jint)_inline_depth)); ; } @@ -3338,8 +3338,8 @@ bool TypeOopPtr::eq( const Type *t ) const { // Type-specific hashing function. int TypeOopPtr::hash(void) const { return - java_add(java_add(const_oop() ? const_oop()->hash() : 0, _klass_is_exact), - java_add(_instance_id, TypePtr::hash())); + java_add(java_add((jint)(const_oop() ? const_oop()->hash() : 0), (jint)_klass_is_exact), + java_add((jint)_instance_id, (jint)TypePtr::hash())); } //------------------------------dump2------------------------------------------ @@ -3946,7 +3946,7 @@ bool TypeInstPtr::eq( const Type *t ) const { //------------------------------hash------------------------------------------- // Type-specific hashing function. int TypeInstPtr::hash(void) const { - int hash = java_add(klass()->hash(), TypeOopPtr::hash()); + int hash = java_add((jint)klass()->hash(), (jint)TypeOopPtr::hash()); return hash; } @@ -4871,7 +4871,7 @@ bool TypeKlassPtr::eq( const Type *t ) const { //------------------------------hash------------------------------------------- // Type-specific hashing function. int TypeKlassPtr::hash(void) const { - return java_add(klass()->hash(), TypePtr::hash()); + return java_add((jint)klass()->hash(), (jint)TypePtr::hash()); } //------------------------------singleton-------------------------------------- diff --git a/src/hotspot/share/precompiled/precompiled.hpp b/src/hotspot/share/precompiled/precompiled.hpp index f3b26b49dcc..dd76a0b9c93 100644 --- a/src/hotspot/share/precompiled/precompiled.hpp +++ b/src/hotspot/share/precompiled/precompiled.hpp @@ -27,6 +27,7 @@ #ifndef DONT_USE_PRECOMPILED_HEADER # include "jni.h" +# include "jvm.h" # include "asm/assembler.hpp" # include "asm/assembler.inline.hpp" # include "asm/codeBuffer.hpp" @@ -160,7 +161,6 @@ # include "oops/symbol.hpp" # include "oops/typeArrayKlass.hpp" # include "oops/typeArrayOop.hpp" -# include "prims/jvm.h" # include "prims/jvmtiExport.hpp" # include "prims/methodHandles.hpp" # include "runtime/arguments.hpp" diff --git a/src/hotspot/share/prims/jni.cpp b/src/hotspot/share/prims/jni.cpp index b4854477784..ca0f634d7f5 100644 --- a/src/hotspot/share/prims/jni.cpp +++ b/src/hotspot/share/prims/jni.cpp @@ -25,6 +25,7 @@ #include "precompiled.hpp" #include "jni.h" +#include "jvm.h" #include "ci/ciReplay.hpp" #include "classfile/altHashing.hpp" #include "classfile/classFileStream.hpp" @@ -55,7 +56,6 @@ #include "prims/jniCheck.hpp" #include "prims/jniExport.hpp" #include "prims/jniFastGetField.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "prims/jvmtiExport.hpp" #include "prims/jvmtiThreadState.hpp" @@ -263,7 +263,7 @@ void jfieldIDWorkaround::verify_instance_jfieldID(Klass* k, jfieldID id) { #ifdef ASSERT Histogram* JNIHistogram; - static volatile jint JNIHistogram_lock = 0; + static volatile int JNIHistogram_lock = 0; class JNIHistogramElement : public HistogramElement { public: @@ -3277,9 +3277,9 @@ JNI_END // Initialization state for three routines below relating to // java.nio.DirectBuffers -static jint directBufferSupportInitializeStarted = 0; -static volatile jint directBufferSupportInitializeEnded = 0; -static volatile jint directBufferSupportInitializeFailed = 0; +static int directBufferSupportInitializeStarted = 0; +static volatile int directBufferSupportInitializeEnded = 0; +static volatile int directBufferSupportInitializeFailed = 0; static jclass bufferClass = NULL; static jclass directBufferClass = NULL; static jclass directByteBufferClass = NULL; @@ -3844,9 +3844,9 @@ struct JNINativeInterface_* jni_functions_nocheck() { extern const struct JNIInvokeInterface_ jni_InvokeInterface; // Global invocation API vars -volatile jint vm_created = 0; +volatile int vm_created = 0; // Indicate whether it is safe to recreate VM -volatile jint safe_to_recreate_vm = 1; +volatile int safe_to_recreate_vm = 1; struct JavaVM_ main_vm = {&jni_InvokeInterface}; @@ -4045,7 +4045,7 @@ _JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_GetCreatedJavaVMs(JavaVM **vm_buf, jsize HOTSPOT_JNI_GETCREATEDJAVAVMS_ENTRY((void **) vm_buf, bufLen, (uintptr_t *) numVMs); - if (vm_created) { + if (vm_created == 1) { if (numVMs != NULL) *numVMs = 1; if (bufLen > 0) *vm_buf = (JavaVM *)(&main_vm); } else { @@ -4065,7 +4065,7 @@ static jint JNICALL jni_DestroyJavaVM_inner(JavaVM *vm) { jint res = JNI_ERR; DT_RETURN_MARK(DestroyJavaVM, jint, (const jint&)res); - if (!vm_created) { + if (vm_created == 0) { res = JNI_ERR; return res; } @@ -4086,7 +4086,7 @@ static jint JNICALL jni_DestroyJavaVM_inner(JavaVM *vm) { ThreadStateTransition::transition_from_native(thread, _thread_in_vm); if (Threads::destroy_vm()) { // Should not change thread state, VM is gone - vm_created = false; + vm_created = 0; res = JNI_OK; return res; } else { @@ -4226,7 +4226,7 @@ static jint attach_current_thread(JavaVM *vm, void **penv, void *_args, bool dae jint JNICALL jni_AttachCurrentThread(JavaVM *vm, void **penv, void *_args) { HOTSPOT_JNI_ATTACHCURRENTTHREAD_ENTRY(vm, penv, _args); - if (!vm_created) { + if (vm_created == 0) { HOTSPOT_JNI_ATTACHCURRENTTHREAD_RETURN((uint32_t) JNI_ERR); return JNI_ERR; } @@ -4285,7 +4285,7 @@ jint JNICALL jni_GetEnv(JavaVM *vm, void **penv, jint version) { jint ret = JNI_ERR; DT_RETURN_MARK(GetEnv, jint, (const jint&)ret); - if (!vm_created) { + if (vm_created == 0) { *penv = NULL; ret = JNI_EDETACHED; return ret; @@ -4336,7 +4336,7 @@ jint JNICALL jni_GetEnv(JavaVM *vm, void **penv, jint version) { jint JNICALL jni_AttachCurrentThreadAsDaemon(JavaVM *vm, void **penv, void *_args) { HOTSPOT_JNI_ATTACHCURRENTTHREADASDAEMON_ENTRY(vm, penv, _args); - if (!vm_created) { + if (vm_created == 0) { HOTSPOT_JNI_ATTACHCURRENTTHREADASDAEMON_RETURN((uint32_t) JNI_ERR); return JNI_ERR; } diff --git a/src/hotspot/share/prims/jniCheck.cpp b/src/hotspot/share/prims/jniCheck.cpp index 6d8d27ccccd..34281a7a071 100644 --- a/src/hotspot/share/prims/jniCheck.cpp +++ b/src/hotspot/share/prims/jniCheck.cpp @@ -24,6 +24,7 @@ #include "precompiled.hpp" #include "jni.h" +#include "jvm.h" #include "classfile/javaClasses.inline.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -32,7 +33,6 @@ #include "oops/oop.inline.hpp" #include "oops/symbol.hpp" #include "prims/jniCheck.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "runtime/fieldDescriptor.hpp" #include "runtime/handles.hpp" diff --git a/src/hotspot/share/prims/jni_md.h b/src/hotspot/share/prims/jni_md.h deleted file mode 100644 index 6a078ed51c8..00000000000 --- a/src/hotspot/share/prims/jni_md.h +++ /dev/null @@ -1,39 +0,0 @@ -/* - * Copyright (c) 1997, 2016, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ - -#include "utilities/macros.hpp" - -/* Switch to the correct jni_md.h file without reliance on -I options. */ -#include CPU_HEADER_H(jni) - -/* - The local copies of JNI header files may be refreshed - from a JDK distribution by means of these commands: - - cp ${JDK_DIST}/solaris/include/solaris/jni_md.h ./jni_sparc.h - cp ${JDK_DIST}/win32/include/win32/jni_md.h ./jni_i486.h - cp ${JDK_DIST}/win32/include/jni.h ./jni.h - -*/ diff --git a/src/hotspot/share/prims/jvm.cpp b/src/hotspot/share/prims/jvm.cpp index faca8621afa..8dbaab493bb 100644 --- a/src/hotspot/share/prims/jvm.cpp +++ b/src/hotspot/share/prims/jvm.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classFileStream.hpp" #include "classfile/classLoader.hpp" #include "classfile/classLoaderData.inline.hpp" @@ -46,7 +47,6 @@ #include "oops/objArrayKlass.hpp" #include "oops/objArrayOop.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "prims/jvmtiExport.hpp" #include "prims/jvmtiThreadState.hpp" @@ -215,7 +215,7 @@ void trace_class_resolution(Klass* to_class) { #ifdef ASSERT Histogram* JVMHistogram; - volatile jint JVMHistogram_lock = 0; + volatile int JVMHistogram_lock = 0; class JVMHistogramElement : public HistogramElement { public: @@ -3766,7 +3766,7 @@ JVM_ENTRY(void, JVM_GetVersionInfo(JNIEnv* env, jvm_version_info* info, size_t i // when we add a new capability in the jvm_version_info struct, we should also // consider to expose this new capability in the sun.rt.jvmCapabilities jvmstat // counter defined in runtimeService.cpp. - info->is_attachable = AttachListener::is_attach_supported(); + info->is_attach_supported = AttachListener::is_attach_supported(); } JVM_END diff --git a/src/hotspot/share/prims/jvm.h b/src/hotspot/share/prims/jvm.h deleted file mode 100644 index 1103c7628f5..00000000000 --- a/src/hotspot/share/prims/jvm.h +++ /dev/null @@ -1,1373 +0,0 @@ -/* - * Copyright (c) 1997, 2017, 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. - * - */ - -#ifndef SHARE_VM_PRIMS_JVM_H -#define SHARE_VM_PRIMS_JVM_H - -#include "jni.h" -#include "utilities/macros.hpp" - -#include OS_HEADER_H(jvm) - -#ifndef _JAVASOFT_JVM_H_ -#define _JAVASOFT_JVM_H_ - -// HotSpot integration note: -// -// This file and jvm.h used with the JDK are identical, -// except for the three includes removed below - -// #include -// #include "jni.h" -// #include "jvm_md.h" - - -#ifdef __cplusplus -extern "C" { -#endif - -/* - * This file contains additional functions exported from the VM. - * These functions are complementary to the standard JNI support. - * There are three parts to this file: - * - * First, this file contains the VM-related functions needed by native - * libraries in the standard Java API. For example, the java.lang.Object - * class needs VM-level functions that wait for and notify monitors. - * - * Second, this file contains the functions and constant definitions - * needed by the byte code verifier and class file format checker. - * These functions allow the verifier and format checker to be written - * in a VM-independent way. - * - * Third, this file contains various I/O and nerwork operations needed - * by the standard Java I/O and network APIs. - */ - -/* - * Bump the version number when either of the following happens: - * - * 1. There is a change in JVM_* functions. - * - * 2. There is a change in the contract between VM and Java classes. - * For example, if the VM relies on a new private field in Thread - * class. - */ - -#define JVM_INTERFACE_VERSION 5 - -JNIEXPORT jobjectArray JNICALL -JVM_GetMethodParameters(JNIEnv *env, jobject method); - -JNIEXPORT jint JNICALL -JVM_GetInterfaceVersion(void); - -/************************************************************************* - PART 1: Functions for Native Libraries - ************************************************************************/ -/* - * java.lang.Object - */ -JNIEXPORT jint JNICALL -JVM_IHashCode(JNIEnv *env, jobject obj); - -JNIEXPORT void JNICALL -JVM_MonitorWait(JNIEnv *env, jobject obj, jlong ms); - -JNIEXPORT void JNICALL -JVM_MonitorNotify(JNIEnv *env, jobject obj); - -JNIEXPORT void JNICALL -JVM_MonitorNotifyAll(JNIEnv *env, jobject obj); - -JNIEXPORT jobject JNICALL -JVM_Clone(JNIEnv *env, jobject obj); - -/* - * java.lang.String - */ -JNIEXPORT jstring JNICALL -JVM_InternString(JNIEnv *env, jstring str); - -/* - * java.lang.System - */ -JNIEXPORT jlong JNICALL -JVM_CurrentTimeMillis(JNIEnv *env, jclass ignored); - -JNIEXPORT jlong JNICALL -JVM_NanoTime(JNIEnv *env, jclass ignored); - -JNIEXPORT jlong JNICALL -JVM_GetNanoTimeAdjustment(JNIEnv *env, jclass ignored, jlong offset_secs); - -JNIEXPORT void JNICALL -JVM_ArrayCopy(JNIEnv *env, jclass ignored, jobject src, jint src_pos, - jobject dst, jint dst_pos, jint length); - -JNIEXPORT jobject JNICALL -JVM_InitProperties(JNIEnv *env, jobject p); - -/* - * java.lang.Runtime - */ - -JNIEXPORT void JNICALL -JVM_Halt(jint code); - -JNIEXPORT void JNICALL -JVM_GC(void); - -/* Returns the number of real-time milliseconds that have elapsed since the - * least-recently-inspected heap object was last inspected by the garbage - * collector. - * - * For simple stop-the-world collectors this value is just the time - * since the most recent collection. For generational collectors it is the - * time since the oldest generation was most recently collected. Other - * collectors are free to return a pessimistic estimate of the elapsed time, or - * simply the time since the last full collection was performed. - * - * Note that in the presence of reference objects, a given object that is no - * longer strongly reachable may have to be inspected multiple times before it - * can be reclaimed. - */ -JNIEXPORT jlong JNICALL -JVM_MaxObjectInspectionAge(void); - -JNIEXPORT jlong JNICALL -JVM_TotalMemory(void); - -JNIEXPORT jlong JNICALL -JVM_FreeMemory(void); - -JNIEXPORT jlong JNICALL -JVM_MaxMemory(void); - -JNIEXPORT jint JNICALL -JVM_ActiveProcessorCount(void); - -JNIEXPORT void * JNICALL -JVM_LoadLibrary(const char *name); - -JNIEXPORT void JNICALL -JVM_UnloadLibrary(void * handle); - -JNIEXPORT void * JNICALL -JVM_FindLibraryEntry(void *handle, const char *name); - -JNIEXPORT jboolean JNICALL -JVM_IsSupportedJNIVersion(jint version); - -JNIEXPORT jobjectArray JNICALL -JVM_GetVmArguments(JNIEnv *env); - -/* - * java.lang.Throwable - */ -JNIEXPORT void JNICALL -JVM_FillInStackTrace(JNIEnv *env, jobject throwable); - -/* - * java.lang.StackTraceElement - */ -JNIEXPORT void JNICALL -JVM_InitStackTraceElementArray(JNIEnv *env, jobjectArray elements, jobject throwable); - -JNIEXPORT void JNICALL -JVM_InitStackTraceElement(JNIEnv* env, jobject element, jobject stackFrameInfo); - -/* - * java.lang.StackWalker - */ -enum { - JVM_STACKWALK_FILL_CLASS_REFS_ONLY = 0x02, - JVM_STACKWALK_GET_CALLER_CLASS = 0x04, - JVM_STACKWALK_SHOW_HIDDEN_FRAMES = 0x20, - JVM_STACKWALK_FILL_LIVE_STACK_FRAMES = 0x100 -}; - -JNIEXPORT jobject JNICALL -JVM_CallStackWalk(JNIEnv *env, jobject stackStream, jlong mode, - jint skip_frames, jint frame_count, jint start_index, - jobjectArray frames); - -JNIEXPORT jint JNICALL -JVM_MoreStackWalk(JNIEnv *env, jobject stackStream, jlong mode, jlong anchor, - jint frame_count, jint start_index, - jobjectArray frames); - -/* - * java.lang.Thread - */ -JNIEXPORT void JNICALL -JVM_StartThread(JNIEnv *env, jobject thread); - -JNIEXPORT void JNICALL -JVM_StopThread(JNIEnv *env, jobject thread, jobject exception); - -JNIEXPORT jboolean JNICALL -JVM_IsThreadAlive(JNIEnv *env, jobject thread); - -JNIEXPORT void JNICALL -JVM_SuspendThread(JNIEnv *env, jobject thread); - -JNIEXPORT void JNICALL -JVM_ResumeThread(JNIEnv *env, jobject thread); - -JNIEXPORT void JNICALL -JVM_SetThreadPriority(JNIEnv *env, jobject thread, jint prio); - -JNIEXPORT void JNICALL -JVM_Yield(JNIEnv *env, jclass threadClass); - -JNIEXPORT void JNICALL -JVM_Sleep(JNIEnv *env, jclass threadClass, jlong millis); - -JNIEXPORT jobject JNICALL -JVM_CurrentThread(JNIEnv *env, jclass threadClass); - -JNIEXPORT jint JNICALL -JVM_CountStackFrames(JNIEnv *env, jobject thread); - -JNIEXPORT void JNICALL -JVM_Interrupt(JNIEnv *env, jobject thread); - -JNIEXPORT jboolean JNICALL -JVM_IsInterrupted(JNIEnv *env, jobject thread, jboolean clearInterrupted); - -JNIEXPORT jboolean JNICALL -JVM_HoldsLock(JNIEnv *env, jclass threadClass, jobject obj); - -JNIEXPORT void JNICALL -JVM_DumpAllStacks(JNIEnv *env, jclass unused); - -JNIEXPORT jobjectArray JNICALL -JVM_GetAllThreads(JNIEnv *env, jclass dummy); - -JNIEXPORT void JNICALL -JVM_SetNativeThreadName(JNIEnv *env, jobject jthread, jstring name); - -/* getStackTrace() and getAllStackTraces() method */ -JNIEXPORT jobjectArray JNICALL -JVM_DumpThreads(JNIEnv *env, jclass threadClass, jobjectArray threads); - -/* - * java.lang.SecurityManager - */ -JNIEXPORT jclass JNICALL -JVM_CurrentLoadedClass(JNIEnv *env); - -JNIEXPORT jobject JNICALL -JVM_CurrentClassLoader(JNIEnv *env); - -JNIEXPORT jobjectArray JNICALL -JVM_GetClassContext(JNIEnv *env); - -JNIEXPORT jint JNICALL -JVM_ClassDepth(JNIEnv *env, jstring name); - -JNIEXPORT jint JNICALL -JVM_ClassLoaderDepth(JNIEnv *env); - -/* - * java.lang.Package - */ -JNIEXPORT jstring JNICALL -JVM_GetSystemPackage(JNIEnv *env, jstring name); - -JNIEXPORT jobjectArray JNICALL -JVM_GetSystemPackages(JNIEnv *env); - -/* - * java.lang.ref.Reference - */ -JNIEXPORT jobject JNICALL -JVM_GetAndClearReferencePendingList(JNIEnv *env); - -JNIEXPORT jboolean JNICALL -JVM_HasReferencePendingList(JNIEnv *env); - -JNIEXPORT void JNICALL -JVM_WaitForReferencePendingList(JNIEnv *env); - -/* - * java.io.ObjectInputStream - */ -JNIEXPORT jobject JNICALL -JVM_LatestUserDefinedLoader(JNIEnv *env); - -/* - * java.lang.reflect.Array - */ -JNIEXPORT jint JNICALL -JVM_GetArrayLength(JNIEnv *env, jobject arr); - -JNIEXPORT jobject JNICALL -JVM_GetArrayElement(JNIEnv *env, jobject arr, jint index); - -JNIEXPORT jvalue JNICALL -JVM_GetPrimitiveArrayElement(JNIEnv *env, jobject arr, jint index, jint wCode); - -JNIEXPORT void JNICALL -JVM_SetArrayElement(JNIEnv *env, jobject arr, jint index, jobject val); - -JNIEXPORT void JNICALL -JVM_SetPrimitiveArrayElement(JNIEnv *env, jobject arr, jint index, jvalue v, - unsigned char vCode); - -JNIEXPORT jobject JNICALL -JVM_NewArray(JNIEnv *env, jclass eltClass, jint length); - -JNIEXPORT jobject JNICALL -JVM_NewMultiArray(JNIEnv *env, jclass eltClass, jintArray dim); - -/* - * java.lang.Class and java.lang.ClassLoader - */ - -#define JVM_CALLER_DEPTH -1 - -/* - * Returns the class in which the code invoking the native method - * belongs. - * - * Note that in JDK 1.1, native methods did not create a frame. - * In 1.2, they do. Therefore native methods like Class.forName - * can no longer look at the current frame for the caller class. - */ -JNIEXPORT jclass JNICALL -JVM_GetCallerClass(JNIEnv *env, int n); - -/* - * Find primitive classes - * utf: class name - */ -JNIEXPORT jclass JNICALL -JVM_FindPrimitiveClass(JNIEnv *env, const char *utf); - -/* - * Find a class from a boot class loader. Returns NULL if class not found. - */ -JNIEXPORT jclass JNICALL -JVM_FindClassFromBootLoader(JNIEnv *env, const char *name); - -/* - * Find a class from a given class loader. Throws ClassNotFoundException. - * name: name of class - * init: whether initialization is done - * loader: class loader to look up the class. This may not be the same as the caller's - * class loader. - * caller: initiating class. The initiating class may be null when a security - * manager is not installed. - */ -JNIEXPORT jclass JNICALL -JVM_FindClassFromCaller(JNIEnv *env, const char *name, jboolean init, - jobject loader, jclass caller); - -/* - * Find a class from a given class. - */ -JNIEXPORT jclass JNICALL -JVM_FindClassFromClass(JNIEnv *env, const char *name, jboolean init, - jclass from); - -/* Find a loaded class cached by the VM */ -JNIEXPORT jclass JNICALL -JVM_FindLoadedClass(JNIEnv *env, jobject loader, jstring name); - -/* Define a class */ -JNIEXPORT jclass JNICALL -JVM_DefineClass(JNIEnv *env, const char *name, jobject loader, const jbyte *buf, - jsize len, jobject pd); - -/* Define a class with a source (added in JDK1.5) */ -JNIEXPORT jclass JNICALL -JVM_DefineClassWithSource(JNIEnv *env, const char *name, jobject loader, - const jbyte *buf, jsize len, jobject pd, - const char *source); - -/* - * Module support funcions - */ - -/* - * Define a module with the specified packages and bind the module to the - * given class loader. - * module: module to define - * is_open: specifies if module is open (currently ignored) - * version: the module version - * location: the module location - * packages: list of packages in the module - * num_packages: number of packages in the module - */ -JNIEXPORT void JNICALL -JVM_DefineModule(JNIEnv *env, jobject module, jboolean is_open, jstring version, - jstring location, const char* const* packages, jsize num_packages); - -/* - * Set the boot loader's unnamed module. - * module: boot loader's unnamed module - */ -JNIEXPORT void JNICALL -JVM_SetBootLoaderUnnamedModule(JNIEnv *env, jobject module); - -/* - * Do a qualified export of a package. - * from_module: module containing the package to export - * package: name of the package to export - * to_module: module to export the package to - */ -JNIEXPORT void JNICALL -JVM_AddModuleExports(JNIEnv *env, jobject from_module, const char* package, jobject to_module); - -/* - * Do an export of a package to all unnamed modules. - * from_module: module containing the package to export - * package: name of the package to export to all unnamed modules - */ -JNIEXPORT void JNICALL -JVM_AddModuleExportsToAllUnnamed(JNIEnv *env, jobject from_module, const char* package); - -/* - * Do an unqualified export of a package. - * from_module: module containing the package to export - * package: name of the package to export - */ -JNIEXPORT void JNICALL -JVM_AddModuleExportsToAll(JNIEnv *env, jobject from_module, const char* package); - -/* - * Add a module to the list of modules that a given module can read. - * from_module: module requesting read access - * source_module: module that from_module wants to read - */ -JNIEXPORT void JNICALL -JVM_AddReadsModule(JNIEnv *env, jobject from_module, jobject source_module); - -/* - * Reflection support functions - */ - -JNIEXPORT jstring JNICALL -JVM_GetClassName(JNIEnv *env, jclass cls); - -JNIEXPORT jobjectArray JNICALL -JVM_GetClassInterfaces(JNIEnv *env, jclass cls); - -JNIEXPORT jboolean JNICALL -JVM_IsInterface(JNIEnv *env, jclass cls); - -JNIEXPORT jobjectArray JNICALL -JVM_GetClassSigners(JNIEnv *env, jclass cls); - -JNIEXPORT void JNICALL -JVM_SetClassSigners(JNIEnv *env, jclass cls, jobjectArray signers); - -JNIEXPORT jobject JNICALL -JVM_GetProtectionDomain(JNIEnv *env, jclass cls); - -JNIEXPORT jboolean JNICALL -JVM_IsArrayClass(JNIEnv *env, jclass cls); - -JNIEXPORT jboolean JNICALL -JVM_IsPrimitiveClass(JNIEnv *env, jclass cls); - -JNIEXPORT jint JNICALL -JVM_GetClassModifiers(JNIEnv *env, jclass cls); - -JNIEXPORT jobjectArray JNICALL -JVM_GetDeclaredClasses(JNIEnv *env, jclass ofClass); - -JNIEXPORT jclass JNICALL -JVM_GetDeclaringClass(JNIEnv *env, jclass ofClass); - -JNIEXPORT jstring JNICALL -JVM_GetSimpleBinaryName(JNIEnv *env, jclass ofClass); - -/* Generics support (JDK 1.5) */ -JNIEXPORT jstring JNICALL -JVM_GetClassSignature(JNIEnv *env, jclass cls); - -/* Annotations support (JDK 1.5) */ -JNIEXPORT jbyteArray JNICALL -JVM_GetClassAnnotations(JNIEnv *env, jclass cls); - -/* Annotations support (JDK 1.6) */ - -/* Type use annotations support (JDK 1.8) */ - -JNIEXPORT jbyteArray JNICALL -JVM_GetClassTypeAnnotations(JNIEnv *env, jclass cls); - -// field is a handle to a java.lang.reflect.Field object -JNIEXPORT jbyteArray JNICALL -JVM_GetFieldTypeAnnotations(JNIEnv *env, jobject field); - -// method is a handle to a java.lang.reflect.Method object -JNIEXPORT jbyteArray JNICALL -JVM_GetMethodTypeAnnotations(JNIEnv *env, jobject method); - -/* - * New (JDK 1.4) reflection implementation - */ - -JNIEXPORT jobjectArray JNICALL -JVM_GetClassDeclaredMethods(JNIEnv *env, jclass ofClass, jboolean publicOnly); - -JNIEXPORT jobjectArray JNICALL -JVM_GetClassDeclaredFields(JNIEnv *env, jclass ofClass, jboolean publicOnly); - -JNIEXPORT jobjectArray JNICALL -JVM_GetClassDeclaredConstructors(JNIEnv *env, jclass ofClass, jboolean publicOnly); - -/* Differs from JVM_GetClassModifiers in treatment of inner classes. - This returns the access flags for the class as specified in the - class file rather than searching the InnerClasses attribute (if - present) to find the source-level access flags. Only the values of - the low 13 bits (i.e., a mask of 0x1FFF) are guaranteed to be - valid. */ -JNIEXPORT jint JNICALL -JVM_GetClassAccessFlags(JNIEnv *env, jclass cls); - -/* - * Constant pool access; currently used to implement reflective access to annotations (JDK 1.5) - */ - -JNIEXPORT jobject JNICALL -JVM_GetClassConstantPool(JNIEnv *env, jclass cls); - -JNIEXPORT jint JNICALL JVM_ConstantPoolGetSize -(JNIEnv *env, jobject obj, jobject unused); - -JNIEXPORT jclass JNICALL JVM_ConstantPoolGetClassAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jclass JNICALL JVM_ConstantPoolGetClassAtIfLoaded -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jobject JNICALL JVM_ConstantPoolGetMethodAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jobject JNICALL JVM_ConstantPoolGetMethodAtIfLoaded -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jobject JNICALL JVM_ConstantPoolGetFieldAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jobject JNICALL JVM_ConstantPoolGetFieldAtIfLoaded -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jobjectArray JNICALL JVM_ConstantPoolGetMemberRefInfoAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jobjectArray JNICALL JVM_ConstantPoolGetNameAndTypeRefInfoAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jint JNICALL JVM_ConstantPoolGetNameAndTypeRefIndexAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jint JNICALL JVM_ConstantPoolGetClassRefIndexAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jint JNICALL JVM_ConstantPoolGetIntAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jlong JNICALL JVM_ConstantPoolGetLongAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jfloat JNICALL JVM_ConstantPoolGetFloatAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jdouble JNICALL JVM_ConstantPoolGetDoubleAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jstring JNICALL JVM_ConstantPoolGetStringAt -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jstring JNICALL JVM_ConstantPoolGetUTF8At -(JNIEnv *env, jobject obj, jobject unused, jint index); - -JNIEXPORT jbyte JNICALL JVM_ConstantPoolGetTagAt -(JNIEnv *env, jobject unused, jobject jcpool, jint index); - -/* - * java.security.* - */ - -JNIEXPORT jobject JNICALL -JVM_DoPrivileged(JNIEnv *env, jclass cls, - jobject action, jobject context, jboolean wrapException); - -JNIEXPORT jobject JNICALL -JVM_GetInheritedAccessControlContext(JNIEnv *env, jclass cls); - -JNIEXPORT jobject JNICALL -JVM_GetStackAccessControlContext(JNIEnv *env, jclass cls); - -/* - * Signal support, used to implement the shutdown sequence. Every VM must - * support JVM_SIGINT and JVM_SIGTERM, raising the former for user interrupts - * (^C) and the latter for external termination (kill, system shutdown, etc.). - * Other platform-dependent signal values may also be supported. - */ - -JNIEXPORT void * JNICALL -JVM_RegisterSignal(jint sig, void *handler); - -JNIEXPORT jboolean JNICALL -JVM_RaiseSignal(jint sig); - -JNIEXPORT jint JNICALL -JVM_FindSignal(const char *name); - -/* - * Retrieve the assertion directives for the specified class. - */ -JNIEXPORT jboolean JNICALL -JVM_DesiredAssertionStatus(JNIEnv *env, jclass unused, jclass cls); - -/* - * Retrieve the assertion directives from the VM. - */ -JNIEXPORT jobject JNICALL -JVM_AssertionStatusDirectives(JNIEnv *env, jclass unused); - -/* - * java.util.concurrent.atomic.AtomicLong - */ -JNIEXPORT jboolean JNICALL -JVM_SupportsCX8(void); - -/************************************************************************* - PART 2: Support for the Verifier and Class File Format Checker - ************************************************************************/ -/* - * Return the class name in UTF format. The result is valid - * until JVM_ReleaseUTf is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetClassNameUTF(JNIEnv *env, jclass cb); - -/* - * Returns the constant pool types in the buffer provided by "types." - */ -JNIEXPORT void JNICALL -JVM_GetClassCPTypes(JNIEnv *env, jclass cb, unsigned char *types); - -/* - * Returns the number of Constant Pool entries. - */ -JNIEXPORT jint JNICALL -JVM_GetClassCPEntriesCount(JNIEnv *env, jclass cb); - -/* - * Returns the number of *declared* fields or methods. - */ -JNIEXPORT jint JNICALL -JVM_GetClassFieldsCount(JNIEnv *env, jclass cb); - -JNIEXPORT jint JNICALL -JVM_GetClassMethodsCount(JNIEnv *env, jclass cb); - -/* - * Returns the CP indexes of exceptions raised by a given method. - * Places the result in the given buffer. - * - * The method is identified by method_index. - */ -JNIEXPORT void JNICALL -JVM_GetMethodIxExceptionIndexes(JNIEnv *env, jclass cb, jint method_index, - unsigned short *exceptions); -/* - * Returns the number of exceptions raised by a given method. - * The method is identified by method_index. - */ -JNIEXPORT jint JNICALL -JVM_GetMethodIxExceptionsCount(JNIEnv *env, jclass cb, jint method_index); - -/* - * Returns the byte code sequence of a given method. - * Places the result in the given buffer. - * - * The method is identified by method_index. - */ -JNIEXPORT void JNICALL -JVM_GetMethodIxByteCode(JNIEnv *env, jclass cb, jint method_index, - unsigned char *code); - -/* - * Returns the length of the byte code sequence of a given method. - * The method is identified by method_index. - */ -JNIEXPORT jint JNICALL -JVM_GetMethodIxByteCodeLength(JNIEnv *env, jclass cb, jint method_index); - -/* - * A structure used to a capture exception table entry in a Java method. - */ -typedef struct { - jint start_pc; - jint end_pc; - jint handler_pc; - jint catchType; -} JVM_ExceptionTableEntryType; - -/* - * Returns the exception table entry at entry_index of a given method. - * Places the result in the given buffer. - * - * The method is identified by method_index. - */ -JNIEXPORT void JNICALL -JVM_GetMethodIxExceptionTableEntry(JNIEnv *env, jclass cb, jint method_index, - jint entry_index, - JVM_ExceptionTableEntryType *entry); - -/* - * Returns the length of the exception table of a given method. - * The method is identified by method_index. - */ -JNIEXPORT jint JNICALL -JVM_GetMethodIxExceptionTableLength(JNIEnv *env, jclass cb, int index); - -/* - * Returns the modifiers of a given field. - * The field is identified by field_index. - */ -JNIEXPORT jint JNICALL -JVM_GetFieldIxModifiers(JNIEnv *env, jclass cb, int index); - -/* - * Returns the modifiers of a given method. - * The method is identified by method_index. - */ -JNIEXPORT jint JNICALL -JVM_GetMethodIxModifiers(JNIEnv *env, jclass cb, int index); - -/* - * Returns the number of local variables of a given method. - * The method is identified by method_index. - */ -JNIEXPORT jint JNICALL -JVM_GetMethodIxLocalsCount(JNIEnv *env, jclass cb, int index); - -/* - * Returns the number of arguments (including this pointer) of a given method. - * The method is identified by method_index. - */ -JNIEXPORT jint JNICALL -JVM_GetMethodIxArgsSize(JNIEnv *env, jclass cb, int index); - -/* - * Returns the maximum amount of stack (in words) used by a given method. - * The method is identified by method_index. - */ -JNIEXPORT jint JNICALL -JVM_GetMethodIxMaxStack(JNIEnv *env, jclass cb, int index); - -/* - * Is a given method a constructor. - * The method is identified by method_index. - */ -JNIEXPORT jboolean JNICALL -JVM_IsConstructorIx(JNIEnv *env, jclass cb, int index); - -/* - * Is the given method generated by the VM. - * The method is identified by method_index. - */ -JNIEXPORT jboolean JNICALL -JVM_IsVMGeneratedMethodIx(JNIEnv *env, jclass cb, int index); - -/* - * Returns the name of a given method in UTF format. - * The result remains valid until JVM_ReleaseUTF is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetMethodIxNameUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the signature of a given method in UTF format. - * The result remains valid until JVM_ReleaseUTF is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetMethodIxSignatureUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the name of the field refered to at a given constant pool - * index. - * - * The result is in UTF format and remains valid until JVM_ReleaseUTF - * is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetCPFieldNameUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the name of the method refered to at a given constant pool - * index. - * - * The result is in UTF format and remains valid until JVM_ReleaseUTF - * is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetCPMethodNameUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the signature of the method refered to at a given constant pool - * index. - * - * The result is in UTF format and remains valid until JVM_ReleaseUTF - * is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetCPMethodSignatureUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the signature of the field refered to at a given constant pool - * index. - * - * The result is in UTF format and remains valid until JVM_ReleaseUTF - * is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetCPFieldSignatureUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the class name refered to at a given constant pool index. - * - * The result is in UTF format and remains valid until JVM_ReleaseUTF - * is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetCPClassNameUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the class name refered to at a given constant pool index. - * - * The constant pool entry must refer to a CONSTANT_Fieldref. - * - * The result is in UTF format and remains valid until JVM_ReleaseUTF - * is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetCPFieldClassNameUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the class name refered to at a given constant pool index. - * - * The constant pool entry must refer to CONSTANT_Methodref or - * CONSTANT_InterfaceMethodref. - * - * The result is in UTF format and remains valid until JVM_ReleaseUTF - * is called. - * - * The caller must treat the string as a constant and not modify it - * in any way. - */ -JNIEXPORT const char * JNICALL -JVM_GetCPMethodClassNameUTF(JNIEnv *env, jclass cb, jint index); - -/* - * Returns the modifiers of a field in calledClass. The field is - * referred to in class cb at constant pool entry index. - * - * The caller must treat the string as a constant and not modify it - * in any way. - * - * Returns -1 if the field does not exist in calledClass. - */ -JNIEXPORT jint JNICALL -JVM_GetCPFieldModifiers(JNIEnv *env, jclass cb, int index, jclass calledClass); - -/* - * Returns the modifiers of a method in calledClass. The method is - * referred to in class cb at constant pool entry index. - * - * Returns -1 if the method does not exist in calledClass. - */ -JNIEXPORT jint JNICALL -JVM_GetCPMethodModifiers(JNIEnv *env, jclass cb, int index, jclass calledClass); - -/* - * Releases the UTF string obtained from the VM. - */ -JNIEXPORT void JNICALL -JVM_ReleaseUTF(const char *utf); - -/* - * Compare if two classes are in the same package. - */ -JNIEXPORT jboolean JNICALL -JVM_IsSameClassPackage(JNIEnv *env, jclass class1, jclass class2); - -/* Constants in class files */ - -#define JVM_ACC_PUBLIC 0x0001 /* visible to everyone */ -#define JVM_ACC_PRIVATE 0x0002 /* visible only to the defining class */ -#define JVM_ACC_PROTECTED 0x0004 /* visible to subclasses */ -#define JVM_ACC_STATIC 0x0008 /* instance variable is static */ -#define JVM_ACC_FINAL 0x0010 /* no further subclassing, overriding */ -#define JVM_ACC_SYNCHRONIZED 0x0020 /* wrap method call in monitor lock */ -#define JVM_ACC_SUPER 0x0020 /* funky handling of invokespecial */ -#define JVM_ACC_VOLATILE 0x0040 /* can not cache in registers */ -#define JVM_ACC_BRIDGE 0x0040 /* bridge method generated by compiler */ -#define JVM_ACC_TRANSIENT 0x0080 /* not persistent */ -#define JVM_ACC_VARARGS 0x0080 /* method declared with variable number of args */ -#define JVM_ACC_NATIVE 0x0100 /* implemented in C */ -#define JVM_ACC_INTERFACE 0x0200 /* class is an interface */ -#define JVM_ACC_ABSTRACT 0x0400 /* no definition provided */ -#define JVM_ACC_STRICT 0x0800 /* strict floating point */ -#define JVM_ACC_SYNTHETIC 0x1000 /* compiler-generated class, method or field */ -#define JVM_ACC_ANNOTATION 0x2000 /* annotation type */ -#define JVM_ACC_ENUM 0x4000 /* field is declared as element of enum */ -#define JVM_ACC_MODULE 0x8000 /* module-info class file */ - -#define JVM_ACC_PUBLIC_BIT 0 -#define JVM_ACC_PRIVATE_BIT 1 -#define JVM_ACC_PROTECTED_BIT 2 -#define JVM_ACC_STATIC_BIT 3 -#define JVM_ACC_FINAL_BIT 4 -#define JVM_ACC_SYNCHRONIZED_BIT 5 -#define JVM_ACC_SUPER_BIT 5 -#define JVM_ACC_VOLATILE_BIT 6 -#define JVM_ACC_BRIDGE_BIT 6 -#define JVM_ACC_TRANSIENT_BIT 7 -#define JVM_ACC_VARARGS_BIT 7 -#define JVM_ACC_NATIVE_BIT 8 -#define JVM_ACC_INTERFACE_BIT 9 -#define JVM_ACC_ABSTRACT_BIT 10 -#define JVM_ACC_STRICT_BIT 11 -#define JVM_ACC_SYNTHETIC_BIT 12 -#define JVM_ACC_ANNOTATION_BIT 13 -#define JVM_ACC_ENUM_BIT 14 - -// NOTE: replicated in SA in vm/agent/sun/jvm/hotspot/utilities/ConstantTag.java -enum { - JVM_CONSTANT_Utf8 = 1, - JVM_CONSTANT_Unicode, /* unused */ - JVM_CONSTANT_Integer, - JVM_CONSTANT_Float, - JVM_CONSTANT_Long, - JVM_CONSTANT_Double, - JVM_CONSTANT_Class, - JVM_CONSTANT_String, - JVM_CONSTANT_Fieldref, - JVM_CONSTANT_Methodref, - JVM_CONSTANT_InterfaceMethodref, - JVM_CONSTANT_NameAndType, - JVM_CONSTANT_MethodHandle = 15, // JSR 292 - JVM_CONSTANT_MethodType = 16, // JSR 292 - //JVM_CONSTANT_(unused) = 17, // JSR 292 early drafts only - JVM_CONSTANT_InvokeDynamic = 18, // JSR 292 - JVM_CONSTANT_ExternalMax = 18 // Last tag found in classfiles -}; - -/* JVM_CONSTANT_MethodHandle subtypes */ -enum { - JVM_REF_getField = 1, - JVM_REF_getStatic = 2, - JVM_REF_putField = 3, - JVM_REF_putStatic = 4, - JVM_REF_invokeVirtual = 5, - JVM_REF_invokeStatic = 6, - JVM_REF_invokeSpecial = 7, - JVM_REF_newInvokeSpecial = 8, - JVM_REF_invokeInterface = 9 -}; - -/* Used in the newarray instruction. */ - -#define JVM_T_BOOLEAN 4 -#define JVM_T_CHAR 5 -#define JVM_T_FLOAT 6 -#define JVM_T_DOUBLE 7 -#define JVM_T_BYTE 8 -#define JVM_T_SHORT 9 -#define JVM_T_INT 10 -#define JVM_T_LONG 11 - -/* JVM method signatures */ - -#define JVM_SIGNATURE_ARRAY '[' -#define JVM_SIGNATURE_BYTE 'B' -#define JVM_SIGNATURE_CHAR 'C' -#define JVM_SIGNATURE_CLASS 'L' -#define JVM_SIGNATURE_ENDCLASS ';' -#define JVM_SIGNATURE_ENUM 'E' -#define JVM_SIGNATURE_FLOAT 'F' -#define JVM_SIGNATURE_DOUBLE 'D' -#define JVM_SIGNATURE_FUNC '(' -#define JVM_SIGNATURE_ENDFUNC ')' -#define JVM_SIGNATURE_INT 'I' -#define JVM_SIGNATURE_LONG 'J' -#define JVM_SIGNATURE_SHORT 'S' -#define JVM_SIGNATURE_VOID 'V' -#define JVM_SIGNATURE_BOOLEAN 'Z' - -/* - * A function defined by the byte-code verifier and called by the VM. - * This is not a function implemented in the VM. - * - * Returns JNI_FALSE if verification fails. A detailed error message - * will be places in msg_buf, whose length is specified by buf_len. - */ -typedef jboolean (*verifier_fn_t)(JNIEnv *env, - jclass cb, - char * msg_buf, - jint buf_len); - - -/* - * Support for a VM-independent class format checker. - */ -typedef struct { - unsigned long code; /* byte code */ - unsigned long excs; /* exceptions */ - unsigned long etab; /* catch table */ - unsigned long lnum; /* line number */ - unsigned long lvar; /* local vars */ -} method_size_info; - -typedef struct { - unsigned int constants; /* constant pool */ - unsigned int fields; - unsigned int methods; - unsigned int interfaces; - unsigned int fields2; /* number of static 2-word fields */ - unsigned int innerclasses; /* # of records in InnerClasses attr */ - - method_size_info clinit; /* memory used in clinit */ - method_size_info main; /* used everywhere else */ -} class_size_info; - -/* - * Functions defined in libjava.so to perform string conversions. - * - */ - -typedef jstring (*to_java_string_fn_t)(JNIEnv *env, char *str); - -typedef char *(*to_c_string_fn_t)(JNIEnv *env, jstring s, jboolean *b); - -/* This is the function defined in libjava.so that performs class - * format checks. This functions fills in size information about - * the class file and returns: - * - * 0: good - * -1: out of memory - * -2: bad format - * -3: unsupported version - * -4: bad class name - */ - -typedef jint (*check_format_fn_t)(char *class_name, - unsigned char *data, - unsigned int data_size, - class_size_info *class_size, - char *message_buffer, - jint buffer_length, - jboolean measure_only, - jboolean check_relaxed); - -#define JVM_RECOGNIZED_CLASS_MODIFIERS (JVM_ACC_PUBLIC | \ - JVM_ACC_FINAL | \ - JVM_ACC_SUPER | \ - JVM_ACC_INTERFACE | \ - JVM_ACC_ABSTRACT | \ - JVM_ACC_ANNOTATION | \ - JVM_ACC_ENUM | \ - JVM_ACC_SYNTHETIC) - -#define JVM_RECOGNIZED_FIELD_MODIFIERS (JVM_ACC_PUBLIC | \ - JVM_ACC_PRIVATE | \ - JVM_ACC_PROTECTED | \ - JVM_ACC_STATIC | \ - JVM_ACC_FINAL | \ - JVM_ACC_VOLATILE | \ - JVM_ACC_TRANSIENT | \ - JVM_ACC_ENUM | \ - JVM_ACC_SYNTHETIC) - -#define JVM_RECOGNIZED_METHOD_MODIFIERS (JVM_ACC_PUBLIC | \ - JVM_ACC_PRIVATE | \ - JVM_ACC_PROTECTED | \ - JVM_ACC_STATIC | \ - JVM_ACC_FINAL | \ - JVM_ACC_SYNCHRONIZED | \ - JVM_ACC_BRIDGE | \ - JVM_ACC_VARARGS | \ - JVM_ACC_NATIVE | \ - JVM_ACC_ABSTRACT | \ - JVM_ACC_STRICT | \ - JVM_ACC_SYNTHETIC) - -/* - * This is the function defined in libjava.so to perform path - * canonicalization. VM call this function before opening jar files - * to load system classes. - * - */ - -typedef int (*canonicalize_fn_t)(JNIEnv *env, char *orig, char *out, int len); - -/************************************************************************* - PART 3: I/O and Network Support - ************************************************************************/ - -/* - * Convert a pathname into native format. This function does syntactic - * cleanup, such as removing redundant separator characters. It modifies - * the given pathname string in place. - */ -JNIEXPORT char * JNICALL -JVM_NativePath(char *); - -/* - * The standard printing functions supported by the Java VM. (Should they - * be renamed to JVM_* in the future? - */ - -/* jio_snprintf() and jio_vsnprintf() behave like snprintf(3) and vsnprintf(3), - * respectively, with the following differences: - * - The string written to str is always zero-terminated, also in case of - * truncation (count is too small to hold the result string), unless count - * is 0. In case of truncation count-1 characters are written and '\0' - * appendend. - * - If count is too small to hold the whole string, -1 is returned across - * all platforms. */ -JNIEXPORT int -jio_vsnprintf(char *str, size_t count, const char *fmt, va_list args); - -JNIEXPORT int -jio_snprintf(char *str, size_t count, const char *fmt, ...); - -JNIEXPORT int -jio_fprintf(FILE *, const char *fmt, ...); - -JNIEXPORT int -jio_vfprintf(FILE *, const char *fmt, va_list args); - - -JNIEXPORT void * JNICALL -JVM_RawMonitorCreate(void); - -JNIEXPORT void JNICALL -JVM_RawMonitorDestroy(void *mon); - -JNIEXPORT jint JNICALL -JVM_RawMonitorEnter(void *mon); - -JNIEXPORT void JNICALL -JVM_RawMonitorExit(void *mon); - -/* - * java.lang.reflect.Method - */ -JNIEXPORT jobject JNICALL -JVM_InvokeMethod(JNIEnv *env, jobject method, jobject obj, jobjectArray args0); - -/* - * java.lang.reflect.Constructor - */ -JNIEXPORT jobject JNICALL -JVM_NewInstanceFromConstructor(JNIEnv *env, jobject c, jobjectArray args0); - -/* - * java.lang.management support - */ -JNIEXPORT void* JNICALL -JVM_GetManagement(jint version); - -/* - * com.sun.tools.attach.VirtualMachine support - * - * Initialize the agent properties with the properties maintained in the VM. - */ -JNIEXPORT jobject JNICALL -JVM_InitAgentProperties(JNIEnv *env, jobject agent_props); - -JNIEXPORT jstring JNICALL -JVM_GetTemporaryDirectory(JNIEnv *env); - -/* Generics reflection support. - * - * Returns information about the given class's EnclosingMethod - * attribute, if present, or null if the class had no enclosing - * method. - * - * If non-null, the returned array contains three elements. Element 0 - * is the java.lang.Class of which the enclosing method is a member, - * and elements 1 and 2 are the java.lang.Strings for the enclosing - * method's name and descriptor, respectively. - */ -JNIEXPORT jobjectArray JNICALL -JVM_GetEnclosingMethodInfo(JNIEnv* env, jclass ofClass); - -/* ========================================================================= - * The following defines a private JVM interface that the JDK can query - * for the JVM version and capabilities. sun.misc.Version defines - * the methods for getting the VM version and its capabilities. - * - * When a new bit is added, the following should be updated to provide - * access to the new capability: - * HS: JVM_GetVersionInfo and Abstract_VM_Version class - * SDK: Version class - * - * Similary, a private JDK interface JDK_GetVersionInfo0 is defined for - * JVM to query for the JDK version and capabilities. - * - * When a new bit is added, the following should be updated to provide - * access to the new capability: - * HS: JDK_Version class - * SDK: JDK_GetVersionInfo0 - * - * ========================================================================== - */ -typedef struct { - unsigned int jvm_version; /* Encoded $VNUM as defined by JEP-223 */ - unsigned int patch_version : 8; /* JEP-223 patch version */ - unsigned int reserved3 : 8; - unsigned int reserved1 : 16; - unsigned int reserved2; - - /* The following bits represents JVM supports that JDK has dependency on. - * JDK can use these bits to determine which JVM version - * and support it has to maintain runtime compatibility. - * - * When a new bit is added in a minor or update release, make sure - * the new bit is also added in the main/baseline. - */ - unsigned int is_attachable : 1; - unsigned int : 31; - unsigned int : 32; - unsigned int : 32; -} jvm_version_info; - -#define JVM_VERSION_MAJOR(version) ((version & 0xFF000000) >> 24) -#define JVM_VERSION_MINOR(version) ((version & 0x00FF0000) >> 16) -#define JVM_VERSION_SECURITY(version) ((version & 0x0000FF00) >> 8) -#define JVM_VERSION_BUILD(version) ((version & 0x000000FF)) - -JNIEXPORT void JNICALL -JVM_GetVersionInfo(JNIEnv* env, jvm_version_info* info, size_t info_size); - -typedef struct { - unsigned int jdk_version; /* Encoded $VNUM as defined by JEP-223 */ - unsigned int patch_version : 8; /* JEP-223 patch version */ - unsigned int reserved3 : 8; - unsigned int reserved1 : 16; - unsigned int reserved2; - - /* The following bits represents new JDK supports that VM has dependency on. - * VM implementation can use these bits to determine which JDK version - * and support it has to maintain runtime compatibility. - * - * When a new bit is added in a minor or update release, make sure - * the new bit is also added in the main/baseline. - */ - unsigned int thread_park_blocker : 1; - unsigned int post_vm_init_hook_enabled : 1; - unsigned int pending_list_uses_discovered_field : 1; - unsigned int : 29; - unsigned int : 32; - unsigned int : 32; -} jdk_version_info; - -#define JDK_VERSION_MAJOR(version) ((version & 0xFF000000) >> 24) -#define JDK_VERSION_MINOR(version) ((version & 0x00FF0000) >> 16) -#define JDK_VERSION_SECURITY(version) ((version & 0x0000FF00) >> 8) -#define JDK_VERSION_BUILD(version) ((version & 0x000000FF)) - -/* - * This is the function JDK_GetVersionInfo0 defined in libjava.so - * that is dynamically looked up by JVM. - */ -typedef void (*jdk_version_info_fn_t)(jdk_version_info* info, size_t info_size); - -/* - * This structure is used by the launcher to get the default thread - * stack size from the VM using JNI_GetDefaultJavaVMInitArgs() with a - * version of 1.1. As it is not supported otherwise, it has been removed - * from jni.h - */ -typedef struct JDK1_1InitArgs { - jint version; - - char **properties; - jint checkSource; - jint nativeStackSize; - jint javaStackSize; - jint minHeapSize; - jint maxHeapSize; - jint verifyMode; - char *classpath; - - jint (JNICALL *vfprintf)(FILE *fp, const char *format, va_list args); - void (JNICALL *exit)(jint code); - void (JNICALL *abort)(void); - - jint enableClassGC; - jint enableVerboseGC; - jint disableAsyncGC; - jint verbose; - jboolean debugging; - jint debugPort; -} JDK1_1InitArgs; - -#ifdef __cplusplus -} /* extern "C" */ -#endif /* __cplusplus */ - -#endif /* !_JAVASOFT_JVM_H_ */ - -#endif // SHARE_VM_PRIMS_JVM_H diff --git a/src/hotspot/share/prims/jvmtiEnvBase.cpp b/src/hotspot/share/prims/jvmtiEnvBase.cpp index 47e7672f36a..2e8918c57e1 100644 --- a/src/hotspot/share/prims/jvmtiEnvBase.cpp +++ b/src/hotspot/share/prims/jvmtiEnvBase.cpp @@ -741,7 +741,7 @@ JvmtiEnvBase::get_owned_monitors(JavaThread *calling_thread, JavaThread* java_th // Save JNI local handles for any objects that this frame owns. jvmtiError JvmtiEnvBase::get_locked_objects_in_frame(JavaThread* calling_thread, JavaThread* java_thread, - javaVFrame *jvf, GrowableArray* owned_monitors_list, int stack_depth) { + javaVFrame *jvf, GrowableArray* owned_monitors_list, jint stack_depth) { jvmtiError err = JVMTI_ERROR_NONE; ResourceMark rm; diff --git a/src/hotspot/share/prims/jvmtiImpl.cpp b/src/hotspot/share/prims/jvmtiImpl.cpp index 8265b0b15a4..4869a653144 100644 --- a/src/hotspot/share/prims/jvmtiImpl.cpp +++ b/src/hotspot/share/prims/jvmtiImpl.cpp @@ -523,7 +523,7 @@ void JvmtiCurrentBreakpoints::gc_epilogue() { // // Constructor for non-object getter -VM_GetOrSetLocal::VM_GetOrSetLocal(JavaThread* thread, jint depth, int index, BasicType type) +VM_GetOrSetLocal::VM_GetOrSetLocal(JavaThread* thread, jint depth, jint index, BasicType type) : _thread(thread) , _calling_thread(NULL) , _depth(depth) @@ -536,7 +536,7 @@ VM_GetOrSetLocal::VM_GetOrSetLocal(JavaThread* thread, jint depth, int index, Ba } // Constructor for object or non-object setter -VM_GetOrSetLocal::VM_GetOrSetLocal(JavaThread* thread, jint depth, int index, BasicType type, jvalue value) +VM_GetOrSetLocal::VM_GetOrSetLocal(JavaThread* thread, jint depth, jint index, BasicType type, jvalue value) : _thread(thread) , _calling_thread(NULL) , _depth(depth) diff --git a/src/hotspot/share/prims/jvmtiTagMap.cpp b/src/hotspot/share/prims/jvmtiTagMap.cpp index 25807a2a626..3a45b2c8eff 100644 --- a/src/hotspot/share/prims/jvmtiTagMap.cpp +++ b/src/hotspot/share/prims/jvmtiTagMap.cpp @@ -2006,7 +2006,7 @@ inline bool CallbackInvoker::invoke_basic_stack_ref_callback(jvmtiHeapRootKind r jlong thread_tag, jint depth, jmethodID method, - jint slot, + int slot, oop obj) { assert(ServiceUtil::visible_oop(obj), "checking"); diff --git a/src/hotspot/share/prims/methodHandles.hpp b/src/hotspot/share/prims/methodHandles.hpp index a2b37460090..df83f23daf3 100644 --- a/src/hotspot/share/prims/methodHandles.hpp +++ b/src/hotspot/share/prims/methodHandles.hpp @@ -27,7 +27,6 @@ #include "classfile/javaClasses.hpp" #include "classfile/vmSymbols.hpp" -#include "prims/jvm.h" #include "runtime/frame.inline.hpp" #include "runtime/globals.hpp" #include "runtime/interfaceSupport.hpp" diff --git a/src/hotspot/share/prims/perf.cpp b/src/hotspot/share/prims/perf.cpp index afb26639e52..ba019e04cb2 100644 --- a/src/hotspot/share/prims/perf.cpp +++ b/src/hotspot/share/prims/perf.cpp @@ -24,11 +24,11 @@ #include "precompiled.hpp" #include "jni.h" +#include "jvm.h" #include "classfile/vmSymbols.hpp" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/interfaceSupport.hpp" #include "runtime/perfData.hpp" #include "runtime/perfMemory.hpp" diff --git a/src/hotspot/share/prims/unsafe.cpp b/src/hotspot/share/prims/unsafe.cpp index 63436feaa74..889bcb1317a 100644 --- a/src/hotspot/share/prims/unsafe.cpp +++ b/src/hotspot/share/prims/unsafe.cpp @@ -24,6 +24,7 @@ #include "precompiled.hpp" #include "jni.h" +#include "jvm.h" #include "classfile/classFileStream.hpp" #include "classfile/vmSymbols.hpp" #include "memory/allocation.inline.hpp" @@ -31,7 +32,6 @@ #include "oops/fieldStreams.hpp" #include "oops/objArrayOop.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/unsafe.hpp" #include "runtime/atomic.hpp" #include "runtime/globals.hpp" diff --git a/src/hotspot/share/runtime/arguments.cpp b/src/hotspot/share/runtime/arguments.cpp index a7ca6ce09bd..e3fe0d287d7 100644 --- a/src/hotspot/share/runtime/arguments.cpp +++ b/src/hotspot/share/runtime/arguments.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/javaAssertions.hpp" #include "classfile/moduleEntry.hpp" @@ -39,7 +40,6 @@ #include "memory/allocation.inline.hpp" #include "memory/universe.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/jvmtiExport.hpp" #include "runtime/arguments.hpp" #include "runtime/arguments_ext.hpp" diff --git a/src/hotspot/share/runtime/commandLineFlagRangeList.cpp b/src/hotspot/share/runtime/commandLineFlagRangeList.cpp index d96eb14b8db..9580a171a4b 100644 --- a/src/hotspot/share/runtime/commandLineFlagRangeList.cpp +++ b/src/hotspot/share/runtime/commandLineFlagRangeList.cpp @@ -23,10 +23,10 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/stringTable.hpp" #include "classfile/symbolTable.hpp" #include "gc/shared/referenceProcessor.hpp" -#include "prims/jvm.h" #include "oops/markOop.hpp" #include "runtime/arguments.hpp" #include "runtime/commandLineFlagConstraintList.hpp" diff --git a/src/hotspot/share/runtime/deoptimization.cpp b/src/hotspot/share/runtime/deoptimization.cpp index 9e14d5f166f..4197e87e723 100644 --- a/src/hotspot/share/runtime/deoptimization.cpp +++ b/src/hotspot/share/runtime/deoptimization.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/systemDictionary.hpp" #include "code/codeCache.hpp" #include "code/debugInfoRec.hpp" @@ -40,7 +41,6 @@ #include "oops/oop.inline.hpp" #include "oops/fieldStreams.hpp" #include "oops/verifyOopClosure.hpp" -#include "prims/jvm.h" #include "prims/jvmtiThreadState.hpp" #include "runtime/biasedLocking.hpp" #include "runtime/compilationPolicy.hpp" diff --git a/src/hotspot/share/runtime/globals.cpp b/src/hotspot/share/runtime/globals.cpp index 5ac9f55a880..cc87c59c77a 100644 --- a/src/hotspot/share/runtime/globals.cpp +++ b/src/hotspot/share/runtime/globals.cpp @@ -23,9 +23,9 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "memory/allocation.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/arguments.hpp" #include "runtime/globals.hpp" #include "runtime/globals_extension.hpp" diff --git a/src/hotspot/share/runtime/interfaceSupport.cpp b/src/hotspot/share/runtime/interfaceSupport.cpp index db7ad9ce905..a734460201b 100644 --- a/src/hotspot/share/runtime/interfaceSupport.cpp +++ b/src/hotspot/share/runtime/interfaceSupport.cpp @@ -47,7 +47,7 @@ long InterfaceSupport::_fullgc_alot_invocation = 0; Histogram* RuntimeHistogram; RuntimeHistogramElement::RuntimeHistogramElement(const char* elementName) { - static volatile jint RuntimeHistogram_lock = 0; + static volatile int RuntimeHistogram_lock = 0; _name = elementName; uintx count = 0; diff --git a/src/hotspot/share/runtime/java.cpp b/src/hotspot/share/runtime/java.cpp index 16957fd3919..11d84f00ee8 100644 --- a/src/hotspot/share/runtime/java.cpp +++ b/src/hotspot/share/runtime/java.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "aot/aotLoader.hpp" #include "classfile/classLoader.hpp" #include "classfile/stringTable.hpp" @@ -49,7 +50,6 @@ #include "oops/objArrayOop.hpp" #include "oops/oop.inline.hpp" #include "oops/symbol.hpp" -#include "prims/jvm.h" #include "prims/jvmtiExport.hpp" #include "runtime/arguments.hpp" #include "runtime/biasedLocking.hpp" diff --git a/src/hotspot/share/runtime/os.cpp b/src/hotspot/share/runtime/os.cpp index f3718b06f0c..0306faa5515 100644 --- a/src/hotspot/share/runtime/os.cpp +++ b/src/hotspot/share/runtime/os.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/javaClasses.hpp" #include "classfile/moduleEntry.hpp" @@ -41,7 +42,6 @@ #endif #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "prims/privilegedStack.hpp" #include "runtime/arguments.hpp" diff --git a/src/hotspot/share/runtime/os.hpp b/src/hotspot/share/runtime/os.hpp index b29c0c3d03a..9dd53398712 100644 --- a/src/hotspot/share/runtime/os.hpp +++ b/src/hotspot/share/runtime/os.hpp @@ -25,8 +25,8 @@ #ifndef SHARE_VM_RUNTIME_OS_HPP #define SHARE_VM_RUNTIME_OS_HPP +#include "jvm.h" #include "jvmtifiles/jvmti.h" -#include "prims/jvm.h" #include "runtime/extendedPC.hpp" #include "runtime/handles.hpp" #include "utilities/macros.hpp" diff --git a/src/hotspot/share/runtime/perfData.cpp b/src/hotspot/share/runtime/perfData.cpp index 719fd02643c..d79bbc66b48 100644 --- a/src/hotspot/share/runtime/perfData.cpp +++ b/src/hotspot/share/runtime/perfData.cpp @@ -23,10 +23,10 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/vmSymbols.hpp" #include "logging/log.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/handles.inline.hpp" #include "runtime/java.hpp" #include "runtime/mutex.hpp" @@ -420,11 +420,11 @@ PerfLongConstant* PerfDataManager::create_long_constant(CounterNS ns, PerfStringVariable* PerfDataManager::create_string_variable(CounterNS ns, const char* name, - jint max_length, + int max_length, const char* s, TRAPS) { - if (max_length == 0 && s != NULL) max_length = (jint)strlen(s); + if (max_length == 0 && s != NULL) max_length = (int)strlen(s); assert(max_length != 0, "PerfStringVariable with length 0"); diff --git a/src/hotspot/share/runtime/perfMemory.cpp b/src/hotspot/share/runtime/perfMemory.cpp index 58868d22706..fcae80cb217 100644 --- a/src/hotspot/share/runtime/perfMemory.cpp +++ b/src/hotspot/share/runtime/perfMemory.cpp @@ -23,9 +23,9 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/log.hpp" #include "memory/allocation.inline.hpp" -#include "prims/jvm.h" #include "runtime/arguments.hpp" #include "runtime/java.hpp" #include "runtime/mutex.hpp" @@ -51,7 +51,7 @@ char* PerfMemory::_start = NULL; char* PerfMemory::_end = NULL; char* PerfMemory::_top = NULL; size_t PerfMemory::_capacity = 0; -jint PerfMemory::_initialized = false; +int PerfMemory::_initialized = false; PerfDataPrologue* PerfMemory::_prologue = NULL; bool PerfMemory::_destroyed = false; diff --git a/src/hotspot/share/runtime/perfMemory.hpp b/src/hotspot/share/runtime/perfMemory.hpp index c0684ca9422..19d00ce1804 100644 --- a/src/hotspot/share/runtime/perfMemory.hpp +++ b/src/hotspot/share/runtime/perfMemory.hpp @@ -120,7 +120,7 @@ class PerfMemory : AllStatic { static char* _top; static size_t _capacity; static PerfDataPrologue* _prologue; - static jint _initialized; + static int _initialized; static bool _destroyed; static void create_memory_region(size_t sizep); diff --git a/src/hotspot/share/runtime/reflection.cpp b/src/hotspot/share/runtime/reflection.cpp index 7377596cb04..8fa6e9cb5ab 100644 --- a/src/hotspot/share/runtime/reflection.cpp +++ b/src/hotspot/share/runtime/reflection.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/javaClasses.hpp" #include "classfile/moduleEntry.hpp" #include "classfile/packageEntry.hpp" @@ -38,7 +39,6 @@ #include "oops/objArrayKlass.hpp" #include "oops/objArrayOop.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/jvmtiExport.hpp" #include "runtime/arguments.hpp" #include "runtime/handles.inline.hpp" diff --git a/src/hotspot/share/runtime/sharedRuntime.cpp b/src/hotspot/share/runtime/sharedRuntime.cpp index b08af7bd746..7cb09c201e4 100644 --- a/src/hotspot/share/runtime/sharedRuntime.cpp +++ b/src/hotspot/share/runtime/sharedRuntime.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "aot/aotLoader.hpp" #include "classfile/stringTable.hpp" #include "classfile/systemDictionary.hpp" @@ -45,7 +46,6 @@ #include "oops/objArrayKlass.hpp" #include "oops/oop.inline.hpp" #include "prims/forte.hpp" -#include "prims/jvm.h" #include "prims/jvmtiExport.hpp" #include "prims/methodHandles.hpp" #include "prims/nativeLookup.hpp" diff --git a/src/hotspot/share/runtime/thread.cpp b/src/hotspot/share/runtime/thread.cpp index 991dce7505f..9ffff8a0405 100644 --- a/src/hotspot/share/runtime/thread.cpp +++ b/src/hotspot/share/runtime/thread.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classLoader.hpp" #include "classfile/javaClasses.hpp" #include "classfile/moduleEntry.hpp" @@ -51,7 +52,6 @@ #include "oops/oop.inline.hpp" #include "oops/symbol.hpp" #include "oops/verifyOopClosure.hpp" -#include "prims/jvm.h" #include "prims/jvm_misc.hpp" #include "prims/jvmtiExport.hpp" #include "prims/jvmtiThreadState.hpp" @@ -762,7 +762,7 @@ bool Thread::is_interrupted(Thread* thread, bool clear_interrupted) { // GC Support bool Thread::claim_oops_do_par_case(int strong_roots_parity) { - jint thread_parity = _oops_do_parity; + int thread_parity = _oops_do_parity; if (thread_parity != strong_roots_parity) { jint res = Atomic::cmpxchg(strong_roots_parity, &_oops_do_parity, thread_parity); if (res == thread_parity) { diff --git a/src/hotspot/share/runtime/thread.hpp b/src/hotspot/share/runtime/thread.hpp index feeb0cf281a..c8c06da96d8 100644 --- a/src/hotspot/share/runtime/thread.hpp +++ b/src/hotspot/share/runtime/thread.hpp @@ -244,7 +244,7 @@ class Thread: public ThreadShadow { // The parity of the last strong_roots iteration in which this thread was // claimed as a task. - jint _oops_do_parity; + int _oops_do_parity; public: void set_last_handle_mark(HandleMark* mark) { _last_handle_mark = mark; } diff --git a/src/hotspot/share/runtime/vmStructs.cpp b/src/hotspot/share/runtime/vmStructs.cpp index 30d46168d7d..8d4c4c235bd 100644 --- a/src/hotspot/share/runtime/vmStructs.cpp +++ b/src/hotspot/share/runtime/vmStructs.cpp @@ -356,7 +356,7 @@ typedef PaddedEnd PaddedObjectMonitor; nonstatic_field(Symbol, _length, unsigned short) \ unchecked_nonstatic_field(Symbol, _body, sizeof(jbyte)) /* NOTE: no type */ \ nonstatic_field(Symbol, _body[0], jbyte) \ - nonstatic_field(TypeArrayKlass, _max_length, int) \ + nonstatic_field(TypeArrayKlass, _max_length, jint) \ \ /***********************/ \ /* Constant Pool Cache */ \ @@ -579,7 +579,7 @@ typedef PaddedEnd PaddedObjectMonitor; static_field(PerfMemory, _top, char*) \ static_field(PerfMemory, _capacity, size_t) \ static_field(PerfMemory, _prologue, PerfDataPrologue*) \ - static_field(PerfMemory, _initialized, jint) \ + static_field(PerfMemory, _initialized, int) \ \ /***************/ \ /* SymbolTable */ \ diff --git a/src/hotspot/share/services/diagnosticArgument.cpp b/src/hotspot/share/services/diagnosticArgument.cpp index 8c386ce202d..691bd797dd1 100644 --- a/src/hotspot/share/services/diagnosticArgument.cpp +++ b/src/hotspot/share/services/diagnosticArgument.cpp @@ -23,9 +23,9 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/thread.hpp" #include "services/diagnosticArgument.hpp" diff --git a/src/hotspot/share/services/diagnosticCommand.cpp b/src/hotspot/share/services/diagnosticCommand.cpp index 4d91e7da8a7..d2557f557c5 100644 --- a/src/hotspot/share/services/diagnosticCommand.cpp +++ b/src/hotspot/share/services/diagnosticCommand.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/classLoaderStats.hpp" #include "classfile/compactHashtable.hpp" #include "compiler/compileBroker.hpp" @@ -30,7 +31,6 @@ #include "gc/shared/vmGCOperations.hpp" #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/globals.hpp" #include "runtime/javaCalls.hpp" #include "runtime/os.hpp" diff --git a/src/hotspot/share/services/diagnosticFramework.cpp b/src/hotspot/share/services/diagnosticFramework.cpp index 3c9beebfb4b..0c4827ebdb7 100644 --- a/src/hotspot/share/services/diagnosticFramework.cpp +++ b/src/hotspot/share/services/diagnosticFramework.cpp @@ -23,10 +23,10 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "memory/oopFactory.hpp" #include "memory/resourceArea.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/javaCalls.hpp" #include "runtime/mutexLocker.hpp" #include "services/diagnosticArgument.hpp" diff --git a/src/hotspot/share/services/heapDumper.cpp b/src/hotspot/share/services/heapDumper.cpp index 3df6d0a8494..e931b2ec334 100644 --- a/src/hotspot/share/services/heapDumper.cpp +++ b/src/hotspot/share/services/heapDumper.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/symbolTable.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -34,7 +35,6 @@ #include "oops/objArrayKlass.hpp" #include "oops/objArrayOop.inline.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/javaCalls.hpp" #include "runtime/jniHandles.hpp" #include "runtime/os.hpp" diff --git a/src/hotspot/share/services/mallocSiteTable.cpp b/src/hotspot/share/services/mallocSiteTable.cpp index 472b461f491..4c6c7fe933d 100644 --- a/src/hotspot/share/services/mallocSiteTable.cpp +++ b/src/hotspot/share/services/mallocSiteTable.cpp @@ -235,8 +235,8 @@ bool MallocSiteTable::walk_malloc_site(MallocSiteWalker* walker) { void MallocSiteTable::AccessLock::exclusiveLock() { - jint target; - jint val; + int target; + int val; assert(_lock_state != ExclusiveLock, "Can only call once"); assert(*_lock >= 0, "Can not content exclusive lock"); diff --git a/src/hotspot/share/services/memTracker.cpp b/src/hotspot/share/services/memTracker.cpp index bc2f0ee71e5..c930da6a439 100644 --- a/src/hotspot/share/services/memTracker.cpp +++ b/src/hotspot/share/services/memTracker.cpp @@ -22,8 +22,8 @@ * */ #include "precompiled.hpp" +#include "jvm.h" -#include "prims/jvm.h" #include "runtime/mutex.hpp" #include "services/memBaseline.hpp" #include "services/memReporter.hpp" diff --git a/src/hotspot/share/utilities/accessFlags.hpp b/src/hotspot/share/utilities/accessFlags.hpp index f11eab322e4..b565330008c 100644 --- a/src/hotspot/share/utilities/accessFlags.hpp +++ b/src/hotspot/share/utilities/accessFlags.hpp @@ -25,8 +25,8 @@ #ifndef SHARE_VM_UTILITIES_ACCESSFLAGS_HPP #define SHARE_VM_UTILITIES_ACCESSFLAGS_HPP +#include "jvm.h" #include "memory/allocation.hpp" -#include "prims/jvm.h" #include "utilities/macros.hpp" // AccessFlags is an abstraction over Java access flags. diff --git a/src/hotspot/share/utilities/constantTag.hpp b/src/hotspot/share/utilities/constantTag.hpp index c083848c0bd..73cee23a513 100644 --- a/src/hotspot/share/utilities/constantTag.hpp +++ b/src/hotspot/share/utilities/constantTag.hpp @@ -25,8 +25,8 @@ #ifndef SHARE_VM_UTILITIES_CONSTANTTAG_HPP #define SHARE_VM_UTILITIES_CONSTANTTAG_HPP +#include "jvm.h" #include "memory/allocation.hpp" -#include "prims/jvm.h" // constant tags in Java .class files diff --git a/src/hotspot/share/utilities/debug.cpp b/src/hotspot/share/utilities/debug.cpp index ef4ddf0d0bd..ad19b5fd1ae 100644 --- a/src/hotspot/share/utilities/debug.cpp +++ b/src/hotspot/share/utilities/debug.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "classfile/systemDictionary.hpp" #include "code/codeCache.hpp" #include "code/icBuffer.hpp" @@ -36,7 +37,6 @@ #include "memory/resourceArea.hpp" #include "memory/universe.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "prims/privilegedStack.hpp" #include "runtime/arguments.hpp" #include "runtime/atomic.hpp" @@ -276,7 +276,7 @@ void report_untested(const char* file, int line, const char* message) { } void report_java_out_of_memory(const char* message) { - static jint out_of_memory_reported = 0; + static int out_of_memory_reported = 0; // A number of threads may attempt to report OutOfMemoryError at around the // same time. To avoid dumping the heap or executing the data collection diff --git a/src/hotspot/share/utilities/decoder.cpp b/src/hotspot/share/utilities/decoder.cpp index c1790b3325b..61a274c68fb 100644 --- a/src/hotspot/share/utilities/decoder.cpp +++ b/src/hotspot/share/utilities/decoder.cpp @@ -23,7 +23,7 @@ */ #include "precompiled.hpp" -#include "prims/jvm.h" +#include "jvm.h" #include "runtime/os.hpp" #include "utilities/decoder.hpp" #include "utilities/vmError.hpp" diff --git a/src/hotspot/share/utilities/elfStringTable.cpp b/src/hotspot/share/utilities/elfStringTable.cpp index c514e1dfbb0..9da7e238daa 100644 --- a/src/hotspot/share/utilities/elfStringTable.cpp +++ b/src/hotspot/share/utilities/elfStringTable.cpp @@ -26,8 +26,8 @@ #if !defined(_WINDOWS) && !defined(__APPLE__) +#include "jvm.h" #include "memory/allocation.inline.hpp" -#include "prims/jvm.h" #include "runtime/os.hpp" #include "utilities/elfStringTable.hpp" diff --git a/src/hotspot/share/utilities/formatBuffer.cpp b/src/hotspot/share/utilities/formatBuffer.cpp index 64209ddb5f4..b4d5b77d22d 100644 --- a/src/hotspot/share/utilities/formatBuffer.cpp +++ b/src/hotspot/share/utilities/formatBuffer.cpp @@ -23,8 +23,8 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "memory/allocation.hpp" -#include "prims/jvm.h" #include "utilities/formatBuffer.hpp" #include diff --git a/src/hotspot/share/utilities/formatBuffer.hpp b/src/hotspot/share/utilities/formatBuffer.hpp index c9e0d31419b..8726fda8d63 100644 --- a/src/hotspot/share/utilities/formatBuffer.hpp +++ b/src/hotspot/share/utilities/formatBuffer.hpp @@ -25,8 +25,8 @@ #ifndef SHARE_VM_UTILITIES_FORMATBUFFER_HPP #define SHARE_VM_UTILITIES_FORMATBUFFER_HPP +#include "jvm.h" #include "utilities/globalDefinitions.hpp" -#include "prims/jvm.h" #include // Simple class to format the ctor arguments into a fixed-sized buffer. diff --git a/src/hotspot/share/utilities/globalDefinitions_xlc.hpp b/src/hotspot/share/utilities/globalDefinitions_xlc.hpp index 26ec729c137..d22256b44d5 100644 --- a/src/hotspot/share/utilities/globalDefinitions_xlc.hpp +++ b/src/hotspot/share/utilities/globalDefinitions_xlc.hpp @@ -86,6 +86,7 @@ // In this case you need to copy the following defines to a position after #include // (see jmv_aix.h). #ifdef AIX + #include #ifdef _LP64 #undef NULL #define NULL 0L diff --git a/src/hotspot/share/utilities/macros.hpp b/src/hotspot/share/utilities/macros.hpp index bf9ba54ef99..eb04db48dfa 100644 --- a/src/hotspot/share/utilities/macros.hpp +++ b/src/hotspot/share/utilities/macros.hpp @@ -322,7 +322,9 @@ #endif #if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__APPLE__) +#ifndef BSD #define BSD +#endif // BSD defined in #define BSD_ONLY(code) code #define NOT_BSD(code) #else diff --git a/src/hotspot/share/utilities/ostream.cpp b/src/hotspot/share/utilities/ostream.cpp index 4fd752198fe..531c2c9bafd 100644 --- a/src/hotspot/share/utilities/ostream.cpp +++ b/src/hotspot/share/utilities/ostream.cpp @@ -23,9 +23,9 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "compiler/compileLog.hpp" #include "oops/oop.inline.hpp" -#include "prims/jvm.h" #include "runtime/arguments.hpp" #include "runtime/os.hpp" #include "runtime/vm_version.hpp" diff --git a/src/hotspot/share/utilities/vmError.cpp b/src/hotspot/share/utilities/vmError.cpp index bea1dbef648..580fc86e89e 100644 --- a/src/hotspot/share/utilities/vmError.cpp +++ b/src/hotspot/share/utilities/vmError.cpp @@ -23,12 +23,12 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "code/codeCache.hpp" #include "compiler/compileBroker.hpp" #include "compiler/disassembler.hpp" #include "gc/shared/collectedHeap.hpp" #include "logging/logConfiguration.hpp" -#include "prims/jvm.h" #include "prims/whitebox.hpp" #include "runtime/arguments.hpp" #include "runtime/atomic.hpp" diff --git a/src/java.base/macosx/native/include/jni_md.h b/src/java.base/macosx/native/include/jni_md.h deleted file mode 100644 index 9e47bedd61a..00000000000 --- a/src/java.base/macosx/native/include/jni_md.h +++ /dev/null @@ -1,42 +0,0 @@ -/* - * Copyright (c) 1996, 2013, 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. - */ - -#ifndef _JAVASOFT_JNI_MD_H_ -#define _JAVASOFT_JNI_MD_H_ - -#define JNIEXPORT __attribute__((visibility("default"))) -#define JNIIMPORT __attribute__((visibility("default"))) -#define JNICALL - -typedef int jint; -#ifdef _LP64 /* 64-bit */ -typedef long jlong; -#else -typedef long long jlong; -#endif - -typedef signed char jbyte; - -#endif /* !_JAVASOFT_JNI_MD_H_ */ diff --git a/src/java.base/macosx/native/include/jvm_md.h b/src/java.base/macosx/native/include/jvm_md.h deleted file mode 100644 index b139bd298c0..00000000000 --- a/src/java.base/macosx/native/include/jvm_md.h +++ /dev/null @@ -1,80 +0,0 @@ -/* - * Copyright (c) 1997, 2013, 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. - */ - -#ifndef _JAVASOFT_JVM_MD_H_ -#define _JAVASOFT_JVM_MD_H_ - -/* - * This file is currently collecting system-specific dregs for the - * JNI conversion, which should be sorted out later. - */ - -#include /* For DIR */ -#include /* For MAXPATHLEN */ -#include /* For F_OK, R_OK, W_OK */ -#include /* For ptrdiff_t */ -#include /* For uintptr_t */ - -#define JNI_ONLOAD_SYMBOLS {"JNI_OnLoad"} -#define JNI_ONUNLOAD_SYMBOLS {"JNI_OnUnload"} - -#define JNI_LIB_PREFIX "lib" -#define JNI_LIB_SUFFIX ".dylib" -#define VERSIONED_JNI_LIB_NAME(NAME, VERSION) JNI_LIB_PREFIX NAME "." VERSION JNI_LIB_SUFFIX -#define JNI_LIB_NAME(NAME) JNI_LIB_PREFIX NAME JNI_LIB_SUFFIX - -#define JVM_MAXPATHLEN MAXPATHLEN - -#define JVM_R_OK R_OK -#define JVM_W_OK W_OK -#define JVM_X_OK X_OK -#define JVM_F_OK F_OK - -/* - * File I/O - */ - -#include -#include -#include -#include -#include - -/* O Flags */ - -#define JVM_O_RDONLY O_RDONLY -#define JVM_O_WRONLY O_WRONLY -#define JVM_O_RDWR O_RDWR -#define JVM_O_O_APPEND O_APPEND -#define JVM_O_EXCL O_EXCL -#define JVM_O_CREAT O_CREAT - -/* Signals */ - -#define JVM_SIGINT SIGINT -#define JVM_SIGTERM SIGTERM - - -#endif /* !_JAVASOFT_JVM_MD_H_ */ diff --git a/src/java.base/share/native/include/classfile_constants.h b/src/java.base/share/native/include/classfile_constants.h index 94b97f75f2e..e6490a74cda 100644 --- a/src/java.base/share/native/include/classfile_constants.h +++ b/src/java.base/share/native/include/classfile_constants.h @@ -54,9 +54,29 @@ enum { JVM_ACC_STRICT = 0x0800, JVM_ACC_SYNTHETIC = 0x1000, JVM_ACC_ANNOTATION = 0x2000, - JVM_ACC_ENUM = 0x4000 + JVM_ACC_ENUM = 0x4000, + JVM_ACC_MODULE = 0x8000 }; +#define JVM_ACC_PUBLIC_BIT 0 +#define JVM_ACC_PRIVATE_BIT 1 +#define JVM_ACC_PROTECTED_BIT 2 +#define JVM_ACC_STATIC_BIT 3 +#define JVM_ACC_FINAL_BIT 4 +#define JVM_ACC_SYNCHRONIZED_BIT 5 +#define JVM_ACC_SUPER_BIT 5 +#define JVM_ACC_VOLATILE_BIT 6 +#define JVM_ACC_BRIDGE_BIT 6 +#define JVM_ACC_TRANSIENT_BIT 7 +#define JVM_ACC_VARARGS_BIT 7 +#define JVM_ACC_NATIVE_BIT 8 +#define JVM_ACC_INTERFACE_BIT 9 +#define JVM_ACC_ABSTRACT_BIT 10 +#define JVM_ACC_STRICT_BIT 11 +#define JVM_ACC_SYNTHETIC_BIT 12 +#define JVM_ACC_ANNOTATION_BIT 13 +#define JVM_ACC_ENUM_BIT 14 + /* Used in newarray instruction. */ enum { @@ -86,8 +106,9 @@ enum { JVM_CONSTANT_InterfaceMethodref = 11, JVM_CONSTANT_NameAndType = 12, JVM_CONSTANT_MethodHandle = 15, // JSR 292 - JVM_CONSTANT_MethodType = 16, // JSR 292 - JVM_CONSTANT_InvokeDynamic = 18 + JVM_CONSTANT_MethodType = 16, // JSR 292 + JVM_CONSTANT_InvokeDynamic = 18, + JVM_CONSTANT_ExternalMax = 18 }; /* JVM_CONSTANT_MethodHandle subtypes */ diff --git a/src/java.base/share/native/include/jvm.h b/src/java.base/share/native/include/jvm.h index de90fd99dae..109d1d1dbdf 100644 --- a/src/java.base/share/native/include/jvm.h +++ b/src/java.base/share/native/include/jvm.h @@ -1155,20 +1155,25 @@ JVM_NativePath(char *); * be renamed to JVM_* in the future? */ -/* - * BE CAREFUL! The following functions do not implement the - * full feature set of standard C printf formats. - */ -int +/* jio_snprintf() and jio_vsnprintf() behave like snprintf(3) and vsnprintf(3), + * respectively, with the following differences: + * - The string written to str is always zero-terminated, also in case of + * truncation (count is too small to hold the result string), unless count + * is 0. In case of truncation count-1 characters are written and '\0' + * appendend. + * - If count is too small to hold the whole string, -1 is returned across + * all platforms. */ + +JNIEXPORT int jio_vsnprintf(char *str, size_t count, const char *fmt, va_list args); -int +JNIEXPORT int jio_snprintf(char *str, size_t count, const char *fmt, ...); -int +JNIEXPORT int jio_fprintf(FILE *, const char *fmt, ...); -int +JNIEXPORT int jio_vfprintf(FILE *, const char *fmt, va_list args); diff --git a/src/java.base/solaris/native/libjsig/jsig.c b/src/java.base/solaris/native/libjsig/jsig.c index 3c509159a2d..2e98f0363ee 100644 --- a/src/java.base/solaris/native/libjsig/jsig.c +++ b/src/java.base/solaris/native/libjsig/jsig.c @@ -37,7 +37,7 @@ #include #include #include -#include "jvm_solaris.h" +#include "jvm_md.h" #define bool int #define true 1 diff --git a/src/java.base/unix/native/include/jni_md.h b/src/java.base/unix/native/include/jni_md.h index 80eedf33577..6e352038fc5 100644 --- a/src/java.base/unix/native/include/jni_md.h +++ b/src/java.base/unix/native/include/jni_md.h @@ -30,8 +30,13 @@ #define __has_attribute(x) 0 #endif #if (defined(__GNUC__) && ((__GNUC__ > 4) || (__GNUC__ == 4) && (__GNUC_MINOR__ > 2))) || __has_attribute(visibility) - #define JNIEXPORT __attribute__((visibility("default"))) - #define JNIIMPORT __attribute__((visibility("default"))) + #ifdef ARM + #define JNIEXPORT __attribute__((externally_visible,visibility("default"))) + #define JNIIMPORT __attribute__((externally_visible,visibility("default"))) + #else + #define JNIEXPORT __attribute__((visibility("default"))) + #define JNIIMPORT __attribute__((visibility("default"))) + #endif #else #define JNIEXPORT #define JNIIMPORT @@ -40,7 +45,7 @@ #define JNICALL typedef int jint; -#ifdef _LP64 /* 64-bit Solaris */ +#ifdef _LP64 typedef long jlong; #else typedef long long jlong; diff --git a/src/java.base/unix/native/include/jvm_md.h b/src/java.base/unix/native/include/jvm_md.h index ad7e9d62077..13d778fba8f 100644 --- a/src/java.base/unix/native/include/jvm_md.h +++ b/src/java.base/unix/native/include/jvm_md.h @@ -39,6 +39,10 @@ #define JNI_ONLOAD_SYMBOLS {"JNI_OnLoad"} #define JNI_ONUNLOAD_SYMBOLS {"JNI_OnUnload"} +#define JVM_ONLOAD_SYMBOLS {"JVM_OnLoad"} +#define AGENT_ONLOAD_SYMBOLS {"Agent_OnLoad"} +#define AGENT_ONUNLOAD_SYMBOLS {"Agent_OnUnload"} +#define AGENT_ONATTACH_SYMBOLS {"Agent_OnAttach"} #define JNI_LIB_PREFIX "lib" #ifdef __APPLE__ @@ -50,7 +54,15 @@ #endif #define JNI_LIB_NAME(NAME) JNI_LIB_PREFIX NAME JNI_LIB_SUFFIX +#if defined(AIX) || defined(SOLARIS) #define JVM_MAXPATHLEN MAXPATHLEN +#else +// Hack: MAXPATHLEN is 4095 on some Linux and 4096 on others. This may +// cause problems if JVM and the rest of JDK are built on different +// Linux releases. Here we define JVM_MAXPATHLEN to be MAXPATHLEN + 1, +// so buffers declared in VM are always >= 4096. +#define JVM_MAXPATHLEN MAXPATHLEN + 1 +#endif #define JVM_R_OK R_OK #define JVM_W_OK W_OK @@ -67,19 +79,22 @@ #include #include -/* O Flags */ - -#define JVM_O_RDONLY O_RDONLY -#define JVM_O_WRONLY O_WRONLY -#define JVM_O_RDWR O_RDWR -#define JVM_O_O_APPEND O_APPEND -#define JVM_O_EXCL O_EXCL -#define JVM_O_CREAT O_CREAT - /* Signals */ +#include // for socklen_t + #define JVM_SIGINT SIGINT #define JVM_SIGTERM SIGTERM +#define BREAK_SIGNAL SIGQUIT /* Thread dumping support. */ +#ifdef SOLARIS +#define ASYNC_SIGNAL SIGJVM2 /* Event-based suspend/resume support */ +#endif // SOLARIS +#define SHUTDOWN1_SIGNAL SIGHUP /* Shutdown Hooks support. */ +#define SHUTDOWN2_SIGNAL SIGINT +#define SHUTDOWN3_SIGNAL SIGTERM + +/* With 1.4.1 libjsig added versioning: used in os_solaris.cpp and jsig.c */ +#define JSIG_VERSION_1_4_1 0x30140100 #endif /* !_JAVASOFT_JVM_MD_H_ */ diff --git a/src/java.base/unix/native/libjava/UnixFileSystem_md.c b/src/java.base/unix/native/libjava/UnixFileSystem_md.c index e3eae8bac17..315aa92b642 100644 --- a/src/java.base/unix/native/libjava/UnixFileSystem_md.c +++ b/src/java.base/unix/native/libjava/UnixFileSystem_md.c @@ -38,11 +38,13 @@ #include #include #include +#include +#include +#include #include "jni.h" #include "jni_util.h" #include "jlong.h" -#include "jvm.h" #include "io_util.h" #include "io_util_md.h" #include "java_io_FileSystem.h" @@ -97,9 +99,9 @@ Java_java_io_UnixFileSystem_canonicalize0(JNIEnv *env, jobject this, jstring rv = NULL; WITH_PLATFORM_STRING(env, pathname, path) { - char canonicalPath[JVM_MAXPATHLEN]; + char canonicalPath[PATH_MAX]; if (canonicalize((char *)path, - canonicalPath, JVM_MAXPATHLEN) < 0) { + canonicalPath, PATH_MAX) < 0) { JNU_ThrowIOExceptionWithLastError(env, "Bad pathname"); } else { #ifdef MACOSX diff --git a/src/java.base/windows/native/include/jni_md.h b/src/java.base/windows/native/include/jni_md.h index b8dce62c1c7..6c8d6b9e77e 100644 --- a/src/java.base/windows/native/include/jni_md.h +++ b/src/java.base/windows/native/include/jni_md.h @@ -30,6 +30,7 @@ #define JNIIMPORT __declspec(dllimport) #define JNICALL __stdcall +// 'long' is always 32 bit on windows so this matches what jdk expects typedef long jint; typedef __int64 jlong; typedef signed char jbyte; diff --git a/src/java.base/windows/native/include/jvm_md.h b/src/java.base/windows/native/include/jvm_md.h index e9feba32415..e047137c783 100644 --- a/src/java.base/windows/native/include/jvm_md.h +++ b/src/java.base/windows/native/include/jvm_md.h @@ -31,13 +31,20 @@ * JNI conversion, which should be sorted out later. */ +#include #include #include #include "jni.h" +typedef int socklen_t; + #define JNI_ONLOAD_SYMBOLS {"_JNI_OnLoad@8", "JNI_OnLoad"} #define JNI_ONUNLOAD_SYMBOLS {"_JNI_OnUnload@8", "JNI_OnUnload"} +#define JVM_ONLOAD_SYMBOLS {"_JVM_OnLoad@12", "JVM_OnLoad"} +#define AGENT_ONLOAD_SYMBOLS {"_Agent_OnLoad@12", "Agent_OnLoad"} +#define AGENT_ONUNLOAD_SYMBOLS {"_Agent_OnUnload@4", "Agent_OnUnload"} +#define AGENT_ONATTACH_SYMBOLS {"_Agent_OnAttach@12", "Agent_OnAttach"} #define JNI_LIB_PREFIX "" #define JNI_LIB_SUFFIX ".dll" @@ -63,21 +70,16 @@ typedef struct { #define JVM_X_OK 1 #define JVM_F_OK 0 +#ifdef __cplusplus +extern "C" { +#endif + JNIEXPORT void * JNICALL JVM_GetThreadInterruptEvent(); -/* - * These routines are only reentrant on Windows - */ - -JNIEXPORT struct protoent * JNICALL -JVM_GetProtoByName(char* name); - -JNIEXPORT struct hostent* JNICALL -JVM_GetHostByAddr(const char* name, int len, int type); - -JNIEXPORT struct hostent* JNICALL -JVM_GetHostByName(char* name); +#ifdef __cplusplus +} /* extern "C" */ +#endif /* __cplusplus */ /* * File I/O @@ -89,19 +91,12 @@ JVM_GetHostByName(char* name); #include #include -/* O Flags */ - -#define JVM_O_RDONLY O_RDONLY -#define JVM_O_WRONLY O_WRONLY -#define JVM_O_RDWR O_RDWR -#define JVM_O_O_APPEND O_APPEND -#define JVM_O_EXCL O_EXCL -#define JVM_O_CREAT O_CREAT - /* Signals */ #define JVM_SIGINT SIGINT #define JVM_SIGTERM SIGTERM +#define SHUTDOWN1_SIGNAL SIGINT /* Shutdown Hooks support. */ +#define SHUTDOWN2_SIGNAL SIGTERM #endif /* !_JAVASOFT_JVM_MD_H_ */ diff --git a/test/hotspot/gtest/classfile/test_AltHashing.cpp b/test/hotspot/gtest/classfile/test_AltHashing.cpp index 4718bdd97b2..fe65754e297 100644 --- a/test/hotspot/gtest/classfile/test_AltHashing.cpp +++ b/test/hotspot/gtest/classfile/test_AltHashing.cpp @@ -51,7 +51,7 @@ static const juint MURMUR3_32_X86_CHECK_VALUE = 0xB0F57EE3; class AltHashingTest : public ::testing::Test { public: - static juint murmur3_32(const int* data, int len) { + static juint murmur3_32(const jint* data, int len) { return AltHashing::murmur3_32(data, len); } }; diff --git a/test/hotspot/gtest/logging/logTestFixture.cpp b/test/hotspot/gtest/logging/logTestFixture.cpp index 985e1a443bd..3dda7c4464c 100644 --- a/test/hotspot/gtest/logging/logTestFixture.cpp +++ b/test/hotspot/gtest/logging/logTestFixture.cpp @@ -22,11 +22,11 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "logTestFixture.hpp" #include "logTestUtils.inline.hpp" #include "logging/logConfiguration.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "unittest.hpp" #include "utilities/ostream.hpp" diff --git a/test/hotspot/gtest/logging/test_log.cpp b/test/hotspot/gtest/logging/test_log.cpp index 2b1f8041fa8..0700e202b70 100644 --- a/test/hotspot/gtest/logging/test_log.cpp +++ b/test/hotspot/gtest/logging/test_log.cpp @@ -22,11 +22,11 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "gc/shared/gcTraceTime.inline.hpp" #include "logTestFixture.hpp" #include "logTestUtils.inline.hpp" #include "logging/log.hpp" -#include "prims/jvm.h" #include "unittest.hpp" class LogTest : public LogTestFixture { diff --git a/test/hotspot/gtest/logging/test_logConfiguration.cpp b/test/hotspot/gtest/logging/test_logConfiguration.cpp index e894fb7ed18..87c316a6579 100644 --- a/test/hotspot/gtest/logging/test_logConfiguration.cpp +++ b/test/hotspot/gtest/logging/test_logConfiguration.cpp @@ -22,6 +22,7 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "logTestFixture.hpp" #include "logTestUtils.inline.hpp" #include "logging/logConfiguration.hpp" @@ -31,7 +32,6 @@ #include "logging/logTag.hpp" #include "logging/logTagSet.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "unittest.hpp" #include "utilities/ostream.hpp" diff --git a/test/hotspot/gtest/logging/test_logDecorators.cpp b/test/hotspot/gtest/logging/test_logDecorators.cpp index fcb8bfeb56c..09fa5eacccd 100644 --- a/test/hotspot/gtest/logging/test_logDecorators.cpp +++ b/test/hotspot/gtest/logging/test_logDecorators.cpp @@ -22,8 +22,8 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/logDecorators.hpp" -#include "prims/jvm.h" #include "unittest.hpp" static LogDecorators::Decorator decorator_array[] = { diff --git a/test/hotspot/gtest/logging/test_logFileOutput.cpp b/test/hotspot/gtest/logging/test_logFileOutput.cpp index 8a2e6ccea78..24cdf83c15d 100644 --- a/test/hotspot/gtest/logging/test_logFileOutput.cpp +++ b/test/hotspot/gtest/logging/test_logFileOutput.cpp @@ -22,10 +22,10 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "logTestUtils.inline.hpp" #include "logging/logFileOutput.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "runtime/os.hpp" #include "unittest.hpp" #include "utilities/globalDefinitions.hpp" diff --git a/test/hotspot/gtest/logging/test_logMessageTest.cpp b/test/hotspot/gtest/logging/test_logMessageTest.cpp index 171bf4d8275..b3c736f34fa 100644 --- a/test/hotspot/gtest/logging/test_logMessageTest.cpp +++ b/test/hotspot/gtest/logging/test_logMessageTest.cpp @@ -22,11 +22,11 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "logTestFixture.hpp" #include "logTestUtils.inline.hpp" #include "logging/log.hpp" #include "logging/logMessage.hpp" -#include "prims/jvm.h" #include "unittest.hpp" #include "utilities/globalDefinitions.hpp" diff --git a/test/hotspot/gtest/logging/test_logTagLevelExpression.cpp b/test/hotspot/gtest/logging/test_logTagLevelExpression.cpp index 5322d58dec5..560597c482f 100644 --- a/test/hotspot/gtest/logging/test_logTagLevelExpression.cpp +++ b/test/hotspot/gtest/logging/test_logTagLevelExpression.cpp @@ -22,10 +22,10 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "logging/logLevel.hpp" #include "logging/logTagLevelExpression.hpp" #include "logging/logTagSet.hpp" -#include "prims/jvm.h" #include "unittest.hpp" #include "utilities/globalDefinitions.hpp" diff --git a/test/hotspot/gtest/logging/test_logTagSetDescriptions.cpp b/test/hotspot/gtest/logging/test_logTagSetDescriptions.cpp index 268841d8c37..540e8661702 100644 --- a/test/hotspot/gtest/logging/test_logTagSetDescriptions.cpp +++ b/test/hotspot/gtest/logging/test_logTagSetDescriptions.cpp @@ -22,12 +22,12 @@ * */ #include "precompiled.hpp" +#include "jvm.h" #include "logTestUtils.inline.hpp" #include "logging/logConfiguration.hpp" #include "logging/logTagSet.hpp" #include "logging/logTagSetDescriptions.hpp" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "unittest.hpp" #include "utilities/ostream.hpp" diff --git a/test/hotspot/gtest/runtime/test_arguments.cpp b/test/hotspot/gtest/runtime/test_arguments.cpp index 06292623a9c..b6121ba3aba 100644 --- a/test/hotspot/gtest/runtime/test_arguments.cpp +++ b/test/hotspot/gtest/runtime/test_arguments.cpp @@ -22,9 +22,9 @@ */ #include "precompiled.hpp" -#include "prims/jvm.h" -#include "runtime/arguments.hpp" +#include "jvm.h" #include "unittest.hpp" +#include "runtime/arguments.hpp" #include "utilities/align.hpp" #include "utilities/globalDefinitions.hpp" diff --git a/test/hotspot/gtest/utilities/test_json.cpp b/test/hotspot/gtest/utilities/test_json.cpp index e55fed6afc0..7113988adf9 100644 --- a/test/hotspot/gtest/utilities/test_json.cpp +++ b/test/hotspot/gtest/utilities/test_json.cpp @@ -22,8 +22,8 @@ */ #include "precompiled.hpp" +#include "jvm.h" #include "memory/resourceArea.hpp" -#include "prims/jvm.h" #include "utilities/json.hpp" #include "unittest.hpp" From 92254a72a4eca3e0cba753b06e4fb38ed4947678 Mon Sep 17 00:00:00 2001 From: Igor Ignatyev Date: Tue, 31 Oct 2017 16:31:39 -0700 Subject: [PATCH 10/61] 8186618: [TESTBUG] Test applications/ctw/Modules.java doesn't have timeout and hang on windows Reviewed-by: jwilhelm, epavlova --- test/hotspot/jtreg/ProblemList.txt | 3 + .../applications/ctw/modules/generate.bash | 72 +++++++++++++++++++ .../ctw/modules/java_activation.java | 38 ++++++++++ .../applications/ctw/modules/java_base.java | 38 ++++++++++ .../ctw/modules/java_compiler.java | 38 ++++++++++ .../applications/ctw/modules/java_corba.java | 38 ++++++++++ .../ctw/modules/java_datatransfer.java | 38 ++++++++++ .../ctw/modules/java_desktop.java | 38 ++++++++++ .../ctw/modules/java_instrument.java | 38 ++++++++++ .../ctw/modules/java_logging.java | 38 ++++++++++ .../ctw/modules/java_management.java | 38 ++++++++++ .../ctw/modules/java_management_rmi.java | 38 ++++++++++ .../applications/ctw/modules/java_naming.java | 38 ++++++++++ .../applications/ctw/modules/java_prefs.java | 38 ++++++++++ .../applications/ctw/modules/java_rmi.java | 38 ++++++++++ .../ctw/modules/java_scripting.java | 38 ++++++++++ .../ctw/modules/java_security_jgss.java | 38 ++++++++++ .../ctw/modules/java_security_sasl.java | 38 ++++++++++ .../ctw/modules/java_smartcardio.java | 38 ++++++++++ .../applications/ctw/modules/java_sql.java | 38 ++++++++++ .../ctw/modules/java_sql_rowset.java | 38 ++++++++++ .../ctw/modules/java_transaction.java | 38 ++++++++++ .../applications/ctw/modules/java_xml.java | 38 ++++++++++ .../ctw/modules/java_xml_bind.java | 38 ++++++++++ .../ctw/modules/java_xml_crypto.java | 38 ++++++++++ .../applications/ctw/modules/java_xml_ws.java | 38 ++++++++++ .../ctw/modules/java_xml_ws_annotation.java | 38 ++++++++++ .../applications/ctw/modules/javafx_base.java | 38 ++++++++++ .../ctw/modules/javafx_controls.java | 38 ++++++++++ .../applications/ctw/modules/javafx_fxml.java | 38 ++++++++++ .../ctw/modules/javafx_graphics.java | 38 ++++++++++ .../ctw/modules/javafx_media.java | 38 ++++++++++ .../ctw/modules/javafx_swing.java | 38 ++++++++++ .../applications/ctw/modules/javafx_web.java | 38 ++++++++++ .../ctw/modules/jdk_accessibility.java | 38 ++++++++++ .../{Modules.java => modules/jdk_aot.java} | 6 +- .../applications/ctw/modules/jdk_attach.java | 38 ++++++++++ .../ctw/modules/jdk_charsets.java | 38 ++++++++++ .../ctw/modules/jdk_compiler.java | 38 ++++++++++ .../ctw/modules/jdk_crypto_cryptoki.java | 38 ++++++++++ .../ctw/modules/jdk_crypto_ec.java | 38 ++++++++++ .../ctw/modules/jdk_crypto_mscapi.java | 38 ++++++++++ .../ctw/modules/jdk_crypto_ucrypto.java | 38 ++++++++++ .../ctw/modules/jdk_dynalink.java | 38 ++++++++++ .../applications/ctw/modules/jdk_editpad.java | 38 ++++++++++ .../ctw/modules/jdk_hotspot_agent.java | 38 ++++++++++ .../ctw/modules/jdk_httpserver.java | 38 ++++++++++ .../ctw/modules/jdk_incubator_httpclient.java | 38 ++++++++++ .../ctw/modules/jdk_internal_ed.java | 38 ++++++++++ .../ctw/modules/jdk_internal_jvmstat.java | 38 ++++++++++ .../ctw/modules/jdk_internal_le.java | 38 ++++++++++ .../ctw/modules/jdk_internal_opt.java | 38 ++++++++++ .../ctw/modules/jdk_internal_vm_ci.java | 38 ++++++++++ .../ctw/modules/jdk_internal_vm_compiler.java | 38 ++++++++++ .../jdk_internal_vm_compiler_management.java | 38 ++++++++++ .../applications/ctw/modules/jdk_jartool.java | 38 ++++++++++ .../applications/ctw/modules/jdk_javadoc.java | 38 ++++++++++ .../applications/ctw/modules/jdk_jcmd.java | 38 ++++++++++ .../ctw/modules/jdk_jconsole.java | 38 ++++++++++ .../applications/ctw/modules/jdk_jdeps.java | 38 ++++++++++ .../applications/ctw/modules/jdk_jdi.java | 38 ++++++++++ .../applications/ctw/modules/jdk_jlink.java | 38 ++++++++++ .../applications/ctw/modules/jdk_jshell.java | 38 ++++++++++ .../ctw/modules/jdk_jsobject.java | 38 ++++++++++ .../applications/ctw/modules/jdk_jstatd.java | 38 ++++++++++ .../ctw/modules/jdk_localedata.java | 38 ++++++++++ .../ctw/modules/jdk_management.java | 38 ++++++++++ .../ctw/modules/jdk_management_agent.java | 38 ++++++++++ .../ctw/modules/jdk_naming_dns.java | 38 ++++++++++ .../ctw/modules/jdk_naming_rmi.java | 38 ++++++++++ .../applications/ctw/modules/jdk_net.java | 38 ++++++++++ .../ctw/modules/jdk_packager.java | 38 ++++++++++ .../ctw/modules/jdk_packager_services.java | 38 ++++++++++ .../applications/ctw/modules/jdk_rmic.java | 38 ++++++++++ .../ctw/modules/jdk_scripting_nashorn.java | 38 ++++++++++ .../modules/jdk_scripting_nashorn_shell.java | 38 ++++++++++ .../applications/ctw/modules/jdk_sctp.java | 38 ++++++++++ .../ctw/modules/jdk_security_auth.java | 38 ++++++++++ .../ctw/modules/jdk_security_jgss.java | 38 ++++++++++ .../ctw/modules/jdk_unsupported.java | 38 ++++++++++ .../ctw/modules/jdk_xml_bind.java | 38 ++++++++++ .../applications/ctw/modules/jdk_xml_dom.java | 38 ++++++++++ .../applications/ctw/modules/jdk_xml_ws.java | 38 ++++++++++ .../applications/ctw/modules/jdk_zipfs.java | 38 ++++++++++ .../hotspot/tools/ctw/ClassPathDirEntry.java | 4 +- .../tools/ctw/ClassPathJarInDirEntry.java | 1 - .../tools/ctw/ClassPathJimageEntry.java | 3 +- .../hotspot/tools/ctw/CompileTheWorld.java | 26 ++++--- .../src/sun/hotspot/tools/ctw/Compiler.java | 8 +-- .../src/sun/hotspot/tools/ctw/CtwRunner.java | 24 ++++--- .../sun/hotspot/tools/ctw/PathHandler.java | 23 ++++-- .../ctw/src/sun/hotspot/tools/ctw/Utils.java | 7 +- 92 files changed, 3215 insertions(+), 40 deletions(-) create mode 100644 test/hotspot/jtreg/applications/ctw/modules/generate.bash create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_activation.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_base.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_compiler.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_corba.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_datatransfer.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_desktop.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_instrument.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_logging.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_management.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_management_rmi.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_naming.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_prefs.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_rmi.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_scripting.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_security_jgss.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_security_sasl.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_smartcardio.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_sql.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_sql_rowset.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_transaction.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_xml.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_xml_bind.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_xml_crypto.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_xml_ws.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/java_xml_ws_annotation.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/javafx_base.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/javafx_controls.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/javafx_fxml.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/javafx_graphics.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/javafx_media.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/javafx_swing.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/javafx_web.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_accessibility.java rename test/hotspot/jtreg/applications/ctw/{Modules.java => modules/jdk_aot.java} (90%) create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_attach.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_charsets.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_compiler.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_crypto_cryptoki.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_crypto_ec.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_crypto_mscapi.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_crypto_ucrypto.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_dynalink.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_editpad.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_hotspot_agent.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_httpserver.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_incubator_httpclient.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_internal_ed.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_internal_jvmstat.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_internal_le.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_internal_opt.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_internal_vm_ci.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_internal_vm_compiler.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_internal_vm_compiler_management.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jartool.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_javadoc.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jcmd.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jconsole.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jdeps.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jdi.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jlink.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jshell.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jsobject.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_jstatd.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_localedata.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_management.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_management_agent.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_naming_dns.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_naming_rmi.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_net.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_packager.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_packager_services.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_rmic.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_scripting_nashorn.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_scripting_nashorn_shell.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_sctp.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_security_auth.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_security_jgss.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_unsupported.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_xml_bind.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_xml_dom.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_xml_ws.java create mode 100644 test/hotspot/jtreg/applications/ctw/modules/jdk_zipfs.java diff --git a/test/hotspot/jtreg/ProblemList.txt b/test/hotspot/jtreg/ProblemList.txt index 1b118f407b2..ecb170526dc 100644 --- a/test/hotspot/jtreg/ProblemList.txt +++ b/test/hotspot/jtreg/ProblemList.txt @@ -54,6 +54,9 @@ compiler/types/correctness/OffTest.java 8066173 generic-all # aot test intermittently failing in jprt 8175791 compiler/aot/DeoptimizationTest.java 8175791 windows-all +applications/ctw/modules/java_desktop.java 8189604 windows-all +applications/ctw/modules/jdk_jconsole.java 8189604 windows-all + ############################################################################# # :hotspot_gc diff --git a/test/hotspot/jtreg/applications/ctw/modules/generate.bash b/test/hotspot/jtreg/applications/ctw/modules/generate.bash new file mode 100644 index 00000000000..8c7a0c5a316 --- /dev/null +++ b/test/hotspot/jtreg/applications/ctw/modules/generate.bash @@ -0,0 +1,72 @@ +#!/bin/bash +# +# Copyright (c) 2017, 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. +# + +# generates CTW tests for modules passed as argument + +for module in $@ +do + file=${module//./_}.java + echo creating $file for $module... + cat > $file < { @@ -87,10 +91,10 @@ public class CompileTheWorld { } finally { await(executor); } - CompileTheWorld.OUT.printf("Done (%d classes, %d methods, %d ms)%n", + CompileTheWorld.OUT.println(String.format("Done (%d classes, %d methods, %d ms)", PathHandler.getProcessedClassCount(), Compiler.getMethodCount(), - System.currentTimeMillis() - start); + System.currentTimeMillis() - start)); passed = true; } catch (Throwable t){ t.printStackTrace(ERR); diff --git a/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/Compiler.java b/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/Compiler.java index da6ab6aeb31..f5091c7df02 100644 --- a/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/Compiler.java +++ b/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/Compiler.java @@ -100,8 +100,8 @@ public class Compiler { } } } catch (Throwable t) { - CompileTheWorld.OUT.printf("[%d]\t%s\tWARNING preloading failed : %s%n", - id, className, t); + CompileTheWorld.OUT.println(String.format("[%d]\t%s\tWARNING preloading failed : %s", + id, className, t)); t.printStackTrace(CompileTheWorld.ERR); } } @@ -113,8 +113,8 @@ public class Compiler { try { WHITE_BOX.enqueueInitializerForCompilation(aClass, i); } catch (Throwable t) { - CompileTheWorld.OUT.printf("[%d]\t%s::\tERROR at level %d : %s%n", - id, aClass.getName(), i, t); + CompileTheWorld.OUT.println(String.format("[%d]\t%s::\tERROR at level %d : %s", + id, aClass.getName(), i, t)); t.printStackTrace(CompileTheWorld.ERR); } } diff --git a/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/CtwRunner.java b/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/CtwRunner.java index 57d626e0ed2..453c3c0c612 100644 --- a/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/CtwRunner.java +++ b/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/CtwRunner.java @@ -35,8 +35,6 @@ import java.nio.file.Path; import java.nio.file.Paths; import java.util.ArrayList; import java.util.List; -import java.util.concurrent.Executor; -import java.util.concurrent.ExecutorService; import java.util.concurrent.TimeUnit; import java.util.function.Predicate; import java.util.regex.Pattern; @@ -62,19 +60,27 @@ public class CtwRunner { } private final List errors; + private final String target; private final Path targetPath; private final String targetName; private CtwRunner(String target) { - if (target.equals("modules")) { + if (target.startsWith("modules")) { targetPath = Paths .get(Utils.TEST_JDK) .resolve("lib") - .resolve(target); + .resolve("modules"); + if (target.equals("modules")){ + target = targetPath.toString(); + } + targetName = target.replace(':', '_') + .replace('.', '_') + .replace(',', '_'); } else { targetPath = Paths.get(target).toAbsolutePath(); + targetName = targetPath.getFileName().toString(); } - targetName = targetPath.getFileName().toString(); + this.target = target; errors = new ArrayList<>(); } @@ -104,7 +110,7 @@ public class CtwRunner { long classStart = 0L; long classCount = classCount(); Asserts.assertGreaterThan(classCount, 0L, - targetPath + " does not have any classes"); + target + "(at " + targetPath + ") does not have any classes"); boolean done = false; while (!done) { String[] cmd = cmd(classStart); @@ -138,7 +144,7 @@ public class CtwRunner { + "] != classCount[" + classCount + "]")); } else { System.out.println("Executed CTW for all " + classCount - + " classes in " + targetPath); + + " classes in " + target + "(at " + targetPath + ")"); } done = true; } else { @@ -162,7 +168,7 @@ public class CtwRunner { } private long classCount() { - List phs = PathHandler.create(targetPath.toString()); + List phs = PathHandler.create(target); long result = phs.stream() .mapToLong(PathHandler::classCount) .sum(); @@ -215,7 +221,7 @@ public class CtwRunner { "-XX:CompileCommand=exclude,java/lang/invoke/MethodHandle.*", // CTW entry point CompileTheWorld.class.getName(), - targetPath.toString(), + target, }; } diff --git a/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/PathHandler.java b/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/PathHandler.java index 4ee3ec4e240..5ab94b0a356 100644 --- a/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/PathHandler.java +++ b/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/PathHandler.java @@ -24,9 +24,12 @@ package sun.hotspot.tools.ctw; import java.io.Closeable; +import java.net.URI; +import java.nio.file.FileSystems; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; +import java.util.Arrays; import java.util.Collections; import java.util.List; import java.util.Objects; @@ -35,6 +38,7 @@ import java.util.concurrent.atomic.AtomicLong; import java.util.function.Function; import java.util.regex.Matcher; import java.util.regex.Pattern; +import java.util.stream.Collectors; import java.util.stream.Stream; /** @@ -121,6 +125,15 @@ public class PathHandler implements Closeable { path = matcher.group(1); path = path.isEmpty() ? "." : path; return ClassPathJarInDirEntry.create(Paths.get(path)); + } else if (path.startsWith("modules:")) { + Path modules = FileSystems.getFileSystem(URI.create("jrt:/")) + .getPath("modules"); + return Arrays.stream(path.substring("modules:".length()) + .split(",")) + .map(modules::resolve) + .map(ClassPathDirEntry::new) + .map(PathHandler::new) + .collect(Collectors.toList()); } else { path = path.isEmpty() ? "." : path; Path p = Paths.get(path); @@ -180,7 +193,9 @@ public class PathHandler implements Closeable { */ public final void process(Executor executor) { CompileTheWorld.OUT.println(entry.description()); - entry.classes().forEach(s -> processClass(s, executor)); + entry.classes() + .distinct() + .forEach(s -> processClass(s, executor)); } /** @@ -209,12 +224,12 @@ public class PathHandler implements Closeable { Class aClass; Thread.currentThread().setContextClassLoader(entry.loader()); try { - CompileTheWorld.OUT.printf("[%d]\t%s%n", id, name); + CompileTheWorld.OUT.println(String.format("[%d]\t%s", id, name)); aClass = entry.loader().loadClass(name); Compiler.compileClass(aClass, id, executor); } catch (Throwable e) { - CompileTheWorld.OUT.printf("[%d]\t%s\tWARNING skipped: %s%n", - id, name, e); + CompileTheWorld.OUT.println(String.format("[%d]\t%s\tWARNING skipped: %s", + id, name, e)); e.printStackTrace(CompileTheWorld.ERR); } } diff --git a/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/Utils.java b/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/Utils.java index f9a6b7e03d5..ff44618647c 100644 --- a/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/Utils.java +++ b/test/hotspot/jtreg/testlibrary/ctw/src/sun/hotspot/tools/ctw/Utils.java @@ -193,10 +193,11 @@ public class Utils { * @param filename tested filename */ public static boolean isClassFile(String filename) { - // If the filename has a period after removing '.class', it's not valid class file return endsWithIgnoreCase(filename, CLASSFILE_EXT) - && (filename.indexOf('.') - == (filename.length() - CLASSFILE_EXT.length())); + // skip all module-info.class files + && !(filename.substring(filename.lastIndexOf('/') + 1, + filename.lastIndexOf('.')) + .equals("module-info")); } /** From ca70374a3d99252c326ccccae4b36733a070e169 Mon Sep 17 00:00:00 2001 From: Dmitry Chuyko Date: Wed, 1 Nov 2017 16:48:12 +0300 Subject: [PATCH 11/61] 8189745: AARCH64: Use CRC32C intrinsic code in interpreter and C1 Reviewed-by: aph --- .../cpu/aarch64/c1_LIRGenerator_aarch64.cpp | 76 ++++++++++++++++++- .../cpu/aarch64/stubGenerator_aarch64.cpp | 8 +- .../templateInterpreterGenerator_aarch64.cpp | 38 +++++++++- src/hotspot/share/c1/c1_Compiler.cpp | 2 +- 4 files changed, 117 insertions(+), 7 deletions(-) diff --git a/src/hotspot/cpu/aarch64/c1_LIRGenerator_aarch64.cpp b/src/hotspot/cpu/aarch64/c1_LIRGenerator_aarch64.cpp index 1a612b9bac2..c7c4d2dad99 100644 --- a/src/hotspot/cpu/aarch64/c1_LIRGenerator_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/c1_LIRGenerator_aarch64.cpp @@ -1030,7 +1030,81 @@ void LIRGenerator::do_update_CRC32(Intrinsic* x) { } void LIRGenerator::do_update_CRC32C(Intrinsic* x) { - Unimplemented(); + assert(UseCRC32CIntrinsics, "why are we here?"); + // Make all state_for calls early since they can emit code + LIR_Opr result = rlock_result(x); + int flags = 0; + switch (x->id()) { + case vmIntrinsics::_updateBytesCRC32C: + case vmIntrinsics::_updateDirectByteBufferCRC32C: { + bool is_updateBytes = (x->id() == vmIntrinsics::_updateBytesCRC32C); + int offset = is_updateBytes ? arrayOopDesc::base_offset_in_bytes(T_BYTE) : 0; + + LIRItem crc(x->argument_at(0), this); + LIRItem buf(x->argument_at(1), this); + LIRItem off(x->argument_at(2), this); + LIRItem end(x->argument_at(3), this); + + buf.load_item(); + off.load_nonconstant(); + end.load_nonconstant(); + + // len = end - off + LIR_Opr len = end.result(); + LIR_Opr tmpA = new_register(T_INT); + LIR_Opr tmpB = new_register(T_INT); + __ move(end.result(), tmpA); + __ move(off.result(), tmpB); + __ sub(tmpA, tmpB, tmpA); + len = tmpA; + + LIR_Opr index = off.result(); + if(off.result()->is_constant()) { + index = LIR_OprFact::illegalOpr; + offset += off.result()->as_jint(); + } + LIR_Opr base_op = buf.result(); + + if (index->is_valid()) { + LIR_Opr tmp = new_register(T_LONG); + __ convert(Bytecodes::_i2l, index, tmp); + index = tmp; + } + + if (offset) { + LIR_Opr tmp = new_pointer_register(); + __ add(base_op, LIR_OprFact::intConst(offset), tmp); + base_op = tmp; + offset = 0; + } + + LIR_Address* a = new LIR_Address(base_op, + index, + offset, + T_BYTE); + BasicTypeList signature(3); + signature.append(T_INT); + signature.append(T_ADDRESS); + signature.append(T_INT); + CallingConvention* cc = frame_map()->c_calling_convention(&signature); + const LIR_Opr result_reg = result_register_for(x->type()); + + LIR_Opr addr = new_pointer_register(); + __ leal(LIR_OprFact::address(a), addr); + + crc.load_item_force(cc->at(0)); + __ move(addr, cc->at(1)); + __ move(len, cc->at(2)); + + __ call_runtime_leaf(StubRoutines::updateBytesCRC32C(), getThreadTemp(), result_reg, cc->args()); + __ move(result_reg, result); + + break; + } + default: { + ShouldNotReachHere(); + } + } } void LIRGenerator::do_FmaIntrinsic(Intrinsic* x) { diff --git a/src/hotspot/cpu/aarch64/stubGenerator_aarch64.cpp b/src/hotspot/cpu/aarch64/stubGenerator_aarch64.cpp index d9c2cdc4a52..a2515e5774f 100644 --- a/src/hotspot/cpu/aarch64/stubGenerator_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/stubGenerator_aarch64.cpp @@ -4937,6 +4937,10 @@ class StubGenerator: public StubCodeGenerator { StubRoutines::_crc_table_adr = (address)StubRoutines::aarch64::_crc_table; StubRoutines::_updateBytesCRC32 = generate_updateBytesCRC32(); } + + if (UseCRC32CIntrinsics) { + StubRoutines::_updateBytesCRC32C = generate_updateBytesCRC32C(); + } } void generate_all() { @@ -5014,10 +5018,6 @@ class StubGenerator: public StubCodeGenerator { StubRoutines::_sha256_implCompressMB = generate_sha256_implCompress(true, "sha256_implCompressMB"); } - if (UseCRC32CIntrinsics) { - StubRoutines::_updateBytesCRC32C = generate_updateBytesCRC32C(); - } - // generate Adler32 intrinsics code if (UseAdler32Intrinsics) { StubRoutines::_updateBytesAdler32 = generate_updateBytesAdler32(); diff --git a/src/hotspot/cpu/aarch64/templateInterpreterGenerator_aarch64.cpp b/src/hotspot/cpu/aarch64/templateInterpreterGenerator_aarch64.cpp index f22f8d874aa..242a0715e8b 100644 --- a/src/hotspot/cpu/aarch64/templateInterpreterGenerator_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/templateInterpreterGenerator_aarch64.cpp @@ -1061,8 +1061,44 @@ address TemplateInterpreterGenerator::generate_CRC32_updateBytes_entry(AbstractI return NULL; } -// Not supported +/** + * Method entry for intrinsic-candidate (non-native) methods: + * int java.util.zip.CRC32C.updateBytes(int crc, byte[] b, int off, int end) + * int java.util.zip.CRC32C.updateDirectByteBuffer(int crc, long buf, int off, int end) + * Unlike CRC32, CRC32C does not have any methods marked as native + * CRC32C also uses an "end" variable instead of the length variable CRC32 uses + */ address TemplateInterpreterGenerator::generate_CRC32C_updateBytes_entry(AbstractInterpreter::MethodKind kind) { + if (UseCRC32Intrinsics) { + address entry = __ pc(); + + // Prepare jump to stub using parameters from the stack + const Register crc = c_rarg0; // initial crc + const Register buf = c_rarg1; // source java byte array address + const Register len = c_rarg2; // len argument to the kernel + + const Register end = len; // index of last element to process + const Register off = crc; // offset + + __ ldrw(end, Address(esp)); // int end + __ ldrw(off, Address(esp, wordSize)); // int offset + __ sub(len, end, off); + __ ldr(buf, Address(esp, 2*wordSize)); // byte[] buf | long buf + __ add(buf, buf, off); // + offset + if (kind == Interpreter::java_util_zip_CRC32C_updateDirectByteBuffer) { + __ ldrw(crc, Address(esp, 4*wordSize)); // long crc + } else { + __ add(buf, buf, arrayOopDesc::base_offset_in_bytes(T_BYTE)); // + header size + __ ldrw(crc, Address(esp, 3*wordSize)); // long crc + } + + __ andr(sp, r13, -16); // Restore the caller's SP + + // Jump to the stub. + __ b(CAST_FROM_FN_PTR(address, StubRoutines::updateBytesCRC32C())); + + return entry; + } return NULL; } diff --git a/src/hotspot/share/c1/c1_Compiler.cpp b/src/hotspot/share/c1/c1_Compiler.cpp index 5498d9c8e0d..eac28f7fec1 100644 --- a/src/hotspot/share/c1/c1_Compiler.cpp +++ b/src/hotspot/share/c1/c1_Compiler.cpp @@ -213,7 +213,7 @@ bool Compiler::is_intrinsic_supported(const methodHandle& method) { case vmIntrinsics::_updateCRC32: case vmIntrinsics::_updateBytesCRC32: case vmIntrinsics::_updateByteBufferCRC32: -#if defined(SPARC) || defined(S390) || defined(PPC64) +#if defined(SPARC) || defined(S390) || defined(PPC64) || defined(AARCH64) case vmIntrinsics::_updateBytesCRC32C: case vmIntrinsics::_updateDirectByteBufferCRC32C: #endif From 426e7951aa16ba362c452a93a309a48c419dd852 Mon Sep 17 00:00:00 2001 From: Volker Simonis Date: Wed, 1 Nov 2017 12:56:48 -0700 Subject: [PATCH 12/61] 8187091: ReturnBlobToWrongHeapTest fails because of problems in CodeHeap::contains_blob() Reviewed-by: kvn --- src/hotspot/share/code/codeBlob.hpp | 30 +++++++++++++++++++++-------- src/hotspot/share/memory/heap.hpp | 13 ++++++++++++- 2 files changed, 34 insertions(+), 9 deletions(-) diff --git a/src/hotspot/share/code/codeBlob.hpp b/src/hotspot/share/code/codeBlob.hpp index 35b356b7d37..12e702c4f2a 100644 --- a/src/hotspot/share/code/codeBlob.hpp +++ b/src/hotspot/share/code/codeBlob.hpp @@ -49,21 +49,35 @@ struct CodeBlobType { // CodeBlob - superclass for all entries in the CodeCache. // // Subtypes are: -// CompiledMethod : Compiled Java methods (include method that calls to native code) -// nmethod : JIT Compiled Java methods -// RuntimeBlob : Non-compiled method code; generated glue code -// RuntimeStub : Call to VM runtime methods -// DeoptimizationBlob : Used for deoptimization -// ExceptionBlob : Used for stack unrolling -// SafepointBlob : Used to handle illegal instruction exceptions +// CompiledMethod : Compiled Java methods (include method that calls to native code) +// nmethod : JIT Compiled Java methods +// AOTCompiledMethod : AOT Compiled Java methods - Not in the CodeCache! +// AOTCompiledMethod objects are allocated in the C-Heap, the code they +// point to is allocated in the AOTCodeHeap which is in the C-Heap as +// well (i.e. it's the memory where the shared library was loaded to) +// RuntimeBlob : Non-compiled method code; generated glue code +// BufferBlob : Used for non-relocatable code such as interpreter, stubroutines, etc. +// AdapterBlob : Used to hold C2I/I2C adapters +// MethodHandlesAdapterBlob : Used to hold MethodHandles adapters +// RuntimeStub : Call to VM runtime methods +// SingletonBlob : Super-class for all blobs that exist in only one instance +// DeoptimizationBlob : Used for deoptimization +// ExceptionBlob : Used for stack unrolling +// SafepointBlob : Used to handle illegal instruction exceptions +// UncommonTrapBlob : Used to handle uncommon traps // // -// Layout: +// Layout (all except AOTCompiledMethod) : continuous in the CodeCache // - header // - relocation // - content space // - instruction space // - data space +// +// Layout (AOTCompiledMethod) : in the C-Heap +// - header -\ +// ... | +// - code <-/ class CodeBlobLayout; diff --git a/src/hotspot/share/memory/heap.hpp b/src/hotspot/share/memory/heap.hpp index 01e6c47fc13..e25a16133ab 100644 --- a/src/hotspot/share/memory/heap.hpp +++ b/src/hotspot/share/memory/heap.hpp @@ -160,7 +160,18 @@ class CodeHeap : public CHeapObj { char* high_boundary() const { return _memory.high_boundary(); } bool contains(const void* p) const { return low_boundary() <= p && p < high(); } - bool contains_blob(const CodeBlob* blob) const { return contains(blob->code_begin()); } + bool contains_blob(const CodeBlob* blob) const { + // AOT CodeBlobs (i.e. AOTCompiledMethod) objects aren't allocated in the AOTCodeHeap but on the C-Heap. + // Only the code they are pointing to is located in the AOTCodeHeap. All other CodeBlobs are allocated + // directly in their corresponding CodeHeap with their code appended to the actual C++ object. + // So all CodeBlobs except AOTCompiledMethod are continuous in memory with their data and code while + // AOTCompiledMethod and their code/data is distributed in the C-Heap. This means we can use the + // address of a CodeBlob object in order to locate it in its heap while we have to use the address + // of the actual code an AOTCompiledMethod object is pointing to in order to locate it. + // Notice that for an ordinary CodeBlob with code size zero, code_begin() may point beyond the object! + const void* start = AOT_ONLY( (code_blob_type() == CodeBlobType::AOT) ? blob->code_begin() : ) (void*)blob; + return contains(start); + } virtual void* find_start(void* p) const; // returns the block containing p or NULL virtual CodeBlob* find_blob_unsafe(void* start) const; From e57652bb8eb15ac8a3fa2e68eec0d03f4b2e1287 Mon Sep 17 00:00:00 2001 From: Coleen Phillimore Date: Wed, 1 Nov 2017 19:43:13 -0400 Subject: [PATCH 13/61] 8190491: SA tests failed after 8189610 changes Change maxLength to an IntField Reviewed-by: hseigel, jiangli --- .../share/classes/sun/jvm/hotspot/oops/TypeArrayKlass.java | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/oops/TypeArrayKlass.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/oops/TypeArrayKlass.java index e4f1f98f7d5..011bed4a2c6 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/oops/TypeArrayKlass.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/oops/TypeArrayKlass.java @@ -44,14 +44,14 @@ public class TypeArrayKlass extends ArrayKlass { private static synchronized void initialize(TypeDataBase db) throws WrongTypeException { Type t = db.lookupType("TypeArrayKlass"); - maxLength = new CIntField(t.getCIntegerField("_max_length"), 0); + maxLength = new IntField(t.getJIntField("_max_length"), 0); } public TypeArrayKlass(Address addr) { super(addr); } - private static CIntField maxLength; + private static IntField maxLength; public long getMaxLength() { return maxLength.getValue(this); } @@ -98,7 +98,7 @@ public class TypeArrayKlass extends ArrayKlass { public void iterateFields(MetadataVisitor visitor) { super.iterateFields(visitor); - visitor.doCInt(maxLength, true); + visitor.doInt(maxLength, true); } public Klass arrayKlassImpl(boolean orNull, int n) { From e78d4c3cb6786bfc3568830419e2094e8ae783de Mon Sep 17 00:00:00 2001 From: Ioi Lam Date: Thu, 26 Oct 2017 11:35:09 -0700 Subject: [PATCH 14/61] 8179624: [REDO] Avoid repeated calls to JavaThread::last_frame in InterpreterRuntime Fixed typo in the original patch (8179305) Reviewed-by: dlong, coleenp --- .../share/interpreter/interpreterRuntime.cpp | 152 ++++++++++-------- .../share/interpreter/interpreterRuntime.hpp | 71 +++++--- 2 files changed, 132 insertions(+), 91 deletions(-) diff --git a/src/hotspot/share/interpreter/interpreterRuntime.cpp b/src/hotspot/share/interpreter/interpreterRuntime.cpp index 15a7b7502c2..4288922afa3 100644 --- a/src/hotspot/share/interpreter/interpreterRuntime.cpp +++ b/src/hotspot/share/interpreter/interpreterRuntime.cpp @@ -87,14 +87,15 @@ class UnlockFlagSaver { // State accessors void InterpreterRuntime::set_bcp_and_mdp(address bcp, JavaThread *thread) { - last_frame(thread).interpreter_frame_set_bcp(bcp); + LastFrameAccessor last_frame(thread); + last_frame.set_bcp(bcp); if (ProfileInterpreter) { // ProfileTraps uses MDOs independently of ProfileInterpreter. // That is why we must check both ProfileInterpreter and mdo != NULL. - MethodData* mdo = last_frame(thread).interpreter_frame_method()->method_data(); + MethodData* mdo = last_frame.method()->method_data(); if (mdo != NULL) { NEEDS_CLEANUP; - last_frame(thread).interpreter_frame_set_mdp(mdo->bci_to_dp(last_frame(thread).interpreter_frame_bci())); + last_frame.set_mdp(mdo->bci_to_dp(last_frame.bci())); } } } @@ -105,8 +106,9 @@ void InterpreterRuntime::set_bcp_and_mdp(address bcp, JavaThread *thread) { IRT_ENTRY(void, InterpreterRuntime::ldc(JavaThread* thread, bool wide)) // access constant pool - ConstantPool* pool = method(thread)->constants(); - int index = wide ? get_index_u2(thread, Bytecodes::_ldc_w) : get_index_u1(thread, Bytecodes::_ldc); + LastFrameAccessor last_frame(thread); + ConstantPool* pool = last_frame.method()->constants(); + int index = wide ? last_frame.get_index_u2(Bytecodes::_ldc_w) : last_frame.get_index_u1(Bytecodes::_ldc); constantTag tag = pool->tag_at(index); assert (tag.is_unresolved_klass() || tag.is_klass(), "wrong ldc call"); @@ -119,13 +121,14 @@ IRT_ENTRY(void, InterpreterRuntime::resolve_ldc(JavaThread* thread, Bytecodes::C assert(bytecode == Bytecodes::_fast_aldc || bytecode == Bytecodes::_fast_aldc_w, "wrong bc"); ResourceMark rm(thread); - methodHandle m (thread, method(thread)); - Bytecode_loadconstant ldc(m, bci(thread)); + LastFrameAccessor last_frame(thread); + methodHandle m (thread, last_frame.method()); + Bytecode_loadconstant ldc(m, last_frame.bci()); oop result = ldc.resolve_constant(CHECK); #ifdef ASSERT { // The bytecode wrappers aren't GC-safe so construct a new one - Bytecode_loadconstant ldc2(m, bci(thread)); + Bytecode_loadconstant ldc2(m, last_frame.bci()); oop coop = m->constants()->resolved_references()->obj_at(ldc2.cache_index()); assert(result == coop, "expected result for assembly code"); } @@ -182,10 +185,11 @@ IRT_END IRT_ENTRY(void, InterpreterRuntime::multianewarray(JavaThread* thread, jint* first_size_address)) // We may want to pass in more arguments - could make this slightly faster - ConstantPool* constants = method(thread)->constants(); - int i = get_index_u2(thread, Bytecodes::_multianewarray); - Klass* klass = constants->klass_at(i, CHECK); - int nof_dims = number_of_dimensions(thread); + LastFrameAccessor last_frame(thread); + ConstantPool* constants = last_frame.method()->constants(); + int i = last_frame.get_index_u2(Bytecodes::_multianewarray); + Klass* klass = constants->klass_at(i, CHECK); + int nof_dims = last_frame.number_of_dimensions(); assert(klass->is_klass(), "not a class"); assert(nof_dims >= 1, "multianewarray rank must be nonzero"); @@ -217,8 +221,9 @@ IRT_END // Quicken instance-of and check-cast bytecodes IRT_ENTRY(void, InterpreterRuntime::quicken_io_cc(JavaThread* thread)) // Force resolving; quicken the bytecode - int which = get_index_u2(thread, Bytecodes::_checkcast); - ConstantPool* cpool = method(thread)->constants(); + LastFrameAccessor last_frame(thread); + int which = last_frame.get_index_u2(Bytecodes::_checkcast); + ConstantPool* cpool = last_frame.method()->constants(); // We'd expect to assert that we're only here to quicken bytecodes, but in a multithreaded // program we might have seen an unquick'd bytecode in the interpreter but have another // thread quicken the bytecode before we get here. @@ -257,8 +262,9 @@ void InterpreterRuntime::note_trap_inner(JavaThread* thread, int reason, // If necessary, create an MDO to hold the information, and record it. void InterpreterRuntime::note_trap(JavaThread* thread, int reason, TRAPS) { assert(ProfileTraps, "call me only if profiling"); - methodHandle trap_method(thread, method(thread)); - int trap_bci = trap_method->bci_from(bcp(thread)); + LastFrameAccessor last_frame(thread); + methodHandle trap_method(thread, last_frame.method()); + int trap_bci = trap_method->bci_from(last_frame.bcp()); note_trap_inner(thread, reason, trap_method, trap_bci, THREAD); } @@ -391,12 +397,13 @@ IRT_END // invoke w/o arguments (i.e., as if one were inside the call). IRT_ENTRY(address, InterpreterRuntime::exception_handler_for_exception(JavaThread* thread, oopDesc* exception)) + LastFrameAccessor last_frame(thread); Handle h_exception(thread, exception); - methodHandle h_method (thread, method(thread)); + methodHandle h_method (thread, last_frame.method()); constantPoolHandle h_constants(thread, h_method->constants()); bool should_repeat; int handler_bci; - int current_bci = bci(thread); + int current_bci = last_frame.bci(); if (thread->frames_to_pop_failed_realloc() > 0) { // Allocation of scalar replaced object used in this frame @@ -493,7 +500,7 @@ IRT_ENTRY(address, InterpreterRuntime::exception_handler_for_exception(JavaThrea // notify JVMTI of an exception throw; JVMTI will detect if this is a first // time throw or a stack unwinding throw and accordingly notify the debugger if (JvmtiExport::can_post_on_exceptions()) { - JvmtiExport::post_exception_throw(thread, h_method(), bcp(thread), h_exception()); + JvmtiExport::post_exception_throw(thread, h_method(), last_frame.bcp(), h_exception()); } #ifdef CC_INTERP @@ -556,20 +563,21 @@ void InterpreterRuntime::resolve_get_put(JavaThread* thread, Bytecodes::Code byt Thread* THREAD = thread; // resolve field fieldDescriptor info; - constantPoolHandle pool(thread, method(thread)->constants()); - methodHandle m(thread, method(thread)); + LastFrameAccessor last_frame(thread); + constantPoolHandle pool(thread, last_frame.method()->constants()); + methodHandle m(thread, last_frame.method()); bool is_put = (bytecode == Bytecodes::_putfield || bytecode == Bytecodes::_nofast_putfield || bytecode == Bytecodes::_putstatic); bool is_static = (bytecode == Bytecodes::_getstatic || bytecode == Bytecodes::_putstatic); { JvmtiHideSingleStepping jhss(thread); - LinkResolver::resolve_field_access(info, pool, get_index_u2_cpcache(thread, bytecode), + LinkResolver::resolve_field_access(info, pool, last_frame.get_index_u2_cpcache(bytecode), m, bytecode, CHECK); } // end JvmtiHideSingleStepping // check if link resolution caused cpCache to be updated - ConstantPoolCacheEntry* cp_cache_entry = cache_entry(thread); + ConstantPoolCacheEntry* cp_cache_entry = last_frame.cache_entry(); if (cp_cache_entry->is_resolved(bytecode)) return; // compute auxiliary field attributes @@ -718,16 +726,17 @@ IRT_END void InterpreterRuntime::resolve_invoke(JavaThread* thread, Bytecodes::Code bytecode) { Thread* THREAD = thread; + LastFrameAccessor last_frame(thread); // extract receiver from the outgoing argument list if necessary Handle receiver(thread, NULL); if (bytecode == Bytecodes::_invokevirtual || bytecode == Bytecodes::_invokeinterface || bytecode == Bytecodes::_invokespecial) { ResourceMark rm(thread); - methodHandle m (thread, method(thread)); - Bytecode_invoke call(m, bci(thread)); + methodHandle m (thread, last_frame.method()); + Bytecode_invoke call(m, last_frame.bci()); Symbol* signature = call.signature(); - receiver = Handle(thread, - thread->last_frame().interpreter_callee_receiver(signature)); + receiver = Handle(thread, last_frame.callee_receiver(signature)); + assert(Universe::heap()->is_in_reserved_or_null(receiver()), "sanity check"); assert(receiver.is_null() || @@ -737,12 +746,12 @@ void InterpreterRuntime::resolve_invoke(JavaThread* thread, Bytecodes::Code byte // resolve method CallInfo info; - constantPoolHandle pool(thread, method(thread)->constants()); + constantPoolHandle pool(thread, last_frame.method()->constants()); { JvmtiHideSingleStepping jhss(thread); LinkResolver::resolve_invoke(info, receiver, pool, - get_index_u2_cpcache(thread, bytecode), bytecode, + last_frame.get_index_u2_cpcache(bytecode), bytecode, CHECK); if (JvmtiExport::can_hotswap_or_post_breakpoint()) { int retry_count = 0; @@ -754,14 +763,14 @@ void InterpreterRuntime::resolve_invoke(JavaThread* thread, Bytecodes::Code byte "Could not resolve to latest version of redefined method"); // method is redefined in the middle of resolve so re-try. LinkResolver::resolve_invoke(info, receiver, pool, - get_index_u2_cpcache(thread, bytecode), bytecode, + last_frame.get_index_u2_cpcache(bytecode), bytecode, CHECK); } } } // end JvmtiHideSingleStepping // check if link resolution caused cpCache to be updated - ConstantPoolCacheEntry* cp_cache_entry = cache_entry(thread); + ConstantPoolCacheEntry* cp_cache_entry = last_frame.cache_entry(); if (cp_cache_entry->is_resolved(bytecode)) return; #ifdef ASSERT @@ -825,33 +834,35 @@ void InterpreterRuntime::resolve_invoke(JavaThread* thread, Bytecodes::Code byte void InterpreterRuntime::resolve_invokehandle(JavaThread* thread) { Thread* THREAD = thread; const Bytecodes::Code bytecode = Bytecodes::_invokehandle; + LastFrameAccessor last_frame(thread); // resolve method CallInfo info; - constantPoolHandle pool(thread, method(thread)->constants()); + constantPoolHandle pool(thread, last_frame.method()->constants()); { JvmtiHideSingleStepping jhss(thread); LinkResolver::resolve_invoke(info, Handle(), pool, - get_index_u2_cpcache(thread, bytecode), bytecode, + last_frame.get_index_u2_cpcache(bytecode), bytecode, CHECK); } // end JvmtiHideSingleStepping - ConstantPoolCacheEntry* cp_cache_entry = cache_entry(thread); + ConstantPoolCacheEntry* cp_cache_entry = last_frame.cache_entry(); cp_cache_entry->set_method_handle(pool, info); } // First time execution: Resolve symbols, create a permanent CallSite object. void InterpreterRuntime::resolve_invokedynamic(JavaThread* thread) { Thread* THREAD = thread; + LastFrameAccessor last_frame(thread); const Bytecodes::Code bytecode = Bytecodes::_invokedynamic; //TO DO: consider passing BCI to Java. - // int caller_bci = method(thread)->bci_from(bcp(thread)); + // int caller_bci = last_frame.method()->bci_from(last_frame.bcp()); // resolve method CallInfo info; - constantPoolHandle pool(thread, method(thread)->constants()); - int index = get_index_u4(thread, bytecode); + constantPoolHandle pool(thread, last_frame.method()->constants()); + int index = last_frame.get_index_u4(bytecode); { JvmtiHideSingleStepping jhss(thread); LinkResolver::resolve_invoke(info, Handle(), pool, @@ -905,9 +916,9 @@ nmethod* InterpreterRuntime::frequency_counter_overflow(JavaThread* thread, addr // nm could have been unloaded so look it up again. It's unsafe // to examine nm directly since it might have been freed and used // for something else. - frame fr = thread->last_frame(); - Method* method = fr.interpreter_frame_method(); - int bci = method->bci_from(fr.interpreter_frame_bcp()); + LastFrameAccessor last_frame(thread); + Method* method = last_frame.method(); + int bci = method->bci_from(last_frame.bcp()); nm = method->lookup_osr_nmethod_for(bci, CompLevel_none, false); } #ifndef PRODUCT @@ -927,11 +938,11 @@ IRT_ENTRY(nmethod*, // flag, in case this method triggers classloading which will call into Java. UnlockFlagSaver fs(thread); - frame fr = thread->last_frame(); - assert(fr.is_interpreted_frame(), "must come from interpreter"); - methodHandle method(thread, fr.interpreter_frame_method()); + LastFrameAccessor last_frame(thread); + assert(last_frame.is_interpreted_frame(), "must come from interpreter"); + methodHandle method(thread, last_frame.method()); const int branch_bci = branch_bcp != NULL ? method->bci_from(branch_bcp) : InvocationEntryBci; - const int bci = branch_bcp != NULL ? method->bci_from(fr.interpreter_frame_bcp()) : InvocationEntryBci; + const int bci = branch_bcp != NULL ? method->bci_from(last_frame.bcp()) : InvocationEntryBci; assert(!HAS_PENDING_EXCEPTION, "Should not have any exceptions pending"); nmethod* osr_nm = CompilationPolicy::policy()->event(method, method, branch_bci, bci, CompLevel_none, NULL, thread); @@ -947,9 +958,9 @@ IRT_ENTRY(nmethod*, if (UseBiasedLocking) { ResourceMark rm; GrowableArray* objects_to_revoke = new GrowableArray(); - for( BasicObjectLock *kptr = fr.interpreter_frame_monitor_end(); - kptr < fr.interpreter_frame_monitor_begin(); - kptr = fr.next_monitor_in_interpreter_frame(kptr) ) { + for( BasicObjectLock *kptr = last_frame.monitor_end(); + kptr < last_frame.monitor_begin(); + kptr = last_frame.next_monitor(kptr) ) { if( kptr->obj() != NULL ) { objects_to_revoke->append(Handle(THREAD, kptr->obj())); } @@ -974,9 +985,9 @@ IRT_ENTRY(void, InterpreterRuntime::profile_method(JavaThread* thread)) UnlockFlagSaver fs(thread); assert(ProfileInterpreter, "must be profiling interpreter"); - frame fr = thread->last_frame(); - assert(fr.is_interpreted_frame(), "must come from interpreter"); - methodHandle method(thread, fr.interpreter_frame_method()); + LastFrameAccessor last_frame(thread); + assert(last_frame.is_interpreted_frame(), "must come from interpreter"); + methodHandle method(thread, last_frame.method()); Method::build_interpreter_method_data(method, THREAD); if (HAS_PENDING_EXCEPTION) { assert((PENDING_EXCEPTION->is_a(SystemDictionary::OutOfMemoryError_klass())), "we expect only an OOM error here"); @@ -1021,9 +1032,9 @@ IRT_ENTRY(void, InterpreterRuntime::update_mdp_for_ret(JavaThread* thread, int r assert(ProfileInterpreter, "must be profiling interpreter"); ResourceMark rm(thread); HandleMark hm(thread); - frame fr = thread->last_frame(); - assert(fr.is_interpreted_frame(), "must come from interpreter"); - MethodData* h_mdo = fr.interpreter_frame_method()->method_data(); + LastFrameAccessor last_frame(thread); + assert(last_frame.is_interpreted_frame(), "must come from interpreter"); + MethodData* h_mdo = last_frame.method()->method_data(); // Grab a lock to ensure atomic access to setting the return bci and // the displacement. This can block and GC, invalidating all naked oops. @@ -1031,11 +1042,11 @@ IRT_ENTRY(void, InterpreterRuntime::update_mdp_for_ret(JavaThread* thread, int r // ProfileData is essentially a wrapper around a derived oop, so we // need to take the lock before making any ProfileData structures. - ProfileData* data = h_mdo->data_at(h_mdo->dp_to_di(fr.interpreter_frame_mdp())); + ProfileData* data = h_mdo->data_at(h_mdo->dp_to_di(last_frame.mdp())); guarantee(data != NULL, "profile data must be valid"); RetData* rdata = data->as_RetData(); address new_mdp = rdata->fixup_ret(return_bci, h_mdo); - fr.interpreter_frame_set_mdp(new_mdp); + last_frame.set_mdp(new_mdp); IRT_END IRT_ENTRY(MethodCounters*, InterpreterRuntime::build_method_counters(JavaThread* thread, Method* m)) @@ -1060,7 +1071,8 @@ IRT_ENTRY(void, InterpreterRuntime::at_safepoint(JavaThread* thread)) // We are called during regular safepoints and when the VM is // single stepping. If any thread is marked for single stepping, // then we may have JVMTI work to do. - JvmtiExport::at_single_stepping_point(thread, method(thread), bcp(thread)); + LastFrameAccessor last_frame(thread); + JvmtiExport::at_single_stepping_point(thread, last_frame.method(), last_frame.bcp()); } IRT_END @@ -1083,7 +1095,8 @@ ConstantPoolCacheEntry *cp_entry)) } InstanceKlass* cp_entry_f1 = InstanceKlass::cast(cp_entry->f1_as_klass()); jfieldID fid = jfieldIDWorkaround::to_jfieldID(cp_entry_f1, cp_entry->f2_as_index(), is_static); - JvmtiExport::post_field_access(thread, method(thread), bcp(thread), cp_entry_f1, h_obj, fid); + LastFrameAccessor last_frame(thread); + JvmtiExport::post_field_access(thread, last_frame.method(), last_frame.bcp(), cp_entry_f1, h_obj, fid); IRT_END IRT_ENTRY(void, InterpreterRuntime::post_field_modification(JavaThread *thread, @@ -1138,17 +1151,20 @@ IRT_ENTRY(void, InterpreterRuntime::post_field_modification(JavaThread *thread, h_obj = Handle(thread, obj); } - JvmtiExport::post_raw_field_modification(thread, method(thread), bcp(thread), ik, h_obj, + LastFrameAccessor last_frame(thread); + JvmtiExport::post_raw_field_modification(thread, last_frame.method(), last_frame.bcp(), ik, h_obj, fid, sig_type, &fvalue); IRT_END IRT_ENTRY(void, InterpreterRuntime::post_method_entry(JavaThread *thread)) - JvmtiExport::post_method_entry(thread, InterpreterRuntime::method(thread), InterpreterRuntime::last_frame(thread)); + LastFrameAccessor last_frame(thread); + JvmtiExport::post_method_entry(thread, last_frame.method(), last_frame.get_frame()); IRT_END IRT_ENTRY(void, InterpreterRuntime::post_method_exit(JavaThread *thread)) - JvmtiExport::post_method_exit(thread, InterpreterRuntime::method(thread), InterpreterRuntime::last_frame(thread)); + LastFrameAccessor last_frame(thread); + JvmtiExport::post_method_exit(thread, last_frame.method(), last_frame.get_frame()); IRT_END IRT_LEAF(int, InterpreterRuntime::interpreter_contains(address pc)) @@ -1372,10 +1388,10 @@ IRT_LEAF(void, InterpreterRuntime::popframe_move_outgoing_args(JavaThread* threa ResetNoHandleMark rnm; // In a LEAF entry. HandleMark hm; ResourceMark rm; - frame fr = thread->last_frame(); - assert(fr.is_interpreted_frame(), ""); - jint bci = fr.interpreter_frame_bci(); - methodHandle mh(thread, fr.interpreter_frame_method()); + LastFrameAccessor last_frame(thread); + assert(last_frame.is_interpreted_frame(), ""); + jint bci = last_frame.bci(); + methodHandle mh(thread, last_frame.method()); Bytecode_invoke invoke(mh, bci); ArgumentSizeComputer asc(invoke.signature()); int size_of_arguments = (asc.size() + (invoke.has_receiver() ? 1 : 0)); // receiver @@ -1421,10 +1437,10 @@ IRT_END // The generated code still uses call_VM because that will set up the frame pointer for // bcp and method. IRT_LEAF(intptr_t, InterpreterRuntime::trace_bytecode(JavaThread* thread, intptr_t preserve_this_value, intptr_t tos, intptr_t tos2)) - const frame f = thread->last_frame(); - assert(f.is_interpreted_frame(), "must be an interpreted frame"); - methodHandle mh(thread, f.interpreter_frame_method()); - BytecodeTracer::trace(mh, f.interpreter_frame_bcp(), tos, tos2); + LastFrameAccessor last_frame(thread); + assert(last_frame.is_interpreted_frame(), "must be an interpreted frame"); + methodHandle mh(thread, last_frame.method()); + BytecodeTracer::trace(mh, last_frame.bcp(), tos, tos2); return preserve_this_value; IRT_END #endif // !PRODUCT diff --git a/src/hotspot/share/interpreter/interpreterRuntime.hpp b/src/hotspot/share/interpreter/interpreterRuntime.hpp index 39fc137ed33..49e00f9da2b 100644 --- a/src/hotspot/share/interpreter/interpreterRuntime.hpp +++ b/src/hotspot/share/interpreter/interpreterRuntime.hpp @@ -42,29 +42,54 @@ class InterpreterRuntime: AllStatic { friend class PrintingClosure; // for method and bcp private: - // Helper functions to access current interpreter state - static frame last_frame(JavaThread *thread) { return thread->last_frame(); } - static Method* method(JavaThread *thread) { return last_frame(thread).interpreter_frame_method(); } - static address bcp(JavaThread *thread) { return last_frame(thread).interpreter_frame_bcp(); } - static int bci(JavaThread *thread) { return last_frame(thread).interpreter_frame_bci(); } - static void set_bcp_and_mdp(address bcp, JavaThread*thread); - static Bytecodes::Code code(JavaThread *thread) { - // pass method to avoid calling unsafe bcp_to_method (partial fix 4926272) - return Bytecodes::code_at(method(thread), bcp(thread)); - } - static Bytecode bytecode(JavaThread *thread) { return Bytecode(method(thread), bcp(thread)); } - static int get_index_u1(JavaThread *thread, Bytecodes::Code bc) - { return bytecode(thread).get_index_u1(bc); } - static int get_index_u2(JavaThread *thread, Bytecodes::Code bc) - { return bytecode(thread).get_index_u2(bc); } - static int get_index_u2_cpcache(JavaThread *thread, Bytecodes::Code bc) - { return bytecode(thread).get_index_u2_cpcache(bc); } - static int get_index_u4(JavaThread *thread, Bytecodes::Code bc) - { return bytecode(thread).get_index_u4(bc); } - static int number_of_dimensions(JavaThread *thread) { return bcp(thread)[3]; } + // Helper class to access current interpreter state + class LastFrameAccessor : public StackObj { + frame _last_frame; + public: + LastFrameAccessor(JavaThread* thread) { + assert(thread == Thread::current(), "sanity"); + _last_frame = thread->last_frame(); + } + bool is_interpreted_frame() const { return _last_frame.is_interpreted_frame(); } + Method* method() const { return _last_frame.interpreter_frame_method(); } + address bcp() const { return _last_frame.interpreter_frame_bcp(); } + int bci() const { return _last_frame.interpreter_frame_bci(); } + address mdp() const { return _last_frame.interpreter_frame_mdp(); } - static ConstantPoolCacheEntry* cache_entry_at(JavaThread *thread, int i) { return method(thread)->constants()->cache()->entry_at(i); } - static ConstantPoolCacheEntry* cache_entry(JavaThread *thread) { return cache_entry_at(thread, Bytes::get_native_u2(bcp(thread) + 1)); } + void set_bcp(address bcp) { _last_frame.interpreter_frame_set_bcp(bcp); } + void set_mdp(address dp) { _last_frame.interpreter_frame_set_mdp(dp); } + + // pass method to avoid calling unsafe bcp_to_method (partial fix 4926272) + Bytecodes::Code code() const { return Bytecodes::code_at(method(), bcp()); } + + Bytecode bytecode() const { return Bytecode(method(), bcp()); } + int get_index_u1(Bytecodes::Code bc) const { return bytecode().get_index_u1(bc); } + int get_index_u2(Bytecodes::Code bc) const { return bytecode().get_index_u2(bc); } + int get_index_u2_cpcache(Bytecodes::Code bc) const + { return bytecode().get_index_u2_cpcache(bc); } + int get_index_u4(Bytecodes::Code bc) const { return bytecode().get_index_u4(bc); } + int number_of_dimensions() const { return bcp()[3]; } + ConstantPoolCacheEntry* cache_entry_at(int i) const + { return method()->constants()->cache()->entry_at(i); } + ConstantPoolCacheEntry* cache_entry() const { return cache_entry_at(Bytes::get_native_u2(bcp() + 1)); } + + oop callee_receiver(Symbol* signature) { + return _last_frame.interpreter_callee_receiver(signature); + } + BasicObjectLock* monitor_begin() const { + return _last_frame.interpreter_frame_monitor_begin(); + } + BasicObjectLock* monitor_end() const { + return _last_frame.interpreter_frame_monitor_end(); + } + BasicObjectLock* next_monitor(BasicObjectLock* current) const { + return _last_frame.next_monitor_in_interpreter_frame(current); + } + + frame& get_frame() { return _last_frame; } + }; + + static void set_bcp_and_mdp(address bcp, JavaThread*thread); static void note_trap_inner(JavaThread* thread, int reason, const methodHandle& trap_method, int trap_bci, TRAPS); static void note_trap(JavaThread *thread, int reason, TRAPS); @@ -139,7 +164,7 @@ class InterpreterRuntime: AllStatic { static void _breakpoint(JavaThread* thread, Method* method, address bcp); static Bytecodes::Code get_original_bytecode_at(JavaThread* thread, Method* method, address bcp); static void set_original_bytecode_at(JavaThread* thread, Method* method, address bcp, Bytecodes::Code new_code); - static bool is_breakpoint(JavaThread *thread) { return Bytecodes::code_or_bp_at(bcp(thread)) == Bytecodes::_breakpoint; } + static bool is_breakpoint(JavaThread *thread) { return Bytecodes::code_or_bp_at(LastFrameAccessor(thread).bcp()) == Bytecodes::_breakpoint; } // Safepoints static void at_safepoint(JavaThread* thread); From de5686fe49e9a19911abf0c369b9dd5eb8e3fbcd Mon Sep 17 00:00:00 2001 From: Patric Hedlin Date: Thu, 28 Sep 2017 14:55:45 +0200 Subject: [PATCH 15/61] 8172232: SPARC ISA/CPU feature detection is broken/insufficient (on Linux) Reimplementation extending feature detection to SPARC Core C3&C4. Reviewed-by: kvn, neliasso --- .../linux_sparc/vm_version_linux_sparc.cpp | 352 +++++++++++++++--- 1 file changed, 310 insertions(+), 42 deletions(-) diff --git a/src/hotspot/os_cpu/linux_sparc/vm_version_linux_sparc.cpp b/src/hotspot/os_cpu/linux_sparc/vm_version_linux_sparc.cpp index d3cb4914795..c1f981afa18 100644 --- a/src/hotspot/os_cpu/linux_sparc/vm_version_linux_sparc.cpp +++ b/src/hotspot/os_cpu/linux_sparc/vm_version_linux_sparc.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2006, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2006, 2017, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -27,58 +27,326 @@ #include "runtime/os.hpp" #include "vm_version_sparc.hpp" -static bool cpuinfo_field_contains(const char* field, const char* value) { - char line[1024]; - bool rv = false; - FILE* fp = fopen("/proc/cpuinfo", "r"); - if (fp == NULL) { - return rv; - } +#define CPUINFO_LINE_SIZE 1024 - while (fgets(line, sizeof(line), fp) != NULL) { - assert(strlen(line) < sizeof(line) - 1, "buffer line[1024] is too small."); - if (strncmp(line, field, strlen(field)) == 0) { - if (strstr(line, value) != NULL) { - rv = true; + +class CPUinfo { +public: + CPUinfo(const char* field) : _string(NULL) { + + char line[CPUINFO_LINE_SIZE]; + FILE* fp = fopen("/proc/cpuinfo", "r"); + + if (fp != NULL) { + while (fgets(line, sizeof(line), fp) != NULL) { + assert(strlen(line) < sizeof(line) - 1, + "buffer too small (%d)", CPUINFO_LINE_SIZE); + + const char* vstr = match_field(line, field); + + if (vstr != NULL) { + // We have a matching line and a valid starting point to the value of + // the field, copy the string for keeps. + _string = strdup(vstr); + break; + } } - break; + fclose(fp); } } - fclose(fp); - return rv; -} + ~CPUinfo() { os::free((void*)_string); } -static bool detect_niagara() { - return cpuinfo_field_contains("cpu", "Niagara"); -} + const char* value() const { return _string; } -static bool detect_M_family() { - return cpuinfo_field_contains("cpu", "SPARC-M"); -} + bool valid() const { return _string != NULL; } -static bool detect_blkinit() { - return cpuinfo_field_contains("cpucaps", "blkinit"); -} - -int VM_Version::platform_features(int features) { - // Default to generic v9 - features = generic_v9_m; - - if (detect_niagara()) { - log_info(os, cpu)("Detected Linux on Niagara"); - features = niagara1_m | T_family_m; + bool match(const char* s) const { + return valid() ? strcmp(_string, s) == 0 : false; } - if (detect_M_family()) { - log_info(os, cpu)("Detected Linux on M family"); - features = sun4v_m | generic_v9_m | M_family_m | T_family_m; - } +private: + const char* _string; - if (detect_blkinit()) { - features |= blk_init_instructions_m; - } + const char* match_field(char line[CPUINFO_LINE_SIZE], const char* field); + const char* match_alo(const char* text, const char* exp); + const char* match_seq(const char* text, const char* seq); +}; - return features; +/* Given a line of text read from /proc/cpuinfo, determine if the property header + * matches the field specified, according to the following regexp: ""\W+:\W+ + * + * If we have a matching expression, return a pointer to the first character after + * the matching pattern, i.e. the "value", otherwise return NULL. + */ +const char* CPUinfo::match_field(char line[CPUINFO_LINE_SIZE], const char* field) { + return match_alo(match_seq(match_alo(match_seq(line, field), "\t "), ":"), "\t "); +} + +/* Match a sequence of at-least-one character in the string expression (exp) to + * the text input. + */ +const char* CPUinfo::match_alo(const char* text, const char* exp) { + if (text == NULL) return NULL; + + const char* chp; + + for (chp = &text[0]; *chp != '\0'; chp++) { + if (strchr(exp, *chp) == NULL) break; + } + + return text < chp ? chp : NULL; +} + +/* Match an exact sequence of characters as specified by the string expression + * (seq) to the text input. + */ +const char* CPUinfo::match_seq(const char* text, const char* seq) { + if (text == NULL) return NULL; + + while (*seq != '\0') { + if (*seq != *text++) break; else seq++; + } + + return *seq == '\0' ? text : NULL; +} + + +typedef struct { + const uint32_t hash; + bool seen; + const char* const name; + const uint64_t mask; +} FeatureEntry; + + +static uint64_t parse_features(FeatureEntry feature_tbl[], const char input[]); + + +void VM_Version::platform_features() { + + // Some of the features reported via "cpucaps", such as; 'flush', 'stbar', + // 'swap', 'muldiv', 'ultra3', 'blkinit', 'n2', 'mul32', 'div32', 'fsmuld' + // and 'v8plus', are either SPARC V8, supported by all HW or simply nonsense + // (the 'ultra3' "property"). + // + // Entries marked as 'NYI' are not yet supported via "cpucaps" but are + // expected to have the names used in the table below (these are SPARC M7 + // features or more recent). + // + // NOTE: Table sorted on lookup/hash ID. + + static FeatureEntry s_feature_tbl[] = { + { 0x006f, false, "v9", ISA_v9_msk }, // Mandatory + { 0x00a6, false, "md5", ISA_md5_msk }, + { 0x00ce, false, "adi", ISA_adi_msk }, // NYI + { 0x00d7, false, "ima", ISA_ima_msk }, + { 0x00d9, false, "aes", ISA_aes_msk }, + { 0x00db, false, "hpc", ISA_hpc_msk }, + { 0x00dc, false, "des", ISA_des_msk }, + { 0x00ed, false, "sha1", ISA_sha1_msk }, + { 0x00f2, false, "vis", ISA_vis1_msk }, + { 0x0104, false, "vis2", ISA_vis2_msk }, + { 0x0105, false, "vis3", ISA_vis3_msk }, + { 0x0114, false, "sha512", ISA_sha512_msk }, + { 0x0119, false, "sha256", ISA_sha256_msk }, + { 0x011a, false, "fmaf", ISA_fmaf_msk }, + { 0x0132, false, "popc", ISA_popc_msk }, + { 0x0140, false, "crc32c", ISA_crc32c_msk }, + { 0x0147, false, "vis3b", ISA_vis3b_msk }, // NYI + { 0x017e, false, "pause", ISA_pause_msk }, + { 0x0182, false, "mwait", ISA_mwait_msk }, // NYI + { 0x018b, false, "mpmul", ISA_mpmul_msk }, + { 0x018e, false, "sparc5", ISA_sparc5_msk }, // NYI + { 0x01a9, false, "cbcond", ISA_cbcond_msk }, + { 0x01c3, false, "vamask", ISA_vamask_msk }, // NYI + { 0x01ca, false, "kasumi", ISA_kasumi_msk }, + { 0x01e3, false, "xmpmul", ISA_xmpmul_msk }, // NYI + { 0x022c, false, "montmul", ISA_mont_msk }, + { 0x0234, false, "montsqr", ISA_mont_msk }, + { 0x0238, false, "camellia", ISA_camellia_msk }, + { 0x024a, false, "ASIBlkInit", ISA_blk_init_msk }, + { 0x0284, false, "xmontmul", ISA_xmont_msk }, // NYI + { 0x02e6, false, "pause_nsec", ISA_pause_nsec_msk }, // NYI + + { 0x0000, false, NULL, 0 } + }; + + CPUinfo caps("cpucaps"); // Read "cpucaps" from /proc/cpuinfo. + + assert(caps.valid(), "must be"); + + _features = parse_features(s_feature_tbl, caps.value()); + + assert(has_v9(), "must be"); // Basic SPARC-V9 required (V8 not supported). + + CPUinfo type("type"); + + bool is_sun4v = type.match("sun4v"); // All Oracle SPARC + Fujitsu Athena+ + bool is_sun4u = type.match("sun4u"); // All other Fujitsu + + uint64_t synthetic = 0; + + if (is_sun4v) { + // Indirect and direct branches are equally fast. + synthetic = CPU_fast_ind_br_msk; + // Fast IDIV, BIS and LD available on Niagara Plus. + if (has_vis2()) { + synthetic |= (CPU_fast_idiv_msk | CPU_fast_ld_msk); + // ...on Core C4 however, we prefer not to use BIS. + if (!has_sparc5()) { + synthetic |= CPU_fast_bis_msk; + } + } + // Niagara Core C3 supports fast RDPC and block zeroing. + if (has_ima()) { + synthetic |= (CPU_fast_rdpc_msk | CPU_blk_zeroing_msk); + } + // Niagara Core C3 and C4 have slow CMOVE. + if (!has_ima()) { + synthetic |= CPU_fast_cmove_msk; + } + } else if (is_sun4u) { + // SPARC64 only have fast IDIV and RDPC. + synthetic |= (CPU_fast_idiv_msk | CPU_fast_rdpc_msk); + } else { + log_info(os, cpu)("Unable to derive CPU features: %s", type.value()); + } + + _features += synthetic; // Including CPU derived/synthetic features. +} + + +//////////////////////////////////////////////////////////////////////////////// + +static uint32_t uhash32(const char name[]); + +static void update_table(FeatureEntry feature_tbl[], uint32_t hv, + const char* ch1p, + const char* endp); + +/* Given a feature table, parse the input text holding the string value of + * 'cpucaps' as reported by '/proc/cpuinfo', in order to complete the table + * with information on each admissible feature (whether present or not). + * + * Return the composite bit-mask representing the features found. + */ +static uint64_t parse_features(FeatureEntry feature_tbl[], const char input[]) { + log_info(os, cpu)("Parse CPU features: %s\n", input); + +#ifdef ASSERT + // Verify that hash value entries in the table are unique and ordered. + + uint32_t prev = 0; + + for (uint k = 0; feature_tbl[k].name != NULL; k++) { + feature_tbl[k].seen = false; + + assert(feature_tbl[k].hash == uhash32(feature_tbl[k].name), + "feature '%s' has mismatching hash 0x%08x (expected 0x%08x).\n", + feature_tbl[k].name, + feature_tbl[k].hash, + uhash32(feature_tbl[k].name)); + + assert(prev < feature_tbl[k].hash, + "feature '%s' has invalid hash 0x%08x (previous is 0x%08x).\n", + feature_tbl[k].name, + feature_tbl[k].hash, + prev); + + prev = feature_tbl[k].hash; + } +#endif + // Identify features from the input, consisting of a string with features + // separated by commas (or whitespace), e.g. "flush,muldiv,v9,mul32,div32, + // v8plus,popc,vis". + + uint32_t hv = 0; + const char* ch1p = &input[0]; + uint i = 0; + + do { + char ch = input[i]; + + if (isalnum(ch) || ch == '_') { + hv += (ch - 32u); + } + else if (isspace(ch) || ch == ',' || ch == '\0') { // end-of-token + if (ch1p < &input[i]) { + update_table(feature_tbl, hv, ch1p, &input[i]); + } + ch1p = &input[i + 1]; hv = 0; + } else { + // Handle non-accepted input robustly. + log_info(os, cpu)("Bad token in feature string: '%c' (0x%02x).\n", ch, ch); + ch1p = &input[i + 1]; hv = 0; + } + } + while (input[i++] != '\0'); + + // Compute actual bit-mask representation. + + uint64_t mask = 0; + + for (uint k = 0; feature_tbl[k].name != NULL; k++) { + mask |= feature_tbl[k].seen ? feature_tbl[k].mask : 0; + } + + return mask; +} + +static uint32_t uhash32(const char name[]) { + uint32_t hv = 0; + + for (uint i = 0; name[i] != '\0'; i++) { + hv += (name[i] - 32u); + } + + return hv; +} + +static bool verify_match(const char name[], const char* ch1p, const char* endp); + +static void update_table(FeatureEntry feature_tbl[], uint32_t hv, const char* ch1p, const char* endp) { + assert(ch1p < endp, "at least one character"); + + // Look for a hash value in the table. Since this table is a small one (and + // is expected to stay small), we use a simple linear search (iff the table + // grows large, we may consider to adopt a binary ditto, or a perfect hash). + + for (uint k = 0; feature_tbl[k].name != NULL; k++) { + uint32_t hash = feature_tbl[k].hash; + + if (hash < hv) continue; + + if (hash == hv) { + const char* name = feature_tbl[k].name; + + if (verify_match(name, ch1p, endp)) { + feature_tbl[k].seen = true; + break; + } + } + + // Either a non-matching feature (when hash == hv) or hash > hv. In either + // case we break out of the loop and terminate the search (note that the + // table is assumed to be uniquely sorted on the hash). + + break; + } +} + +static bool verify_match(const char name[], const char* ch1p, const char* endp) { + size_t len = strlen(name); + + if (len != static_cast(endp - ch1p)) { + return false; + } + + for (uint i = 0; ch1p + i < endp; i++) { + if (name[i] != ch1p[i]) return false; + } + + return true; } From 40cdb9fd60177b53f30bc9ff5d89939d75ee2fd4 Mon Sep 17 00:00:00 2001 From: Doug Simon Date: Mon, 6 Nov 2017 21:38:42 +0100 Subject: [PATCH 16/61] 8187315: [JVMCI] hosted use of JVMCI can crash VM under -Xint Reviewed-by: kvn, dlong --- src/hotspot/share/jvmci/jvmciCompiler.hpp | 5 ++++- src/hotspot/share/jvmci/jvmciCompilerToVM.cpp | 4 ++-- src/hotspot/share/jvmci/jvmciRuntime.cpp | 2 +- src/hotspot/share/prims/jni.cpp | 2 +- 4 files changed, 8 insertions(+), 5 deletions(-) diff --git a/src/hotspot/share/jvmci/jvmciCompiler.hpp b/src/hotspot/share/jvmci/jvmciCompiler.hpp index b1401e6d75f..fd246325f07 100644 --- a/src/hotspot/share/jvmci/jvmciCompiler.hpp +++ b/src/hotspot/share/jvmci/jvmciCompiler.hpp @@ -55,10 +55,13 @@ private: public: JVMCICompiler(); - static JVMCICompiler* instance(TRAPS) { + static JVMCICompiler* instance(bool require_non_null, TRAPS) { if (!EnableJVMCI) { THROW_MSG_NULL(vmSymbols::java_lang_InternalError(), "JVMCI is not enabled") } + if (_instance == NULL && require_non_null) { + THROW_MSG_NULL(vmSymbols::java_lang_InternalError(), "The JVMCI compiler instance has not been created"); + } return _instance; } diff --git a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp index a01da7b52cc..c0509df788b 100644 --- a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp +++ b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp @@ -1018,7 +1018,7 @@ C2V_VMENTRY(jint, installCode, (JNIEnv *jniEnv, jobject, jobject target, jobject Handle installed_code_handle(THREAD, JNIHandles::resolve(installed_code)); Handle speculation_log_handle(THREAD, JNIHandles::resolve(speculation_log)); - JVMCICompiler* compiler = JVMCICompiler::instance(CHECK_JNI_ERR); + JVMCICompiler* compiler = JVMCICompiler::instance(true, CHECK_JNI_ERR); TraceTime install_time("installCode", JVMCICompiler::codeInstallTimer()); bool is_immutable_PIC = HotSpotCompiledCode::isImmutablePIC(compiled_code_handle) > 0; @@ -1136,7 +1136,7 @@ C2V_VMENTRY(jint, getMetadata, (JNIEnv *jniEnv, jobject, jobject target, jobject C2V_END C2V_VMENTRY(void, resetCompilationStatistics, (JNIEnv *jniEnv, jobject)) - JVMCICompiler* compiler = JVMCICompiler::instance(CHECK); + JVMCICompiler* compiler = JVMCICompiler::instance(true, CHECK); CompilerStatistics* stats = compiler->stats(); stats->_standard.reset(); stats->_osr.reset(); diff --git a/src/hotspot/share/jvmci/jvmciRuntime.cpp b/src/hotspot/share/jvmci/jvmciRuntime.cpp index 2f9678d1119..134ea685c83 100644 --- a/src/hotspot/share/jvmci/jvmciRuntime.cpp +++ b/src/hotspot/share/jvmci/jvmciRuntime.cpp @@ -821,7 +821,7 @@ void JVMCIRuntime::shutdown(TRAPS) { } CompLevel JVMCIRuntime::adjust_comp_level_inner(const methodHandle& method, bool is_osr, CompLevel level, JavaThread* thread) { - JVMCICompiler* compiler = JVMCICompiler::instance(thread); + JVMCICompiler* compiler = JVMCICompiler::instance(false, thread); if (compiler != NULL && compiler->is_bootstrapping()) { return level; } diff --git a/src/hotspot/share/prims/jni.cpp b/src/hotspot/share/prims/jni.cpp index ca0f634d7f5..15bc070fbfd 100644 --- a/src/hotspot/share/prims/jni.cpp +++ b/src/hotspot/share/prims/jni.cpp @@ -3949,7 +3949,7 @@ static jint JNI_CreateJavaVM_inner(JavaVM **vm, void **penv, void *args) { // JVMCI is initialized on a CompilerThread if (BootstrapJVMCI) { JavaThread* THREAD = thread; - JVMCICompiler* compiler = JVMCICompiler::instance(CATCH); + JVMCICompiler* compiler = JVMCICompiler::instance(true, CATCH); compiler->bootstrap(THREAD); if (HAS_PENDING_EXCEPTION) { HandleMark hm; From 545e8eb333c2a5add86a994134f649b514b52cfc Mon Sep 17 00:00:00 2001 From: Jiangli Zhou Date: Mon, 6 Nov 2017 14:12:37 -0500 Subject: [PATCH 17/61] 8184206: Resolve all string constants in shared classes at CDS dump time Change ConstantPool::resolve_class_constants() to resolve all string constants. Reviewed-by: iklam, coleenp --- src/hotspot/share/oops/constantPool.cpp | 11 +++-------- 1 file changed, 3 insertions(+), 8 deletions(-) diff --git a/src/hotspot/share/oops/constantPool.cpp b/src/hotspot/share/oops/constantPool.cpp index f1aec2bff2e..9e1eefd5251 100644 --- a/src/hotspot/share/oops/constantPool.cpp +++ b/src/hotspot/share/oops/constantPool.cpp @@ -305,14 +305,9 @@ void ConstantPool::resolve_class_constants(TRAPS) { constantPoolHandle cp(THREAD, this); for (int index = 1; index < length(); index++) { // Index 0 is unused - if (tag_at(index).is_string()) { - Symbol* sym = cp->unresolved_string_at(index); - // Look up only. Only resolve references to already interned strings. - oop str = StringTable::lookup(sym); - if (str != NULL) { - int cache_index = cp->cp_to_object_index(index); - cp->string_at_put(index, cache_index, str); - } + if (tag_at(index).is_string() && !cp->is_pseudo_string_at(index)) { + int cache_index = cp->cp_to_object_index(index); + string_at_impl(cp, index, cache_index, CHECK); } } } From c8aacd39724c83cfa25ecc80ae9e9f083e5f9e4a Mon Sep 17 00:00:00 2001 From: Dean Long Date: Mon, 6 Nov 2017 20:29:49 -0800 Subject: [PATCH 18/61] 8190710: Update Graal Reviewed-by: kvn --- make/CompileToolsHotspot.gmk | 2 + src/hotspot/share/aot/aotCodeHeap.cpp | 2 + .../jaotc/binformat/BinaryContainer.java | 2 + .../test/BlackholeDirectiveTest.java | 8 + .../test/ControlFlowAnchorDirectiveTest.java | 8 + .../directives/test/OpaqueDirectiveTest.java | 8 + .../compiler/api/replacements/Snippet.java | 7 + .../compiler/asm/amd64/AMD64Assembler.java | 90 +- .../asm/amd64/AMD64MacroAssembler.java | 12 +- .../amd64/test/AMD64AddressLoweringTest.java | 126 +++ .../core/amd64/AMD64AddressLowering.java | 180 +++- .../core/amd64/AMD64LIRGenerator.java | 91 +- .../compiler/core/amd64/AMD64LIRKindTool.java | 10 +- .../core/common/CompressEncoding.java | 19 +- .../core/common/calc/FloatConvert.java | 30 +- .../compiler/core/common/cfg/Loop.java | 1 - .../core/common/type/ArithmeticOpTable.java | 21 +- .../compiler/core/common/type/FloatStamp.java | 170 +++- .../core/common/type/IntegerStamp.java | 42 +- .../core/test/CheckGraalInvariants.java | 6 +- .../test/ConditionalEliminationTest14.java | 91 ++ .../test/ConditionalEliminationTestBase.java | 12 + .../compiler/core/test/DumpPathTest.java | 111 +++ .../core/test/FinalizableSubclassTest.java | 2 +- .../compiler/core/test/GraalCompilerTest.java | 11 +- .../test/GraalDebugHandlersFactoryTest.java | 21 +- .../core/test/GuardedIntrinsicTest.java | 5 + .../compiler/core/test/HashCodeTest.java | 6 + .../core/test/MarkUnsafeAccessTest.java | 25 +- .../core/test/MergeCanonicalizerTest.java | 11 + .../compiler/core/test/ProfilingInfoTest.java | 18 +- .../compiler/core/test/SubWordReturnTest.java | 143 +++ .../core/test/UnbalancedMonitorsTest.java | 2 +- .../core/test/UnsafeVirtualizationTest.java | 92 +- .../compiler/core/test/ea/EATestBase.java | 12 +- .../core/test/ea/PEAAssertionsTest.java | 11 + .../compiler/core/test/ea/TrufflePEATest.java | 16 +- .../compiler/core/test/ea/UnsafeEATest.java | 180 +++- .../NestedLoopEffectsPhaseComplexityTest.java | 2 +- .../compiler/core/GraalCompilerOptions.java | 2 +- .../compiler/core/gen/DebugInfoBuilder.java | 37 +- .../compiler/core/gen/NodeLIRBuilder.java | 7 +- .../graalvm/compiler/core/target/Backend.java | 9 +- .../graalvm/compiler/debug/DebugContext.java | 45 + .../compiler/debug/DebugHandlersFactory.java | 3 + .../graalvm/compiler/debug/DebugOptions.java | 2 - .../compiler/debug/IgvDumpChannel.java | 119 +++ .../graalvm/compiler/debug/PathUtilities.java | 163 ++-- .../graph/test/graphio/GraphSnippetTest.java | 51 ++ .../graph/test/graphio/NodeEncodingTest.java | 311 +++++++ .../src/org/graalvm/compiler/graph/Graph.java | 91 +- .../src/org/graalvm/compiler/graph/Node.java | 4 +- .../aarch64/AArch64HotSpotNodeLIRBuilder.java | 3 +- .../amd64/AMD64HotSpotAddressLowering.java | 88 +- .../amd64/AMD64HotSpotLIRGenerator.java | 42 +- .../amd64/AMD64HotSpotLIRKindTool.java | 39 + .../hotspot/amd64/AMD64HotSpotMove.java | 90 -- .../amd64/AMD64HotSpotNodeLIRBuilder.java | 3 +- .../hotspot/sparc/SPARCHotSpotMove.java | 3 +- .../sparc/SPARCHotSpotNodeLIRBuilder.java | 3 +- .../sparc/SPARCHotSpotStrategySwitchOp.java | 3 +- .../test/ArrayCopyIntrinsificationTest.java | 2 +- .../hotspot/test/CompilationWrapperTest.java | 97 +- .../hotspot/test/ExplicitExceptionTest.java | 25 +- .../hotspot/test/GraalOSRLockTest.java | 40 +- .../hotspot/test/GraalOSRTestBase.java | 9 + .../test/WriteBarrierVerificationTest.java | 7 - .../hotspot/GraalHotSpotVMConfig.java | 15 +- .../compiler/hotspot/HotSpotBackend.java | 2 + .../hotspot/HotSpotGraalCompiler.java | 2 +- .../hotspot/HotSpotGraalCompilerFactory.java | 9 + .../compiler/hotspot/HotSpotGraalMBean.java | 6 +- .../compiler/hotspot/HotSpotGraalRuntime.java | 2 +- .../meta/DefaultHotSpotLoweringProvider.java | 49 +- .../meta/HotSpotGraphBuilderPlugins.java | 6 +- .../meta/HotSpotHostForeignCallsProvider.java | 6 +- .../nodes/aot/ResolveDynamicConstantNode.java | 3 +- .../nodes/profiling/ProfileBranchNode.java | 11 +- .../nodes/profiling/ProfileInvokeNode.java | 11 +- .../hotspot/nodes/profiling/ProfileNode.java | 49 +- .../ProfileWithNotificationNode.java | 4 +- .../phases/OnStackReplacementPhase.java | 66 +- .../replacements/HotSpotReplacementsUtil.java | 5 + .../hotspot/replacements/HubGetClassNode.java | 2 +- .../replacements/IdentityHashCodeNode.java | 2 +- .../arraycopy/ArrayCopySnippets.java | 842 ++++++++---------- .../arraycopy/ArrayCopyUnrollNode.java | 148 --- ...de.java => ArrayCopyWithSlowPathNode.java} | 53 +- .../arraycopy/CheckcastArrayCopyCallNode.java | 10 +- .../arraycopy/GenericArrayCopyCallNode.java | 122 +++ .../arraycopy/UnsafeArrayCopyNode.java | 153 ---- .../arraycopy/UnsafeArrayCopySnippets.java | 324 ------- .../ProbabilisticProfileSnippets.java | 35 +- .../profiling/ProfileSnippets.java | 36 +- .../compiler/hotspot/stubs/NewArrayStub.java | 3 +- .../hotspot/stubs/NewInstanceStub.java | 3 +- .../graalvm/compiler/java/BytecodeParser.java | 123 ++- .../compiler/java/FrameStateBuilder.java | 4 +- .../optimize/ConditionalElimination02.java | 11 + .../compiler/jtt/optimize/Fold_Double04.java | 66 ++ .../compiler/jtt/optimize/Fold_Float03.java | 66 ++ .../lir/amd64/AMD64ArrayEqualsOp.java | 10 +- .../compiler/lir/amd64/AMD64ControlFlow.java | 61 +- .../graalvm/compiler/lir/amd64/AMD64Move.java | 113 +++ .../sparc/SPARCLoadConstantTableBaseOp.java | 4 +- .../graalvm/compiler/lir/sparc/SPARCMove.java | 33 +- .../graalvm/compiler/lir/LIRValueUtil.java | 10 + .../lir/RedundantMoveElimination.java | 7 +- .../lir/alloc/SaveCalleeSaveRegisters.java | 4 +- .../lir/asm/CompilationResultBuilder.java | 25 +- .../lir/gen/ArithmeticLIRGenerator.java | 53 +- .../compiler/lir/gen/LIRGenerationResult.java | 5 + .../compiler/lir/gen/LIRGenerator.java | 12 +- .../compiler/lir/gen/LIRGeneratorTool.java | 3 + .../loop/test/LoopPartialUnrollTest.java | 2 +- .../compiler/loop/CountedLoopInfo.java | 6 +- .../compiler/loop/DefaultLoopPolicies.java | 31 +- .../graalvm/compiler/nodeinfo/NodeCycles.java | 4 + .../compiler/nodes/test/IntegerStampTest.java | 17 + .../test/PrimitiveStampBoundaryTest.java | 326 +++++++ .../compiler/nodes/CompressionNode.java | 16 +- .../graalvm/compiler/nodes/ConstantNode.java | 5 +- .../org/graalvm/compiler/nodes/IfNode.java | 11 +- .../org/graalvm/compiler/nodes/PhiNode.java | 12 + .../org/graalvm/compiler/nodes/PiNode.java | 43 +- .../graalvm/compiler/nodes/ValuePhiNode.java | 16 +- .../nodes/calc/BinaryArithmeticNode.java | 4 +- .../compiler/nodes/calc/ConditionalNode.java | 152 ++-- .../calc/{DivNode.java => FloatDivNode.java} | 37 +- .../nodes/calc/IntegerConvertNode.java | 2 +- .../nodes/calc/IntegerLessThanNode.java | 83 +- .../compiler/nodes/calc/IsNullNode.java | 3 +- .../compiler/nodes/calc/SignedDivNode.java | 54 +- .../compiler/nodes/calc/SignedRemNode.java | 7 +- .../compiler/nodes/calc/UnaryNode.java | 5 + .../nodes/calc/UnpackEndianHalfNode.java | 90 ++ .../compiler/nodes/cfg/ControlFlowGraph.java | 6 +- .../nodes/extended/BranchProbabilityNode.java | 3 + .../compiler/nodes/extended/RawLoadNode.java | 2 +- .../compiler/nodes/extended/RawStoreNode.java | 40 +- .../graphbuilderconf/GraphBuilderContext.java | 2 +- .../graphbuilderconf/InlineInvokePlugin.java | 6 + .../graphbuilderconf/IntrinsicContext.java | 39 +- .../compiler/nodes/java/InstanceOfNode.java | 19 +- .../compiler/nodes/java/LoadFieldNode.java | 6 +- .../compiler/nodes/java/StoreFieldNode.java | 13 +- .../compiler/nodes/java/StoreIndexedNode.java | 2 +- .../compiler/nodes/memory/MemoryPhiNode.java | 5 + .../memory/address/OffsetAddressNode.java | 6 + .../compiler/nodes/spi/VirtualizerTool.java | 13 +- .../compiler/nodes/util/GraphUtil.java | 2 +- .../nodes/util/JavaConstantFormattable.java | 31 + .../nodes/util/JavaConstantFormatter.java | 29 + .../nodes/virtual/VirtualArrayNode.java | 4 +- .../nodes/virtual/VirtualInstanceNode.java | 4 +- .../options/processor/OptionProcessor.java | 3 + .../common/ConditionalEliminationPhase.java | 28 +- .../compiler/phases/common/FixReadsPhase.java | 7 +- .../common/ProfileCompiledMethodsPhase.java | 4 +- .../inlining/info/elem/InlineableGraph.java | 5 +- .../common/util/HashSetNodeEventListener.java | 4 +- .../graalvm/compiler/phases/BasePhase.java | 4 +- .../phases/schedule/SchedulePhase.java | 2 +- .../phases/verify/VerifyGetOptionsUsage.java | 86 ++ .../phases/verify/VerifyGraphAddUsage.java | 117 +++ .../printer/BasicIdealGraphPrinter.java | 366 -------- .../compiler/printer/BinaryGraphPrinter.java | 41 +- .../compiler/printer/CFGPrinterObserver.java | 6 +- .../printer/CanonicalStringGraphPrinter.java | 10 +- .../printer/GraalDebugHandlersFactory.java | 194 +--- .../compiler/printer/GraphPrinter.java | 47 +- .../printer/GraphPrinterDumpHandler.java | 16 +- .../compiler/printer/IdealGraphPrinter.java | 356 -------- .../amd64/AMD64StringIndexOfNode.java | 4 +- .../ConstantBindingParameterPlugin.java | 2 +- .../DefaultJavaLoweringProvider.java | 55 +- .../compiler/replacements/GraphKit.java | 11 +- .../replacements/ReplacementsImpl.java | 5 +- .../compiler/replacements/SnippetCounter.java | 7 +- .../replacements/SnippetIntegerHistogram.java | 153 ++++ .../replacements/nodes/ArrayEqualsNode.java | 6 +- .../nodes/BasicArrayCopyNode.java | 8 +- .../nodes/BasicObjectCloneNode.java | 4 +- .../nodes/BinaryMathIntrinsicNode.java | 4 +- .../org/graalvm/compiler/test/GraalTest.java | 23 +- .../virtual/phases/ea/EffectList.java | 7 +- .../virtual/phases/ea/EffectsClosure.java | 2 - .../virtual/phases/ea/EffectsPhase.java | 4 +- .../virtual/phases/ea/ObjectState.java | 25 + .../phases/ea/PartialEscapeClosure.java | 1 + .../phases/ea/VirtualizerToolImpl.java | 79 +- .../src/org/graalvm/graphio/GraphOutput.java | 41 +- .../org/graalvm/graphio/GraphProtocol.java | 147 ++- .../org/graalvm/graphio/GraphSnippets.java | 285 ++++++ .../org/graalvm/graphio/GraphStructure.java | 37 +- .../src/org/graalvm/graphio/ProtocolImpl.java | 23 +- .../org/graalvm/graphio/doc-files/diamond.png | Bin 0 -> 48209 bytes .../src/org/graalvm/graphio/package-info.java | 86 ++ .../org.graalvm.word/.checkstyle_checks.xml | 240 +++++ 199 files changed, 5903 insertions(+), 3303 deletions(-) create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64.test/src/org/graalvm/compiler/core/amd64/test/AMD64AddressLoweringTest.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ConditionalEliminationTest14.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/DumpPathTest.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/SubWordReturnTest.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/IgvDumpChannel.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph.test/src/org/graalvm/compiler/graph/test/graphio/GraphSnippetTest.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph.test/src/org/graalvm/compiler/graph/test/graphio/NodeEncodingTest.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotLIRKindTool.java delete mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopyUnrollNode.java rename src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/{ArrayCopySlowPathNode.java => ArrayCopyWithSlowPathNode.java} (54%) create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/GenericArrayCopyCallNode.java delete mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/UnsafeArrayCopyNode.java delete mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/UnsafeArrayCopySnippets.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/Fold_Double04.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/Fold_Float03.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes.test/src/org/graalvm/compiler/nodes/test/PrimitiveStampBoundaryTest.java rename src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/{DivNode.java => FloatDivNode.java} (69%) create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/UnpackEndianHalfNode.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/JavaConstantFormattable.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/JavaConstantFormatter.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/verify/VerifyGetOptionsUsage.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/verify/VerifyGraphAddUsage.java delete mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/BasicIdealGraphPrinter.java delete mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/IdealGraphPrinter.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/SnippetIntegerHistogram.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphSnippets.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/doc-files/diamond.png create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/package-info.java create mode 100644 src/jdk.internal.vm.compiler/share/classes/org.graalvm.word/.checkstyle_checks.xml diff --git a/make/CompileToolsHotspot.gmk b/make/CompileToolsHotspot.gmk index 87d8bd5b3fc..1ccc3e2a31b 100644 --- a/make/CompileToolsHotspot.gmk +++ b/make/CompileToolsHotspot.gmk @@ -67,6 +67,7 @@ ifeq ($(INCLUDE_GRAAL), true) $(SRC_DIR)/org.graalvm.compiler.phases.common/src \ $(SRC_DIR)/org.graalvm.compiler.serviceprovider/src \ $(SRC_DIR)/org.graalvm.compiler.virtual/src \ + $(SRC_DIR)/org.graalvm.graphio/src \ $(SRC_DIR)/org.graalvm.util/src \ $(VM_CI_SRC_DIR)/jdk.vm.ci.code/src \ $(VM_CI_SRC_DIR)/jdk.vm.ci.common/src \ @@ -125,6 +126,7 @@ ifeq ($(INCLUDE_GRAAL), true) $(SRC_DIR)/org.graalvm.compiler.nodeinfo/src \ $(SRC_DIR)/org.graalvm.compiler.options/src \ $(SRC_DIR)/org.graalvm.compiler.serviceprovider/src \ + $(SRC_DIR)/org.graalvm.graphio/src \ $(SRC_DIR)/org.graalvm.util/src \ $(VM_CI_SRC_DIR)/jdk.vm.ci.code/src \ $(VM_CI_SRC_DIR)/jdk.vm.ci.common/src \ diff --git a/src/hotspot/share/aot/aotCodeHeap.cpp b/src/hotspot/share/aot/aotCodeHeap.cpp index 15c1e696ca4..2a9b3982cd2 100644 --- a/src/hotspot/share/aot/aotCodeHeap.cpp +++ b/src/hotspot/share/aot/aotCodeHeap.cpp @@ -490,6 +490,8 @@ void AOTCodeHeap::link_stub_routines_symbols() { SET_AOT_GLOBAL_SYMBOL_VALUE("_aot_stub_routines_checkcast_arraycopy", address, StubRoutines::_checkcast_arraycopy); + SET_AOT_GLOBAL_SYMBOL_VALUE("_aot_stub_routines_generic_arraycopy", address, StubRoutines::_generic_arraycopy); + SET_AOT_GLOBAL_SYMBOL_VALUE("_aot_stub_routines_aescrypt_encryptBlock", address, StubRoutines::_aescrypt_encryptBlock); SET_AOT_GLOBAL_SYMBOL_VALUE("_aot_stub_routines_aescrypt_decryptBlock", address, StubRoutines::_aescrypt_decryptBlock); SET_AOT_GLOBAL_SYMBOL_VALUE("_aot_stub_routines_cipherBlockChaining_encryptAESCrypt", address, StubRoutines::_cipherBlockChaining_encryptAESCrypt); diff --git a/src/jdk.aot/share/classes/jdk.tools.jaotc.binformat/src/jdk/tools/jaotc/binformat/BinaryContainer.java b/src/jdk.aot/share/classes/jdk.tools.jaotc.binformat/src/jdk/tools/jaotc/binformat/BinaryContainer.java index 9e3aa91f4cf..0971479577e 100644 --- a/src/jdk.aot/share/classes/jdk.tools.jaotc.binformat/src/jdk/tools/jaotc/binformat/BinaryContainer.java +++ b/src/jdk.aot/share/classes/jdk.tools.jaotc.binformat/src/jdk/tools/jaotc/binformat/BinaryContainer.java @@ -192,6 +192,8 @@ public final class BinaryContainer implements SymbolTable { {"StubRoutines::_checkcast_arraycopy", "_aot_stub_routines_checkcast_arraycopy"}, + {"StubRoutines::_generic_arraycopy", "_aot_stub_routines_generic_arraycopy"}, + {"StubRoutines::_aescrypt_encryptBlock", "_aot_stub_routines_aescrypt_encryptBlock"}, {"StubRoutines::_aescrypt_decryptBlock", "_aot_stub_routines_aescrypt_decryptBlock"}, {"StubRoutines::_cipherBlockChaining_encryptAESCrypt", "_aot_stub_routines_cipherBlockChaining_encryptAESCrypt"}, diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/BlackholeDirectiveTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/BlackholeDirectiveTest.java index e6ed6ecd205..ead9c2544ec 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/BlackholeDirectiveTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/BlackholeDirectiveTest.java @@ -34,6 +34,9 @@ import org.graalvm.compiler.api.directives.GraalDirectives; import org.graalvm.compiler.core.test.GraalCompilerTest; import org.graalvm.compiler.nodes.ParameterNode; import org.graalvm.compiler.nodes.StructuredGraph; +import org.graalvm.compiler.phases.OptimisticOptimizations; +import org.graalvm.compiler.phases.OptimisticOptimizations.Optimization; +import org.graalvm.compiler.phases.tiers.HighTierContext; /** * Tests for {@link GraalDirectives#blackhole}. @@ -128,6 +131,11 @@ public class BlackholeDirectiveTest extends GraalCompilerTest { test("blackholeObjectSnippet", 37); } + @Override + protected HighTierContext getDefaultHighTierContext() { + return new HighTierContext(getProviders(), getDefaultGraphBuilderSuite(), OptimisticOptimizations.ALL.remove(Optimization.RemoveNeverExecutedCode)); + } + @Override protected boolean checkLowTierGraph(StructuredGraph graph) { BlackholeSnippet snippet = graph.method().getAnnotation(BlackholeSnippet.class); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/ControlFlowAnchorDirectiveTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/ControlFlowAnchorDirectiveTest.java index 1eaf803fc73..b2ff51fd76f 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/ControlFlowAnchorDirectiveTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/ControlFlowAnchorDirectiveTest.java @@ -46,6 +46,9 @@ import org.graalvm.compiler.nodes.StructuredGraph.AllowAssumptions; import org.graalvm.compiler.nodes.debug.ControlFlowAnchorNode; import jdk.vm.ci.meta.ResolvedJavaMethod; +import org.graalvm.compiler.phases.OptimisticOptimizations; +import org.graalvm.compiler.phases.OptimisticOptimizations.Optimization; +import org.graalvm.compiler.phases.tiers.HighTierContext; public class ControlFlowAnchorDirectiveTest extends GraalCompilerTest { @@ -238,6 +241,11 @@ public class ControlFlowAnchorDirectiveTest extends GraalCompilerTest { return Collections.emptyList(); } + @Override + protected HighTierContext getDefaultHighTierContext() { + return new HighTierContext(getProviders(), getDefaultGraphBuilderSuite(), OptimisticOptimizations.ALL.remove(Optimization.RemoveNeverExecutedCode)); + } + @Override protected boolean checkLowTierGraph(StructuredGraph graph) { List anchors = graph.getNodes().filter(ControlFlowAnchorNode.class).snapshot(); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/OpaqueDirectiveTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/OpaqueDirectiveTest.java index 64e9cc42f98..6ec736902c9 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/OpaqueDirectiveTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.directives.test/src/org/graalvm/compiler/api/directives/test/OpaqueDirectiveTest.java @@ -37,6 +37,9 @@ import org.graalvm.compiler.nodes.ReturnNode; import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.calc.AddNode; import org.graalvm.compiler.nodes.calc.ConditionalNode; +import org.graalvm.compiler.phases.OptimisticOptimizations; +import org.graalvm.compiler.phases.OptimisticOptimizations.Optimization; +import org.graalvm.compiler.phases.tiers.HighTierContext; /** * Tests for {@link GraalDirectives#opaque}. @@ -127,6 +130,11 @@ public class OpaqueDirectiveTest extends GraalCompilerTest { test("opaqueObjectSnippet"); } + @Override + protected HighTierContext getDefaultHighTierContext() { + return new HighTierContext(getProviders(), getDefaultGraphBuilderSuite(), OptimisticOptimizations.ALL.remove(Optimization.RemoveNeverExecutedCode)); + } + @Override protected boolean checkLowTierGraph(StructuredGraph graph) { OpaqueSnippet snippet = graph.method().getAnnotation(OpaqueSnippet.class); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.replacements/src/org/graalvm/compiler/api/replacements/Snippet.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.replacements/src/org/graalvm/compiler/api/replacements/Snippet.java index 08cbae1ce6c..39d756eded2 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.replacements/src/org/graalvm/compiler/api/replacements/Snippet.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.api.replacements/src/org/graalvm/compiler/api/replacements/Snippet.java @@ -35,6 +35,13 @@ import java.lang.annotation.Target; @Target(ElementType.METHOD) public @interface Snippet { + /** + * A partial intrinsic exits by (effectively) calling the intrinsified method. Normally, this + * call must use exactly the same arguments as the call that is being intrinsified. For well + * known snippets that are used after frame state assignment, we want to relax this restriction. + */ + boolean allowPartialIntrinsicArgumentMismatch() default false; + /** * Denotes a snippet parameter representing 0 or more arguments that will be bound during * snippet template instantiation. During snippet template creation, its value must be an array diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.asm.amd64/src/org/graalvm/compiler/asm/amd64/AMD64Assembler.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.asm.amd64/src/org/graalvm/compiler/asm/amd64/AMD64Assembler.java index 56090fa7af3..04e0e22182d 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.asm.amd64/src/org/graalvm/compiler/asm/amd64/AMD64Assembler.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.asm.amd64/src/org/graalvm/compiler/asm/amd64/AMD64Assembler.java @@ -22,10 +22,14 @@ */ package org.graalvm.compiler.asm.amd64; -import static org.graalvm.compiler.core.common.NumUtil.isByte; -import static org.graalvm.compiler.core.common.NumUtil.isInt; -import static org.graalvm.compiler.core.common.NumUtil.isShiftCount; -import static org.graalvm.compiler.core.common.NumUtil.isUByte; +import static jdk.vm.ci.amd64.AMD64.CPU; +import static jdk.vm.ci.amd64.AMD64.XMM; +import static jdk.vm.ci.amd64.AMD64.r12; +import static jdk.vm.ci.amd64.AMD64.r13; +import static jdk.vm.ci.amd64.AMD64.rbp; +import static jdk.vm.ci.amd64.AMD64.rip; +import static jdk.vm.ci.amd64.AMD64.rsp; +import static jdk.vm.ci.code.MemoryBarriers.STORE_LOAD; import static org.graalvm.compiler.asm.amd64.AMD64AsmOptions.UseAddressNop; import static org.graalvm.compiler.asm.amd64.AMD64AsmOptions.UseNormalNop; import static org.graalvm.compiler.asm.amd64.AMD64Assembler.AMD64BinaryArithmetic.ADD; @@ -47,25 +51,24 @@ import static org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize.QWORD; import static org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize.SD; import static org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize.SS; import static org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize.WORD; -import static jdk.vm.ci.amd64.AMD64.CPU; -import static jdk.vm.ci.amd64.AMD64.XMM; -import static jdk.vm.ci.amd64.AMD64.r12; -import static jdk.vm.ci.amd64.AMD64.r13; -import static jdk.vm.ci.amd64.AMD64.rbp; -import static jdk.vm.ci.amd64.AMD64.rip; -import static jdk.vm.ci.amd64.AMD64.rsp; -import static jdk.vm.ci.code.MemoryBarriers.STORE_LOAD; +import static org.graalvm.compiler.core.common.NumUtil.isByte; +import static org.graalvm.compiler.core.common.NumUtil.isInt; +import static org.graalvm.compiler.core.common.NumUtil.isShiftCount; +import static org.graalvm.compiler.core.common.NumUtil.isUByte; import org.graalvm.compiler.asm.Assembler; import org.graalvm.compiler.asm.Label; -import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.asm.amd64.AMD64Address.Scale; +import org.graalvm.compiler.core.common.NumUtil; +import org.graalvm.compiler.debug.GraalError; import jdk.vm.ci.amd64.AMD64; import jdk.vm.ci.amd64.AMD64.CPUFeature; +import jdk.vm.ci.amd64.AMD64Kind; import jdk.vm.ci.code.Register; import jdk.vm.ci.code.Register.RegisterCategory; import jdk.vm.ci.code.TargetDescription; +import jdk.vm.ci.meta.PlatformKind; /** * This class implements an assembler that can encode most X86 instructions. @@ -225,7 +228,7 @@ public class AMD64Assembler extends Assembler { * The x86 operand sizes. */ public enum OperandSize { - BYTE(1) { + BYTE(1, AMD64Kind.BYTE) { @Override protected void emitImmediate(AMD64Assembler asm, int imm) { assert imm == (byte) imm; @@ -238,7 +241,7 @@ public class AMD64Assembler extends Assembler { } }, - WORD(2, 0x66) { + WORD(2, AMD64Kind.WORD, 0x66) { @Override protected void emitImmediate(AMD64Assembler asm, int imm) { assert imm == (short) imm; @@ -251,7 +254,7 @@ public class AMD64Assembler extends Assembler { } }, - DWORD(4) { + DWORD(4, AMD64Kind.DWORD) { @Override protected void emitImmediate(AMD64Assembler asm, int imm) { asm.emitInt(imm); @@ -263,7 +266,7 @@ public class AMD64Assembler extends Assembler { } }, - QWORD(8) { + QWORD(8, AMD64Kind.QWORD) { @Override protected void emitImmediate(AMD64Assembler asm, int imm) { asm.emitInt(imm); @@ -275,34 +278,35 @@ public class AMD64Assembler extends Assembler { } }, - SS(4, 0xF3, true), + SS(4, AMD64Kind.SINGLE, 0xF3, true), - SD(8, 0xF2, true), + SD(8, AMD64Kind.DOUBLE, 0xF2, true), - PS(16, true), + PS(16, AMD64Kind.V128_SINGLE, true), - PD(16, 0x66, true); + PD(16, AMD64Kind.V128_DOUBLE, 0x66, true); private final int sizePrefix; - private final int bytes; private final boolean xmm; + private final AMD64Kind kind; - OperandSize(int bytes) { - this(bytes, 0); + OperandSize(int bytes, AMD64Kind kind) { + this(bytes, kind, 0); } - OperandSize(int bytes, int sizePrefix) { - this(bytes, sizePrefix, false); + OperandSize(int bytes, AMD64Kind kind, int sizePrefix) { + this(bytes, kind, sizePrefix, false); } - OperandSize(int bytes, boolean xmm) { - this(bytes, 0, xmm); + OperandSize(int bytes, AMD64Kind kind, boolean xmm) { + this(bytes, kind, 0, xmm); } - OperandSize(int bytes, int sizePrefix, boolean xmm) { + OperandSize(int bytes, AMD64Kind kind, int sizePrefix, boolean xmm) { this.sizePrefix = sizePrefix; this.bytes = bytes; + this.kind = kind; this.xmm = xmm; } @@ -314,6 +318,19 @@ public class AMD64Assembler extends Assembler { return xmm; } + public AMD64Kind getKind() { + return kind; + } + + public static OperandSize get(PlatformKind kind) { + for (OperandSize operandSize : OperandSize.values()) { + if (operandSize.kind.equals(kind)) { + return operandSize; + } + } + throw GraalError.shouldNotReachHere("Unexpected kind: " + kind.toString()); + } + /** * Emit an immediate of this size. Note that immediate {@link #QWORD} operands are encoded * as sign-extended 32-bit values. @@ -2230,6 +2247,14 @@ public class AMD64Assembler extends Assembler { emitOperandHelper(dst, src, 0); } + public final void movzbl(Register dst, Register src) { + AMD64RMOp.MOVZXB.emit(this, OperandSize.DWORD, dst, src); + } + + public final void movzbq(Register dst, Register src) { + AMD64RMOp.MOVZXB.emit(this, OperandSize.QWORD, dst, src); + } + public final void movzwl(Register dst, AMD64Address src) { prefix(src, dst); emitByte(0x0F); @@ -3198,6 +3223,13 @@ public class AMD64Assembler extends Assembler { emitByte(0xC0 | encode); } + public final void setb(ConditionFlag cc, Register dst) { + int encode = prefixAndEncode(dst.encoding, true); + emitByte(0x0F); + emitByte(0x90 | cc.getValue()); + emitByte(0xC0 | encode); + } + public final void cmovq(ConditionFlag cc, Register dst, AMD64Address src) { prefixq(src, dst); emitByte(0x0F); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.asm.amd64/src/org/graalvm/compiler/asm/amd64/AMD64MacroAssembler.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.asm.amd64/src/org/graalvm/compiler/asm/amd64/AMD64MacroAssembler.java index 36995bf6853..d91ddae485a 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.asm.amd64/src/org/graalvm/compiler/asm/amd64/AMD64MacroAssembler.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.asm.amd64/src/org/graalvm/compiler/asm/amd64/AMD64MacroAssembler.java @@ -31,8 +31,8 @@ import static org.graalvm.compiler.asm.amd64.AMD64AsmOptions.UseXmmLoadAndClearU import static org.graalvm.compiler.asm.amd64.AMD64AsmOptions.UseXmmRegToRegMoveAll; import org.graalvm.compiler.asm.Label; -import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.asm.amd64.AMD64Address.Scale; +import org.graalvm.compiler.core.common.NumUtil; import jdk.vm.ci.amd64.AMD64; import jdk.vm.ci.amd64.AMD64Kind; @@ -281,6 +281,16 @@ public class AMD64MacroAssembler extends AMD64Assembler { } + public final void setl(ConditionFlag cc, Register dst) { + setb(cc, dst); + movzbl(dst, dst); + } + + public final void setq(ConditionFlag cc, Register dst) { + setb(cc, dst); + movzbq(dst, dst); + } + public final void flog(Register dest, Register value, boolean base10) { if (base10) { fldlg2(); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64.test/src/org/graalvm/compiler/core/amd64/test/AMD64AddressLoweringTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64.test/src/org/graalvm/compiler/core/amd64/test/AMD64AddressLoweringTest.java new file mode 100644 index 00000000000..99fc5192292 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64.test/src/org/graalvm/compiler/core/amd64/test/AMD64AddressLoweringTest.java @@ -0,0 +1,126 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.core.amd64.test; + +import static org.junit.Assume.assumeTrue; + +import org.graalvm.compiler.asm.amd64.AMD64Address.Scale; +import org.graalvm.compiler.core.amd64.AMD64AddressLowering; +import org.graalvm.compiler.core.amd64.AMD64AddressNode; +import org.graalvm.compiler.core.test.GraalCompilerTest; +import org.graalvm.compiler.nodes.ConstantNode; +import org.graalvm.compiler.nodes.StructuredGraph; +import org.graalvm.compiler.nodes.ValueNode; +import org.graalvm.compiler.nodes.calc.AddNode; +import org.graalvm.compiler.nodes.calc.LeftShiftNode; +import org.graalvm.compiler.nodes.calc.NegateNode; +import org.graalvm.compiler.nodes.memory.address.AddressNode; +import org.junit.Assert; +import org.junit.Before; +import org.junit.Test; + +import jdk.vm.ci.amd64.AMD64; + +public class AMD64AddressLoweringTest extends GraalCompilerTest { + + private StructuredGraph graph; + private AMD64AddressLowering lowering; + + @Before + public void checkAMD64() { + assumeTrue("skipping AMD64 specific test", getTarget().arch instanceof AMD64); + graph = new StructuredGraph.Builder(getInitialOptions(), getDebugContext()).build(); + lowering = new AMD64AddressLowering(); + } + + @Test + public void convertBaseAndIndexToDisplacement() { + ValueNode base = graph.unique(const64(1000)); + ValueNode index = graph.unique(const64(10)); + AddressNode result = lowering.lower(base, index); + assertAddress(result, null, null, Scale.Times1, 1010); + } + + @Test + public void convertBaseToDisplacement() { + ValueNode constantAddress = graph.addOrUniqueWithInputs(const64(1000)); + AddressNode result = lowering.lower(constantAddress, null); + assertAddress(result, null, null, Scale.Times1, 1000); + } + + @Test + public void convertBaseAndShiftedIndexToDisplacement() { + ValueNode base = graph.addOrUniqueWithInputs(const64(1000)); + ValueNode index = graph.addOrUniqueWithInputs(new LeftShiftNode(const64(10), const32(1))); + AddressNode result = lowering.lower(base, index); + assertAddress(result, null, null, Scale.Times2, 1020); + } + + @Test + public void convertBaseAndNegatedShiftedIndexToDisplacement() { + ValueNode base = graph.addOrUniqueWithInputs(const64(1000)); + ValueNode index = graph.addOrUniqueWithInputs(new NegateNode(new LeftShiftNode(const64(10), const32(2)))); + AddressNode result = lowering.lower(base, index); + assertAddress(result, null, null, Scale.Times4, 960); + } + + @Test + public void convertNegatedBaseAndNegatedShiftedIndexToDisplacement() { + ValueNode base = graph.addOrUniqueWithInputs(new NegateNode(const64(1000))); + ValueNode index = graph.addOrUniqueWithInputs(new NegateNode(new LeftShiftNode(const64(10), const32(2)))); + AddressNode result = lowering.lower(base, index); + assertAddress(result, null, null, Scale.Times4, -1040); + } + + @Test + public void convertNegatedShiftedBaseAndNegatedIndexToDisplacement() { + ValueNode base = graph.addOrUniqueWithInputs(new NegateNode(new LeftShiftNode(const64(10), const32(2)))); + ValueNode index = graph.addOrUniqueWithInputs(new NegateNode(const64(1000))); + AddressNode result = lowering.lower(base, index); + assertAddress(result, null, null, Scale.Times4, -1040); + } + + @Test + public void convertTwoLevelsOfNegatedShiftedBaseAndNegatedIndexToDisplacement() { + ValueNode base = graph.addOrUniqueWithInputs(new NegateNode(new LeftShiftNode(new NegateNode(new LeftShiftNode(const64(500), const32(1))), const32(1)))); + ValueNode index = graph.addOrUniqueWithInputs(new NegateNode(new AddNode(new NegateNode(const64(13)), const64(3)))); + AddressNode result = lowering.lower(base, index); + assertAddress(result, null, null, Scale.Times4, 2010); + } + + private static ConstantNode const64(long value) { + return ConstantNode.forIntegerBits(Long.SIZE, value); + } + + private static ConstantNode const32(long value) { + return ConstantNode.forIntegerBits(Integer.SIZE, value); + } + + private static void assertAddress(AddressNode actual, ValueNode expectedBase, ValueNode expectedIndex, Scale expectedScale, int expectedDisplacement) { + AMD64AddressNode address = (AMD64AddressNode) actual; + Assert.assertEquals(expectedBase, address.getBase()); + Assert.assertEquals(expectedIndex, address.getIndex()); + Assert.assertEquals(expectedScale, address.getScale()); + Assert.assertEquals(expectedDisplacement, address.getDisplacement()); + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64AddressLowering.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64AddressLowering.java index 6a827f8c1fa..63e535bc853 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64AddressLowering.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64AddressLowering.java @@ -23,21 +23,22 @@ package org.graalvm.compiler.core.amd64; -import jdk.vm.ci.meta.JavaConstant; - -import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.asm.amd64.AMD64Address.Scale; +import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.core.common.type.IntegerStamp; import org.graalvm.compiler.debug.DebugContext; +import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.ValueNode; import org.graalvm.compiler.nodes.calc.AddNode; import org.graalvm.compiler.nodes.calc.LeftShiftNode; +import org.graalvm.compiler.nodes.calc.NegateNode; import org.graalvm.compiler.nodes.calc.ZeroExtendNode; import org.graalvm.compiler.nodes.memory.address.AddressNode; import org.graalvm.compiler.phases.common.AddressLoweringPhase.AddressLowering; -public class AMD64AddressLowering extends AddressLowering { +import jdk.vm.ci.meta.JavaConstant; +public class AMD64AddressLowering extends AddressLowering { @Override public AddressNode lower(ValueNode address) { return lower(address, null); @@ -46,24 +47,37 @@ public class AMD64AddressLowering extends AddressLowering { @Override public AddressNode lower(ValueNode base, ValueNode offset) { AMD64AddressNode ret = new AMD64AddressNode(base, offset); + StructuredGraph graph = base.graph(); + boolean changed; do { - changed = improve(base.getDebug(), ret); + changed = improve(graph, base.getDebug(), ret, false, false); } while (changed); - return base.graph().unique(ret); + + return graph.unique(ret); } /** - * @param debug + * Tries to optimize addresses so that they match the AMD64-specific addressing mode better + * (base + index * scale + displacement). + * + * @param graph the current graph + * @param debug the current debug context + * @param ret the address that should be optimized + * @param isBaseNegated determines if the address base is negated. if so, all values that are + * extracted from the base will be negated as well + * @param isIndexNegated determines if the index is negated. if so, all values that are + * extracted from the index will be negated as well + * @return true if the address was modified */ - protected boolean improve(DebugContext debug, AMD64AddressNode ret) { - ValueNode newBase = improveInput(ret, ret.getBase(), 0); + protected boolean improve(StructuredGraph graph, DebugContext debug, AMD64AddressNode ret, boolean isBaseNegated, boolean isIndexNegated) { + ValueNode newBase = improveInput(ret, ret.getBase(), 0, isBaseNegated); if (newBase != ret.getBase()) { ret.setBase(newBase); return true; } - ValueNode newIdx = improveInput(ret, ret.getIndex(), ret.getScale().log2); + ValueNode newIdx = improveInput(ret, ret.getIndex(), ret.getScale().log2, isIndexNegated); if (newIdx != ret.getIndex()) { ret.setIndex(newIdx); return true; @@ -83,55 +97,122 @@ public class AMD64AddressLowering extends AddressLowering { } if (ret.getScale() == Scale.Times1) { - if (ret.getBase() == null || ret.getIndex() == null) { - if (ret.getBase() instanceof AddNode) { - AddNode add = (AddNode) ret.getBase(); - ret.setBase(add.getX()); - ret.setIndex(add.getY()); - return true; - } else if (ret.getIndex() instanceof AddNode) { - AddNode add = (AddNode) ret.getIndex(); - ret.setBase(add.getX()); - ret.setIndex(add.getY()); - return true; - } + if (ret.getIndex() == null && ret.getBase() instanceof AddNode) { + AddNode add = (AddNode) ret.getBase(); + ret.setBase(add.getX()); + ret.setIndex(considerNegation(graph, add.getY(), isBaseNegated)); + return true; + } else if (ret.getBase() == null && ret.getIndex() instanceof AddNode) { + AddNode add = (AddNode) ret.getIndex(); + ret.setBase(considerNegation(graph, add.getX(), isIndexNegated)); + ret.setIndex(add.getY()); + return true; } if (ret.getBase() instanceof LeftShiftNode && !(ret.getIndex() instanceof LeftShiftNode)) { ValueNode tmp = ret.getBase(); - ret.setBase(ret.getIndex()); - ret.setIndex(tmp); + ret.setBase(considerNegation(graph, ret.getIndex(), isIndexNegated != isBaseNegated)); + ret.setIndex(considerNegation(graph, tmp, isIndexNegated != isBaseNegated)); return true; } } + return improveNegation(graph, debug, ret, isBaseNegated, isIndexNegated); + } + + private boolean improveNegation(StructuredGraph graph, DebugContext debug, AMD64AddressNode ret, boolean originalBaseNegated, boolean originalIndexNegated) { + boolean baseNegated = originalBaseNegated; + boolean indexNegated = originalIndexNegated; + + ValueNode originalBase = ret.getBase(); + ValueNode originalIndex = ret.getIndex(); + + if (ret.getBase() instanceof NegateNode) { + NegateNode negate = (NegateNode) ret.getBase(); + ret.setBase(negate.getValue()); + baseNegated = !baseNegated; + } + + if (ret.getIndex() instanceof NegateNode) { + NegateNode negate = (NegateNode) ret.getIndex(); + ret.setIndex(negate.getValue()); + indexNegated = !indexNegated; + } + + if (baseNegated != originalBaseNegated || indexNegated != originalIndexNegated) { + ValueNode base = ret.getBase(); + ValueNode index = ret.getIndex(); + + boolean improved = improve(graph, debug, ret, baseNegated, indexNegated); + if (baseNegated != originalBaseNegated) { + if (base == ret.getBase()) { + ret.setBase(originalBase); + } else if (ret.getBase() != null) { + ret.setBase(graph.maybeAddOrUnique(NegateNode.create(ret.getBase()))); + } + } + + if (indexNegated != originalIndexNegated) { + if (index == ret.getIndex()) { + ret.setIndex(originalIndex); + } else if (ret.getIndex() != null) { + ret.setIndex(graph.maybeAddOrUnique(NegateNode.create(ret.getIndex()))); + } + } + return improved; + } else { + assert ret.getBase() == originalBase && ret.getIndex() == originalIndex; + } return false; } - private static ValueNode improveInput(AMD64AddressNode address, ValueNode node, int shift) { + private static ValueNode considerNegation(StructuredGraph graph, ValueNode value, boolean negate) { + if (negate && value != null) { + return graph.maybeAddOrUnique(NegateNode.create(value)); + } + return value; + } + + private ValueNode improveInput(AMD64AddressNode address, ValueNode node, int shift, boolean negateExtractedDisplacement) { if (node == null) { return null; } JavaConstant c = node.asJavaConstant(); if (c != null) { - return improveConstDisp(address, node, c, null, shift); + return improveConstDisp(address, node, c, null, shift, negateExtractedDisplacement); } else { - if (node.stamp() instanceof IntegerStamp && ((IntegerStamp) node.stamp()).getBits() == 64) { - if (node instanceof ZeroExtendNode) { - if (((ZeroExtendNode) node).getInputBits() == 32) { - /* - * We can just swallow a zero-extend from 32 bit to 64 bit because the upper - * half of the register will always be zero. - */ - return ((ZeroExtendNode) node).getValue(); + if (node.stamp() instanceof IntegerStamp) { + if (node instanceof ZeroExtendNode && (((ZeroExtendNode) node).getInputBits() == 32)) { + /* + * we can't just swallow all zero-extends as we might encounter something like + * the following: ZeroExtend(Add(negativeValue, positiveValue)). + * + * if we swallow the zero-extend in this case and subsequently optimize the add, + * we might end up with a negative value that has less than 64 bits in base or + * index. such a value would require sign extension instead of zero-extension + * but the backend can only do zero-extension. if we ever want to optimize that + * further, we would also need to be careful about over-/underflows. + * + * furthermore, we also can't swallow zero-extends with less than 32 bits as + * most of these values are immediately sign-extended to 32 bit by the backend + * (therefore, the subsequent implicit zero-extension to 64 bit won't do what we + * expect). + */ + ValueNode value = ((ZeroExtendNode) node).getValue(); + if (!mightBeOptimized(value)) { + // if the value is not optimized further by the address lowering, then we + // can safely rely on the backend doing the implicitly zero-extension. + return value; } - } else if (node instanceof AddNode) { + } + + if (node instanceof AddNode) { AddNode add = (AddNode) node; if (add.getX().isConstant()) { - return improveConstDisp(address, node, add.getX().asJavaConstant(), add.getY(), shift); + return improveConstDisp(address, node, add.getX().asJavaConstant(), add.getY(), shift, negateExtractedDisplacement); } else if (add.getY().isConstant()) { - return improveConstDisp(address, node, add.getY().asJavaConstant(), add.getX(), shift); + return improveConstDisp(address, node, add.getY().asJavaConstant(), add.getX(), shift, negateExtractedDisplacement); } } } @@ -140,15 +221,30 @@ public class AMD64AddressLowering extends AddressLowering { return node; } - private static ValueNode improveConstDisp(AMD64AddressNode address, ValueNode original, JavaConstant c, ValueNode other, int shift) { + /** + * This method returns true for all nodes that might be optimized by the address lowering. + */ + protected boolean mightBeOptimized(ValueNode value) { + return value instanceof AddNode || value instanceof LeftShiftNode || value instanceof NegateNode || value instanceof ZeroExtendNode; + } + + private static ValueNode improveConstDisp(AMD64AddressNode address, ValueNode original, JavaConstant c, ValueNode other, int shift, boolean negateExtractedDisplacement) { if (c.getJavaKind().isNumericInteger()) { - long disp = address.getDisplacement(); - disp += c.asLong() << shift; - if (NumUtil.isInt(disp)) { - address.setDisplacement((int) disp); + long delta = c.asLong() << shift; + if (updateDisplacement(address, delta, negateExtractedDisplacement)) { return other; } } return original; } + + protected static boolean updateDisplacement(AMD64AddressNode address, long displacementDelta, boolean negateDelta) { + long sign = negateDelta ? -1 : 1; + long disp = address.getDisplacement() + displacementDelta * sign; + if (NumUtil.isInt(disp)) { + address.setDisplacement((int) disp); + return true; + } + return false; + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64LIRGenerator.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64LIRGenerator.java index e207f830464..6e7258ae62f 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64LIRGenerator.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64LIRGenerator.java @@ -33,15 +33,17 @@ import static org.graalvm.compiler.core.common.GraalOptions.GeneratePIC; import static org.graalvm.compiler.lir.LIRValueUtil.asConstantValue; import static org.graalvm.compiler.lir.LIRValueUtil.asJavaConstant; import static org.graalvm.compiler.lir.LIRValueUtil.isConstantValue; +import static org.graalvm.compiler.lir.LIRValueUtil.isIntConstant; import static org.graalvm.compiler.lir.LIRValueUtil.isJavaConstant; -import org.graalvm.compiler.core.common.NumUtil; +import org.graalvm.compiler.asm.amd64.AMD64Assembler.AMD64BinaryArithmetic; import org.graalvm.compiler.asm.amd64.AMD64Assembler.AMD64MIOp; import org.graalvm.compiler.asm.amd64.AMD64Assembler.AMD64RMOp; import org.graalvm.compiler.asm.amd64.AMD64Assembler.ConditionFlag; import org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize; import org.graalvm.compiler.asm.amd64.AMD64Assembler.SSEOp; import org.graalvm.compiler.core.common.LIRKind; +import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.core.common.calc.Condition; import org.graalvm.compiler.core.common.spi.ForeignCallLinkage; import org.graalvm.compiler.core.common.spi.LIRKindTool; @@ -58,13 +60,17 @@ import org.graalvm.compiler.lir.Variable; import org.graalvm.compiler.lir.amd64.AMD64AddressValue; import org.graalvm.compiler.lir.amd64.AMD64ArithmeticLIRGeneratorTool; import org.graalvm.compiler.lir.amd64.AMD64ArrayEqualsOp; +import org.graalvm.compiler.lir.amd64.AMD64Binary; import org.graalvm.compiler.lir.amd64.AMD64BinaryConsumer; import org.graalvm.compiler.lir.amd64.AMD64ByteSwapOp; import org.graalvm.compiler.lir.amd64.AMD64Call; +import org.graalvm.compiler.lir.amd64.AMD64ControlFlow; import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.BranchOp; import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.CondMoveOp; +import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.CondSetOp; import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.FloatBranchOp; import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.FloatCondMoveOp; +import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.FloatCondSetOp; import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.ReturnOp; import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.StrategySwitchOp; import org.graalvm.compiler.lir.amd64.AMD64ControlFlow.TableSwitchOp; @@ -257,8 +263,7 @@ public abstract class AMD64LIRGenerator extends LIRGenerator { @Override public void emitCompareBranch(PlatformKind cmpKind, Value left, Value right, Condition cond, boolean unorderedIsTrue, LabelRef trueLabel, LabelRef falseLabel, double trueLabelProbability) { - boolean mirrored = emitCompare(cmpKind, left, right); - Condition finalCondition = mirrored ? cond.mirror() : cond; + Condition finalCondition = emitCompare(cmpKind, left, right, cond); if (cmpKind == AMD64Kind.SINGLE || cmpKind == AMD64Kind.DOUBLE) { append(new FloatBranchOp(finalCondition, unorderedIsTrue, trueLabel, falseLabel, trueLabelProbability)); } else { @@ -290,14 +295,60 @@ public abstract class AMD64LIRGenerator extends LIRGenerator { @Override public Variable emitConditionalMove(PlatformKind cmpKind, Value left, Value right, Condition cond, boolean unorderedIsTrue, Value trueValue, Value falseValue) { - boolean mirrored = emitCompare(cmpKind, left, right); - Condition finalCondition = mirrored ? cond.mirror() : cond; + boolean isFloatComparison = cmpKind == AMD64Kind.SINGLE || cmpKind == AMD64Kind.DOUBLE; - Variable result = newVariable(trueValue.getValueKind()); - if (cmpKind == AMD64Kind.SINGLE || cmpKind == AMD64Kind.DOUBLE) { - append(new FloatCondMoveOp(result, finalCondition, unorderedIsTrue, load(trueValue), load(falseValue))); + Condition finalCondition = cond; + Value finalTrueValue = trueValue; + Value finalFalseValue = falseValue; + if (isFloatComparison) { + // eliminate the parity check in case of a float comparison + Value finalLeft = left; + Value finalRight = right; + if (unorderedIsTrue != AMD64ControlFlow.trueOnUnordered(finalCondition)) { + if (unorderedIsTrue == AMD64ControlFlow.trueOnUnordered(finalCondition.mirror())) { + finalCondition = finalCondition.mirror(); + finalLeft = right; + finalRight = left; + } else if (finalCondition != Condition.EQ && finalCondition != Condition.NE) { + // negating EQ and NE does not make any sense as we would need to negate + // unorderedIsTrue as well (otherwise, we would no longer fulfill the Java + // NaN semantics) + assert unorderedIsTrue == AMD64ControlFlow.trueOnUnordered(finalCondition.negate()); + finalCondition = finalCondition.negate(); + finalTrueValue = falseValue; + finalFalseValue = trueValue; + } + } + emitRawCompare(cmpKind, finalLeft, finalRight); } else { - append(new CondMoveOp(result, finalCondition, load(trueValue), loadNonConst(falseValue))); + finalCondition = emitCompare(cmpKind, left, right, cond); + } + + boolean isParityCheckNecessary = isFloatComparison && unorderedIsTrue != AMD64ControlFlow.trueOnUnordered(finalCondition); + Variable result = newVariable(finalTrueValue.getValueKind()); + if (!isParityCheckNecessary && isIntConstant(finalTrueValue, 1) && isIntConstant(finalFalseValue, 0)) { + if (isFloatComparison) { + append(new FloatCondSetOp(result, finalCondition)); + } else { + append(new CondSetOp(result, finalCondition)); + } + } else if (!isParityCheckNecessary && isIntConstant(finalTrueValue, 0) && isIntConstant(finalFalseValue, 1)) { + if (isFloatComparison) { + if (unorderedIsTrue == AMD64ControlFlow.trueOnUnordered(finalCondition.negate())) { + append(new FloatCondSetOp(result, finalCondition.negate())); + } else { + append(new FloatCondSetOp(result, finalCondition)); + Variable negatedResult = newVariable(result.getValueKind()); + append(new AMD64Binary.ConstOp(AMD64BinaryArithmetic.XOR, OperandSize.get(result.getPlatformKind()), negatedResult, result, 1)); + result = negatedResult; + } + } else { + append(new CondSetOp(result, finalCondition.negate())); + } + } else if (isFloatComparison) { + append(new FloatCondMoveOp(result, finalCondition, unorderedIsTrue, load(finalTrueValue), load(finalFalseValue))); + } else { + append(new CondMoveOp(result, finalCondition, load(finalTrueValue), loadNonConst(finalFalseValue))); } return result; } @@ -394,23 +445,21 @@ public abstract class AMD64LIRGenerator extends LIRGenerator { * * @param a the left operand of the comparison * @param b the right operand of the comparison + * @param cond the condition of the comparison * @return true if the left and right operands were switched, false otherwise */ - private boolean emitCompare(PlatformKind cmpKind, Value a, Value b) { - Variable left; - Value right; - boolean mirrored; + private Condition emitCompare(PlatformKind cmpKind, Value a, Value b, Condition cond) { if (LIRValueUtil.isVariable(b)) { - left = load(b); - right = loadNonConst(a); - mirrored = true; + emitRawCompare(cmpKind, b, a); + return cond.mirror(); } else { - left = load(a); - right = loadNonConst(b); - mirrored = false; + emitRawCompare(cmpKind, a, b); + return cond; } - ((AMD64ArithmeticLIRGeneratorTool) arithmeticLIRGen).emitCompareOp((AMD64Kind) cmpKind, left, right); - return mirrored; + } + + private void emitRawCompare(PlatformKind cmpKind, Value left, Value right) { + ((AMD64ArithmeticLIRGeneratorTool) arithmeticLIRGen).emitCompareOp((AMD64Kind) cmpKind, load(left), loadNonConst(right)); } @Override diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64LIRKindTool.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64LIRKindTool.java index a2ff74c5f66..da1d919b0ea 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64LIRKindTool.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.amd64/src/org/graalvm/compiler/core/amd64/AMD64LIRKindTool.java @@ -28,7 +28,7 @@ import org.graalvm.compiler.debug.GraalError; import jdk.vm.ci.amd64.AMD64Kind; -public class AMD64LIRKindTool implements LIRKindTool { +public abstract class AMD64LIRKindTool implements LIRKindTool { @Override public LIRKind getIntegerKind(int bits) { @@ -67,12 +67,8 @@ public class AMD64LIRKindTool implements LIRKindTool { } @Override - public LIRKind getNarrowOopKind() { - return LIRKind.reference(AMD64Kind.DWORD); - } + public abstract LIRKind getNarrowOopKind(); @Override - public LIRKind getNarrowPointerKind() { - return LIRKind.value(AMD64Kind.DWORD); - } + public abstract LIRKind getNarrowPointerKind(); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/CompressEncoding.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/CompressEncoding.java index 4eac53452cb..5356c139188 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/CompressEncoding.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/CompressEncoding.java @@ -34,14 +34,6 @@ public final class CompressEncoding { this.shift = shift; } - public int compress(long ptr) { - if (ptr == 0L) { - return 0; - } else { - return (int) ((ptr - base) >>> shift); - } - } - public boolean hasBase() { return base != 0; } @@ -58,14 +50,6 @@ public final class CompressEncoding { return shift; } - public long uncompress(int ptr) { - if (ptr == 0) { - return 0L; - } else { - return ((ptr & 0xFFFFFFFFL) << shift) + base; - } - } - @Override public String toString() { return "base: " + base + " shift: " + shift; @@ -85,8 +69,7 @@ public final class CompressEncoding { if (obj instanceof CompressEncoding) { CompressEncoding other = (CompressEncoding) obj; return base == other.base && shift == other.shift; - } else { - return false; } + return false; } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/calc/FloatConvert.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/calc/FloatConvert.java index a7f430010dd..cc0316f4fa2 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/calc/FloatConvert.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/calc/FloatConvert.java @@ -25,21 +25,23 @@ package org.graalvm.compiler.core.common.calc; import org.graalvm.compiler.debug.GraalError; public enum FloatConvert { - F2I(FloatConvertCategory.FloatingPointToInteger), - D2I(FloatConvertCategory.FloatingPointToInteger), - F2L(FloatConvertCategory.FloatingPointToInteger), - D2L(FloatConvertCategory.FloatingPointToInteger), - I2F(FloatConvertCategory.IntegerToFloatingPoint), - L2F(FloatConvertCategory.IntegerToFloatingPoint), - D2F(FloatConvertCategory.FloatingPointToFloatingPoint), - I2D(FloatConvertCategory.IntegerToFloatingPoint), - L2D(FloatConvertCategory.IntegerToFloatingPoint), - F2D(FloatConvertCategory.FloatingPointToFloatingPoint); + F2I(FloatConvertCategory.FloatingPointToInteger, 32), + D2I(FloatConvertCategory.FloatingPointToInteger, 64), + F2L(FloatConvertCategory.FloatingPointToInteger, 32), + D2L(FloatConvertCategory.FloatingPointToInteger, 64), + I2F(FloatConvertCategory.IntegerToFloatingPoint, 32), + L2F(FloatConvertCategory.IntegerToFloatingPoint, 64), + D2F(FloatConvertCategory.FloatingPointToFloatingPoint, 64), + I2D(FloatConvertCategory.IntegerToFloatingPoint, 32), + L2D(FloatConvertCategory.IntegerToFloatingPoint, 64), + F2D(FloatConvertCategory.FloatingPointToFloatingPoint, 32); - private FloatConvertCategory category; + private final FloatConvertCategory category; + private final int inputBits; - FloatConvert(FloatConvertCategory category) { + FloatConvert(FloatConvertCategory category, int inputBits) { this.category = category; + this.inputBits = inputBits; } public FloatConvertCategory getCategory() { @@ -72,4 +74,8 @@ public enum FloatConvert { throw GraalError.shouldNotReachHere(); } } + + public int getInputBits() { + return inputBits; + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/cfg/Loop.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/cfg/Loop.java index af3cb75d9b6..21eed6b98a0 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/cfg/Loop.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/cfg/Loop.java @@ -41,7 +41,6 @@ public abstract class Loop> { this.parent = parent; if (parent != null) { this.depth = parent.getDepth() + 1; - parent.getChildren().add(this); } else { this.depth = 1; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/ArithmeticOpTable.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/ArithmeticOpTable.java index db33cf533d7..fb8c0b90029 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/ArithmeticOpTable.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/ArithmeticOpTable.java @@ -96,6 +96,22 @@ public final class ArithmeticOpTable { } } + public BinaryOp[] getBinaryOps() { + return new BinaryOp[]{add, sub, mul, mulHigh, umulHigh, div, rem, and, or, xor}; + } + + public UnaryOp[] getUnaryOps() { + return new UnaryOp[]{neg, not, abs, sqrt}; + } + + public ShiftOp[] getShiftOps() { + return new ShiftOp[]{shl, shr, ushr}; + } + + public IntegerConvertOp[] getIntegerConvertOps() { + return new IntegerConvertOp[]{zeroExtend, signExtend, narrow}; + } + public static final ArithmeticOpTable EMPTY = new ArithmeticOpTable(null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null); public interface ArithmeticOpWrapper { @@ -562,7 +578,10 @@ public final class ArithmeticOpTable { } /** - * Apply the operation to two {@linkplain Constant Constants}. + * Applies this operation to {@code a} and {@code b}. + * + * @return the result of applying this operation or {@code null} if applying it would raise + * an exception (e.g., {@link ArithmeticException} for dividing by 0) */ public abstract Constant foldConstant(Constant a, Constant b); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/FloatStamp.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/FloatStamp.java index 2aa19dbd7a4..b9e057ef153 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/FloatStamp.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/FloatStamp.java @@ -291,7 +291,7 @@ public class FloatStamp extends PrimitiveStamp { @Override public JavaConstant asConstant() { - if (nonNaN && Double.compare(lowerBound, upperBound) == 0) { + if (isConstant()) { switch (getBits()) { case 32: return JavaConstant.forFloat((float) lowerBound); @@ -302,6 +302,68 @@ public class FloatStamp extends PrimitiveStamp { return null; } + private boolean isConstant() { + /* + * There are many forms of NaNs and any operations on them can silently convert them into + * the canonical NaN. + */ + return (Double.compare(lowerBound, upperBound) == 0 && nonNaN); + } + + private static FloatStamp stampForConstant(Constant constant) { + FloatStamp result; + PrimitiveConstant value = (PrimitiveConstant) constant; + switch (value.getJavaKind()) { + case Float: + if (Float.isNaN(value.asFloat())) { + result = new FloatStamp(32, Double.NaN, Double.NaN, false); + } else { + result = new FloatStamp(32, value.asFloat(), value.asFloat(), !Float.isNaN(value.asFloat())); + } + break; + case Double: + if (Double.isNaN(value.asDouble())) { + result = new FloatStamp(64, Double.NaN, Double.NaN, false); + } else { + result = new FloatStamp(64, value.asDouble(), value.asDouble(), !Double.isNaN(value.asDouble())); + } + break; + default: + throw GraalError.shouldNotReachHere(); + } + if (result.isConstant()) { + return result; + } + return null; + } + + private static Stamp maybeFoldConstant(UnaryOp op, FloatStamp stamp) { + if (stamp.isConstant()) { + JavaConstant constant = stamp.asConstant(); + Constant folded = op.foldConstant(constant); + if (folded != null) { + return FloatStamp.stampForConstant(folded); + } + } + return null; + } + + private static Stamp maybeFoldConstant(BinaryOp op, FloatStamp stamp1, FloatStamp stamp2) { + if (stamp1.isConstant() && stamp2.isConstant()) { + JavaConstant constant1 = stamp1.asConstant(); + JavaConstant constant2 = stamp2.asConstant(); + Constant folded = op.foldConstant(constant1, constant2); + if (folded != null) { + FloatStamp stamp = stampForConstant(folded); + if (stamp != null && stamp.isConstant()) { + assert stamp.asConstant().equals(folded); + return stamp; + } + } + } + return null; + } + public static final ArithmeticOpTable OPS = new ArithmeticOpTable( new UnaryOp.Neg() { @@ -322,8 +384,13 @@ public class FloatStamp extends PrimitiveStamp { @Override public Stamp foldStamp(Stamp s) { FloatStamp stamp = (FloatStamp) s; + Stamp folded = maybeFoldConstant(this, stamp); + if (folded != null) { + return folded; + } return new FloatStamp(stamp.getBits(), -stamp.upperBound(), -stamp.lowerBound(), stamp.isNonNaN()); } + }, new BinaryOp.Add(false, true) { @@ -344,8 +411,13 @@ public class FloatStamp extends PrimitiveStamp { } @Override - public Stamp foldStamp(Stamp stamp1, Stamp stamp2) { - // TODO + public Stamp foldStamp(Stamp s1, Stamp s2) { + FloatStamp stamp1 = (FloatStamp) s1; + FloatStamp stamp2 = (FloatStamp) s2; + Stamp folded = maybeFoldConstant(this, stamp1, stamp2); + if (folded != null) { + return folded; + } return stamp1.unrestricted(); } @@ -381,8 +453,13 @@ public class FloatStamp extends PrimitiveStamp { } @Override - public Stamp foldStamp(Stamp stamp1, Stamp stamp2) { - // TODO + public Stamp foldStamp(Stamp s1, Stamp s2) { + FloatStamp stamp1 = (FloatStamp) s1; + FloatStamp stamp2 = (FloatStamp) s2; + Stamp folded = maybeFoldConstant(this, stamp1, stamp2); + if (folded != null) { + return folded; + } return stamp1.unrestricted(); } @@ -418,9 +495,14 @@ public class FloatStamp extends PrimitiveStamp { } @Override - public Stamp foldStamp(Stamp a, Stamp b) { - // TODO - return a.unrestricted(); + public Stamp foldStamp(Stamp s1, Stamp s2) { + FloatStamp stamp1 = (FloatStamp) s1; + FloatStamp stamp2 = (FloatStamp) s2; + Stamp folded = maybeFoldConstant(this, stamp1, stamp2); + if (folded != null) { + return folded; + } + return stamp1.unrestricted(); } @Override @@ -450,17 +532,24 @@ public class FloatStamp extends PrimitiveStamp { assert a.getJavaKind() == b.getJavaKind(); switch (a.getJavaKind()) { case Float: - return JavaConstant.forFloat(a.asFloat() / b.asFloat()); + float floatDivisor = b.asFloat(); + return (floatDivisor == 0) ? null : JavaConstant.forFloat(a.asFloat() / floatDivisor); case Double: - return JavaConstant.forDouble(a.asDouble() / b.asDouble()); + double doubleDivisor = b.asDouble(); + return (doubleDivisor == 0) ? null : JavaConstant.forDouble(a.asDouble() / doubleDivisor); default: throw GraalError.shouldNotReachHere(); } } @Override - public Stamp foldStamp(Stamp stamp1, Stamp stamp2) { - // TODO + public Stamp foldStamp(Stamp s1, Stamp s2) { + FloatStamp stamp1 = (FloatStamp) s1; + FloatStamp stamp2 = (FloatStamp) s2; + Stamp folded = maybeFoldConstant(this, stamp1, stamp2); + if (folded != null) { + return folded; + } return stamp1.unrestricted(); } @@ -496,8 +585,13 @@ public class FloatStamp extends PrimitiveStamp { } @Override - public Stamp foldStamp(Stamp stamp1, Stamp stamp2) { - // TODO + public Stamp foldStamp(Stamp s1, Stamp s2) { + FloatStamp stamp1 = (FloatStamp) s1; + FloatStamp stamp2 = (FloatStamp) s2; + Stamp folded = maybeFoldConstant(this, stamp1, stamp2); + if (folded != null) { + return folded; + } return stamp1.unrestricted(); } }, @@ -521,6 +615,17 @@ public class FloatStamp extends PrimitiveStamp { @Override public Stamp foldStamp(Stamp s) { + FloatStamp stamp = (FloatStamp) s; + JavaConstant constant = stamp.asConstant(); + if (constant != null) { + Constant folded = foldConstant(constant); + if (folded != null) { + FloatStamp result = stampForConstant(folded); + if (result != null && result.isConstant()) { + return result; + } + } + } return s.unrestricted(); } }, @@ -547,7 +652,13 @@ public class FloatStamp extends PrimitiveStamp { } @Override - public Stamp foldStamp(Stamp stamp1, Stamp stamp2) { + public Stamp foldStamp(Stamp s1, Stamp s2) { + FloatStamp stamp1 = (FloatStamp) s1; + FloatStamp stamp2 = (FloatStamp) s2; + Stamp folded = maybeFoldConstant(this, stamp1, stamp2); + if (folded != null) { + return folded; + } return stamp1.unrestricted(); } @@ -576,7 +687,9 @@ public class FloatStamp extends PrimitiveStamp { case Float: int fa = Float.floatToRawIntBits(a.asFloat()); int fb = Float.floatToRawIntBits(b.asFloat()); - return JavaConstant.forFloat(Float.intBitsToFloat(fa | fb)); + float floatOr = Float.intBitsToFloat(fa | fb); + assert (fa | fb) == Float.floatToRawIntBits((floatOr)); + return JavaConstant.forFloat(floatOr); case Double: long da = Double.doubleToRawLongBits(a.asDouble()); long db = Double.doubleToRawLongBits(b.asDouble()); @@ -587,7 +700,13 @@ public class FloatStamp extends PrimitiveStamp { } @Override - public Stamp foldStamp(Stamp stamp1, Stamp stamp2) { + public Stamp foldStamp(Stamp s1, Stamp s2) { + FloatStamp stamp1 = (FloatStamp) s1; + FloatStamp stamp2 = (FloatStamp) s2; + Stamp folded = maybeFoldConstant(this, stamp1, stamp2); + if (folded != null) { + return folded; + } return stamp1.unrestricted(); } @@ -627,7 +746,13 @@ public class FloatStamp extends PrimitiveStamp { } @Override - public Stamp foldStamp(Stamp stamp1, Stamp stamp2) { + public Stamp foldStamp(Stamp s1, Stamp s2) { + FloatStamp stamp1 = (FloatStamp) s1; + FloatStamp stamp2 = (FloatStamp) s2; + Stamp folded = maybeFoldConstant(this, stamp1, stamp2); + if (folded != null) { + return folded; + } return stamp1.unrestricted(); } @@ -665,6 +790,10 @@ public class FloatStamp extends PrimitiveStamp { @Override public Stamp foldStamp(Stamp s) { FloatStamp stamp = (FloatStamp) s; + Stamp folded = maybeFoldConstant(this, stamp); + if (folded != null) { + return folded; + } if (stamp.isNaN()) { return stamp; } @@ -689,6 +818,11 @@ public class FloatStamp extends PrimitiveStamp { @Override public Stamp foldStamp(Stamp s) { + FloatStamp stamp = (FloatStamp) s; + Stamp folded = maybeFoldConstant(this, stamp); + if (folded != null) { + return folded; + } return s.unrestricted(); } }, diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/IntegerStamp.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/IntegerStamp.java index 36e40d54f8f..2ca05ce0a77 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/IntegerStamp.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.common/src/org/graalvm/compiler/core/common/type/IntegerStamp.java @@ -597,6 +597,10 @@ public final class IntegerStamp extends PrimitiveStamp { public Stamp foldStamp(Stamp s) { IntegerStamp stamp = (IntegerStamp) s; int bits = stamp.getBits(); + if (stamp.lowerBound == stamp.upperBound) { + long value = CodeUtil.convert(-stamp.lowerBound(), stamp.getBits(), false); + return StampFactory.forInteger(stamp.getBits(), value, value); + } if (stamp.lowerBound() != CodeUtil.minValue(bits)) { // TODO(ls) check if the mask calculation is correct... return StampFactory.forInteger(bits, -stamp.upperBound(), -stamp.lowerBound()); @@ -624,6 +628,11 @@ public final class IntegerStamp extends PrimitiveStamp { int bits = a.getBits(); assert bits == b.getBits(); + if (a.lowerBound == a.upperBound && b.lowerBound == b.upperBound) { + long value = CodeUtil.convert(a.lowerBound() + b.lowerBound(), a.getBits(), false); + return StampFactory.forInteger(a.getBits(), value, value); + } + if (a.isUnrestricted()) { return a; } else if (b.isUnrestricted()) { @@ -711,6 +720,12 @@ public final class IntegerStamp extends PrimitiveStamp { int bits = a.getBits(); assert bits == b.getBits(); + + if (a.lowerBound == a.upperBound && b.lowerBound == b.upperBound) { + long value = CodeUtil.convert(a.lowerBound() * b.lowerBound(), a.getBits(), false); + return StampFactory.forInteger(a.getBits(), value, value); + } + // if a==0 or b==0 result of a*b is always 0 if (a.upMask() == 0) { return a; @@ -791,7 +806,7 @@ public final class IntegerStamp extends PrimitiveStamp { long maxPosB = b.upperBound(); // multiplication has shift semantics - long newUpMask = ~CodeUtil.mask(Long.numberOfTrailingZeros(a.upMask) + Long.numberOfTrailingZeros(b.upMask)) & CodeUtil.mask(bits); + long newUpMask = ~CodeUtil.mask(Math.min(64, Long.numberOfTrailingZeros(a.upMask) + Long.numberOfTrailingZeros(b.upMask))) & CodeUtil.mask(bits); if (a.canBePositive()) { if (b.canBePositive()) { @@ -1023,6 +1038,9 @@ public final class IntegerStamp extends PrimitiveStamp { PrimitiveConstant a = (PrimitiveConstant) const1; PrimitiveConstant b = (PrimitiveConstant) const2; assert a.getJavaKind() == b.getJavaKind(); + if (b.asLong() == 0) { + return null; + } return JavaConstant.forIntegerKind(a.getJavaKind(), a.asLong() / b.asLong()); } @@ -1031,9 +1049,12 @@ public final class IntegerStamp extends PrimitiveStamp { IntegerStamp a = (IntegerStamp) stamp1; IntegerStamp b = (IntegerStamp) stamp2; assert a.getBits() == b.getBits(); - if (b.isStrictlyPositive()) { - long newLowerBound = a.lowerBound() / b.upperBound(); - long newUpperBound = a.upperBound() / b.lowerBound(); + if (a.lowerBound == a.upperBound && b.lowerBound == b.upperBound && b.lowerBound != 0) { + long value = CodeUtil.convert(a.lowerBound() / b.lowerBound(), a.getBits(), false); + return StampFactory.forInteger(a.getBits(), value, value); + } else if (b.isStrictlyPositive()) { + long newLowerBound = a.lowerBound() < 0 ? a.lowerBound() / b.lowerBound() : a.lowerBound() / b.upperBound(); + long newUpperBound = a.upperBound() < 0 ? a.upperBound() / b.upperBound() : a.upperBound() / b.lowerBound(); return StampFactory.forInteger(a.getBits(), newLowerBound, newUpperBound); } else { return a.unrestricted(); @@ -1054,6 +1075,9 @@ public final class IntegerStamp extends PrimitiveStamp { PrimitiveConstant a = (PrimitiveConstant) const1; PrimitiveConstant b = (PrimitiveConstant) const2; assert a.getJavaKind() == b.getJavaKind(); + if (b.asLong() == 0) { + return null; + } return JavaConstant.forIntegerKind(a.getJavaKind(), a.asLong() % b.asLong()); } @@ -1062,6 +1086,12 @@ public final class IntegerStamp extends PrimitiveStamp { IntegerStamp a = (IntegerStamp) stamp1; IntegerStamp b = (IntegerStamp) stamp2; assert a.getBits() == b.getBits(); + + if (a.lowerBound == a.upperBound && b.lowerBound == b.upperBound && b.lowerBound != 0) { + long value = CodeUtil.convert(a.lowerBound() % b.lowerBound(), a.getBits(), false); + return StampFactory.forInteger(a.getBits(), value, value); + } + // zero is always possible long newLowerBound = Math.min(a.lowerBound(), 0); long newUpperBound = Math.max(a.upperBound(), 0); @@ -1364,6 +1394,10 @@ public final class IntegerStamp extends PrimitiveStamp { public Stamp foldStamp(Stamp input) { IntegerStamp stamp = (IntegerStamp) input; int bits = stamp.getBits(); + if (stamp.lowerBound == stamp.upperBound) { + long value = CodeUtil.convert(Math.abs(stamp.lowerBound()), stamp.getBits(), false); + return StampFactory.forInteger(stamp.getBits(), value, value); + } if (stamp.lowerBound() == CodeUtil.minValue(bits)) { return input.unrestricted(); } else { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/CheckGraalInvariants.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/CheckGraalInvariants.java index 10f0a91e853..7ddf8c24b10 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/CheckGraalInvariants.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/CheckGraalInvariants.java @@ -49,8 +49,8 @@ import org.graalvm.compiler.core.CompilerThreadFactory; import org.graalvm.compiler.core.common.LIRKind; import org.graalvm.compiler.core.common.type.ArithmeticOpTable; import org.graalvm.compiler.debug.DebugCloseable; -import org.graalvm.compiler.debug.DebugHandlersFactory; import org.graalvm.compiler.debug.DebugContext; +import org.graalvm.compiler.debug.DebugHandlersFactory; import org.graalvm.compiler.debug.GraalError; import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.graph.NodeClass; @@ -73,6 +73,8 @@ import org.graalvm.compiler.phases.util.Providers; import org.graalvm.compiler.phases.verify.VerifyBailoutUsage; import org.graalvm.compiler.phases.verify.VerifyCallerSensitiveMethods; import org.graalvm.compiler.phases.verify.VerifyDebugUsage; +import org.graalvm.compiler.phases.verify.VerifyGetOptionsUsage; +import org.graalvm.compiler.phases.verify.VerifyGraphAddUsage; import org.graalvm.compiler.phases.verify.VerifyInstanceOfUsage; import org.graalvm.compiler.phases.verify.VerifyUpdateUsages; import org.graalvm.compiler.phases.verify.VerifyUsageWithEquals; @@ -381,6 +383,8 @@ public class CheckGraalInvariants extends GraalCompilerTest { new VerifyUpdateUsages().apply(graph, context); new VerifyBailoutUsage().apply(graph, context); new VerifyInstanceOfUsage().apply(graph, context); + new VerifyGraphAddUsage().apply(graph, context); + new VerifyGetOptionsUsage().apply(graph, context); if (graph.method().isBridge()) { BridgeMethodUtils.getBridgedMethod(graph.method()); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ConditionalEliminationTest14.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ConditionalEliminationTest14.java new file mode 100644 index 00000000000..fbba0820ba8 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ConditionalEliminationTest14.java @@ -0,0 +1,91 @@ +/* + * Copyright (c) 2015, 2015, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.core.test; + +import org.graalvm.compiler.graph.iterators.NodeIterable; +import org.graalvm.compiler.nodes.ConstantNode; +import org.graalvm.compiler.nodes.FixedGuardNode; +import org.graalvm.compiler.nodes.GuardNode; +import org.graalvm.compiler.nodes.LogicNode; +import org.graalvm.compiler.nodes.StructuredGraph; +import org.graalvm.compiler.nodes.StructuredGraph.AllowAssumptions; +import org.graalvm.compiler.nodes.calc.IntegerBelowNode; +import org.graalvm.compiler.nodes.java.LoadIndexedNode; +import org.graalvm.compiler.nodes.memory.FloatingReadNode; +import org.graalvm.compiler.nodes.memory.ReadNode; +import org.graalvm.compiler.nodes.spi.LoweringTool; +import org.graalvm.compiler.phases.common.CanonicalizerPhase; +import org.graalvm.compiler.phases.common.FloatingReadPhase; +import org.graalvm.compiler.phases.common.IterativeConditionalEliminationPhase; +import org.graalvm.compiler.phases.common.LoweringPhase; +import org.graalvm.compiler.phases.tiers.PhaseContext; +import org.junit.Assert; +import org.junit.Test; + +import jdk.vm.ci.meta.DeoptimizationReason; + +/** + * Check that multiple bounds checks are correctly grouped together. + */ +public class ConditionalEliminationTest14 extends ConditionalEliminationTestBase { + + public static void test1Snippet(Object[] args) { + Object a5 = args[5]; + Object a7 = args[7]; + Object a6 = args[6]; + + /* + * The order of the conditions matters: The scheduler processes the floating reads for the + * array loads in the order of the conditions here, and we want the index 7 access to be + * processed before the index 6 access. + */ + if (a5 != null && a7 != null && a6 != null) { + sink1 = 1; + } + sink0 = 0; + } + + @Test + public void test1() { + StructuredGraph graph = parseEager("test1Snippet", AllowAssumptions.YES); + CanonicalizerPhase canonicalizer = new CanonicalizerPhase(); + PhaseContext context = new PhaseContext(getProviders()); + + /* Convert the LoadIndexNode to ReadNode with floating guards. */ + new LoweringPhase(canonicalizer, LoweringTool.StandardLoweringStage.HIGH_TIER).apply(graph, context); + /* Convert the ReadNode to FloatingReadNode. */ + new FloatingReadPhase().apply(graph); + /* Apply the phase that we want to test. */ + new IterativeConditionalEliminationPhase(canonicalizer, true).apply(graph, context); + + Assert.assertEquals("All guards must be floating", 0, graph.getNodes(FixedGuardNode.TYPE).count()); + Assert.assertEquals("All array accesses must have been lowered", 0, graph.getNodes().filter(LoadIndexedNode.class).count()); + Assert.assertEquals("All reads must be floating", 0, graph.getNodes().filter(ReadNode.class).count()); + Assert.assertEquals("Must have floating reads (3 array accesses, 1 array length)", 4, graph.getNodes().filter(FloatingReadNode.class).count()); + + NodeIterable boundsChecks = graph.getNodes(GuardNode.TYPE).filter(n -> ((GuardNode) n).getReason() == DeoptimizationReason.BoundsCheckException); + Assert.assertEquals("Must have only 1 bounds check remaining", 1, boundsChecks.count()); + LogicNode condition = boundsChecks.first().getCondition(); + Assert.assertTrue("Bounds check must check for array length 8", condition instanceof IntegerBelowNode && ((IntegerBelowNode) condition).getY().valueEquals(ConstantNode.forInt(8))); + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ConditionalEliminationTestBase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ConditionalEliminationTestBase.java index 0140fe6e5de..dfc8aecff31 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ConditionalEliminationTestBase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ConditionalEliminationTestBase.java @@ -27,12 +27,15 @@ import org.graalvm.compiler.nodes.ProxyNode; import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.StructuredGraph.AllowAssumptions; import org.graalvm.compiler.nodes.spi.LoweringTool; +import org.graalvm.compiler.phases.OptimisticOptimizations; +import org.graalvm.compiler.phases.OptimisticOptimizations.Optimization; import org.graalvm.compiler.phases.common.CanonicalizerPhase; import org.graalvm.compiler.phases.common.ConditionalEliminationPhase; import org.graalvm.compiler.phases.common.ConvertDeoptimizeToGuardPhase; import org.graalvm.compiler.phases.common.IterativeConditionalEliminationPhase; import org.graalvm.compiler.phases.common.LoweringPhase; import org.graalvm.compiler.phases.schedule.SchedulePhase; +import org.graalvm.compiler.phases.tiers.HighTierContext; import org.graalvm.compiler.phases.tiers.PhaseContext; import org.junit.Assert; @@ -45,6 +48,15 @@ public class ConditionalEliminationTestBase extends GraalCompilerTest { protected static int sink1; protected static int sink2; + /** + * These tests assume all code paths in called routines are reachable so disable removal of dead + * code based on method profiles. + */ + @Override + protected HighTierContext getDefaultHighTierContext() { + return new HighTierContext(getProviders(), getDefaultGraphBuilderSuite(), OptimisticOptimizations.ALL.remove(Optimization.RemoveNeverExecutedCode)); + } + protected void testConditionalElimination(String snippet, String referenceSnippet) { testConditionalElimination(snippet, referenceSnippet, false, false); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/DumpPathTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/DumpPathTest.java new file mode 100644 index 00000000000..8380c9a4a5e --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/DumpPathTest.java @@ -0,0 +1,111 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.core.test; + +import java.io.IOException; +import java.nio.file.DirectoryStream; +import java.nio.file.Files; +import java.nio.file.Path; + +import org.graalvm.compiler.debug.DebugOptions; +import org.graalvm.compiler.options.OptionKey; +import org.graalvm.compiler.options.OptionValues; +import org.graalvm.util.EconomicMap; +import org.junit.Test; + +/** + * Check that setting the dump path results in files ending up in the right directory with matching + * names. + */ +public class DumpPathTest extends GraalCompilerTest { + + public static Object snippet() { + return new String("snippet"); + } + + @Test + public void testDump() throws IOException { + Path dumpDirectoryPath = Files.createTempDirectory("DumpPathTest"); + String[] extensions = new String[]{".cfg", ".bgv", ".graph-strings"}; + EconomicMap, Object> overrides = OptionValues.newOptionMap(); + overrides.put(DebugOptions.DumpPath, dumpDirectoryPath.toString()); + overrides.put(DebugOptions.PrintGraphFile, true); + overrides.put(DebugOptions.PrintCanonicalGraphStrings, true); + overrides.put(DebugOptions.Dump, "*"); + + // Generate dump files. + test(new OptionValues(getInitialOptions(), overrides), "snippet"); + // Check that Ideal files got created, in the right place. + checkForFiles(dumpDirectoryPath, extensions); + + // Clean up the generated files. + scrubDirectory(dumpDirectoryPath); + } + + /** + * Check that the given directory contains file or directory names with all the given + * extensions. + */ + private static void checkForFiles(Path directoryPath, String[] extensions) throws IOException { + String[] paths = new String[extensions.length]; + try (DirectoryStream stream = Files.newDirectoryStream(directoryPath)) { + for (Path filePath : stream) { + String fileName = filePath.getFileName().toString(); + for (int i = 0; i < extensions.length; i++) { + String extension = extensions[i]; + if (fileName.endsWith(extensions[i])) { + assertTrue(paths[i] == null, "multiple files found for %s in %s", extension, directoryPath); + paths[i] = fileName.replace(extensions[i], ""); + } + } + } + } + for (int i = 0; i < paths.length; i++) { + assertTrue(paths[i] != null, "missing file for extension %s in %s", extensions[i], directoryPath); + } + // Ensure that all file names are the same. + for (int i = 1; i < paths.length; i++) { + assertTrue(paths[0].equals(paths[i]), paths[0] + " != " + paths[i]); + } + } + + /** + * Remove the temporary directory. + */ + private static void scrubDirectory(Path directoryPath) { + try { + try (DirectoryStream stream = Files.newDirectoryStream(directoryPath)) { + for (Path filePath : stream) { + if (Files.isRegularFile(filePath)) { + Files.delete(filePath); + } else if (Files.isDirectory(filePath)) { + scrubDirectory(filePath); + } + } + } + Files.delete(directoryPath); + } catch (IOException ioe) { + ioe.printStackTrace(); + } + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/FinalizableSubclassTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/FinalizableSubclassTest.java index e6e802ce4c1..c01acf87524 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/FinalizableSubclassTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/FinalizableSubclassTest.java @@ -72,7 +72,7 @@ public class FinalizableSubclassTest extends GraalCompilerTest { Assert.assertTrue(constructors.length == 1); final ResolvedJavaMethod javaMethod = getMetaAccess().lookupJavaMethod(constructors[0]); OptionValues options = getInitialOptions(); - StructuredGraph graph = new StructuredGraph.Builder(options, getDebugContext(options), allowAssumptions).method(javaMethod).build(); + StructuredGraph graph = new StructuredGraph.Builder(options, getDebugContext(options, null, javaMethod), allowAssumptions).method(javaMethod).build(); GraphBuilderConfiguration conf = GraphBuilderConfiguration.getSnippetDefault(getDefaultGraphBuilderPlugins()); new GraphBuilderPhase.Instance(getMetaAccess(), getProviders().getStampProvider(), getProviders().getConstantReflection(), getProviders().getConstantFieldProvider(), conf, diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GraalCompilerTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GraalCompilerTest.java index 00887afda2a..4941269fdb3 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GraalCompilerTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GraalCompilerTest.java @@ -389,7 +389,7 @@ public abstract class GraalCompilerTest extends GraalTest { * {@link DebugDumpHandler}s closed in {@link #afterTest()}. */ protected DebugContext getDebugContext() { - return getDebugContext(getInitialOptions()); + return getDebugContext(getInitialOptions(), null, null); } @Override @@ -862,7 +862,7 @@ public abstract class GraalCompilerTest extends GraalTest { Result actual = executeActual(options, method, receiver, args); profile = method.getProfilingInfo(); // profile can change after execution for (DeoptimizationReason reason : shouldNotDeopt) { - Assert.assertEquals((int) deoptCounts.get(reason), profile.getDeoptimizationCount(reason)); + Assert.assertEquals("wrong number of deopt counts for " + reason, (int) deoptCounts.get(reason), profile.getDeoptimizationCount(reason)); } return actual; } @@ -1216,15 +1216,15 @@ public abstract class GraalCompilerTest extends GraalTest { protected final Builder builder(ResolvedJavaMethod method, AllowAssumptions allowAssumptions) { OptionValues options = getInitialOptions(); - return new Builder(options, getDebugContext(options), allowAssumptions).method(method).compilationId(getCompilationId(method)); + return new Builder(options, getDebugContext(options, null, method), allowAssumptions).method(method).compilationId(getCompilationId(method)); } protected final Builder builder(ResolvedJavaMethod method, AllowAssumptions allowAssumptions, CompilationIdentifier compilationId, OptionValues options) { - return new Builder(options, getDebugContext(options), allowAssumptions).method(method).compilationId(compilationId); + return new Builder(options, getDebugContext(options, compilationId.toString(CompilationIdentifier.Verbosity.ID), method), allowAssumptions).method(method).compilationId(compilationId); } protected final Builder builder(ResolvedJavaMethod method, AllowAssumptions allowAssumptions, OptionValues options) { - return new Builder(options, getDebugContext(options), allowAssumptions).method(method).compilationId(getCompilationId(method)); + return new Builder(options, getDebugContext(options, null, method), allowAssumptions).method(method).compilationId(getCompilationId(method)); } protected PhaseSuite getDebugGraphBuilderSuite() { @@ -1234,6 +1234,7 @@ public abstract class GraalCompilerTest extends GraalTest { @SuppressWarnings("try") protected StructuredGraph parse(StructuredGraph.Builder builder, PhaseSuite graphBuilderSuite) { ResolvedJavaMethod javaMethod = builder.getMethod(); + builder.speculationLog(getSpeculationLog()); if (builder.getCancellable() == null) { builder.cancellable(getCancellable(javaMethod)); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GraalDebugHandlersFactoryTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GraalDebugHandlersFactoryTest.java index 723d127618a..42d058d4bd6 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GraalDebugHandlersFactoryTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GraalDebugHandlersFactoryTest.java @@ -31,8 +31,12 @@ import java.nio.file.Path; import java.nio.file.Paths; import java.util.Comparator; -import org.graalvm.compiler.printer.GraalDebugHandlersFactory; +import org.graalvm.compiler.debug.DebugOptions; +import org.graalvm.compiler.debug.PathUtilities; +import org.graalvm.compiler.options.OptionKey; +import org.graalvm.compiler.options.OptionValues; import org.graalvm.compiler.test.AddExports; +import org.junit.Assume; import org.junit.Test; @AddExports("jdk.internal.vm.compiler/org.graalvm.compiler.printer") @@ -40,23 +44,28 @@ public class GraalDebugHandlersFactoryTest extends GraalCompilerTest { @Test public void createUniqueTest() throws Exception { - Field maxFileNameLengthField = GraalDebugHandlersFactory.class.getDeclaredField("MAX_FILE_NAME_LENGTH"); - maxFileNameLengthField.setAccessible(true); + Field maxFileNameLengthField = PathUtilities.class.getDeclaredField("MAX_FILE_NAME_LENGTH"); + try { + maxFileNameLengthField.setAccessible(true); + } catch (RuntimeException ex) { + Assume.assumeFalse("If InaccessibleObjectException is thrown, skip the test, we are on JDK9", ex.getClass().getSimpleName().equals("InaccessibleObjectException")); + } int maxFileNameLength = maxFileNameLengthField.getInt(null); - Method createUniqueMethod = GraalDebugHandlersFactory.class.getDeclaredMethod("createUnique", Path.class, String.class, String.class, String.class, boolean.class); + Method createUniqueMethod = PathUtilities.class.getDeclaredMethod("createUnique", OptionValues.class, OptionKey.class, String.class, String.class, String.class, boolean.class); createUniqueMethod.setAccessible(true); Path tmpDir = Files.createTempDirectory(Paths.get("."), "createUniqueTest"); + OptionValues options = new OptionValues(OptionValues.asMap(DebugOptions.DumpPath, tmpDir.toString())); try { for (boolean createDirectory : new boolean[]{true, false}) { for (String ext : new String[]{"", ".bgv", ".graph-strings"}) { for (int i = 0; i < maxFileNameLength + 5; i++) { String id = new String(new char[i]).replace('\0', 'i'); String label = ""; - createUniqueMethod.invoke(null, tmpDir, id, label, ext, createDirectory); + createUniqueMethod.invoke(null, options, null, id, label, ext, createDirectory); id = ""; label = new String(new char[i]).replace('\0', 'l'); - createUniqueMethod.invoke(null, tmpDir, id, label, ext, createDirectory); + createUniqueMethod.invoke(null, options, null, id, label, ext, createDirectory); } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GuardedIntrinsicTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GuardedIntrinsicTest.java index fd0e6c2e956..a2a0f48ed2b 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GuardedIntrinsicTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/GuardedIntrinsicTest.java @@ -144,6 +144,11 @@ public class GuardedIntrinsicTest extends GraalCompilerTest { public void test01() { Super inheritsHC = new Super(); Person overridesHC = new Person(0); + + // Ensure the profile for getSuperAge includes both receiver types + getSuperAge(inheritsHC); + getSuperAge(overridesHC); + test("getSuperAge", inheritsHC); test("getSuperAge", overridesHC); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/HashCodeTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/HashCodeTest.java index 6ee99a34f14..6891c666b7b 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/HashCodeTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/HashCodeTest.java @@ -22,6 +22,8 @@ */ package org.graalvm.compiler.core.test; +import java.util.HashMap; + import org.graalvm.compiler.core.phases.HighTier; import org.graalvm.compiler.core.phases.MidTier; import org.graalvm.compiler.nodes.InvokeNode; @@ -139,6 +141,10 @@ public class HashCodeTest extends GraalCompilerTest { public void test08() { initialize(Appendable.class); checkForGuardedIntrinsicPattern("hashCodeInterface"); + + // Ensure the profile for the dispatch in hashCodeSnippet01 + // has a receiver type that does not select Object.hashCode intrinsic + hashCodeSnippet01(new HashMap<>()); checkForGuardedIntrinsicPattern("hashCodeSnippet01"); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/MarkUnsafeAccessTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/MarkUnsafeAccessTest.java index 1d07d3b9375..2e747addf82 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/MarkUnsafeAccessTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/MarkUnsafeAccessTest.java @@ -25,6 +25,7 @@ package org.graalvm.compiler.core.test; import static java.nio.file.StandardOpenOption.READ; import static java.nio.file.StandardOpenOption.WRITE; +import java.io.File; import java.io.IOException; import java.nio.ByteBuffer; import java.nio.MappedByteBuffer; @@ -33,22 +34,20 @@ import java.nio.channels.FileChannel.MapMode; import java.nio.file.Files; import java.nio.file.Path; -import jdk.vm.ci.code.InstalledCode; -import jdk.vm.ci.code.InvalidInstalledCodeException; -import jdk.vm.ci.meta.ResolvedJavaMethod; -import jdk.vm.ci.meta.ResolvedJavaType; - -import org.junit.Assert; -import org.junit.Assume; -import org.junit.Test; - -import sun.misc.Unsafe; - import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.phases.common.CanonicalizerPhase; import org.graalvm.compiler.phases.common.inlining.InliningPhase; import org.graalvm.compiler.phases.common.inlining.policy.InlineEverythingPolicy; import org.graalvm.compiler.phases.tiers.HighTierContext; +import org.junit.Assert; +import org.junit.Assume; +import org.junit.Test; + +import jdk.vm.ci.code.InstalledCode; +import jdk.vm.ci.code.InvalidInstalledCodeException; +import jdk.vm.ci.meta.ResolvedJavaMethod; +import jdk.vm.ci.meta.ResolvedJavaType; +import sun.misc.Unsafe; public class MarkUnsafeAccessTest extends GraalCompilerTest { @@ -170,7 +169,9 @@ public class MarkUnsafeAccessTest extends GraalCompilerTest { try { mbb.position(BLOCK_SIZE); getter.get(mbb); - System.currentTimeMillis(); // materialize async exception + + // Make a call that goes into native code to materialize async exception + new File("").exists(); } catch (InternalError e) { return; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/MergeCanonicalizerTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/MergeCanonicalizerTest.java index 08f83bbe2cf..5c6605e4e6d 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/MergeCanonicalizerTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/MergeCanonicalizerTest.java @@ -26,12 +26,23 @@ import org.graalvm.compiler.debug.DebugContext; import org.graalvm.compiler.nodes.ReturnNode; import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.StructuredGraph.AllowAssumptions; +import org.graalvm.compiler.phases.OptimisticOptimizations; +import org.graalvm.compiler.phases.OptimisticOptimizations.Optimization; import org.graalvm.compiler.phases.common.CanonicalizerPhase; +import org.graalvm.compiler.phases.tiers.HighTierContext; import org.graalvm.compiler.phases.tiers.PhaseContext; import org.junit.Test; public class MergeCanonicalizerTest extends GraalCompilerTest { + /** + * These tests assume all code paths are reachable so disable profile based dead code removal. + */ + @Override + protected HighTierContext getDefaultHighTierContext() { + return new HighTierContext(getProviders(), getDefaultGraphBuilderSuite(), OptimisticOptimizations.ALL.remove(Optimization.RemoveNeverExecutedCode)); + } + public static int staticField; private int field; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ProfilingInfoTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ProfilingInfoTest.java index 684d4a1c362..7d881183c7e 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ProfilingInfoTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ProfilingInfoTest.java @@ -24,15 +24,17 @@ package org.graalvm.compiler.core.test; import java.io.Serializable; +import org.graalvm.compiler.test.SubprocessUtil; +import org.junit.Assert; +import org.junit.Assume; +import org.junit.Test; + import jdk.vm.ci.meta.JavaTypeProfile; import jdk.vm.ci.meta.ProfilingInfo; import jdk.vm.ci.meta.ResolvedJavaMethod; import jdk.vm.ci.meta.ResolvedJavaType; import jdk.vm.ci.meta.TriState; -import org.junit.Assert; -import org.junit.Test; - /** * Tests profiling information provided by the runtime. *

@@ -40,7 +42,7 @@ import org.junit.Test; * information may be gathered for any given method. For example, HotSpot's advanced compilation * policy can decide to only gather partial profiles in a first level compilation (see * AdvancedThresholdPolicy::common(...) in advancedThresholdPolicy.cpp). Because of this, - * occasionally tests for {@link ProfilingInfo#getNullSeen(int)} can fail since HotSpot only set's + * occasionally tests for {@link ProfilingInfo#getNullSeen(int)} can fail since HotSpot only sets * the null_seen bit when doing full profiling. */ public class ProfilingInfoTest extends GraalCompilerTest { @@ -182,6 +184,14 @@ public class ProfilingInfoTest extends GraalCompilerTest { Assert.assertNull(typeProfile); } + public ProfilingInfoTest() { + // These tests are explicitly testing the profiling behavior of the + // interpreter. C1-based profiling differs slightly and when -Xcomp + // is present, profiles will be created by C1 compiled code, not the + // interpreter. + Assume.assumeTrue(!SubprocessUtil.getVMCommandLine().contains("-Xcomp")); + } + @Test public void testExceptionSeen() { // NullPointerException diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/SubWordReturnTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/SubWordReturnTest.java new file mode 100644 index 00000000000..d8e38370a3f --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/SubWordReturnTest.java @@ -0,0 +1,143 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.core.test; + +import java.util.ArrayList; +import java.util.List; +import jdk.vm.ci.meta.JavaKind; +import jdk.vm.ci.meta.ResolvedJavaMethod; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.junit.runners.Parameterized; +import org.junit.runners.Parameterized.Parameters; +import org.objectweb.asm.ClassWriter; +import org.objectweb.asm.FieldVisitor; +import org.objectweb.asm.MethodVisitor; +import org.objectweb.asm.Opcodes; + +@RunWith(Parameterized.class) +public class SubWordReturnTest extends GraalCompilerTest { + + private final JavaKind kind; + private final int value; + + private final String generatedClassName; + private final String generatedClassNameInternal; + + private final String testMethodName; + + /** + * The {@link AsmLoader} generates a class looking like this for the types byte, short, int and + * char. + */ + static class ByteGetter { + + // private static int intField = 1000000; + + private static byte get() { + // GETSTATIC intField + // IRETURN + return 0; + } + + public static int testByteSnippet() { + return get(); + } + } + + @Parameters(name = "{0}, {1}") + public static List data() { + ArrayList ret = new ArrayList<>(); + for (int i : new int[]{1000000, 1000001, -1000000, -1}) { + ret.add(new Object[]{JavaKind.Boolean, i}); + ret.add(new Object[]{JavaKind.Byte, i}); + ret.add(new Object[]{JavaKind.Short, i}); + ret.add(new Object[]{JavaKind.Char, i}); + } + return ret; + } + + public SubWordReturnTest(JavaKind kind, int value) { + this.kind = kind; + this.value = value; + + this.generatedClassName = SubWordReturnTest.class.getName() + "$" + kind.toString() + "Getter"; + this.generatedClassNameInternal = generatedClassName.replace('.', '/'); + this.testMethodName = "test" + kind.name() + "Snippet"; + } + + @Test + public void test() throws ClassNotFoundException { + Class testClass = new AsmLoader(SubWordReturnTest.class.getClassLoader()).findClass(generatedClassName); + ResolvedJavaMethod method = getResolvedJavaMethod(testClass, testMethodName); + test(method, null); + } + + class AsmLoader extends ClassLoader implements Opcodes { + + Class loaded; + + AsmLoader(ClassLoader parent) { + super(parent); + } + + @Override + protected Class findClass(String name) throws ClassNotFoundException { + if (name.equals(generatedClassName)) { + if (loaded == null) { + byte[] gen = generateClass(); + loaded = defineClass(name, gen, 0, gen.length); + } + return loaded; + } else { + return super.findClass(name); + } + } + + private byte[] generateClass() { + ClassWriter cw = new ClassWriter(0); + cw.visit(52, ACC_SUPER | ACC_PUBLIC, generatedClassNameInternal, null, "java/lang/Object", null); + + FieldVisitor intField = cw.visitField(ACC_PRIVATE | ACC_STATIC, "intField", "I", null, value); + intField.visitEnd(); + + MethodVisitor get = cw.visitMethod(ACC_PRIVATE | ACC_STATIC, "get", "()" + kind.getTypeChar(), null, null); + get.visitCode(); + get.visitFieldInsn(GETSTATIC, generatedClassNameInternal, "intField", "I"); + get.visitInsn(IRETURN); + get.visitMaxs(1, 0); + get.visitEnd(); + + MethodVisitor snippet = cw.visitMethod(ACC_PUBLIC | ACC_STATIC, testMethodName, "()I", null, null); + snippet.visitCode(); + snippet.visitMethodInsn(INVOKESTATIC, generatedClassNameInternal, "get", "()" + kind.getTypeChar(), false); + snippet.visitInsn(IRETURN); + snippet.visitMaxs(1, 0); + snippet.visitEnd(); + + cw.visitEnd(); + return cw.toByteArray(); + } + } + +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/UnbalancedMonitorsTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/UnbalancedMonitorsTest.java index c7825439a07..eea935cfefe 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/UnbalancedMonitorsTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/UnbalancedMonitorsTest.java @@ -87,7 +87,7 @@ public class UnbalancedMonitorsTest extends GraalCompilerTest { ResolvedJavaMethod method = getResolvedJavaMethod(LOADER.findClass(INNER_CLASS_NAME), name); try { OptionValues options = getInitialOptions(); - StructuredGraph graph = new StructuredGraph.Builder(options, getDebugContext(options)).method(method).build(); + StructuredGraph graph = new StructuredGraph.Builder(options, getDebugContext(options, null, method)).method(method).build(); Plugins plugins = new Plugins(new InvocationPlugins()); GraphBuilderConfiguration graphBuilderConfig = GraphBuilderConfiguration.getDefault(plugins).withEagerResolving(true).withUnresolvedIsError(true); OptimisticOptimizations optimisticOpts = OptimisticOptimizations.NONE; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/UnsafeVirtualizationTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/UnsafeVirtualizationTest.java index 236be5cff40..9b09e2522b5 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/UnsafeVirtualizationTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/UnsafeVirtualizationTest.java @@ -32,9 +32,20 @@ import org.graalvm.compiler.phases.tiers.PhaseContext; import org.graalvm.compiler.virtual.phases.ea.PartialEscapePhase; import org.junit.Test; +import jdk.vm.ci.code.InstalledCode; +import jdk.vm.ci.meta.ResolvedJavaMethod; + public class UnsafeVirtualizationTest extends GraalCompilerTest { - public static class A { + public static class Base { + /* + * This padding ensure that the size of the Base class ends up as a multiple of 8, which + * makes the first field of the subclass 8-byte aligned. + */ + double padding; + } + + public static class A extends Base { int f1; int f2; } @@ -56,39 +67,96 @@ public class UnsafeVirtualizationTest extends GraalCompilerTest { AF2Offset = o2; } - public static int unsafeSnippet0(int i1, int i2) { + public static int unsafeSnippet1(double i1) { A a = new A(); - UNSAFE.putDouble(a, AF1Offset, i1 + i2); + UNSAFE.putDouble(a, AF1Offset, i1); return UNSAFE.getInt(a, AF1Offset) + UNSAFE.getInt(a, AF2Offset); } - public static int unsafeSnippet1(int i1, int i2) { + public static long unsafeSnippet2a(int i1) { A a = new A(); - UNSAFE.putDouble(a, AF1Offset, i1 + i2); + UNSAFE.putDouble(a, AF1Offset, i1); + a.f1 = i1; + return UNSAFE.getLong(a, AF1Offset); + } + + public static long unsafeSnippet2b(int i1) { + A a = new A(); + UNSAFE.putDouble(a, AF1Offset, i1); a.f2 = i1; - return (int) UNSAFE.getDouble(a, AF1Offset); + return UNSAFE.getLong(a, AF1Offset); + } + + public static long unsafeSnippet3a(int i1) { + A a = new A(); + UNSAFE.putDouble(a, AF1Offset, i1); + UNSAFE.putInt(a, AF1Offset, i1); + return UNSAFE.getLong(a, AF1Offset); + } + + public static long unsafeSnippet3b(int i1) { + A a = new A(); + UNSAFE.putDouble(a, AF1Offset, i1); + UNSAFE.putInt(a, AF2Offset, i1); + return UNSAFE.getLong(a, AF1Offset); + } + + public static int unsafeSnippet4(double i1) { + A a = new A(); + UNSAFE.putDouble(a, AF1Offset, i1); + UNSAFE.putDouble(a, AF1Offset, i1); + return UNSAFE.getInt(a, AF1Offset) + UNSAFE.getInt(a, AF2Offset); } @Test public void testUnsafePEA01() { - testPartialEscapeReadElimination(parseEager("unsafeSnippet0", AllowAssumptions.NO), false); - testPartialEscapeReadElimination(parseEager("unsafeSnippet0", AllowAssumptions.NO), true); + testPartialEscapeReadElimination("unsafeSnippet1", false, 1.0); + testPartialEscapeReadElimination("unsafeSnippet1", true, 1.0); } @Test public void testUnsafePEA02() { - testPartialEscapeReadElimination(parseEager("unsafeSnippet1", AllowAssumptions.NO), false); - testPartialEscapeReadElimination(parseEager("unsafeSnippet1", AllowAssumptions.NO), true); + testPartialEscapeReadElimination("unsafeSnippet2a", false, 1); + testPartialEscapeReadElimination("unsafeSnippet2a", true, 1); + + testPartialEscapeReadElimination("unsafeSnippet2b", false, 1); + testPartialEscapeReadElimination("unsafeSnippet2b", true, 1); } - public void testPartialEscapeReadElimination(StructuredGraph graph, boolean canonicalizeBefore) { + @Test + public void testUnsafePEA03() { + testPartialEscapeReadElimination("unsafeSnippet3a", false, 1); + testPartialEscapeReadElimination("unsafeSnippet3a", true, 1); + + testPartialEscapeReadElimination("unsafeSnippet3b", false, 1); + testPartialEscapeReadElimination("unsafeSnippet3b", true, 1); + } + + @Test + public void testUnsafePEA04() { + testPartialEscapeReadElimination("unsafeSnippet4", false, 1.0); + testPartialEscapeReadElimination("unsafeSnippet4", true, 1.0); + } + + public void testPartialEscapeReadElimination(String snippet, boolean canonicalizeBefore, Object... args) { + assert AF1Offset % 8 == 0 : "First of the two int-fields must be 8-byte aligned"; + + ResolvedJavaMethod method = getResolvedJavaMethod(snippet); + StructuredGraph graph = parseEager(snippet, AllowAssumptions.NO); OptionValues options = graph.getOptions(); PhaseContext context = getDefaultHighTierContext(); CanonicalizerPhase canonicalizer = new CanonicalizerPhase(); if (canonicalizeBefore) { canonicalizer.apply(graph, context); } + Result r = executeExpected(method, null, args); new PartialEscapePhase(true, true, canonicalizer, null, options).apply(graph, context); + try { + InstalledCode code = getCode(method, graph); + Object result = code.executeVarargs(args); + assertEquals(r, new Result(result, null)); + } catch (Throwable e) { + assertFalse(true, e.toString()); + } } - } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/EATestBase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/EATestBase.java index f429d53ec52..55db5ec5b85 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/EATestBase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/EATestBase.java @@ -77,7 +77,7 @@ public class EATestBase extends GraalCompilerTest { @Override public String toString() { - return "{" + x + "," + y + "}"; + return "{" + x + "," + y + "," + z + "}"; } @Override @@ -158,11 +158,19 @@ public class EATestBase extends GraalCompilerTest { context = getDefaultHighTierContext(); new InliningPhase(new CanonicalizerPhase()).apply(graph, context); new DeadCodeEliminationPhase().apply(graph); - new CanonicalizerPhase().apply(graph, context); + canonicalizeGraph(); new PartialEscapePhase(iterativeEscapeAnalysis, false, new CanonicalizerPhase(), null, graph.getOptions()).apply(graph, context); + postEACanonicalizeGraph(); returnNodes = graph.getNodes(ReturnNode.TYPE).snapshot(); } catch (Throwable e) { throw debug.handle(e); } } + + protected void postEACanonicalizeGraph() { + } + + protected void canonicalizeGraph() { + new CanonicalizerPhase().apply(graph, context); + } } 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 8c3f70273a1..69c7cc21e68 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 @@ -27,9 +27,20 @@ import org.junit.Test; import org.graalvm.compiler.api.directives.GraalDirectives; import org.graalvm.compiler.code.SourceStackTraceBailoutException; import org.graalvm.compiler.core.test.GraalCompilerTest; +import org.graalvm.compiler.phases.OptimisticOptimizations; +import org.graalvm.compiler.phases.OptimisticOptimizations.Optimization; +import org.graalvm.compiler.phases.tiers.HighTierContext; public class PEAAssertionsTest extends GraalCompilerTest { + /** + * These tests assume all code paths are reachable so disable profile based dead code removal. + */ + @Override + protected HighTierContext getDefaultHighTierContext() { + return new HighTierContext(getProviders(), getDefaultGraphBuilderSuite(), OptimisticOptimizations.ALL.remove(Optimization.RemoveNeverExecutedCode)); + } + public static Object field; public static void snippet1(int i) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/TrufflePEATest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/TrufflePEATest.java index 81ee0d6eccc..b08c1adf12e 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/TrufflePEATest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/TrufflePEATest.java @@ -22,6 +22,8 @@ */ package org.graalvm.compiler.core.test.ea; +import java.lang.reflect.Field; + import org.graalvm.compiler.core.common.GraalOptions; import org.graalvm.compiler.core.test.GraalCompilerTest; import org.graalvm.compiler.nodes.StructuredGraph; @@ -33,9 +35,8 @@ import org.graalvm.compiler.phases.common.inlining.InliningPhase; import org.graalvm.compiler.phases.tiers.HighTierContext; import org.graalvm.compiler.virtual.phases.ea.PartialEscapePhase; import org.junit.Test; -import sun.misc.Unsafe; -import java.lang.reflect.Field; +import sun.misc.Unsafe; public class TrufflePEATest extends GraalCompilerTest { @@ -56,6 +57,7 @@ public class TrufflePEATest extends GraalCompilerTest { static class DynamicObject { int primitiveField0; int primitiveField1; + int primitiveField2; } private static final long offsetLong1 = Unsafe.ARRAY_LONG_BASE_OFFSET + Unsafe.ARRAY_LONG_INDEX_SCALE * 1; @@ -66,7 +68,15 @@ public class TrufflePEATest extends GraalCompilerTest { static { try { Field primitiveField0 = DynamicObject.class.getDeclaredField("primitiveField0"); - primitiveField0Offset = UNSAFE.objectFieldOffset(primitiveField0); + long offset = UNSAFE.objectFieldOffset(primitiveField0); + if (offset % 8 == 0) { + primitiveField0Offset = offset; + } else { + Field primitiveField1 = DynamicObject.class.getDeclaredField("primitiveField1"); + offset = UNSAFE.objectFieldOffset(primitiveField1); + assert offset % 8 == 0; + primitiveField0Offset = offset; + } } catch (NoSuchFieldException | SecurityException e) { throw new AssertionError(e); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/UnsafeEATest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/UnsafeEATest.java index 726f42adf8b..4ce92748330 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/UnsafeEATest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/ea/UnsafeEATest.java @@ -22,14 +22,26 @@ */ package org.graalvm.compiler.core.test.ea; -import jdk.vm.ci.meta.JavaConstant; +import java.nio.ByteBuffer; +import org.graalvm.compiler.api.directives.GraalDirectives; +import org.graalvm.compiler.graph.Graph; +import org.graalvm.compiler.graph.Node; +import org.graalvm.compiler.nodes.NamedLocationIdentity; +import org.graalvm.compiler.nodes.PhiNode; +import org.graalvm.compiler.nodes.ValuePhiNode; +import org.graalvm.compiler.nodes.calc.UnpackEndianHalfNode; +import org.graalvm.compiler.nodes.extended.RawLoadNode; +import org.graalvm.compiler.nodes.extended.RawStoreNode; +import org.graalvm.compiler.nodes.extended.UnsafeAccessNode; +import org.graalvm.compiler.nodes.java.LoadFieldNode; +import org.graalvm.compiler.phases.common.CanonicalizerPhase; import org.junit.Assert; import org.junit.Test; -import org.graalvm.compiler.nodes.PhiNode; -import org.graalvm.compiler.nodes.ValuePhiNode; -import org.graalvm.compiler.nodes.java.LoadFieldNode; +import jdk.vm.ci.meta.JavaConstant; +import jdk.vm.ci.meta.JavaKind; +import jdk.vm.ci.meta.ResolvedJavaMethod; public class UnsafeEATest extends EATestBase { @@ -56,6 +68,64 @@ public class UnsafeEATest extends EATestBase { } } + @Override + protected void testEscapeAnalysis(String snippet, JavaConstant expectedConstantResult, boolean iterativeEscapeAnalysis) { + // Exercise both a graph containing UnsafeAccessNodes and one which has been possibly been + // canonicalized into AccessFieldNodes. + testingUnsafe = true; + super.testEscapeAnalysis(snippet, expectedConstantResult, iterativeEscapeAnalysis); + testingUnsafe = false; + super.testEscapeAnalysis(snippet, expectedConstantResult, iterativeEscapeAnalysis); + if (expectedConstantResult != null) { + // Check that a compiled version of this method returns the same value if we expect a + // constant result. + ResolvedJavaMethod method = getResolvedJavaMethod(snippet); + JavaKind[] javaKinds = method.getSignature().toParameterKinds(false); + Object[] args = new Object[javaKinds.length]; + int i = 0; + for (JavaKind k : javaKinds) { + args[i++] = JavaConstant.defaultForKind(k).asBoxedPrimitive(); + } + Result result = executeExpected(method, null, args); + assertTrue(result.returnValue.equals(expectedConstantResult.asBoxedPrimitive())); + } + } + + @Override + protected void canonicalizeGraph() { + if (testingUnsafe) { + // For testing purposes we'd like to ensure that our raw unsafe operations stay as + // unsafe nodes, so force them to appear to have LocationIdentity.any to disable + // transformation into field access nodes. + for (Node node : graph.getNodes().filter(x -> x instanceof UnsafeAccessNode).snapshot()) { + if (node instanceof RawStoreNode) { + RawStoreNode store = (RawStoreNode) node; + RawStoreNode newStore = graph.add(new RawStoreNode(store.object(), store.offset(), store.value(), store.accessKind(), NamedLocationIdentity.any(), + store.needsBarrier(), store.stateAfter(), true)); + graph.replaceFixedWithFixed(store, newStore); + } else if (node instanceof RawLoadNode) { + RawLoadNode load = (RawLoadNode) node; + RawLoadNode newLoad = graph.add(new RawLoadNode(load.object(), load.offset(), load.accessKind(), NamedLocationIdentity.any(), + true)); + graph.replaceFixedWithFixed(load, newLoad); + } + } + } + super.canonicalizeGraph(); + } + + @Override + protected void postEACanonicalizeGraph() { + // Simplify any UnpackEndianHalfNode so we end up with constants. + Graph.Mark mark = graph.getMark(); + for (UnpackEndianHalfNode node : graph.getNodes().filter(UnpackEndianHalfNode.class)) { + node.lower(getTarget().arch.getByteOrder()); + } + new CanonicalizerPhase().applyIncremental(graph, context, mark); + } + + private boolean testingUnsafe; + @Test public void testSimpleInt() { testEscapeAnalysis("testSimpleIntSnippet", JavaConstant.forInt(101), false); @@ -89,6 +159,82 @@ public class UnsafeEATest extends EATestBase { return UNSAFE.getDouble(x, fieldOffset1); } + @Test + public void testSimpleDoubleOverwriteWithInt() { + testEscapeAnalysis("testSimpleDoubleOverwriteWithIntSnippet", JavaConstant.forInt(10), false); + } + + public static int testSimpleDoubleOverwriteWithIntSnippet() { + TestClassInt x = new TestClassInt(); + UNSAFE.putDouble(x, fieldOffset1, 10.1); + UNSAFE.putInt(x, fieldOffset1, 10); + return UNSAFE.getInt(x, fieldOffset1); + } + + @Test + public void testSimpleDoubleOverwriteWithSecondInt() { + ByteBuffer bb = ByteBuffer.allocate(8).order(getTarget().arch.getByteOrder()); + bb.putDouble(10.1); + int value = bb.getInt(4); + + testEscapeAnalysis("testSimpleDoubleOverwriteWithSecondIntSnippet", JavaConstant.forInt(value), false); + } + + public static int testSimpleDoubleOverwriteWithSecondIntSnippet() { + TestClassInt x = new TestClassInt(); + UNSAFE.putDouble(x, fieldOffset1, 10.1); + UNSAFE.putInt(x, fieldOffset1, 10); + return UNSAFE.getInt(x, fieldOffset2); + } + + @Test + public void testSimpleDoubleOverwriteWithFirstInt() { + ByteBuffer bb = ByteBuffer.allocate(8).order(getTarget().arch.getByteOrder()); + bb.putDouble(10.1); + int value = bb.getInt(0); + + testEscapeAnalysis("testSimpleDoubleOverwriteWithFirstIntSnippet", JavaConstant.forInt(value), false); + } + + public static int testSimpleDoubleOverwriteWithFirstIntSnippet() { + TestClassInt x = new TestClassInt(); + UNSAFE.putDouble(x, fieldOffset1, 10.1); + UNSAFE.putInt(x, fieldOffset2, 10); + return UNSAFE.getInt(x, fieldOffset1); + } + + @Test + public void testSimpleLongOverwriteWithSecondInt() { + ByteBuffer bb = ByteBuffer.allocate(8).order(getTarget().arch.getByteOrder()); + bb.putLong(0, 0x1122334455667788L); + int value = bb.getInt(4); + + testEscapeAnalysis("testSimpleLongOverwriteWithSecondIntSnippet", JavaConstant.forInt(value), false); + } + + public static int testSimpleLongOverwriteWithSecondIntSnippet() { + TestClassInt x = new TestClassInt(); + UNSAFE.putLong(x, fieldOffset1, 0x1122334455667788L); + UNSAFE.putInt(x, fieldOffset1, 10); + return UNSAFE.getInt(x, fieldOffset2); + } + + @Test + public void testSimpleLongOverwriteWithFirstInt() { + ByteBuffer bb = ByteBuffer.allocate(8).order(getTarget().arch.getByteOrder()); + bb.putLong(0, 0x1122334455667788L); + int value = bb.getInt(0); + + testEscapeAnalysis("testSimpleLongOverwriteWithFirstIntSnippet", JavaConstant.forInt(value), false); + } + + public static int testSimpleLongOverwriteWithFirstIntSnippet() { + TestClassInt x = new TestClassInt(); + UNSAFE.putLong(x, fieldOffset1, 0x1122334455667788L); + UNSAFE.putInt(x, fieldOffset2, 10); + return UNSAFE.getInt(x, fieldOffset1); + } + @Test public void testMergedDouble() { testEscapeAnalysis("testMergedDoubleSnippet", null, false); @@ -111,6 +257,32 @@ public class UnsafeEATest extends EATestBase { return UNSAFE.getDouble(x, fieldOffset1); } + static class ExtendedTestClassInt extends TestClassInt { + public long l; + } + + @Test + public void testMergedVirtualObjects() { + testEscapeAnalysis("testMergedVirtualObjectsSnippet", null, false); + } + + public static TestClassInt testMergedVirtualObjectsSnippet(int value) { + TestClassInt x; + if (value == 1) { + x = new TestClassInt(); + UNSAFE.putDouble(x, fieldOffset1, 10); + } else { + x = new TestClassInt(); + UNSAFE.putInt(x, fieldOffset1, 0); + } + UNSAFE.putInt(x, fieldOffset1, 0); + if (value == 2) { + UNSAFE.putInt(x, fieldOffset2, 0); + } + GraalDirectives.deoptimizeAndInvalidate(); + return x; + } + @Test public void testMaterializedDouble() { test("testMaterializedDoubleSnippet"); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/inlining/NestedLoopEffectsPhaseComplexityTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/inlining/NestedLoopEffectsPhaseComplexityTest.java index 10505f2de92..0b1468bd7d1 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/inlining/NestedLoopEffectsPhaseComplexityTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core.test/src/org/graalvm/compiler/core/test/inlining/NestedLoopEffectsPhaseComplexityTest.java @@ -146,7 +146,7 @@ public class NestedLoopEffectsPhaseComplexityTest extends GraalCompilerTest { private StructuredGraph parseBytecodes(ResolvedJavaMethod method, HighTierContext context, CanonicalizerPhase canonicalizer) { OptionValues options = getInitialOptions(); - StructuredGraph newGraph = new StructuredGraph.Builder(options, getDebugContext(options), AllowAssumptions.NO).method(method).build(); + StructuredGraph newGraph = new StructuredGraph.Builder(options, getDebugContext(options, null, method), AllowAssumptions.NO).method(method).build(); context.getGraphBuilderSuite().apply(newGraph, context); new DeadCodeEliminationPhase(Optional).apply(newGraph); canonicalizer.apply(newGraph, context); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/GraalCompilerOptions.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/GraalCompilerOptions.java index 44adbe6c112..f19c83ac810 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/GraalCompilerOptions.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/GraalCompilerOptions.java @@ -47,7 +47,7 @@ public class GraalCompilerOptions { public static final EnumOptionKey CompilationFailureAction = new EnumOptionKey<>(ExceptionAction.Diagnose); @Option(help = "The maximum number of compilation failures or bailouts to handle with the action specified " + "by CompilationFailureAction or CompilationBailoutAction before changing to a less verbose action.", type = OptionType.User) - public static final OptionKey MaxCompilationProblemsPerAction = new OptionKey<>(5); + public static final OptionKey MaxCompilationProblemsPerAction = new OptionKey<>(2); @Option(help = "Alias for CompilationFailureAction=ExitVM.", type = OptionType.User) public static final OptionKey ExitVMOnException = new OptionKey<>(false); // @formatter:on diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/gen/DebugInfoBuilder.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/gen/DebugInfoBuilder.java index a48abd8cb62..63dd347b546 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/gen/DebugInfoBuilder.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/gen/DebugInfoBuilder.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2011, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2017, 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 @@ -132,8 +132,17 @@ public class DebugInfoBuilder { slotKinds[pos] = toSlotKind(value); pos++; } else { - assert currentField.values().get(i - 1).getStackKind() == JavaKind.Double || currentField.values().get(i - 1).getStackKind() == JavaKind.Long : vobjNode + " " + i + " " + - currentField.values().get(i - 1); + assert value.getStackKind() == JavaKind.Illegal; + ValueNode previousValue = currentField.values().get(i - 1); + assert (previousValue != null && previousValue.getStackKind().needsTwoSlots()) : vobjNode + " " + i + + " " + previousValue + " " + currentField.values().snapshot(); + if (previousValue == null || !previousValue.getStackKind().needsTwoSlots()) { + // Don't allow the IllegalConstant to leak into the debug info + JavaKind entryKind = vobjNode.entryKind(i); + values[pos] = JavaConstant.defaultForKind(entryKind.getStackKind()); + slotKinds[pos] = entryKind.getStackKind(); + pos++; + } } } if (pos != entryCount) { @@ -164,19 +173,19 @@ public class DebugInfoBuilder { if (!type.isArray()) { ResolvedJavaField[] fields = type.getInstanceFields(true); int fieldIndex = 0; - for (int i = 0; i < values.length; i++) { - ResolvedJavaField field = fields[fieldIndex++]; - JavaKind valKind = slotKinds[i].getStackKind(); + for (int valueIndex = 0; valueIndex < values.length; valueIndex++, fieldIndex++) { + ResolvedJavaField field = fields[fieldIndex]; + JavaKind valKind = slotKinds[valueIndex].getStackKind(); JavaKind fieldKind = storageKind(field.getType()); - if (fieldKind == JavaKind.Object) { - assert valKind.isObject() : field + ": " + valKind + " != " + fieldKind; + if ((valKind == JavaKind.Double || valKind == JavaKind.Long) && fieldKind == JavaKind.Int) { + assert fieldIndex + 1 < fields.length : String.format("Not enough fields for fieldIndex = %d valueIndex = %d %s %s", fieldIndex, valueIndex, Arrays.toString(fields), + Arrays.toString(values)); + assert storageKind(fields[fieldIndex + 1].getType()) == JavaKind.Int : String.format("fieldIndex = %d valueIndex = %d %s %s %s", fieldIndex, valueIndex, + storageKind(fields[fieldIndex + 1].getType()), Arrays.toString(fields), + Arrays.toString(values)); + fieldIndex++; } else { - if ((valKind == JavaKind.Double || valKind == JavaKind.Long) && fieldKind == JavaKind.Int) { - assert storageKind(fields[fieldIndex].getType()) == JavaKind.Int; - fieldIndex++; - } else { - assert valKind == fieldKind.getStackKind() : field + ": " + valKind + " != " + fieldKind; - } + assert valKind == fieldKind.getStackKind() : field + ": " + valKind + " != " + fieldKind; } } assert fields.length == fieldIndex : type + ": fields=" + Arrays.toString(fields) + ", field values=" + Arrays.toString(values); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/gen/NodeLIRBuilder.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/gen/NodeLIRBuilder.java index b5ca686a857..fffe121e077 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/gen/NodeLIRBuilder.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/gen/NodeLIRBuilder.java @@ -59,6 +59,7 @@ import org.graalvm.compiler.lir.StandardOp.LabelOp; import org.graalvm.compiler.lir.SwitchStrategy; import org.graalvm.compiler.lir.Variable; import org.graalvm.compiler.lir.debug.LIRGenerationDebugContext; +import org.graalvm.compiler.lir.framemap.FrameMapBuilder; import org.graalvm.compiler.lir.gen.LIRGenerator; import org.graalvm.compiler.lir.gen.LIRGenerator.Options; import org.graalvm.compiler.lir.gen.LIRGeneratorTool; @@ -577,9 +578,9 @@ public abstract class NodeLIRBuilder implements NodeLIRBuilderTool, LIRGeneratio @Override public void emitInvoke(Invoke x) { LoweredCallTargetNode callTarget = (LoweredCallTargetNode) x.callTarget(); - CallingConvention invokeCc = gen.getResult().getFrameMapBuilder().getRegisterConfig().getCallingConvention(callTarget.callType(), x.asNode().stamp().javaType(gen.getMetaAccess()), - callTarget.signature(), gen); - gen.getResult().getFrameMapBuilder().callsMethod(invokeCc); + FrameMapBuilder frameMapBuilder = gen.getResult().getFrameMapBuilder(); + CallingConvention invokeCc = frameMapBuilder.getRegisterConfig().getCallingConvention(callTarget.callType(), x.asNode().stamp().javaType(gen.getMetaAccess()), callTarget.signature(), gen); + frameMapBuilder.callsMethod(invokeCc); Value[] parameters = visitInvokeArguments(invokeCc, callTarget.arguments()); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/target/Backend.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/target/Backend.java index c5d39002411..820a221e2e7 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/target/Backend.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.core/src/org/graalvm/compiler/core/target/Backend.java @@ -195,9 +195,12 @@ public abstract class Backend implements TargetProvider, ValueKindFactory parentOutput; /** * Stores the {@link MetricKey} values. @@ -111,6 +116,19 @@ public final class DebugContext implements AutoCloseable { return immutable.scopesEnabled; } + public GraphOutput buildOutput(GraphOutput.Builder builder) throws IOException { + if (parentOutput != null) { + return builder.build(parentOutput); + } else { + if (sharedChannel == null) { + sharedChannel = new IgvDumpChannel(() -> getDumpPath(".bgv", false), immutable.options); + } + final GraphOutput output = builder.build(sharedChannel); + parentOutput = output; + return output; + } + } + /** * The immutable configuration that can be shared between {@link DebugContext} objects. */ @@ -323,6 +341,14 @@ public final class DebugContext implements AutoCloseable { String compilableName = compilable instanceof JavaMethod ? ((JavaMethod) compilable).format("%H.%n(%p)%R") : String.valueOf(compilable); return identifier + ":" + compilableName; } + + final String getLabel() { + if (compilable instanceof JavaMethod) { + JavaMethod method = (JavaMethod) compilable; + return method.format("%h.%n(%p)%r"); + } + return String.valueOf(compilable); + } } private final Description description; @@ -394,6 +420,20 @@ public final class DebugContext implements AutoCloseable { } } + public Path getDumpPath(String extension, boolean directory) { + try { + String id = description == null ? null : description.identifier; + String label = description == null ? null : description.getLabel(); + Path result = PathUtilities.createUnique(immutable.options, DumpPath, id, label, extension, directory); + if (ShowDumpFiles.getValue(immutable.options)) { + TTY.println("Dumping debug output to %s", result.toAbsolutePath().toString()); + } + return result; + } catch (IOException ex) { + throw rethrowSilently(RuntimeException.class, ex); + } + } + /** * A special dump level that indicates the dumping machinery is enabled but no dumps will be * produced except through other options. @@ -2043,4 +2083,9 @@ public final class DebugContext implements AutoCloseable { } out.println(); } + + @SuppressWarnings({"unused", "unchecked"}) + private static E rethrowSilently(Class type, Throwable ex) throws E { + throw (E) ex; + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/DebugHandlersFactory.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/DebugHandlersFactory.java index 62fb5668198..169a4418e32 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/DebugHandlersFactory.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/DebugHandlersFactory.java @@ -35,6 +35,9 @@ public interface DebugHandlersFactory { /** * Creates {@link DebugHandler}s based on {@code options}. + * + * @param options options to control type and name of the channel + * @return list of debug handers that have been created */ List createHandlers(OptionValues options); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/DebugOptions.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/DebugOptions.java index 5aee4ccecb0..0df3e7311b1 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/DebugOptions.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/DebugOptions.java @@ -128,8 +128,6 @@ public class DebugOptions { public static final OptionKey PrintGraphProbabilities = new OptionKey<>(false); @Option(help = "Enable dumping to the IdealGraphVisualizer.", type = OptionType.Debug) public static final OptionKey PrintGraph = new OptionKey<>(true); - @Option(help = "Dump graphs in binary format instead of XML format.", type = OptionType.Debug) - public static final OptionKey PrintBinaryGraphs = new OptionKey<>(true); @Option(help = "Print graphs to files instead of sending them over the network.", type = OptionType.Debug) public static final OptionKey PrintGraphFile = new OptionKey<>(false); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/IgvDumpChannel.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/IgvDumpChannel.java new file mode 100644 index 00000000000..e7dc72c7ab7 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/IgvDumpChannel.java @@ -0,0 +1,119 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.debug; + +import java.io.IOException; +import java.io.InterruptedIOException; +import java.net.InetSocketAddress; +import java.nio.ByteBuffer; +import java.nio.channels.ClosedByInterruptException; +import java.nio.channels.FileChannel; +import java.nio.channels.SocketChannel; +import java.nio.channels.WritableByteChannel; +import java.nio.file.Path; +import java.nio.file.StandardOpenOption; +import java.util.function.Supplier; +import static org.graalvm.compiler.debug.DebugOptions.PrintBinaryGraphPort; +import static org.graalvm.compiler.debug.DebugOptions.PrintGraphHost; +import org.graalvm.compiler.options.OptionValues; + +final class IgvDumpChannel implements WritableByteChannel { + private final Supplier pathProvider; + private final OptionValues options; + private WritableByteChannel sharedChannel; + private boolean closed; + + IgvDumpChannel(Supplier pathProvider, OptionValues options) { + this.pathProvider = pathProvider; + this.options = options; + } + + @Override + public int write(ByteBuffer src) throws IOException { + return channel().write(src); + } + + @Override + public boolean isOpen() { + return !closed; + } + + @Override + public void close() throws IOException { + } + + void realClose() throws IOException { + closed = true; + if (sharedChannel != null) { + sharedChannel.close(); + sharedChannel = null; + } + } + + WritableByteChannel channel() throws IOException { + if (closed) { + throw new IOException(); + } + if (sharedChannel == null) { + if (DebugOptions.PrintGraphFile.getValue(options)) { + sharedChannel = createFileChannel(pathProvider); + } else { + sharedChannel = createNetworkChannel(pathProvider, options); + } + } + return sharedChannel; + } + + private static WritableByteChannel createNetworkChannel(Supplier pathProvider, OptionValues options) throws IOException { + String host = PrintGraphHost.getValue(options); + int port = PrintBinaryGraphPort.getValue(options); + try { + WritableByteChannel channel = SocketChannel.open(new InetSocketAddress(host, port)); + TTY.println("Connected to the IGV on %s:%d", host, port); + return channel; + } catch (ClosedByInterruptException | InterruptedIOException e) { + /* + * Interrupts should not count as errors because they may be caused by a cancelled Graal + * compilation. ClosedByInterruptException occurs if the SocketChannel could not be + * opened. InterruptedIOException occurs if new Socket(..) was interrupted. + */ + return null; + } catch (IOException e) { + if (!DebugOptions.PrintGraphFile.hasBeenSet(options)) { + return createFileChannel(pathProvider); + } else { + throw new IOException(String.format("Could not connect to the IGV on %s:%d", host, port), e); + } + } + } + + private static WritableByteChannel createFileChannel(Supplier pathProvider) throws IOException { + Path path = pathProvider.get(); + try { + return FileChannel.open(path, StandardOpenOption.WRITE, StandardOpenOption.CREATE); + } catch (IOException e) { + throw new IOException(String.format("Failed to open %s to dump IGV graphs", path), e); + } + } + +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/PathUtilities.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/PathUtilities.java index 54bb4e78144..e9f1e216faf 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/PathUtilities.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.debug/src/org/graalvm/compiler/debug/PathUtilities.java @@ -22,12 +22,13 @@ */ package org.graalvm.compiler.debug; +import java.io.File; import java.io.IOException; +import java.nio.file.FileAlreadyExistsException; +import java.nio.file.Files; import java.nio.file.InvalidPathException; import java.nio.file.Path; import java.nio.file.Paths; -import java.util.HashMap; -import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicLong; import org.graalvm.compiler.options.OptionKey; @@ -39,54 +40,6 @@ import org.graalvm.compiler.options.OptionValues; public class PathUtilities { private static final AtomicLong globalTimeStamp = new AtomicLong(); - /** - * This generates a per thread persistent id to aid mapping related dump files with each other. - */ - private static final ThreadLocal threadDumpId = new ThreadLocal<>(); - private static final AtomicInteger dumpId = new AtomicInteger(); - - static class PerThreadSequence { - final int threadID; - HashMap sequences = new HashMap<>(2); - - PerThreadSequence(int threadID) { - this.threadID = threadID; - } - - String generateID(String extension) { - Integer box = sequences.get(extension); - if (box == null) { - sequences.put(extension, 1); - return Integer.toString(threadID); - } else { - sequences.put(extension, box + 1); - return Integer.toString(threadID) + '-' + box; - } - } - } - - private static String getThreadDumpId(String extension) { - PerThreadSequence id = threadDumpId.get(); - if (id == null) { - id = new PerThreadSequence(dumpId.incrementAndGet()); - threadDumpId.set(id); - } - return id.generateID(extension); - } - - /** - * Prepends a period (i.e., {@code '.'}) to an non-null, non-empty string representation a file - * extension if the string does not already start with a period. - * - * @return {@code ext} unmodified if it is null, empty or already starts with a period other - * {@code "." + ext} - */ - public static String formatExtension(String ext) { - if (ext == null || ext.length() == 0) { - return ""; - } - return "." + ext; - } /** * Gets a time stamp for the current process. This method will always return the same value for @@ -99,43 +52,6 @@ public class PathUtilities { return globalTimeStamp.get(); } - /** - * Generates a {@link Path} using the format "%s-%d_%d%s" with the {@code baseNameOption}, a - * {@link #getGlobalTimeStamp() global timestamp} , {@link #getThreadDumpId a per thread unique - * id} and an optional {@code extension}. - * - * @return the output file path or null if the flag is null - */ - public static Path getPath(OptionValues options, OptionKey baseNameOption, String extension) throws IOException { - return getPath(options, baseNameOption, extension, true); - } - - /** - * Generate a {@link Path} using the format "%s-%d_%s" with the {@code baseNameOption}, a - * {@link #getGlobalTimeStamp() global timestamp} and an optional {@code extension} . - * - * @return the output file path or null if the flag is null - */ - public static Path getPathGlobal(OptionValues options, OptionKey baseNameOption, String extension) throws IOException { - return getPath(options, baseNameOption, extension, false); - } - - private static Path getPath(OptionValues options, OptionKey baseNameOption, String extension, boolean includeThreadId) throws IOException { - if (baseNameOption.getValue(options) == null) { - return null; - } - String ext = formatExtension(extension); - final String name = includeThreadId - ? String.format("%s-%d_%s%s", baseNameOption.getValue(options), getGlobalTimeStamp(), getThreadDumpId(ext), ext) - : String.format("%s-%d%s", baseNameOption.getValue(options), getGlobalTimeStamp(), ext); - Path result = Paths.get(name); - if (result.isAbsolute()) { - return result; - } - Path dumpDir = DebugOptions.getDumpDirectory(options); - return dumpDir.resolve(name).normalize(); - } - /** * Gets a value based on {@code name} that can be passed to {@link Paths#get(String, String...)} * without causing an {@link InvalidPathException}. @@ -145,21 +61,80 @@ public class PathUtilities { */ public static String sanitizeFileName(String name) { try { - Paths.get(name); - return name; + Path path = Paths.get(name); + if (path.getNameCount() == 0) { + return name; + } } catch (InvalidPathException e) { // fall through } StringBuilder buf = new StringBuilder(name.length()); for (int i = 0; i < name.length(); i++) { char c = name.charAt(i); - try { - Paths.get(String.valueOf(c)); - } catch (InvalidPathException e) { - buf.append('_'); + if (c != File.separatorChar && c != ' ' && !Character.isISOControl(c)) { + try { + Paths.get(String.valueOf(c)); + buf.append(c); + continue; + } catch (InvalidPathException e) { + } } - buf.append(c); + buf.append('_'); } return buf.toString(); } + + /** + * A maximum file name length supported by most file systems. There is no platform independent + * way to get this in Java. + */ + private static final int MAX_FILE_NAME_LENGTH = 255; + + private static final String ELLIPSIS = "..."; + + static Path createUnique(OptionValues options, OptionKey baseNameOption, String id, String label, String ext, boolean createDirectory) throws IOException { + String uniqueTag = ""; + int dumpCounter = 1; + String prefix; + if (id == null) { + prefix = baseNameOption.getValue(options); + int slash = prefix.lastIndexOf(File.separatorChar); + prefix = prefix.substring(slash + 1); + } else { + prefix = id; + } + for (;;) { + int fileNameLengthWithoutLabel = uniqueTag.length() + ext.length() + prefix.length() + "[]".length(); + int labelLengthLimit = MAX_FILE_NAME_LENGTH - fileNameLengthWithoutLabel; + String fileName; + if (labelLengthLimit < ELLIPSIS.length()) { + // This means `id` is very long + String suffix = uniqueTag + ext; + int idLengthLimit = Math.min(MAX_FILE_NAME_LENGTH - suffix.length(), prefix.length()); + fileName = sanitizeFileName(prefix.substring(0, idLengthLimit) + suffix); + } else { + if (label == null) { + fileName = sanitizeFileName(prefix + uniqueTag + ext); + } else { + String adjustedLabel = label; + if (label.length() > labelLengthLimit) { + adjustedLabel = label.substring(0, labelLengthLimit - ELLIPSIS.length()) + ELLIPSIS; + } + fileName = sanitizeFileName(prefix + '[' + adjustedLabel + ']' + uniqueTag + ext); + } + } + Path dumpDir = DebugOptions.getDumpDirectory(options); + Path result = Paths.get(dumpDir.toString(), fileName); + try { + if (createDirectory) { + return Files.createDirectory(result); + } else { + return Files.createFile(result); + } + } catch (FileAlreadyExistsException e) { + uniqueTag = "_" + dumpCounter++; + } + } + } + } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph.test/src/org/graalvm/compiler/graph/test/graphio/GraphSnippetTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph.test/src/org/graalvm/compiler/graph/test/graphio/GraphSnippetTest.java new file mode 100644 index 00000000000..b3c790e3926 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph.test/src/org/graalvm/compiler/graph/test/graphio/GraphSnippetTest.java @@ -0,0 +1,51 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.graph.test.graphio; + +import java.io.File; +import java.lang.reflect.Method; +import static org.junit.Assert.assertTrue; +import org.junit.Assume; +import org.junit.Test; + +public class GraphSnippetTest { + @Test + public void dumpTheFile() throws Exception { + Class snippets = null; + try { + snippets = Class.forName("org.graalvm.graphio.GraphSnippets"); + } catch (ClassNotFoundException notFound) { + Assume.assumeNoException("The snippets class has to be around", notFound); + } + Method dump = null; + try { + dump = snippets.getDeclaredMethod("dump", File.class); + dump.setAccessible(true); + } catch (RuntimeException ex) { + Assume.assumeTrue("Only run the test, if the method is accessible", dump != null && dump.isAccessible()); + } + File diamond = File.createTempFile("diamond", ".bgv"); + dump.invoke(null, diamond); + assertTrue("File .bgv created: " + diamond, diamond.length() > 50); + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph.test/src/org/graalvm/compiler/graph/test/graphio/NodeEncodingTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph.test/src/org/graalvm/compiler/graph/test/graphio/NodeEncodingTest.java new file mode 100644 index 00000000000..a5e4c6862ea --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph.test/src/org/graalvm/compiler/graph/test/graphio/NodeEncodingTest.java @@ -0,0 +1,311 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.graph.test.graphio; + +import java.io.ByteArrayOutputStream; +import java.nio.channels.Channels; +import java.nio.channels.WritableByteChannel; +import java.util.Arrays; +import java.util.Collection; +import java.util.Collections; +import java.util.LinkedHashMap; +import java.util.Map; +import org.graalvm.graphio.GraphOutput; +import org.graalvm.graphio.GraphStructure; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; +import static org.junit.Assert.fail; +import org.junit.Before; +import org.junit.Test; + +public final class NodeEncodingTest { + + private ByteArrayOutputStream out; + + @Before + public void initOutput() { + out = new ByteArrayOutputStream(); + } + + @Test + public void version40TheNodeIsntDumpedWithItsID() throws Exception { + runTheNodeIsntDumpedWithItsID(true); + } + + @Test + public void defaultVersionTheNodeIsntDumpedWithItsID() throws Exception { + runTheNodeIsntDumpedWithItsID(false); + } + + private void runTheNodeIsntDumpedWithItsID(boolean explicitVersion) throws Exception { + WritableByteChannel w = Channels.newChannel(out); + MockGraph graph = new MockGraph(); + MockNodeClass clazz = new MockNodeClass("clazz"); + MockNode node = new MockNode(clazz, 33); // random value otherwise not found in the stream + try (GraphOutput dump = explicitVersion ? GraphOutput.newBuilder(new MockStructure()).protocolVersion(4, 0).build(w) : GraphOutput.newBuilder(new MockStructure()).build(w)) { + dump.beginGroup(graph, "test1", "t1", null, 0, Collections.singletonMap("node", node)); + dump.endGroup(); + } + + assertEquals("Node is always requested", 1, node.nodeRequested); + assertEquals("Nobody asks for id of a node in version 4.0", 0, node.idTested); + assertByte(false, out.toByteArray(), 33); + assertEquals("Node class of the node has been requested", 1, node.nodeClassRequested); + assertEquals("Node class template name stored", 1, clazz.nameTemplateQueried); + assertFalse("No to string ops", node.toStringRequested); + } + + @Test + public void dumpingNodeInVersion10() throws Exception { + runTheNodeIsTreatedAsString(true); + } + + private void runTheNodeIsTreatedAsString(boolean explicitVersion) throws Exception { + WritableByteChannel w = Channels.newChannel(out); + MockGraph graph = new MockGraph(); + MockNodeClass clazz = new MockNodeClass("clazz"); + MockNode node = new MockNode(clazz, 33); // random value otherwise not found in the stream + try (GraphOutput dump = explicitVersion ? GraphOutput.newBuilder(new MockStructure()).protocolVersion(1, 0).build(w) : GraphOutput.newBuilder(new MockStructure()).build(w)) { + dump.beginGroup(graph, "test1", "t1", null, 0, Collections.singletonMap("node", node)); + dump.endGroup(); + } + + assertEquals("Node is always requested", 1, node.nodeRequested); + assertEquals("Nobody asks for id of a node in version 1.0", 0, node.idTested); + assertByte(false, out.toByteArray(), 33); + assertEquals("Node class was needed to find out it is not a NodeClass instance", 1, node.nodeClassRequested); + assertEquals("Node class template name wasn't needed however", 0, clazz.nameTemplateQueried); + assertTrue("Node sent as a string version 1.0", node.toStringRequested); + } + + @Test + public void dumpingNodeInVersion15() throws Exception { + runTheNodeIsTreatedPoolEntry(true); + } + + private void runTheNodeIsTreatedPoolEntry(boolean explicitVersion) throws Exception { + WritableByteChannel w = Channels.newChannel(out); + MockGraph graph = new MockGraph(); + MockNodeClass clazz = new MockNodeClass("clazz"); + MockNode node = new MockNode(clazz, 33); // random value otherwise not found in the stream + try (GraphOutput dump = explicitVersion ? GraphOutput.newBuilder(new MockStructure()).protocolVersion(5, 0).build(w) : GraphOutput.newBuilder(new MockStructure()).build(w)) { + dump.beginGroup(graph, "test1", "t1", null, 0, Collections.singletonMap("node", node)); + dump.endGroup(); + } + + assertEquals("Node is always requested", 1, node.nodeRequested); + assertEquals("Id of our node is requested in version 5.0", 1, node.idTested); + assertByte(true, out.toByteArray(), 33); + assertTrue("Node class was needed at least once", 1 <= node.nodeClassRequested); + assertEquals("Node class template name sent to server", 1, clazz.nameTemplateQueried); + assertFalse("Node.toString() isn't needed", node.toStringRequested); + } + + @Test + public void dumpingNodeTwiceInVersion4() throws Exception { + WritableByteChannel w = Channels.newChannel(out); + MockGraph graph = new MockGraph(); + MockNodeClass clazz = new MockNodeClass("clazz"); + MockNode node = new MockNode(clazz, 33); // random value otherwise not found in the stream + try (GraphOutput dump = GraphOutput.newBuilder(new MockStructure()).protocolVersion(4, 0).build(w)) { + Map props = new LinkedHashMap<>(); + props.put("node1", node); + props.put("node2", node); + props.put("node3", node); + dump.beginGroup(graph, "test1", "t1", null, 0, props); + dump.endGroup(); + } + + assertEquals("Node requested three times", 3, node.nodeRequested); + assertEquals("Nobody asks for id of a node in version 4.0", 0, node.idTested); + // check there is no encoded string for object #3 + assertByte(false, out.toByteArray(), 1, 0, 3); + assertEquals("Node class of the node has been requested three times", 3, node.nodeClassRequested); + assertEquals("Node class template name stored", 1, clazz.nameTemplateQueried); + assertFalse("No to string ops", node.toStringRequested); + } + + private static void assertByte(boolean shouldBeFound, byte[] arr, int... value) { + boolean found = false; + int at = 0; + for (int i = 0; i < arr.length; i++) { + if (arr[i] == value[at]) { + if (++at == value.length) { + found = true; + break; + } + } else { + at = 0; + } + } + if (shouldBeFound == found) { + return; + } + if (shouldBeFound) { + fail("Value " + value + " not found in\n" + Arrays.toString(arr)); + } else { + fail("Value " + value + " surprisingly found in\n" + Arrays.toString(arr)); + } + } + + private static final class MockStructure implements GraphStructure { + + @Override + public MockGraph graph(MockGraph currentGraph, Object obj) { + return obj instanceof MockGraph ? (MockGraph) obj : null; + } + + @Override + public Iterable nodes(MockGraph graph) { + return Collections.emptyList(); + } + + @Override + public int nodesCount(MockGraph graph) { + return 0; + } + + @Override + public int nodeId(MockNode node) { + node.idTested++; + return node.id; + } + + @Override + public boolean nodeHasPredecessor(MockNode node) { + return false; + } + + @Override + public void nodeProperties(MockGraph graph, MockNode node, Map properties) { + } + + @Override + public MockNode node(Object obj) { + if (obj instanceof MockNode) { + ((MockNode) obj).nodeRequested++; + return (MockNode) obj; + } + return null; + } + + @Override + public MockNodeClass nodeClass(Object obj) { + if (obj instanceof MockNode) { + ((MockNode) obj).nodeClassRequested++; + } + return obj instanceof MockNodeClass ? (MockNodeClass) obj : null; + } + + @Override + public MockNodeClass classForNode(MockNode n) { + n.nodeClassRequested++; + return n.clazz; + } + + @Override + public String nameTemplate(MockNodeClass nodeClass) { + nodeClass.nameTemplateQueried++; + return ""; + } + + @Override + public Object nodeClassType(MockNodeClass nodeClass) { + return nodeClass.getClass(); + } + + @Override + public MockNodeClass portInputs(MockNodeClass nodeClass) { + return nodeClass; + } + + @Override + public MockNodeClass portOutputs(MockNodeClass nodeClass) { + return nodeClass; + } + + @Override + public int portSize(MockNodeClass port) { + return 0; + } + + @Override + public boolean edgeDirect(MockNodeClass port, int index) { + return false; + } + + @Override + public String edgeName(MockNodeClass port, int index) { + return null; + } + + @Override + public Object edgeType(MockNodeClass port, int index) { + return null; + } + + @Override + public Collection edgeNodes(MockGraph graph, MockNode node, MockNodeClass port, int index) { + return null; + } + } + + private static final class MockGraph { + + } + + private static final class MockNode { + final MockNodeClass clazz; + final int id; + int idTested; + int nodeClassRequested; + int nodeRequested; + boolean toStringRequested; + + MockNode(MockNodeClass clazz, int id) { + this.clazz = clazz; + this.id = id; + } + + @Override + public String toString() { + this.toStringRequested = true; + return "MockNode{" + "id=" + id + ", class=" + clazz + '}'; + } + } + + private static final class MockNodeClass { + final String name; + int nameTemplateQueried; + + MockNodeClass(String name) { + this.name = name; + } + + @Override + public String toString() { + return "MockNodeClass{" + "name=" + name + '}'; + } + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph/src/org/graalvm/compiler/graph/Graph.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph/src/org/graalvm/compiler/graph/Graph.java index 509f11841f6..092a84a3aff 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph/src/org/graalvm/compiler/graph/Graph.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph/src/org/graalvm/compiler/graph/Graph.java @@ -514,30 +514,61 @@ public class Graph { /** * A node was added to a graph. */ - NODE_ADDED; + NODE_ADDED, + + /** + * A node was removed from the graph. + */ + NODE_REMOVED; } /** * Client interested in one or more node related events. */ - public interface NodeEventListener { + public abstract static class NodeEventListener { /** - * Default handler for events. + * A method called when a change event occurs. + * + * This method dispatches the event to user-defined triggers. The methods that change the + * graph (typically in Graph and Node) must call this method to dispatch the event. * * @param e an event * @param node the node related to {@code e} */ - default void event(NodeEvent e, Node node) { + final void event(NodeEvent e, Node node) { + switch (e) { + case INPUT_CHANGED: + inputChanged(node); + break; + case ZERO_USAGES: + usagesDroppedToZero(node); + break; + case NODE_ADDED: + nodeAdded(node); + break; + case NODE_REMOVED: + nodeRemoved(node); + break; + } + changed(e, node); } /** - * Notifies this listener of a change in a node's inputs. + * Notifies this listener about any change event in the graph. + * + * @param e an event + * @param node the node related to {@code e} + */ + public void changed(NodeEvent e, Node node) { + } + + /** + * Notifies this listener about a change in a node's inputs. * * @param node a node who has had one of its inputs changed */ - default void inputChanged(Node node) { - event(NodeEvent.INPUT_CHANGED, node); + public void inputChanged(Node node) { } /** @@ -545,8 +576,7 @@ public class Graph { * * @param node a node whose {@link Node#usages()} just became empty */ - default void usagesDroppedToZero(Node node) { - event(NodeEvent.ZERO_USAGES, node); + public void usagesDroppedToZero(Node node) { } /** @@ -554,8 +584,15 @@ public class Graph { * * @param node a node that was just added to the graph */ - default void nodeAdded(Node node) { - event(NodeEvent.NODE_ADDED, node); + public void nodeAdded(Node node) { + } + + /** + * Notifies this listener of a removed node. + * + * @param node + */ + public void nodeRemoved(Node node) { } } @@ -583,7 +620,7 @@ public class Graph { } } - private static class ChainedNodeEventListener implements NodeEventListener { + private static class ChainedNodeEventListener extends NodeEventListener { NodeEventListener head; NodeEventListener next; @@ -595,20 +632,32 @@ public class Graph { @Override public void nodeAdded(Node node) { - head.nodeAdded(node); - next.nodeAdded(node); + head.event(NodeEvent.NODE_ADDED, node); + next.event(NodeEvent.NODE_ADDED, node); } @Override public void inputChanged(Node node) { - head.inputChanged(node); - next.inputChanged(node); + head.event(NodeEvent.INPUT_CHANGED, node); + next.event(NodeEvent.INPUT_CHANGED, node); } @Override public void usagesDroppedToZero(Node node) { - head.usagesDroppedToZero(node); - next.usagesDroppedToZero(node); + head.event(NodeEvent.ZERO_USAGES, node); + next.event(NodeEvent.ZERO_USAGES, node); + } + + @Override + public void nodeRemoved(Node node) { + head.event(NodeEvent.NODE_REMOVED, node); + next.event(NodeEvent.NODE_REMOVED, node); + } + + @Override + public void changed(NodeEvent e, Node node) { + head.event(e, node); + next.event(e, node); } } @@ -1023,7 +1072,7 @@ public class Graph { updateNodeCaches(node); if (nodeEventListener != null) { - nodeEventListener.nodeAdded(node); + nodeEventListener.event(NodeEvent.NODE_ADDED, node); } afterRegister(node); } @@ -1085,6 +1134,10 @@ public class Graph { nodes[node.id] = null; nodesDeletedSinceLastCompression++; + if (nodeEventListener != null) { + nodeEventListener.event(NodeEvent.NODE_ADDED, node); + } + // nodes aren't removed from the type cache here - they will be removed during iteration } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph/src/org/graalvm/compiler/graph/Node.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph/src/org/graalvm/compiler/graph/Node.java index fa468561191..0f0c41299af 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph/src/org/graalvm/compiler/graph/Node.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.graph/src/org/graalvm/compiler/graph/Node.java @@ -752,7 +752,7 @@ public abstract class Node implements Cloneable, Formattable, NodeInterface { assert !graph.isFrozen(); NodeEventListener listener = graph.nodeEventListener; if (listener != null) { - listener.inputChanged(node); + listener.event(Graph.NodeEvent.INPUT_CHANGED, node); } } } @@ -762,7 +762,7 @@ public abstract class Node implements Cloneable, Formattable, NodeInterface { assert !graph.isFrozen(); NodeEventListener listener = graph.nodeEventListener; if (listener != null && node.isAlive()) { - listener.usagesDroppedToZero(node); + listener.event(Graph.NodeEvent.ZERO_USAGES, node); } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.aarch64/src/org/graalvm/compiler/hotspot/aarch64/AArch64HotSpotNodeLIRBuilder.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.aarch64/src/org/graalvm/compiler/hotspot/aarch64/AArch64HotSpotNodeLIRBuilder.java index 07ea09963ea..7865e886a87 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.aarch64/src/org/graalvm/compiler/hotspot/aarch64/AArch64HotSpotNodeLIRBuilder.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.aarch64/src/org/graalvm/compiler/hotspot/aarch64/AArch64HotSpotNodeLIRBuilder.java @@ -183,8 +183,7 @@ public class AArch64HotSpotNodeLIRBuilder extends AArch64NodeLIRBuilder implemen sig[i] = node.arguments().get(i).stamp().javaType(gen.getMetaAccess()); } - Value[] parameters = visitInvokeArguments(gen.getResult().getFrameMapBuilder().getRegisterConfig().getCallingConvention(HotSpotCallingConventionType.JavaCall, null, sig, gen), - node.arguments()); + Value[] parameters = visitInvokeArguments(gen.getRegisterConfig().getCallingConvention(HotSpotCallingConventionType.JavaCall, null, sig, gen), node.arguments()); append(new AArch64BreakpointOp(parameters)); } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotAddressLowering.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotAddressLowering.java index a26ed27391c..70bf4dae8c9 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotAddressLowering.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotAddressLowering.java @@ -32,7 +32,6 @@ import org.graalvm.compiler.core.amd64.AMD64AddressLowering; import org.graalvm.compiler.core.amd64.AMD64AddressNode; import org.graalvm.compiler.core.common.CompressEncoding; import org.graalvm.compiler.core.common.LIRKind; -import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.core.common.type.ObjectStamp; import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.debug.CounterKey; @@ -44,6 +43,7 @@ import org.graalvm.compiler.hotspot.nodes.type.KlassPointerStamp; import org.graalvm.compiler.nodeinfo.NodeInfo; import org.graalvm.compiler.nodes.CompressionNode; import org.graalvm.compiler.nodes.CompressionNode.CompressionOp; +import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.ValueNode; import org.graalvm.compiler.nodes.calc.FloatingNode; import org.graalvm.compiler.nodes.spi.LIRLowerable; @@ -93,76 +93,76 @@ public class AMD64HotSpotAddressLowering extends AMD64AddressLowering { } @Override - protected boolean improve(DebugContext debug, AMD64AddressNode addr) { - - boolean result = false; - - while (super.improve(debug, addr)) { - result = true; + protected boolean improve(StructuredGraph graph, DebugContext debug, AMD64AddressNode addr, boolean isBaseNegated, boolean isIndexNegated) { + if (super.improve(graph, debug, addr, isBaseNegated, isIndexNegated)) { + return true; } if (addr.getScale() == Scale.Times1) { if (addr.getIndex() instanceof CompressionNode) { - if (improveUncompression(addr, (CompressionNode) addr.getIndex(), addr.getBase())) { + if (improveUncompression(addr, (CompressionNode) addr.getIndex(), addr.getBase(), isBaseNegated, isIndexNegated)) { counterFoldedUncompressDuringAddressLowering.increment(debug); return true; } } if (addr.getBase() instanceof CompressionNode) { - if (improveUncompression(addr, (CompressionNode) addr.getBase(), addr.getIndex())) { + if (improveUncompression(addr, (CompressionNode) addr.getBase(), addr.getIndex(), isBaseNegated, isIndexNegated)) { counterFoldedUncompressDuringAddressLowering.increment(debug); return true; } } } - return result; + return false; } - private boolean improveUncompression(AMD64AddressNode addr, CompressionNode compression, ValueNode other) { - if (compression.getOp() == CompressionOp.Uncompress) { - CompressEncoding encoding = compression.getEncoding(); - Scale scale = Scale.fromShift(encoding.getShift()); - if (scale == null) { + @Override + protected boolean mightBeOptimized(ValueNode value) { + return super.mightBeOptimized(value) || value instanceof CompressionNode; + } + + private boolean improveUncompression(AMD64AddressNode addr, CompressionNode compression, ValueNode other, boolean isBaseNegated, boolean isIndexNegated) { + if (isBaseNegated || isIndexNegated || compression.getOp() != CompressionOp.Uncompress) { + return false; + } + + CompressEncoding encoding = compression.getEncoding(); + Scale scale = Scale.fromShift(encoding.getShift()); + if (scale == null) { + return false; + } + + if (heapBaseRegister != null && encoding.getBase() == heapBase) { + if ((!generatePIC || compression.stamp() instanceof ObjectStamp) && other == null) { + // With PIC it is only legal to do for oops since the base value may be + // different at runtime. + ValueNode base = compression.graph().unique(new HeapBaseNode(heapBaseRegister)); + addr.setBase(base); + } else { return false; } - - if (heapBaseRegister != null && encoding.getBase() == heapBase) { - if ((!generatePIC || compression.stamp() instanceof ObjectStamp) && other == null) { - // With PIC it is only legal to do for oops since the base value may be - // different at runtime. - ValueNode base = compression.graph().unique(new HeapBaseNode(heapBaseRegister)); + } else if (encoding.getBase() != 0 || (generatePIC && compression.stamp() instanceof KlassPointerStamp)) { + if (generatePIC) { + if (other == null) { + ValueNode base = compression.graph().unique(new GraalHotSpotVMConfigNode(config, config.MARKID_NARROW_KLASS_BASE_ADDRESS, JavaKind.Long)); addr.setBase(base); } else { return false; } - } else if (encoding.getBase() != 0 || (generatePIC && compression.stamp() instanceof KlassPointerStamp)) { - if (generatePIC) { - if (other == null) { - ValueNode base = compression.graph().unique(new GraalHotSpotVMConfigNode(config, config.MARKID_NARROW_KLASS_BASE_ADDRESS, JavaKind.Long)); - addr.setBase(base); - } else { - return false; - } - } else { - long disp = addr.getDisplacement() + encoding.getBase(); - if (NumUtil.isInt(disp)) { - addr.setDisplacement((int) disp); - addr.setBase(other); - } else { - return false; - } - } } else { - addr.setBase(other); + if (updateDisplacement(addr, encoding.getBase(), isBaseNegated)) { + addr.setBase(other); + } else { + return false; + } } - - addr.setScale(scale); - addr.setIndex(compression.getValue()); - return true; } else { - return false; + addr.setBase(other); } + + addr.setScale(scale); + addr.setIndex(compression.getValue()); + return true; } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotLIRGenerator.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotLIRGenerator.java index 22844e93006..3a26ff997a7 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotLIRGenerator.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotLIRGenerator.java @@ -39,7 +39,6 @@ import java.util.List; import org.graalvm.compiler.asm.amd64.AMD64Address.Scale; import org.graalvm.compiler.core.amd64.AMD64ArithmeticLIRGenerator; import org.graalvm.compiler.core.amd64.AMD64LIRGenerator; -import org.graalvm.compiler.core.amd64.AMD64LIRKindTool; import org.graalvm.compiler.core.amd64.AMD64MoveFactoryBase.BackupSlotProvider; import org.graalvm.compiler.core.common.CompressEncoding; import org.graalvm.compiler.core.common.LIRKind; @@ -116,7 +115,7 @@ public class AMD64HotSpotLIRGenerator extends AMD64LIRGenerator implements HotSp } private AMD64HotSpotLIRGenerator(HotSpotProviders providers, GraalHotSpotVMConfig config, LIRGenerationResult lirGenRes, BackupSlotProvider backupSlotProvider) { - this(new AMD64LIRKindTool(), new AMD64HotSpotArithmeticLIRGenerator(), new AMD64HotSpotMoveFactory(backupSlotProvider), providers, config, lirGenRes); + this(new AMD64HotSpotLIRKindTool(), new AMD64HotSpotArithmeticLIRGenerator(), new AMD64HotSpotMoveFactory(backupSlotProvider), providers, config, lirGenRes); } protected AMD64HotSpotLIRGenerator(LIRKindTool lirKindTool, AMD64ArithmeticLIRGenerator arithmeticLIRGen, MoveFactory moveFactory, HotSpotProviders providers, GraalHotSpotVMConfig config, @@ -363,7 +362,7 @@ public class AMD64HotSpotLIRGenerator extends AMD64LIRGenerator implements HotSp Stub stub = getStub(); if (destroysRegisters) { if (stub != null && stub.preservesRegisters()) { - Register[] savedRegisters = getResult().getFrameMapBuilder().getRegisterConfig().getAllocatableRegisters().toArray(); + Register[] savedRegisters = getRegisterConfig().getAllocatableRegisters().toArray(); save = emitSaveAllRegisters(savedRegisters, true); } } @@ -567,28 +566,29 @@ public class AMD64HotSpotLIRGenerator extends AMD64LIRGenerator implements HotSp @Override public Value emitCompress(Value pointer, CompressEncoding encoding, boolean nonNull) { LIRKind inputKind = pointer.getValueKind(LIRKind.class); - assert inputKind.getPlatformKind() == AMD64Kind.QWORD; + LIRKindTool lirKindTool = getLIRKindTool(); + assert inputKind.getPlatformKind() == lirKindTool.getObjectKind().getPlatformKind(); if (inputKind.isReference(0)) { // oop - Variable result = newVariable(LIRKind.reference(AMD64Kind.DWORD)); - append(new AMD64HotSpotMove.CompressPointer(result, asAllocatable(pointer), getProviders().getRegisters().getHeapBaseRegister().asValue(), encoding, nonNull)); + Variable result = newVariable(lirKindTool.getNarrowOopKind()); + append(new AMD64Move.CompressPointer(result, asAllocatable(pointer), getProviders().getRegisters().getHeapBaseRegister().asValue(), encoding, nonNull, getLIRKindTool())); return result; } else { // metaspace pointer - Variable result = newVariable(LIRKind.value(AMD64Kind.DWORD)); + Variable result = newVariable(lirKindTool.getNarrowPointerKind()); AllocatableValue base = Value.ILLEGAL; OptionValues options = getResult().getLIR().getOptions(); if (encoding.hasBase() || GeneratePIC.getValue(options)) { if (GeneratePIC.getValue(options)) { - Variable baseAddress = newVariable(LIRKind.value(AMD64Kind.QWORD)); + Variable baseAddress = newVariable(lirKindTool.getWordKind()); AMD64HotSpotMove.BaseMove move = new AMD64HotSpotMove.BaseMove(baseAddress, config); append(move); base = baseAddress; } else { - base = emitLoadConstant(LIRKind.value(AMD64Kind.QWORD), JavaConstant.forLong(encoding.getBase())); + base = emitLoadConstant(lirKindTool.getWordKind(), JavaConstant.forLong(encoding.getBase())); } } - append(new AMD64HotSpotMove.CompressPointer(result, asAllocatable(pointer), base, encoding, nonNull)); + append(new AMD64Move.CompressPointer(result, asAllocatable(pointer), base, encoding, nonNull, getLIRKindTool())); return result; } } @@ -596,35 +596,37 @@ public class AMD64HotSpotLIRGenerator extends AMD64LIRGenerator implements HotSp @Override public Value emitUncompress(Value pointer, CompressEncoding encoding, boolean nonNull) { LIRKind inputKind = pointer.getValueKind(LIRKind.class); - assert inputKind.getPlatformKind() == AMD64Kind.DWORD; + LIRKindTool lirKindTool = getLIRKindTool(); + assert inputKind.getPlatformKind() == lirKindTool.getNarrowOopKind().getPlatformKind(); if (inputKind.isReference(0)) { // oop - Variable result = newVariable(LIRKind.reference(AMD64Kind.QWORD)); - append(new AMD64HotSpotMove.UncompressPointer(result, asAllocatable(pointer), getProviders().getRegisters().getHeapBaseRegister().asValue(), encoding, nonNull)); + Variable result = newVariable(lirKindTool.getObjectKind()); + append(new AMD64Move.UncompressPointer(result, asAllocatable(pointer), getProviders().getRegisters().getHeapBaseRegister().asValue(), encoding, nonNull, lirKindTool)); return result; } else { // metaspace pointer - Variable result = newVariable(LIRKind.value(AMD64Kind.QWORD)); + LIRKind uncompressedKind = lirKindTool.getWordKind(); + Variable result = newVariable(uncompressedKind); AllocatableValue base = Value.ILLEGAL; OptionValues options = getResult().getLIR().getOptions(); if (encoding.hasBase() || GeneratePIC.getValue(options)) { if (GeneratePIC.getValue(options)) { - Variable baseAddress = newVariable(LIRKind.value(AMD64Kind.QWORD)); + Variable baseAddress = newVariable(uncompressedKind); AMD64HotSpotMove.BaseMove move = new AMD64HotSpotMove.BaseMove(baseAddress, config); append(move); base = baseAddress; } else { - base = emitLoadConstant(LIRKind.value(AMD64Kind.QWORD), JavaConstant.forLong(encoding.getBase())); + base = emitLoadConstant(uncompressedKind, JavaConstant.forLong(encoding.getBase())); } } - append(new AMD64HotSpotMove.UncompressPointer(result, asAllocatable(pointer), base, encoding, nonNull)); + append(new AMD64Move.UncompressPointer(result, asAllocatable(pointer), base, encoding, nonNull, lirKindTool)); return result; } } @Override public void emitNullCheck(Value address, LIRFrameState state) { - if (address.getValueKind().getPlatformKind() == AMD64Kind.DWORD) { + if (address.getValueKind().getPlatformKind() == getLIRKindTool().getNarrowOopKind().getPlatformKind()) { CompressEncoding encoding = config.getOopEncoding(); Value uncompressed; if (encoding.getShift() <= 3) { @@ -635,9 +637,9 @@ public class AMD64HotSpotLIRGenerator extends AMD64LIRGenerator implements HotSp uncompressed = emitUncompress(address, encoding, false); } append(new AMD64Move.NullCheckOp(asAddressValue(uncompressed), state)); - } else { - super.emitNullCheck(address, state); + return; } + super.emitNullCheck(address, state); } @Override diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotLIRKindTool.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotLIRKindTool.java new file mode 100644 index 00000000000..095a2a2e056 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotLIRKindTool.java @@ -0,0 +1,39 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.hotspot.amd64; + +import jdk.vm.ci.amd64.AMD64Kind; +import org.graalvm.compiler.core.amd64.AMD64LIRKindTool; +import org.graalvm.compiler.core.common.LIRKind; + +public class AMD64HotSpotLIRKindTool extends AMD64LIRKindTool { + @Override + public LIRKind getNarrowOopKind() { + return LIRKind.reference(AMD64Kind.DWORD); + } + + @Override + public LIRKind getNarrowPointerKind() { + return LIRKind.value(AMD64Kind.DWORD); + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotMove.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotMove.java index a5c75f5958b..4440756c428 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotMove.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotMove.java @@ -24,16 +24,13 @@ package org.graalvm.compiler.hotspot.amd64; import static org.graalvm.compiler.core.common.GraalOptions.GeneratePIC; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.HINT; -import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.ILLEGAL; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.REG; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.STACK; import static jdk.vm.ci.code.ValueUtil.asRegister; import static jdk.vm.ci.code.ValueUtil.isRegister; import static jdk.vm.ci.code.ValueUtil.isStackSlot; -import org.graalvm.compiler.asm.Label; import org.graalvm.compiler.asm.amd64.AMD64Address; -import org.graalvm.compiler.asm.amd64.AMD64Assembler.ConditionFlag; import org.graalvm.compiler.core.common.CompressEncoding; import org.graalvm.compiler.asm.amd64.AMD64MacroAssembler; import org.graalvm.compiler.debug.GraalError; @@ -41,10 +38,8 @@ import org.graalvm.compiler.hotspot.GraalHotSpotVMConfig; import org.graalvm.compiler.lir.LIRInstructionClass; import org.graalvm.compiler.lir.StandardOp.LoadConstantOp; import org.graalvm.compiler.lir.amd64.AMD64LIRInstruction; -import org.graalvm.compiler.lir.amd64.AMD64Move; import org.graalvm.compiler.lir.asm.CompilationResultBuilder; -import jdk.vm.ci.amd64.AMD64Kind; import jdk.vm.ci.code.Register; import jdk.vm.ci.hotspot.HotSpotMetaspaceConstant; import jdk.vm.ci.hotspot.HotSpotObjectConstant; @@ -180,91 +175,6 @@ public class AMD64HotSpotMove { } } - public static final class CompressPointer extends AMD64LIRInstruction { - public static final LIRInstructionClass TYPE = LIRInstructionClass.create(CompressPointer.class); - - private final CompressEncoding encoding; - private final boolean nonNull; - - @Def({REG, HINT}) protected AllocatableValue result; - @Use({REG}) protected AllocatableValue input; - @Alive({REG, ILLEGAL}) protected AllocatableValue baseRegister; - - public CompressPointer(AllocatableValue result, AllocatableValue input, AllocatableValue baseRegister, CompressEncoding encoding, boolean nonNull) { - super(TYPE); - this.result = result; - this.input = input; - this.baseRegister = baseRegister; - this.encoding = encoding; - this.nonNull = nonNull; - } - - @Override - public void emitCode(CompilationResultBuilder crb, AMD64MacroAssembler masm) { - AMD64Move.move(AMD64Kind.QWORD, crb, masm, result, input); - - Register resReg = asRegister(result); - if (encoding.hasBase() || GeneratePIC.getValue(crb.getOptions())) { - Register baseReg = asRegister(baseRegister); - if (!nonNull) { - masm.testq(resReg, resReg); - masm.cmovq(ConditionFlag.Equal, resReg, baseReg); - } - masm.subq(resReg, baseReg); - } - - if (encoding.hasShift()) { - masm.shrq(resReg, encoding.getShift()); - } - } - } - - public static final class UncompressPointer extends AMD64LIRInstruction { - public static final LIRInstructionClass TYPE = LIRInstructionClass.create(UncompressPointer.class); - - private final CompressEncoding encoding; - private final boolean nonNull; - - @Def({REG, HINT}) protected AllocatableValue result; - @Use({REG}) protected AllocatableValue input; - @Alive({REG, ILLEGAL}) protected AllocatableValue baseRegister; - - public UncompressPointer(AllocatableValue result, AllocatableValue input, AllocatableValue baseRegister, CompressEncoding encoding, boolean nonNull) { - super(TYPE); - this.result = result; - this.input = input; - this.baseRegister = baseRegister; - this.encoding = encoding; - this.nonNull = nonNull; - } - - @Override - public void emitCode(CompilationResultBuilder crb, AMD64MacroAssembler masm) { - AMD64Move.move(AMD64Kind.DWORD, crb, masm, result, input); - - Register resReg = asRegister(result); - if (encoding.getShift() != 0) { - masm.shlq(resReg, encoding.getShift()); - } - - if (encoding.hasBase() || GeneratePIC.getValue(crb.getOptions())) { - if (nonNull) { - masm.addq(resReg, asRegister(baseRegister)); - } else { - if (!encoding.hasShift()) { - // if encoding.shift != 0, the flags are already set by the shlq - masm.testq(resReg, resReg); - } - - Label done = new Label(); - masm.jccb(ConditionFlag.Equal, done); - masm.addq(resReg, asRegister(baseRegister)); - masm.bind(done); - } - } - } - } - public static void decodeKlassPointer(CompilationResultBuilder crb, AMD64MacroAssembler masm, Register register, Register scratch, AMD64Address address, GraalHotSpotVMConfig config) { CompressEncoding encoding = config.getKlassEncoding(); masm.movl(register, address); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotNodeLIRBuilder.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotNodeLIRBuilder.java index a81cb6d222e..93a2c360fa8 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotNodeLIRBuilder.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.amd64/src/org/graalvm/compiler/hotspot/amd64/AMD64HotSpotNodeLIRBuilder.java @@ -189,8 +189,7 @@ public class AMD64HotSpotNodeLIRBuilder extends AMD64NodeLIRBuilder implements H sig[i] = node.arguments().get(i).stamp().javaType(gen.getMetaAccess()); } - Value[] parameters = visitInvokeArguments(gen.getResult().getFrameMapBuilder().getRegisterConfig().getCallingConvention(HotSpotCallingConventionType.JavaCall, null, sig, gen), - node.arguments()); + Value[] parameters = visitInvokeArguments(gen.getRegisterConfig().getCallingConvention(HotSpotCallingConventionType.JavaCall, null, sig, gen), node.arguments()); append(new AMD64BreakpointOp(parameters)); } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotMove.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotMove.java index d6f9bdffaea..2175bec863c 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotMove.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotMove.java @@ -113,8 +113,7 @@ public class SPARCHotSpotMove { } else { register = asRegister(result); } - int bytes = result.getPlatformKind().getSizeInBytes(); - loadFromConstantTable(crb, masm, bytes, asRegister(constantTableBase), constant, register, SPARCDelayedControlTransfer.DUMMY); + int bytes = loadFromConstantTable(crb, masm, asRegister(constantTableBase), constant, register, SPARCDelayedControlTransfer.DUMMY); if (isStack) { masm.st(register, (SPARCAddress) crb.asAddress(result), bytes); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotNodeLIRBuilder.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotNodeLIRBuilder.java index 414058f9e67..09dc9cf5f42 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotNodeLIRBuilder.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotNodeLIRBuilder.java @@ -162,8 +162,7 @@ public class SPARCHotSpotNodeLIRBuilder extends SPARCNodeLIRBuilder implements H sig[i] = node.arguments().get(i).stamp().javaType(gen.getMetaAccess()); } - Value[] parameters = visitInvokeArguments(gen.getResult().getFrameMapBuilder().getRegisterConfig().getCallingConvention(HotSpotCallingConventionType.JavaCall, null, sig, gen), - node.arguments()); + Value[] parameters = visitInvokeArguments(gen.getRegisterConfig().getCallingConvention(HotSpotCallingConventionType.JavaCall, null, sig, gen), node.arguments()); append(new SPARCBreakpointOp(parameters)); } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotStrategySwitchOp.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotStrategySwitchOp.java index f4e2c741782..333f8569970 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotStrategySwitchOp.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.sparc/src/org/graalvm/compiler/hotspot/sparc/SPARCHotSpotStrategySwitchOp.java @@ -77,8 +77,7 @@ final class SPARCHotSpotStrategySwitchOp extends SPARCControlFlow.StrategySwitch boolean canUseShortBranch = masm.hasFeature(CPUFeature.CBCOND) && SPARCControlFlow.isShortBranch(masm, cbCondPosition, hint, target); Register scratchRegister = asRegister(scratch); - final int byteCount = constant.isCompressed() ? 4 : 8; - loadFromConstantTable(crb, masm, byteCount, asRegister(constantTableBase), constant, scratchRegister, SPARCDelayedControlTransfer.DUMMY); + loadFromConstantTable(crb, masm, asRegister(constantTableBase), constant, scratchRegister, SPARCDelayedControlTransfer.DUMMY); if (canUseShortBranch) { CBCOND.emit(masm, conditionFlag, conditionCode == CC.Xcc, keyRegister, scratchRegister, target); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/ArrayCopyIntrinsificationTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/ArrayCopyIntrinsificationTest.java index 952dd2bc08e..21b69d4c082 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/ArrayCopyIntrinsificationTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/ArrayCopyIntrinsificationTest.java @@ -160,7 +160,7 @@ public class ArrayCopyIntrinsificationTest extends GraalCompilerTest { } /** - * Tests {@link ArrayCopySnippets#checkcastArraycopyWork}. + * Tests {@link ArrayCopySnippets#arraycopyCheckcastSnippet}. */ @Test public void testArrayStoreException() { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/CompilationWrapperTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/CompilationWrapperTest.java index 669d8d746ab..4f53585d4d4 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/CompilationWrapperTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/CompilationWrapperTest.java @@ -122,7 +122,7 @@ public class CompilationWrapperTest extends GraalCompilerTest { * Tests compilation requested by Truffle. */ @Test - public void testTruffleCompilation() throws IOException, InterruptedException { + public void testTruffleCompilation1() throws IOException, InterruptedException { testHelper(Collections.emptyList(), Arrays.asList( "-Dgraal.CompilationFailureAction=ExitVM", @@ -130,6 +130,22 @@ public class CompilationWrapperTest extends GraalCompilerTest { "org.graalvm.compiler.truffle.test.SLTruffleGraalTestSuite", "test"); } + /** + * Tests that TruffleCompilationExceptionsAreFatal works as expected. + */ + @Test + public void testTruffleCompilation2() throws IOException, InterruptedException { + Probe[] probes = { + new Probe("Exiting VM due to TruffleCompilationExceptionsAreFatal=true", 1), + }; + testHelper(Arrays.asList(probes), + Arrays.asList( + "-Dgraal.CompilationFailureAction=Silent", + "-Dgraal.TruffleCompilationExceptionsAreFatal=true", + "-Dgraal.CrashAt=root test1"), + "org.graalvm.compiler.truffle.test.SLTruffleGraalTestSuite", "test"); + } + private static final boolean VERBOSE = Boolean.getBoolean(CompilationWrapperTest.class.getSimpleName() + ".verbose"); private static void testHelper(List initialProbes, List extraVmArgs, String... mainClassAndArgs) throws IOException, InterruptedException { @@ -149,14 +165,17 @@ public class CompilationWrapperTest extends GraalCompilerTest { } List probes = new ArrayList<>(initialProbes); - Probe diagnosticProbe = new Probe("Graal diagnostic output saved in ", 1); - probes.add(diagnosticProbe); - probes.add(new Probe("Forced crash after compiling", Integer.MAX_VALUE) { - @Override - String test() { - return actualOccurrences > 0 ? null : "expected at least 1 occurrence"; - } - }); + Probe diagnosticProbe = null; + if (!extraVmArgs.contains("-Dgraal.TruffleCompilationExceptionsAreFatal=true")) { + diagnosticProbe = new Probe("Graal diagnostic output saved in ", 1); + probes.add(diagnosticProbe); + probes.add(new Probe("Forced crash after compiling", Integer.MAX_VALUE) { + @Override + String test() { + return actualOccurrences > 0 ? null : "expected at least 1 occurrence"; + } + }); + } for (String line : proc.output) { for (Probe probe : probes) { @@ -171,38 +190,42 @@ public class CompilationWrapperTest extends GraalCompilerTest { Assert.fail(String.format("Did not find expected occurences of '%s' in output of command: %s%n%s", probe.substring, error, proc)); } } + if (diagnosticProbe != null) { + String line = diagnosticProbe.lastMatchingLine; + int substringStart = line.indexOf(diagnosticProbe.substring); + int substringLength = diagnosticProbe.substring.length(); + String diagnosticOutputZip = line.substring(substringStart + substringLength).trim(); - String diagnosticOutputZip = diagnosticProbe.lastMatchingLine.substring(diagnosticProbe.substring.length()).trim(); + List dumpPathEntries = Arrays.asList(dumpPath.list()); - List dumpPathEntries = Arrays.asList(dumpPath.list()); - - File zip = new File(diagnosticOutputZip).getAbsoluteFile(); - Assert.assertTrue(zip.toString(), zip.exists()); - Assert.assertTrue(zip + " not in " + dumpPathEntries, dumpPathEntries.contains(zip.getName())); - try { - int bgv = 0; - int cfg = 0; - ZipFile dd = new ZipFile(diagnosticOutputZip); - List entries = new ArrayList<>(); - for (Enumeration e = dd.entries(); e.hasMoreElements();) { - ZipEntry ze = e.nextElement(); - String name = ze.getName(); - entries.add(name); - if (name.endsWith(".bgv")) { - bgv++; - } else if (name.endsWith(".cfg")) { - cfg++; + File zip = new File(diagnosticOutputZip).getAbsoluteFile(); + Assert.assertTrue(zip.toString(), zip.exists()); + Assert.assertTrue(zip + " not in " + dumpPathEntries, dumpPathEntries.contains(zip.getName())); + try { + int bgv = 0; + int cfg = 0; + ZipFile dd = new ZipFile(diagnosticOutputZip); + List entries = new ArrayList<>(); + for (Enumeration e = dd.entries(); e.hasMoreElements();) { + ZipEntry ze = e.nextElement(); + String name = ze.getName(); + entries.add(name); + if (name.endsWith(".bgv")) { + bgv++; + } else if (name.endsWith(".cfg")) { + cfg++; + } } + if (bgv == 0) { + Assert.fail(String.format("Expected at least one .bgv file in %s: %s%n%s", diagnosticOutputZip, entries, proc)); + } + if (cfg == 0) { + Assert.fail(String.format("Expected at least one .cfg file in %s: %s", diagnosticOutputZip, entries)); + } + } finally { + zip.delete(); + dumpPath.delete(); } - if (bgv == 0) { - Assert.fail(String.format("Expected at least one .bgv file in %s: %s%n%s", diagnosticOutputZip, entries, proc)); - } - if (cfg == 0) { - Assert.fail(String.format("Expected at least one .cfg file in %s: %s", diagnosticOutputZip, entries)); - } - } finally { - zip.delete(); - dumpPath.delete(); } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/ExplicitExceptionTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/ExplicitExceptionTest.java index 44dbfe55e67..0b533e5abe5 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/ExplicitExceptionTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/ExplicitExceptionTest.java @@ -22,23 +22,44 @@ */ package org.graalvm.compiler.hotspot.test; -import org.junit.Test; - import org.graalvm.compiler.core.test.GraalCompilerTest; import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.extended.ForeignCallNode; import org.graalvm.compiler.options.OptionValues; +import org.junit.Assume; +import org.junit.Test; import jdk.vm.ci.code.InstalledCode; +import jdk.vm.ci.meta.ProfilingInfo; import jdk.vm.ci.meta.ResolvedJavaMethod; +import jdk.vm.ci.meta.TriState; public class ExplicitExceptionTest extends GraalCompilerTest { private int expectedForeignCallCount; + /** + * Determines if profiling info for {@code method} indicates an exception was thrown somewhere + * in the method. In the case of the {@code -Xcomp} VM option, interpreter execution can be + * skipped altogether and other execution engines (e.g., C1) may not record seen exceptions in a + * method profile. + */ + private static boolean exceptionWasSeen(ResolvedJavaMethod method) { + ProfilingInfo profilingInfo = method.getProfilingInfo(); + if (profilingInfo != null) { + for (int i = 0; i < profilingInfo.getCodeSize(); i++) { + if (profilingInfo.getExceptionSeen(i) == TriState.TRUE) { + return true; + } + } + } + return false; + } + @Override protected InstalledCode getCode(ResolvedJavaMethod method, StructuredGraph graph, boolean forceCompile, boolean installAsDefault, OptionValues options) { InstalledCode installedCode = super.getCode(method, graph, forceCompile, installAsDefault, options); + Assume.assumeTrue(exceptionWasSeen(method)); assertDeepEquals(expectedForeignCallCount, lastCompiledGraph.getNodes().filter(ForeignCallNode.class).count()); return installedCode; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/GraalOSRLockTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/GraalOSRLockTest.java index bff37edeb79..6b47720d1ac 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/GraalOSRLockTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/GraalOSRLockTest.java @@ -24,11 +24,13 @@ package org.graalvm.compiler.hotspot.test; import java.lang.management.ManagementFactory; import java.lang.management.MonitorInfo; +import java.lang.management.RuntimeMXBean; import java.lang.management.ThreadInfo; import java.lang.management.ThreadMXBean; import java.util.Collection; import java.util.HashMap; import java.util.HashSet; +import java.util.List; import org.graalvm.compiler.api.directives.GraalDirectives; import org.graalvm.compiler.core.phases.HighTier; @@ -44,6 +46,7 @@ import org.junit.Test; import jdk.vm.ci.meta.ResolvedJavaMethod; import org.junit.Assume; +import org.junit.BeforeClass; /** * Test on-stack-replacement with locks. @@ -51,13 +54,28 @@ import org.junit.Assume; public class GraalOSRLockTest extends GraalOSRTestBase { private static boolean TestInSeparateThread = false; + private static final String COMPILE_ONLY_FLAG = "-Xcomp"; - public GraalOSRLockTest() { + @BeforeClass + public static void checkVMArguments() { try { Class.forName("java.lang.management.ManagementFactory"); } catch (ClassNotFoundException ex) { Assume.assumeNoException("cannot check for monitors without java.management JDK9 module", ex); } + /* + * Note: The -Xcomp execution mode of the VM will stop most of the OSR test cases from + * working as every method is compiled at level3 (followed by level4 on the second + * invocation). The tests in this class are written in a way that they expect a method to be + * executed at the invocation BCI with the interpreter and then perform an OSR to an + * installed nmethod at a given BCI. + * + */ + RuntimeMXBean runtimeMxBean = ManagementFactory.getRuntimeMXBean(); + List arguments = runtimeMxBean.getInputArguments(); + for (String arg : arguments) { + Assume.assumeFalse(arg.equals(COMPILE_ONLY_FLAG)); + } } // testing only @@ -438,7 +456,7 @@ public class GraalOSRLockTest extends GraalOSRTestBase { } GraalDirectives.controlFlowAnchor(); if (!GraalDirectives.inCompiledCode()) { - throw new Error("Must part of compiled code"); + throw new Error("Must be part of compiled code"); } return ret; } @@ -449,11 +467,11 @@ public class GraalOSRLockTest extends GraalOSRTestBase { ReturnValue ret = ReturnValue.FAILURE; synchronized (lock) { synchronized (lock1) { - for (int i = 1; i < limit; i++) { + for (int i = 1; i < 10 * limit; i++) { GraalDirectives.blackhole(i); - if (i % 1001 == 0) { + if (i % 33 == 0) { ret = ReturnValue.SUCCESS; - if (GraalDirectives.inCompiledCode() && i + 33 > (limit)) { + if (GraalDirectives.inCompiledCode() && i + 33 > (10 * limit)) { GraalDirectives.blackhole(ret); System.gc(); } @@ -462,7 +480,7 @@ public class GraalOSRLockTest extends GraalOSRTestBase { } GraalDirectives.controlFlowAnchor(); if (!GraalDirectives.inCompiledCode()) { - throw new Error("Must part of compiled code"); + throw new Error("Must be part of compiled code already hereeeeee"); } else { // lock 1 must be free if (isMonitorLockHeld(lock1)) { @@ -519,7 +537,7 @@ public class GraalOSRLockTest extends GraalOSRTestBase { } GraalDirectives.controlFlowAnchor(); if (!GraalDirectives.inCompiledCode()) { - throw new Error("Must part of compiled code"); + throw new Error("Must be part of compiled code"); } return ret; } @@ -543,7 +561,7 @@ public class GraalOSRLockTest extends GraalOSRTestBase { } GraalDirectives.controlFlowAnchor(); if (!GraalDirectives.inCompiledCode()) { - throw new Error("Must part of compiled code"); + throw new Error("Must be part of compiled code"); } return ret; } @@ -568,7 +586,7 @@ public class GraalOSRLockTest extends GraalOSRTestBase { } GraalDirectives.controlFlowAnchor(); if (!GraalDirectives.inCompiledCode()) { - throw new Error("Must part of compiled code"); + throw new Error("Must be part of compiled code"); } return ret; } @@ -646,7 +664,7 @@ public class GraalOSRLockTest extends GraalOSRTestBase { synchronized (monitor) { GraalDirectives.controlFlowAnchor(); if (!GraalDirectives.inCompiledCode()) { - throw new Error("Must part of compiled code"); + throw new Error("Must be part of compiled code"); } } return ret; @@ -670,7 +688,7 @@ public class GraalOSRLockTest extends GraalOSRTestBase { } GraalDirectives.controlFlowAnchor(); if (!GraalDirectives.inCompiledCode()) { - throw new Error("Must part of compiled code"); + throw new Error("Must be part of compiled code"); } return ret; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/GraalOSRTestBase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/GraalOSRTestBase.java index f0d53bb23d7..d73449cd9dd 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/GraalOSRTestBase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/GraalOSRTestBase.java @@ -35,7 +35,9 @@ import org.graalvm.compiler.debug.DebugContext; import org.graalvm.compiler.debug.GraalError; import org.graalvm.compiler.debug.TTY; import org.graalvm.compiler.hotspot.CompilationTask; +import org.graalvm.compiler.hotspot.GraalHotSpotVMConfig; import org.graalvm.compiler.hotspot.HotSpotGraalCompiler; +import org.graalvm.compiler.hotspot.HotSpotGraalRuntimeProvider; import org.graalvm.compiler.java.BciBlockMapping; import org.graalvm.compiler.java.BciBlockMapping.BciBlock; import org.graalvm.compiler.nodes.StructuredGraph; @@ -76,6 +78,13 @@ public abstract class GraalOSRTestBase extends GraalCompilerTest { HotSpotCompilationRequest request = new HotSpotCompilationRequest((HotSpotResolvedJavaMethod) method, bci, jvmciEnv); HotSpotGraalCompiler compiler = (HotSpotGraalCompiler) runtime.getCompiler(); CompilationTask task = new CompilationTask(runtime, compiler, request, true, true, debug.getOptions()); + if (method instanceof HotSpotResolvedJavaMethod) { + HotSpotGraalRuntimeProvider graalRuntime = compiler.getGraalRuntime(); + GraalHotSpotVMConfig config = graalRuntime.getVMConfig(); + if (((HotSpotResolvedJavaMethod) method).hasCodeAtLevel(bci, config.compilationLevelFullOptimization)) { + return; + } + } HotSpotCompilationRequestResult result = task.runCompilation(debug); if (result.getFailure() != null) { throw new GraalError(result.getFailureMessage()); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/WriteBarrierVerificationTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/WriteBarrierVerificationTest.java index 4c3a48468c4..10298107ad4 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/WriteBarrierVerificationTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot.test/src/org/graalvm/compiler/hotspot/test/WriteBarrierVerificationTest.java @@ -37,7 +37,6 @@ import org.graalvm.compiler.hotspot.nodes.SerialArrayRangeWriteBarrier; import org.graalvm.compiler.hotspot.nodes.SerialWriteBarrier; import org.graalvm.compiler.hotspot.phases.WriteBarrierAdditionPhase; import org.graalvm.compiler.hotspot.phases.WriteBarrierVerificationPhase; -import org.graalvm.compiler.hotspot.replacements.arraycopy.UnsafeArrayCopyNode; import org.graalvm.compiler.nodes.AbstractBeginNode; import org.graalvm.compiler.nodes.AbstractMergeNode; import org.graalvm.compiler.nodes.FieldLocationIdentity; @@ -629,12 +628,6 @@ public class WriteBarrierVerificationTest extends HotSpotGraalCompilerTest { System.arraycopy(a, 0, b, 0, a.length); } - @Test - public void test61() { - GraphPredicate checkForUnsafeArrayCopy = graph -> graph.getNodes().filter(UnsafeArrayCopyNode.class).count() > 0 ? 1 : 0; - testPredicate("test13Snippet", checkForUnsafeArrayCopy, new int[]{}); - } - private interface GraphPredicate { int apply(StructuredGraph graph); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/GraalHotSpotVMConfig.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/GraalHotSpotVMConfig.java index f314bbd584f..15e709c3056 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/GraalHotSpotVMConfig.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/GraalHotSpotVMConfig.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2011, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2011, 2016, 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 @@ -280,14 +280,19 @@ public class GraalHotSpotVMConfig extends HotSpotVMConfigAccess { } if (offset == -1) { try { - offset = getFieldOffset(name, Integer.class, "OopHandle"); + offset = getFieldOffset(name, Integer.class, "jobject"); isHandle = true; } catch (JVMCIError e) { - + try { + // JDK-8186777 + offset = getFieldOffset(name, Integer.class, "OopHandle"); + isHandle = true; + } catch (JVMCIError e2) { + } } } if (offset == -1) { - throw new JVMCIError("cannot get offset of field " + name + " with type oop or OopHandle"); + throw new JVMCIError("cannot get offset of field " + name + " with type oop, jobject or OopHandle"); } classMirrorOffset = offset; classMirrorIsHandle = isHandle; @@ -648,6 +653,8 @@ public class GraalHotSpotVMConfig extends HotSpotVMConfigAccess { public final long heapEndAddress = getFieldValue("CompilerToVM::Data::_heap_end_addr", Long.class, "HeapWord**"); public final long heapTopAddress = getFieldValue("CompilerToVM::Data::_heap_top_addr", Long.class, isJDK8 ? "HeapWord**" : "HeapWord* volatile*"); + public final boolean cmsIncrementalMode = getFlag("CMSIncrementalMode", Boolean.class, false); + public final long inlineCacheMissStub = getFieldValue("CompilerToVM::Data::SharedRuntime_ic_miss_stub", Long.class, "address"); public final long handleWrongMethodStub = getFieldValue("CompilerToVM::Data::SharedRuntime_handle_wrong_method_stub", Long.class, "address"); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotBackend.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotBackend.java index 3a6d2ec302b..5191305b3e7 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotBackend.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotBackend.java @@ -266,6 +266,8 @@ public abstract class HotSpotBackend extends Backend implements FrameMap.Referen unsafeArraycopyStub(HotSpotBackend.UNSAFE_ARRAYCOPY, srcAddr, dstAddr, size); } + public static final ForeignCallDescriptor GENERIC_ARRAYCOPY = new ForeignCallDescriptor("generic_arraycopy", int.class, Word.class, int.class, Word.class, int.class, int.class); + @NodeIntrinsic(ForeignCallNode.class) private static native void unsafeArraycopyStub(@ConstantNodeParameter ForeignCallDescriptor descriptor, Word srcAddr, Word dstAddr, Word size); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalCompiler.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalCompiler.java index c77d3eebf18..c160cc0878c 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalCompiler.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalCompiler.java @@ -284,7 +284,7 @@ public class HotSpotGraalCompiler implements GraalJVMCICompiler { public Object mbean() { if (graalRuntime instanceof HotSpotGraalRuntime) { - return ((HotSpotGraalRuntime)graalRuntime).mbean(); + return ((HotSpotGraalRuntime) graalRuntime).getMBean(); } return null; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalCompilerFactory.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalCompilerFactory.java index b4b8e41890f..7146ad77271 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalCompilerFactory.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalCompilerFactory.java @@ -26,6 +26,8 @@ import static jdk.vm.ci.common.InitTimer.timer; import static org.graalvm.compiler.hotspot.HotSpotGraalOptionValues.GRAAL_OPTION_PROPERTY_PREFIX; import java.io.PrintStream; +import java.util.Map; +import java.util.Collections; import org.graalvm.compiler.debug.MethodFilter; import org.graalvm.compiler.options.Option; @@ -190,4 +192,11 @@ public final class HotSpotGraalCompilerFactory extends HotSpotJVMCICompilerFacto } return level; } + + public Map mbeans() { + HotSpotGraalCompiler compiler = createCompiler(HotSpotJVMCIRuntime.runtime()); + String name = "org.graalvm.compiler.hotspot:type=Options"; + Object bean = ((HotSpotGraalRuntime) compiler.getGraalRuntime()).getMBean(); + return Collections.singletonMap(name, bean); + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalMBean.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalMBean.java index c32e26808f1..2d83476c62f 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalMBean.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalMBean.java @@ -283,10 +283,8 @@ public final class HotSpotGraalMBean implements javax.management.DynamicMBean { @Override public javax.management.MBeanInfo getMBeanInfo() { List attrs = new ArrayList<>(); - if (registered != null) { - for (OptionDescriptor descr : allOptionDescriptors()) { - attrs.add(new javax.management.MBeanAttributeInfo(descr.getName(), descr.getType().getName(), descr.getHelp(), true, true, false)); - } + for (OptionDescriptor descr : allOptionDescriptors()) { + attrs.add(new javax.management.MBeanAttributeInfo(descr.getName(), descr.getType().getName(), descr.getHelp(), true, true, false)); } javax.management.MBeanOperationInfo[] ops = { new javax.management.MBeanOperationInfo("dumpMethod", "Enable IGV dumps for provided method", new javax.management.MBeanParameterInfo[]{ diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalRuntime.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalRuntime.java index c56d8105219..e14b268e885 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalRuntime.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/HotSpotGraalRuntime.java @@ -317,7 +317,7 @@ public final class HotSpotGraalRuntime implements HotSpotGraalRuntimeProvider { return compilationProblemsPerAction; } - final Object mbean() { + Object getMBean() { return mBean; } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/DefaultHotSpotLoweringProvider.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/DefaultHotSpotLoweringProvider.java index 268dce0937a..d3c57cfe2c0 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/DefaultHotSpotLoweringProvider.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/DefaultHotSpotLoweringProvider.java @@ -28,8 +28,8 @@ import static org.graalvm.compiler.core.common.GraalOptions.InlineVTableStubs; import static org.graalvm.compiler.core.common.GraalOptions.OmitHotExceptionStacktrace; import static org.graalvm.compiler.hotspot.meta.HotSpotForeignCallsProviderImpl.OSR_MIGRATION_END; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.CLASS_KLASS_LOCATION; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.CLASS_MIRROR_LOCATION; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.CLASS_MIRROR_HANDLE_LOCATION; +import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.CLASS_MIRROR_LOCATION; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.COMPRESSED_HUB_LOCATION; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.DISPLACED_MARK_WORD_LOCATION; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.HUB_LOCATION; @@ -41,6 +41,7 @@ import static org.graalvm.word.LocationIdentity.any; import java.lang.ref.Reference; import org.graalvm.compiler.api.directives.GraalDirectives; +import org.graalvm.compiler.core.common.CompressEncoding; import org.graalvm.compiler.core.common.GraalOptions; import org.graalvm.compiler.core.common.spi.ForeignCallDescriptor; import org.graalvm.compiler.core.common.spi.ForeignCallsProvider; @@ -90,10 +91,8 @@ import org.graalvm.compiler.hotspot.replacements.UnsafeLoadSnippets; import org.graalvm.compiler.hotspot.replacements.WriteBarrierSnippets; import org.graalvm.compiler.hotspot.replacements.aot.ResolveConstantSnippets; import org.graalvm.compiler.hotspot.replacements.arraycopy.ArrayCopyNode; -import org.graalvm.compiler.hotspot.replacements.arraycopy.ArrayCopySlowPathNode; +import org.graalvm.compiler.hotspot.replacements.arraycopy.ArrayCopyWithSlowPathNode; import org.graalvm.compiler.hotspot.replacements.arraycopy.ArrayCopySnippets; -import org.graalvm.compiler.hotspot.replacements.arraycopy.ArrayCopyUnrollNode; -import org.graalvm.compiler.hotspot.replacements.arraycopy.UnsafeArrayCopySnippets; import org.graalvm.compiler.hotspot.replacements.profiling.ProfileSnippets; import org.graalvm.compiler.hotspot.word.KlassPointer; import org.graalvm.compiler.nodes.AbstractBeginNode; @@ -193,7 +192,7 @@ public class DefaultHotSpotLoweringProvider extends DefaultJavaLoweringProvider public DefaultHotSpotLoweringProvider(HotSpotGraalRuntimeProvider runtime, MetaAccessProvider metaAccess, ForeignCallsProvider foreignCalls, HotSpotRegistersProvider registers, HotSpotConstantReflectionProvider constantReflection, TargetDescription target) { - super(metaAccess, foreignCalls, target); + super(metaAccess, foreignCalls, target, runtime.getVMConfig().useCompressedOops); this.runtime = runtime; this.registers = registers; this.constantReflection = constantReflection; @@ -216,7 +215,6 @@ public class DefaultHotSpotLoweringProvider extends DefaultJavaLoweringProvider hashCodeSnippets = new HashCodeSnippets.Templates(options, factories, providers, target); resolveConstantSnippets = new ResolveConstantSnippets.Templates(options, factories, providers, target); profileSnippets = new ProfileSnippets.Templates(options, factories, providers, target); - providers.getReplacements().registerSnippetTemplateCache(new UnsafeArrayCopySnippets.Templates(options, factories, providers, target)); } public MonitorSnippets.Templates getMonitorSnippets() { @@ -315,10 +313,8 @@ public class DefaultHotSpotLoweringProvider extends DefaultJavaLoweringProvider } } else if (n instanceof ArrayCopyNode) { arraycopySnippets.lower((ArrayCopyNode) n, tool); - } else if (n instanceof ArrayCopySlowPathNode) { - arraycopySnippets.lower((ArrayCopySlowPathNode) n, tool); - } else if (n instanceof ArrayCopyUnrollNode) { - arraycopySnippets.lower((ArrayCopyUnrollNode) n, tool); + } else if (n instanceof ArrayCopyWithSlowPathNode) { + arraycopySnippets.lower((ArrayCopyWithSlowPathNode) n, tool); } else if (n instanceof G1PreWriteBarrier) { writeBarrierSnippets.lower((G1PreWriteBarrier) n, registers, tool); } else if (n instanceof G1PostWriteBarrier) { @@ -495,20 +491,18 @@ public class DefaultHotSpotLoweringProvider extends DefaultJavaLoweringProvider } } - @Override - protected Stamp loadStamp(Stamp stamp, JavaKind kind, boolean compressible) { - if (kind == JavaKind.Object && compressible && runtime.getVMConfig().useCompressedOops) { - return HotSpotNarrowOopStamp.compressed((ObjectStamp) stamp, runtime.getVMConfig().getOopEncoding()); - } - return super.loadStamp(stamp, kind, compressible); + private CompressEncoding getOopEncoding() { + return runtime.getVMConfig().getOopEncoding(); } @Override - protected ValueNode implicitLoadConvert(JavaKind kind, ValueNode value, boolean compressible) { - if (kind == JavaKind.Object && compressible && runtime.getVMConfig().useCompressedOops) { - return new HotSpotCompressionNode(CompressionOp.Uncompress, value, runtime.getVMConfig().getOopEncoding()); - } - return super.implicitLoadConvert(kind, value, compressible); + protected Stamp loadCompressedStamp(ObjectStamp stamp) { + return HotSpotNarrowOopStamp.compressed(stamp, getOopEncoding()); + } + + @Override + protected ValueNode newCompressionNode(CompressionOp op, ValueNode value) { + return new HotSpotCompressionNode(op, value, getOopEncoding()); } @Override @@ -518,14 +512,6 @@ public class DefaultHotSpotLoweringProvider extends DefaultJavaLoweringProvider return ConstantNode.forConstant(base, metaAccess, graph); } - @Override - protected ValueNode implicitStoreConvert(JavaKind kind, ValueNode value, boolean compressible) { - if (kind == JavaKind.Object && compressible && runtime.getVMConfig().useCompressedOops) { - return new HotSpotCompressionNode(CompressionOp.Compress, value, runtime.getVMConfig().getOopEncoding()); - } - return super.implicitStoreConvert(kind, value, compressible); - } - @Override protected ValueNode createReadArrayComponentHub(StructuredGraph graph, ValueNode arrayHub, FixedNode anchor) { /* @@ -800,4 +786,9 @@ public class DefaultHotSpotLoweringProvider extends DefaultJavaLoweringProvider public int arrayLengthOffset() { return runtime.getVMConfig().arrayOopDescLengthOffset(); } + + @Override + protected final JavaKind getStorageKind(ResolvedJavaField field) { + return field.getJavaKind(); + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/HotSpotGraphBuilderPlugins.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/HotSpotGraphBuilderPlugins.java index 61fbadca3b6..5e1504ab14c 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/HotSpotGraphBuilderPlugins.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/HotSpotGraphBuilderPlugins.java @@ -39,6 +39,7 @@ import java.util.zip.CRC32; import org.graalvm.compiler.api.replacements.SnippetReflectionProvider; import org.graalvm.compiler.bytecode.BytecodeProvider; import org.graalvm.compiler.core.common.spi.ForeignCallsProvider; +import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.debug.GraalError; import org.graalvm.compiler.hotspot.GraalHotSpotVMConfig; import org.graalvm.compiler.hotspot.nodes.CurrentJavaThreadNode; @@ -68,6 +69,7 @@ import org.graalvm.compiler.nodes.NamedLocationIdentity; import org.graalvm.compiler.nodes.PiNode; import org.graalvm.compiler.nodes.ValueNode; import org.graalvm.compiler.nodes.calc.AddNode; +import org.graalvm.compiler.nodes.calc.IntegerConvertNode; import org.graalvm.compiler.nodes.calc.LeftShiftNode; import org.graalvm.compiler.nodes.graphbuilderconf.ForeignCallPlugin; import org.graalvm.compiler.nodes.graphbuilderconf.GraphBuilderConfiguration.Plugins; @@ -316,8 +318,8 @@ public class HotSpotGraphBuilderPlugins { private static boolean readMetaspaceConstantPoolElement(GraphBuilderContext b, ValueNode constantPoolOop, ValueNode index, JavaKind elementKind, WordTypes wordTypes, GraalHotSpotVMConfig config) { ValueNode constants = getMetaspaceConstantPool(b, constantPoolOop, wordTypes, config); int shift = CodeUtil.log2(wordTypes.getWordKind().getByteCount()); - ValueNode scaledIndex = b.add(new LeftShiftNode(index, b.add(ConstantNode.forInt(shift)))); - ValueNode offset = b.add(new AddNode(scaledIndex, b.add(ConstantNode.forInt(config.constantPoolSize)))); + ValueNode scaledIndex = b.add(new LeftShiftNode(IntegerConvertNode.convert(index, StampFactory.forKind(JavaKind.Long)), b.add(ConstantNode.forInt(shift)))); + ValueNode offset = b.add(new AddNode(scaledIndex, b.add(ConstantNode.forLong(config.constantPoolSize)))); AddressNode elementAddress = b.add(new OffsetAddressNode(constants, offset)); boolean notCompressible = false; ValueNode elementValue = WordOperationPlugin.readOp(b, elementKind, elementAddress, NamedLocationIdentity.getArrayLocation(elementKind), BarrierType.NONE, notCompressible); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/HotSpotHostForeignCallsProvider.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/HotSpotHostForeignCallsProvider.java index 36618783bfc..e18d2fde16d 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/HotSpotHostForeignCallsProvider.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/meta/HotSpotHostForeignCallsProvider.java @@ -34,6 +34,7 @@ import static org.graalvm.compiler.hotspot.HotSpotBackend.DECRYPT_WITH_ORIGINAL_ import static org.graalvm.compiler.hotspot.HotSpotBackend.ENCRYPT; import static org.graalvm.compiler.hotspot.HotSpotBackend.ENCRYPT_BLOCK; import static org.graalvm.compiler.hotspot.HotSpotBackend.EXCEPTION_HANDLER; +import static org.graalvm.compiler.hotspot.HotSpotBackend.GENERIC_ARRAYCOPY; import static org.graalvm.compiler.hotspot.HotSpotBackend.IC_MISS_HANDLER; import static org.graalvm.compiler.hotspot.HotSpotBackend.INITIALIZE_KLASS_BY_SYMBOL; import static org.graalvm.compiler.hotspot.HotSpotBackend.INVOCATION_EVENT; @@ -85,7 +86,6 @@ import static org.graalvm.compiler.hotspot.stubs.NewArrayStub.NEW_ARRAY_C; import static org.graalvm.compiler.hotspot.stubs.NewInstanceStub.NEW_INSTANCE_C; import static org.graalvm.compiler.hotspot.stubs.StubUtil.VM_MESSAGE_C; import static org.graalvm.compiler.hotspot.stubs.UnwindExceptionToCallerStub.EXCEPTION_HANDLER_FOR_RETURN_ADDRESS; -import static org.graalvm.compiler.nodes.NamedLocationIdentity.any; import static org.graalvm.compiler.nodes.java.ForeignCallDescriptors.REGISTER_FINALIZER; import static org.graalvm.compiler.replacements.Log.LOG_OBJECT; import static org.graalvm.compiler.replacements.Log.LOG_PRIMITIVE; @@ -97,6 +97,7 @@ import static org.graalvm.compiler.replacements.nodes.UnaryMathIntrinsicNode.Una import static org.graalvm.compiler.replacements.nodes.UnaryMathIntrinsicNode.UnaryOperation.LOG10; import static org.graalvm.compiler.replacements.nodes.UnaryMathIntrinsicNode.UnaryOperation.SIN; import static org.graalvm.compiler.replacements.nodes.UnaryMathIntrinsicNode.UnaryOperation.TAN; +import static org.graalvm.word.LocationIdentity.any; import java.util.EnumMap; @@ -213,7 +214,7 @@ public abstract class HotSpotHostForeignCallsProvider extends HotSpotForeignCall // c_rarg4 - oop ckval (super_klass) // return: 0 = success, n = number of copied elements xor'd with -1. ForeignCallDescriptor desc = new ForeignCallDescriptor(name, int.class, Word.class, Word.class, Word.class, Word.class, Word.class); - LocationIdentity killed = NamedLocationIdentity.getArrayLocation(JavaKind.Object); + LocationIdentity killed = NamedLocationIdentity.any(); registerForeignCall(desc, routine, NativeCall, DESTROYS_REGISTERS, LEAF_NOFP, NOT_REEXECUTABLE, killed); checkcastArraycopyDescriptors[uninit ? 1 : 0] = desc; } @@ -333,6 +334,7 @@ public abstract class HotSpotHostForeignCallsProvider extends HotSpotForeignCall registerCheckcastArraycopyDescriptor(true, c.checkcastArraycopyUninit); registerCheckcastArraycopyDescriptor(false, c.checkcastArraycopy); + registerForeignCall(GENERIC_ARRAYCOPY, c.genericArraycopy, NativeCall, DESTROYS_REGISTERS, LEAF_NOFP, NOT_REEXECUTABLE, NamedLocationIdentity.any()); registerForeignCall(UNSAFE_ARRAYCOPY, c.unsafeArraycopy, NativeCall, DESTROYS_REGISTERS, LEAF_NOFP, NOT_REEXECUTABLE, NamedLocationIdentity.any()); if (c.useMultiplyToLenIntrinsic()) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/aot/ResolveDynamicConstantNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/aot/ResolveDynamicConstantNode.java index 813055bd6d1..f8a8f92e9e6 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/aot/ResolveDynamicConstantNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/aot/ResolveDynamicConstantNode.java @@ -28,6 +28,7 @@ import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_16; import org.graalvm.word.LocationIdentity; import org.graalvm.compiler.core.common.type.Stamp; import org.graalvm.compiler.graph.NodeClass; +import org.graalvm.compiler.nodeinfo.InputType; import org.graalvm.compiler.nodeinfo.NodeInfo; import org.graalvm.compiler.nodes.DeoptimizingFixedWithNextNode; import org.graalvm.compiler.nodes.ValueNode; @@ -35,7 +36,7 @@ import org.graalvm.compiler.nodes.memory.MemoryCheckpoint; import org.graalvm.compiler.nodes.spi.Lowerable; import org.graalvm.compiler.nodes.spi.LoweringTool; -@NodeInfo(cycles = CYCLES_4, size = SIZE_16) +@NodeInfo(cycles = CYCLES_4, size = SIZE_16, allowedUsageTypes = {InputType.Memory}) public class ResolveDynamicConstantNode extends DeoptimizingFixedWithNextNode implements Lowerable, MemoryCheckpoint.Single { public static final NodeClass TYPE = NodeClass.create(ResolveDynamicConstantNode.class); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileBranchNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileBranchNode.java index d7f475463c3..eb916a17acc 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileBranchNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileBranchNode.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -71,6 +71,15 @@ public class ProfileBranchNode extends ProfileWithNotificationNode { return branchCondition != null; } + @Override + protected boolean canBeMergedWith(ProfileNode p) { + if (p instanceof ProfileBranchNode) { + ProfileBranchNode that = (ProfileBranchNode) p; + return this.method.equals(that.method) && this.bci == that.bci; + } + return false; + } + /** * Gathers all the {@link ProfileBranchNode}s that are inputs to the * {@linkplain StructuredGraph#getNodes() live nodes} in a given graph. diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileInvokeNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileInvokeNode.java index 03f12cdfd08..9ee5c23f81f 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileInvokeNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileInvokeNode.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -37,6 +37,15 @@ public class ProfileInvokeNode extends ProfileWithNotificationNode { super(TYPE, method, freqLog, probabilityLog); } + @Override + protected boolean canBeMergedWith(ProfileNode p) { + if (p instanceof ProfileInvokeNode) { + ProfileInvokeNode that = (ProfileInvokeNode) p; + return this.method.equals(that.method); + } + return false; + } + /** * Gathers all the {@link ProfileInvokeNode}s that are inputs to the * {@linkplain StructuredGraph#getNodes() live nodes} in a given graph. diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileNode.java index c5c09718e6b..9b6d447ca09 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileNode.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -24,11 +24,17 @@ package org.graalvm.compiler.hotspot.nodes.profiling; import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_IGNORED; import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_IGNORED; +import static org.graalvm.compiler.nodes.util.GraphUtil.removeFixedWithUnusedInputs; import org.graalvm.compiler.core.common.type.StampFactory; +import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.graph.NodeClass; import org.graalvm.compiler.graph.iterators.NodeIterable; +import org.graalvm.compiler.graph.spi.Simplifiable; +import org.graalvm.compiler.graph.spi.SimplifierTool; import org.graalvm.compiler.nodeinfo.NodeInfo; +import org.graalvm.compiler.nodes.AbstractMergeNode; +import org.graalvm.compiler.nodes.ControlSplitNode; import org.graalvm.compiler.nodes.DeoptimizingFixedWithNextNode; import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.ValueNode; @@ -41,7 +47,7 @@ import org.graalvm.compiler.options.OptionType; import jdk.vm.ci.meta.ResolvedJavaMethod; @NodeInfo(cycles = CYCLES_IGNORED, cyclesRationale = "profiling should be ignored", size = SIZE_IGNORED, sizeRationale = "profiling should be ignored") -public class ProfileNode extends DeoptimizingFixedWithNextNode implements Lowerable { +public abstract class ProfileNode extends DeoptimizingFixedWithNextNode implements Simplifiable, Lowerable { public static class Options { @Option(help = "Control probabilistic profiling on AMD64", type = OptionType.Expert)// public static final OptionKey ProbabilisticProfiling = new OptionKey<>(true); @@ -54,19 +60,21 @@ public class ProfileNode extends DeoptimizingFixedWithNextNode implements Lowera // Only used if ProbabilisticProfiling == true and may be ignored by lowerer. @OptionalInput protected ValueNode random; - // logarithm base 2 of the profile probability + // Logarithm base 2 of the profile probability. protected int probabilityLog; + // Step value to add to the profile counter. + protected int step; + protected ProfileNode(NodeClass c, ResolvedJavaMethod method, int probabilityLog) { super(c, StampFactory.forVoid()); this.method = method; this.probabilityLog = probabilityLog; + this.step = 1; } public ProfileNode(ResolvedJavaMethod method, int probabilityLog) { - super(TYPE, StampFactory.forVoid()); - this.method = method; - this.probabilityLog = probabilityLog; + this(TYPE, method, probabilityLog); } @Override @@ -92,6 +100,14 @@ public class ProfileNode extends DeoptimizingFixedWithNextNode implements Lowera this.random = r; } + public int getStep() { + return step; + } + + public void setStep(int s) { + step = s; + } + /** * Get the logarithm base 2 of the profile probability. */ @@ -106,4 +122,25 @@ public class ProfileNode extends DeoptimizingFixedWithNextNode implements Lowera public static NodeIterable getProfileNodes(StructuredGraph graph) { return graph.getNodes().filter(ProfileNode.class); } + + protected abstract boolean canBeMergedWith(ProfileNode p); + + @Override + public void simplify(SimplifierTool tool) { + for (Node p = predecessor(); p != null; p = p.predecessor()) { + // Terminate search when we hit a control split or merge. + if (p instanceof ControlSplitNode || p instanceof AbstractMergeNode) { + break; + } + if (p instanceof ProfileNode) { + ProfileNode that = (ProfileNode) p; + if (this.canBeMergedWith(that)) { + that.setStep(this.getStep() + that.getStep()); + removeFixedWithUnusedInputs(this); + tool.addToWorkList(that); + break; + } + } + } + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileWithNotificationNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileWithNotificationNode.java index d0fdcbc2867..1d9281a34da 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileWithNotificationNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/nodes/profiling/ProfileWithNotificationNode.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -28,7 +28,7 @@ import org.graalvm.compiler.nodeinfo.NodeInfo; import jdk.vm.ci.meta.ResolvedJavaMethod; @NodeInfo -public class ProfileWithNotificationNode extends ProfileNode { +public abstract class ProfileWithNotificationNode extends ProfileNode { public static final NodeClass TYPE = NodeClass.create(ProfileWithNotificationNode.class); protected int freqLog; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/phases/OnStackReplacementPhase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/phases/OnStackReplacementPhase.java index df14d7adc37..116ef56436b 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/phases/OnStackReplacementPhase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/phases/OnStackReplacementPhase.java @@ -22,10 +22,16 @@ */ package org.graalvm.compiler.hotspot.phases; +import static jdk.vm.ci.meta.SpeculationLog.SpeculationReason; import static org.graalvm.compiler.phases.common.DeadCodeEliminationPhase.Optionality.Required; +import jdk.vm.ci.meta.DeoptimizationAction; +import jdk.vm.ci.meta.DeoptimizationReason; +import jdk.vm.ci.meta.JavaConstant; +import jdk.vm.ci.meta.JavaKind; import org.graalvm.compiler.core.common.PermanentBailoutException; import org.graalvm.compiler.core.common.cfg.Loop; +import org.graalvm.compiler.core.common.type.ObjectStamp; import org.graalvm.compiler.core.common.type.Stamp; import org.graalvm.compiler.debug.CounterKey; import org.graalvm.compiler.debug.DebugContext; @@ -37,13 +43,15 @@ import org.graalvm.compiler.loop.phases.LoopTransformations; import org.graalvm.compiler.nodeinfo.InputType; import org.graalvm.compiler.nodeinfo.Verbosity; import org.graalvm.compiler.nodes.AbstractBeginNode; -import org.graalvm.compiler.nodes.AbstractLocalNode; import org.graalvm.compiler.nodes.EntryMarkerNode; import org.graalvm.compiler.nodes.EntryProxyNode; +import org.graalvm.compiler.nodes.FixedGuardNode; import org.graalvm.compiler.nodes.FixedNode; import org.graalvm.compiler.nodes.FrameState; +import org.graalvm.compiler.nodes.LogicNode; import org.graalvm.compiler.nodes.LoopBeginNode; import org.graalvm.compiler.nodes.ParameterNode; +import org.graalvm.compiler.nodes.PiNode; import org.graalvm.compiler.nodes.StartNode; import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.ValueNode; @@ -53,6 +61,7 @@ import org.graalvm.compiler.nodes.extended.OSRLockNode; import org.graalvm.compiler.nodes.extended.OSRMonitorEnterNode; import org.graalvm.compiler.nodes.extended.OSRStartNode; import org.graalvm.compiler.nodes.java.AccessMonitorNode; +import org.graalvm.compiler.nodes.java.InstanceOfNode; import org.graalvm.compiler.nodes.java.MonitorEnterNode; import org.graalvm.compiler.nodes.java.MonitorExitNode; import org.graalvm.compiler.nodes.java.MonitorIdNode; @@ -172,15 +181,32 @@ public class OnStackReplacementPhase extends Phase { if (value instanceof EntryProxyNode) { EntryProxyNode proxy = (EntryProxyNode) value; /* - * we need to drop the stamp since the types we see during OSR may be too precise - * (if a branch was not parsed for example). + * We need to drop the stamp since the types we see during OSR may be too precise + * (if a branch was not parsed for example). In cases when this is possible, we + * insert a guard and narrow the OSRLocal stamp at its usages. */ - Stamp s = proxy.stamp().unrestricted(); - AbstractLocalNode osrLocal = null; + Stamp narrowedStamp = proxy.value().stamp(); + Stamp unrestrictedStamp = proxy.stamp().unrestricted(); + ValueNode osrLocal; if (i >= localsSize) { - osrLocal = graph.addOrUnique(new OSRLockNode(i - localsSize, s)); + osrLocal = graph.addOrUnique(new OSRLockNode(i - localsSize, unrestrictedStamp)); } else { - osrLocal = graph.addOrUnique(new OSRLocalNode(i, s)); + osrLocal = graph.addOrUnique(new OSRLocalNode(i, unrestrictedStamp)); + } + // Speculate on the OSRLocal stamps that could be more precise. + OSRLocalSpeculationReason reason = new OSRLocalSpeculationReason(osrState.bci, narrowedStamp, i); + if (graph.getSpeculationLog().maySpeculate(reason) && osrLocal instanceof OSRLocalNode && value.getStackKind().equals(JavaKind.Object) && !narrowedStamp.isUnrestricted()) { + // Add guard. + LogicNode check = graph.addOrUniqueWithInputs(InstanceOfNode.createHelper((ObjectStamp) narrowedStamp, osrLocal, null, null)); + JavaConstant constant = graph.getSpeculationLog().speculate(reason); + FixedGuardNode guard = graph.add(new FixedGuardNode(check, DeoptimizationReason.OptimizedTypeCheckViolated, DeoptimizationAction.InvalidateRecompile, constant, false)); + graph.addAfterFixed(osrStart, guard); + + // Replace with a more specific type at usages. + // We know that we are at the root, + // so we need to replace the proxy in the state. + proxy.replaceAtMatchingUsages(osrLocal, n -> n == osrState); + osrLocal = graph.addOrUnique(new PiNode(osrLocal, narrowedStamp, guard)); } proxy.replaceAndDelete(osrLocal); } else { @@ -268,4 +294,30 @@ public class OnStackReplacementPhase extends Phase { public float codeSizeIncrease() { return 5.0f; } + + private static class OSRLocalSpeculationReason implements SpeculationReason { + private int bci; + private Stamp speculatedStamp; + private int localIndex; + + OSRLocalSpeculationReason(int bci, Stamp speculatedStamp, int localIndex) { + this.bci = bci; + this.speculatedStamp = speculatedStamp; + this.localIndex = localIndex; + } + + @Override + public boolean equals(Object obj) { + if (obj instanceof OSRLocalSpeculationReason) { + OSRLocalSpeculationReason that = (OSRLocalSpeculationReason) obj; + return this.bci == that.bci && this.speculatedStamp.equals(that.speculatedStamp) && this.localIndex == that.localIndex; + } + return false; + } + + @Override + public int hashCode() { + return (bci << 16) ^ speculatedStamp.hashCode() ^ localIndex; + } + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/HotSpotReplacementsUtil.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/HotSpotReplacementsUtil.java index 37e7c1826a8..29a508882d8 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/HotSpotReplacementsUtil.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/HotSpotReplacementsUtil.java @@ -669,6 +669,11 @@ public class HotSpotReplacementsUtil { return config.useG1GC; } + @Fold + public static boolean useCMSIncrementalMode(@InjectedParameter GraalHotSpotVMConfig config) { + return config.cmsIncrementalMode; + } + @Fold public static boolean useCompressedOops(@InjectedParameter GraalHotSpotVMConfig config) { return config.useCompressedOops; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/HubGetClassNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/HubGetClassNode.java index c96dc5c3654..2084027388d 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/HubGetClassNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/HubGetClassNode.java @@ -71,7 +71,7 @@ public final class HubGetClassNode extends FloatingNode implements Lowerable, Ca return null; } else { MetaAccessProvider metaAccess = tool.getMetaAccess(); - if (metaAccess != null && hub.isConstant() && !GraalOptions.ImmutableCode.getValue(graph().getOptions())) { + if (metaAccess != null && hub.isConstant() && !GraalOptions.ImmutableCode.getValue(tool.getOptions())) { ResolvedJavaType exactType = tool.getConstantReflection().asJavaType(hub.asConstant()); if (exactType != null) { return ConstantNode.forConstant(tool.getConstantReflection().asJavaClass(exactType), metaAccess); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/IdentityHashCodeNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/IdentityHashCodeNode.java index 2577987d090..a28f4559f67 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/IdentityHashCodeNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/IdentityHashCodeNode.java @@ -67,7 +67,7 @@ public class IdentityHashCodeNode extends FixedWithNextNode implements Canonical if (object.isConstant()) { assert object.stamp() instanceof AbstractObjectStamp; JavaConstant c = (JavaConstant) object.asConstant(); - if (ImmutableCode.getValue(getOptions())) { + if (ImmutableCode.getValue(tool.getOptions())) { return this; } JavaConstant identityHashCode = null; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopySnippets.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopySnippets.java index bf68c43c48f..a8c59ade39b 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopySnippets.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopySnippets.java @@ -28,11 +28,12 @@ import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil. import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.arrayBaseOffset; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.arrayClassElementOffset; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.arrayIndexScale; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.layoutHelperElementTypePrimitiveInPlace; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.loadHub; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.readLayoutHelper; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.superCheckOffsetOffset; -import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.FAST_PATH_PROBABILITY; +import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.FREQUENT_PROBABILITY; +import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.LIKELY_PROBABILITY; +import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.NOT_FREQUENT_PROBABILITY; import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.SLOW_PATH_PROBABILITY; import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.probability; @@ -47,10 +48,8 @@ import org.graalvm.compiler.debug.DebugHandlersFactory; import org.graalvm.compiler.debug.GraalError; import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.hotspot.meta.HotSpotProviders; -import org.graalvm.compiler.hotspot.nodes.type.KlassPointerStamp; import org.graalvm.compiler.hotspot.word.KlassPointer; import org.graalvm.compiler.nodes.CallTargetNode; -import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.DeoptimizeNode; import org.graalvm.compiler.nodes.Invoke; import org.graalvm.compiler.nodes.InvokeNode; @@ -65,8 +64,10 @@ import org.graalvm.compiler.nodes.spi.LoweringTool; import org.graalvm.compiler.nodes.type.StampTool; import org.graalvm.compiler.nodes.util.GraphUtil; import org.graalvm.compiler.options.OptionValues; +import org.graalvm.compiler.replacements.ReplacementsUtil; import org.graalvm.compiler.replacements.SnippetCounter; import org.graalvm.compiler.replacements.SnippetCounter.Group; +import org.graalvm.compiler.replacements.SnippetIntegerHistogram; import org.graalvm.compiler.replacements.SnippetTemplate; import org.graalvm.compiler.replacements.SnippetTemplate.Arguments; import org.graalvm.compiler.replacements.SnippetTemplate.SnippetInfo; @@ -79,16 +80,208 @@ import org.graalvm.word.LocationIdentity; import org.graalvm.word.WordFactory; import jdk.vm.ci.code.TargetDescription; -import jdk.vm.ci.hotspot.HotSpotResolvedObjectType; import jdk.vm.ci.meta.DeoptimizationAction; import jdk.vm.ci.meta.DeoptimizationReason; -import jdk.vm.ci.meta.JavaConstant; import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.ResolvedJavaMethod; import jdk.vm.ci.meta.ResolvedJavaType; public class ArrayCopySnippets implements Snippets { + private enum ArrayCopyTypeCheck { + UNDEFINED_ARRAY_TYPE_CHECK, + // we know that both objects are arrays and have the same type + NO_ARRAY_TYPE_CHECK, + // can be used when we know that one of the objects is a primitive array + HUB_BASED_ARRAY_TYPE_CHECK, + // must be used when we don't have sufficient information to use one of the others + LAYOUT_HELPER_BASED_ARRAY_TYPE_CHECK + } + + @Snippet + public static void arraycopyZeroLengthSnippet(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter ArrayCopyTypeCheck arrayTypeCheck, + @ConstantParameter Counters counters) { + Object nonNullSrc = GraalDirectives.guardingNonNull(src); + Object nonNullDest = GraalDirectives.guardingNonNull(dest); + checkArrayTypes(nonNullSrc, nonNullDest, arrayTypeCheck); + checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); + counters.zeroLengthStaticCounter.inc(); + } + + @Snippet + public static void arraycopyExactSnippet(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter ArrayCopyTypeCheck arrayTypeCheck, + @ConstantParameter JavaKind elementKind, @ConstantParameter SnippetCounter elementKindCounter, @ConstantParameter SnippetCounter elementKindCopiedCounter, + @ConstantParameter Counters counters) { + Object nonNullSrc = GraalDirectives.guardingNonNull(src); + Object nonNullDest = GraalDirectives.guardingNonNull(dest); + checkArrayTypes(nonNullSrc, nonNullDest, arrayTypeCheck); + checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); + incrementLengthCounter(length, counters); + + elementKindCounter.inc(); + elementKindCopiedCounter.add(length); + ArrayCopyCallNode.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, elementKind); + } + + @Snippet + public static void arraycopyUnrolledSnippet(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter ArrayCopyTypeCheck arrayTypeCheck, + @ConstantParameter JavaKind elementKind, @ConstantParameter int unrolledLength, @ConstantParameter Counters counters) { + Object nonNullSrc = GraalDirectives.guardingNonNull(src); + Object nonNullDest = GraalDirectives.guardingNonNull(dest); + checkArrayTypes(nonNullSrc, nonNullDest, arrayTypeCheck); + checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); + incrementLengthCounter(length, counters); + + unrolledArraycopyWork(nonNullSrc, srcPos, nonNullDest, destPos, unrolledLength, elementKind); + } + + @Snippet + public static void arraycopyCheckcastSnippet(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter ArrayCopyTypeCheck arrayTypeCheck, + @ConstantParameter Counters counters, @ConstantParameter SnippetInfo workSnippet, @ConstantParameter JavaKind elementKind) { + Object nonNullSrc = GraalDirectives.guardingNonNull(src); + Object nonNullDest = GraalDirectives.guardingNonNull(dest); + checkArrayTypes(nonNullSrc, nonNullDest, arrayTypeCheck); + checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); + incrementLengthCounter(length, counters); + + ArrayCopyWithSlowPathNode.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, workSnippet, elementKind); + } + + @Snippet + public static void arraycopyGenericSnippet(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter ArrayCopyTypeCheck arrayTypeCheck, @ConstantParameter Counters counters, + @ConstantParameter SnippetInfo workSnippet, @ConstantParameter JavaKind elementKind) { + Object nonNullSrc = GraalDirectives.guardingNonNull(src); + Object nonNullDest = GraalDirectives.guardingNonNull(dest); + checkArrayTypes(nonNullSrc, nonNullDest, arrayTypeCheck); + checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); + incrementLengthCounter(length, counters); + + ArrayCopyWithSlowPathNode.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, workSnippet, elementKind); + } + + @Snippet + public static void arraycopyNativeSnippet(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter Counters counters) { + // all checks are done in the native method, so no need to emit additional checks here + incrementLengthCounter(length, counters); + counters.systemArraycopyCounter.inc(); + counters.systemArraycopyCopiedCounter.add(length); + + System.arraycopy(src, srcPos, dest, destPos, length); + } + + @Fold + static LocationIdentity getArrayLocation(JavaKind kind) { + return NamedLocationIdentity.getArrayLocation(kind); + } + + private static void unrolledArraycopyWork(Object nonNullSrc, int srcPos, Object nonNullDest, int destPos, int length, JavaKind elementKind) { + int scale = arrayIndexScale(elementKind); + int arrayBaseOffset = arrayBaseOffset(elementKind); + LocationIdentity arrayLocation = getArrayLocation(elementKind); + + long sourceOffset = arrayBaseOffset + (long) srcPos * scale; + long destOffset = arrayBaseOffset + (long) destPos * scale; + long position = 0; + long delta = scale; + if (probability(NOT_FREQUENT_PROBABILITY, nonNullSrc == nonNullDest && srcPos < destPos)) { + // bad aliased case so we need to copy the array from back to front + position = (long) (length - 1) * scale; + delta = -delta; + } + + // the length was already checked before - we can emit unconditional instructions + ExplodeLoopNode.explodeLoop(); + for (int iteration = 0; iteration < length; iteration++) { + Object value = RawLoadNode.load(nonNullSrc, sourceOffset + position, elementKind, arrayLocation); + RawStoreNode.storeObject(nonNullDest, destOffset + position, value, elementKind, arrayLocation, false); + position += delta; + } + } + + @Snippet(allowPartialIntrinsicArgumentMismatch = true) + public static void checkcastArraycopyWithSlowPathWork(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter Counters counters) { + if (probability(FREQUENT_PROBABILITY, length > 0)) { + Object nonNullSrc = PiNode.asNonNullObject(src); + Object nonNullDest = PiNode.asNonNullObject(dest); + KlassPointer srcKlass = loadHub(nonNullSrc); + KlassPointer destKlass = loadHub(nonNullDest); + if (probability(LIKELY_PROBABILITY, srcKlass == destKlass)) { + // no storecheck required. + counters.objectCheckcastSameTypeCounter.inc(); + counters.objectCheckcastSameTypeCopiedCounter.add(length); + ArrayCopyCallNode.arraycopyObjectKillsAny(nonNullSrc, srcPos, nonNullDest, destPos, length); + } else { + KlassPointer destElemKlass = destKlass.readKlassPointer(arrayClassElementOffset(INJECTED_VMCONFIG), OBJ_ARRAY_KLASS_ELEMENT_KLASS_LOCATION); + Word superCheckOffset = WordFactory.signed(destElemKlass.readInt(superCheckOffsetOffset(INJECTED_VMCONFIG), KLASS_SUPER_CHECK_OFFSET_LOCATION)); + + counters.objectCheckcastDifferentTypeCounter.inc(); + counters.objectCheckcastDifferentTypeCopiedCounter.add(length); + + int copiedElements = CheckcastArrayCopyCallNode.checkcastArraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, superCheckOffset, destElemKlass, false); + if (probability(SLOW_PATH_PROBABILITY, copiedElements != 0)) { + /* + * the stub doesn't throw the ArrayStoreException, but returns the number of + * copied elements (xor'd with -1). + */ + copiedElements ^= -1; + System.arraycopy(nonNullSrc, srcPos + copiedElements, nonNullDest, destPos + copiedElements, length - copiedElements); + } + } + } + } + + @Snippet(allowPartialIntrinsicArgumentMismatch = true) + public static void genericArraycopyWithSlowPathWork(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter Counters counters) { + if (probability(FREQUENT_PROBABILITY, length > 0)) { + counters.genericArraycopyDifferentTypeCounter.inc(); + counters.genericArraycopyDifferentTypeCopiedCounter.add(length); + int copiedElements = GenericArrayCopyCallNode.genericArraycopy(src, srcPos, dest, destPos, length); + if (probability(SLOW_PATH_PROBABILITY, copiedElements != 0)) { + /* + * the stub doesn't throw the ArrayStoreException, but returns the number of copied + * elements (xor'd with -1). + */ + copiedElements ^= -1; + System.arraycopy(src, srcPos + copiedElements, dest, destPos + copiedElements, length - copiedElements); + } + } + } + + private static void incrementLengthCounter(int length, Counters counters) { + counters.lengthHistogram.inc(length); + } + + private static void checkLimits(Object src, int srcPos, Object dest, int destPos, int length, Counters counters) { + if (probability(SLOW_PATH_PROBABILITY, srcPos < 0) || + probability(SLOW_PATH_PROBABILITY, destPos < 0) || + probability(SLOW_PATH_PROBABILITY, length < 0) || + probability(SLOW_PATH_PROBABILITY, srcPos > ArrayLengthNode.arrayLength(src) - length) || + probability(SLOW_PATH_PROBABILITY, destPos > ArrayLengthNode.arrayLength(dest) - length)) { + counters.checkAIOOBECounter.inc(); + DeoptimizeNode.deopt(DeoptimizationAction.None, DeoptimizationReason.RuntimeConstraint); + } + counters.checkSuccessCounter.inc(); + } + + private static void checkArrayTypes(Object nonNullSrc, Object nonNullDest, ArrayCopyTypeCheck arrayTypeCheck) { + if (arrayTypeCheck == ArrayCopyTypeCheck.NO_ARRAY_TYPE_CHECK) { + // nothing to do + } else if (arrayTypeCheck == ArrayCopyTypeCheck.HUB_BASED_ARRAY_TYPE_CHECK) { + KlassPointer srcHub = loadHub(nonNullSrc); + KlassPointer destHub = loadHub(nonNullDest); + if (probability(SLOW_PATH_PROBABILITY, srcHub != destHub)) { + DeoptimizeNode.deopt(DeoptimizationAction.None, DeoptimizationReason.RuntimeConstraint); + } + } else if (arrayTypeCheck == ArrayCopyTypeCheck.LAYOUT_HELPER_BASED_ARRAY_TYPE_CHECK) { + KlassPointer srcHub = loadHub(nonNullSrc); + KlassPointer destHub = loadHub(nonNullDest); + checkArrayType(srcHub); + checkArrayType(destHub); + } else { + ReplacementsUtil.staticAssert(false, "unknown array type check"); + } + } + private static int checkArrayType(KlassPointer nonNullHub) { int layoutHelper = readLayoutHelper(nonNullHub); if (probability(SLOW_PATH_PROBABILITY, layoutHelper >= 0)) { @@ -97,528 +290,228 @@ public class ArrayCopySnippets implements Snippets { return layoutHelper; } - private static void checkLimits(Object src, int srcPos, Object dest, int destPos, int length, Counters counters) { - if (probability(SLOW_PATH_PROBABILITY, srcPos < 0)) { - counters.checkAIOOBECounter.inc(); - DeoptimizeNode.deopt(DeoptimizationAction.None, DeoptimizationReason.RuntimeConstraint); - } - if (probability(SLOW_PATH_PROBABILITY, destPos < 0)) { - counters.checkAIOOBECounter.inc(); - DeoptimizeNode.deopt(DeoptimizationAction.None, DeoptimizationReason.RuntimeConstraint); - } - if (probability(SLOW_PATH_PROBABILITY, length < 0)) { - counters.checkAIOOBECounter.inc(); - DeoptimizeNode.deopt(DeoptimizationAction.None, DeoptimizationReason.RuntimeConstraint); - } - if (probability(SLOW_PATH_PROBABILITY, srcPos > ArrayLengthNode.arrayLength(src) - length)) { - counters.checkAIOOBECounter.inc(); - DeoptimizeNode.deopt(DeoptimizationAction.None, DeoptimizationReason.RuntimeConstraint); - } - if (probability(SLOW_PATH_PROBABILITY, destPos > ArrayLengthNode.arrayLength(dest) - length)) { - counters.checkAIOOBECounter.inc(); - DeoptimizeNode.deopt(DeoptimizationAction.None, DeoptimizationReason.RuntimeConstraint); - } - counters.checkSuccessCounter.inc(); - } - - @Snippet - public static void arraycopyZeroLengthIntrinsic(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter Counters counters) { - Object nonNullSrc = GraalDirectives.guardingNonNull(src); - Object nonNullDest = GraalDirectives.guardingNonNull(dest); - KlassPointer srcHub = loadHub(nonNullSrc); - KlassPointer destHub = loadHub(nonNullDest); - checkArrayType(srcHub); - checkArrayType(destHub); - checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); - counters.zeroLengthStaticCounter.inc(); - } - - @Snippet - public static void arraycopyExactIntrinsic(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter JavaKind elementKind, @ConstantParameter SnippetCounter counter, - @ConstantParameter SnippetCounter copiedCounter, @ConstantParameter Counters counters) { - Object nonNullSrc = GraalDirectives.guardingNonNull(src); - Object nonNullDest = GraalDirectives.guardingNonNull(dest); - checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); - counter.inc(); - copiedCounter.add(length); - ArrayCopyCallNode.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, elementKind); - if (length == 0) { - counters.zeroLengthDynamicCounter.inc(); - } else { - counters.nonZeroLengthDynamicCounter.inc(); - counters.nonZeroLengthDynamicCopiedCounter.add(length); - } - } - - /** - * This intrinsic is useful for the case where we know something statically about one of the - * inputs but not the other. - */ - @Snippet - public static void arraycopyPredictedExactIntrinsic(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter JavaKind elementKind, - @ConstantParameter SnippetCounter counter, @ConstantParameter SnippetCounter copiedCounter, @ConstantParameter Counters counters) { - Object nonNullSrc = GraalDirectives.guardingNonNull(src); - Object nonNullDest = GraalDirectives.guardingNonNull(dest); - KlassPointer srcHub = loadHub(nonNullSrc); - KlassPointer destHub = loadHub(nonNullDest); - if (probability(SLOW_PATH_PROBABILITY, srcHub != destHub)) { - DeoptimizeNode.deopt(DeoptimizationAction.None, DeoptimizationReason.RuntimeConstraint); - } - checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); - counter.inc(); - copiedCounter.add(length); - ArrayCopyCallNode.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, elementKind); - if (length == 0) { - counters.zeroLengthDynamicCounter.inc(); - } else { - counters.nonZeroLengthDynamicCounter.inc(); - counters.nonZeroLengthDynamicCopiedCounter.add(length); - } - } - - @Snippet - public static void arraycopyPredictedObjectWork(Object nonNullSrc, int srcPos, Object nonNullDest, int destPos, int length, KlassPointer objectArrayKlass, - @ConstantParameter SnippetCounter counter, @ConstantParameter SnippetCounter copiedCounter, @ConstantParameter Counters counters) { - if (length > 0) { - KlassPointer srcHub = loadHub(PiNode.asNonNullObject(nonNullSrc)); - KlassPointer destHub = loadHub(PiNode.asNonNullObject(nonNullDest)); - if (probability(FAST_PATH_PROBABILITY, srcHub == destHub || destHub == objectArrayKlass)) { - counter.inc(); - copiedCounter.add(length); - counters.predictedObjectArrayCopyFastPathCounter.inc(); - counters.predictedObjectArrayCopyFastPathCopiedCounter.add(length); - ArrayCopyCallNode.arraycopyObjectKillsAny(nonNullSrc, srcPos, nonNullDest, destPos, length); - } else { - counters.predictedObjectArrayCopySlowPathCounter.inc(); - counters.predictedObjectArrayCopySlowPathCopiedCounter.add(length); - System.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length); - } - } - } - - /** - * This is the basic template for the full arraycopy checks, including a check that the - * underlying type is really an array type. - */ - @Snippet - public static void arraycopySlowPathIntrinsic(Object src, int srcPos, Object dest, int destPos, int length, KlassPointer predictedKlass, @ConstantParameter JavaKind elementKind, - @ConstantParameter SnippetInfo slowPath, @ConstantParameter Object slowPathArgument, @ConstantParameter Counters counters) { - Object nonNullSrc = GraalDirectives.guardingNonNull(src); - Object nonNullDest = GraalDirectives.guardingNonNull(dest); - KlassPointer srcHub = loadHub(nonNullSrc); - KlassPointer destHub = loadHub(nonNullDest); - checkArrayType(srcHub); - checkArrayType(destHub); - checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); - if (length == 0) { - counters.zeroLengthDynamicCounter.inc(); - } else { - counters.nonZeroLengthDynamicCounter.inc(); - counters.nonZeroLengthDynamicCopiedCounter.add(length); - } - ArrayCopySlowPathNode.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, predictedKlass, elementKind, slowPath, slowPathArgument); - } - - /** - * Snippet for unrolled arraycopy. - */ - @Snippet - public static void arraycopyUnrolledIntrinsic(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter int unrolledLength, @ConstantParameter JavaKind elementKind, - @ConstantParameter Counters counters) { - Object nonNullSrc = GraalDirectives.guardingNonNull(src); - Object nonNullDest = GraalDirectives.guardingNonNull(dest); - checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); - if (length == 0) { - counters.zeroLengthDynamicCounter.inc(); - } else { - counters.nonZeroLengthDynamicCounter.inc(); - counters.nonZeroLengthDynamicCopiedCounter.add(length); - } - ArrayCopyUnrollNode.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, unrolledLength, elementKind); - } - - @Snippet - public static void checkcastArraycopyWork(Object nonNullSrc, int srcPos, Object nonNullDest, int destPos, int length, @ConstantParameter Counters counters) { - if (length > 0) { - KlassPointer destKlass = loadHub(nonNullDest); - KlassPointer srcKlass = loadHub(nonNullSrc); - if (probability(SLOW_PATH_PROBABILITY, srcKlass == destKlass)) { - // no storecheck required. - counters.objectCheckcastSameTypeCounter.inc(); - counters.objectCheckcastSameTypeCopiedCounter.add(length); - ArrayCopyCallNode.arraycopyObjectKillsAny(nonNullSrc, srcPos, nonNullDest, destPos, length); - } else { - KlassPointer destElemKlass = destKlass.readKlassPointer(arrayClassElementOffset(INJECTED_VMCONFIG), OBJ_ARRAY_KLASS_ELEMENT_KLASS_LOCATION); - Word superCheckOffset = WordFactory.signed(destElemKlass.readInt(superCheckOffsetOffset(INJECTED_VMCONFIG), KLASS_SUPER_CHECK_OFFSET_LOCATION)); - counters.objectCheckcastCounter.inc(); - counters.objectCheckcastCopiedCounter.add(length); - int copiedElements = CheckcastArrayCopyCallNode.checkcastArraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, superCheckOffset, destElemKlass, false); - if (copiedElements != 0) { - /* - * the checkcast stub doesn't throw the ArrayStoreException, but returns the - * number of copied elements (xor'd with -1). - */ - copiedElements ^= -1; - System.arraycopy(nonNullSrc, srcPos + copiedElements, nonNullDest, destPos + copiedElements, length - copiedElements); - } - } - } - } - - @Snippet - public static void arraycopyGeneric(Object src, int srcPos, Object dest, int destPos, int length, @ConstantParameter Counters counters) { - Object nonNullSrc = GraalDirectives.guardingNonNull(src); - Object nonNullDest = GraalDirectives.guardingNonNull(dest); - KlassPointer srcHub = loadHub(nonNullSrc); - KlassPointer destHub = loadHub(nonNullDest); - if (probability(FAST_PATH_PROBABILITY, srcHub.equal(destHub)) && probability(FAST_PATH_PROBABILITY, nonNullSrc != nonNullDest)) { - int layoutHelper = checkArrayType(srcHub); - final boolean isObjectArray = ((layoutHelper & layoutHelperElementTypePrimitiveInPlace(INJECTED_VMCONFIG)) == 0); - checkLimits(nonNullSrc, srcPos, nonNullDest, destPos, length, counters); - if (probability(FAST_PATH_PROBABILITY, isObjectArray)) { - counters.genericObjectExactCallCounter.inc(); - counters.genericObjectExactCallCopiedCounter.add(length); - ArrayCopyCallNode.disjointArraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length, JavaKind.Object); - } else { - counters.genericPrimitiveCallCounter.inc(); - counters.genericPrimitiveCallCopiedCounter.add(length); - UnsafeArrayCopyNode.arraycopyPrimitive(nonNullSrc, srcPos, nonNullDest, destPos, length, layoutHelper); - } - } else { - counters.systemArraycopyCounter.inc(); - counters.systemArraycopyCopiedCounter.add(length); - System.arraycopy(nonNullSrc, srcPos, nonNullDest, destPos, length); - } - } - - @Fold - static LocationIdentity getArrayLocation(JavaKind kind) { - return NamedLocationIdentity.getArrayLocation(kind); - } - - @Snippet - public static void arraycopyUnrolledWork(Object nonNullSrc, int srcPos, Object nonNullDest, int destPos, @ConstantParameter int length, @ConstantParameter JavaKind elementKind) { - final int scale = arrayIndexScale(elementKind); - int arrayBaseOffset = arrayBaseOffset(elementKind); - LocationIdentity arrayLocation = getArrayLocation(elementKind); - if (nonNullSrc == nonNullDest && srcPos < destPos) { // bad aliased case - long start = (long) (length - 1) * scale; - long i = start; - ExplodeLoopNode.explodeLoop(); - for (int iteration = 0; iteration < length; iteration++) { - if (i >= 0) { - Object a = RawLoadNode.load(nonNullSrc, arrayBaseOffset + i + (long) srcPos * scale, elementKind, arrayLocation); - RawStoreNode.storeObject(nonNullDest, arrayBaseOffset + i + (long) destPos * scale, a, elementKind, arrayLocation, false); - i -= scale; - } - } - } else { - long end = (long) length * scale; - long i = 0; - ExplodeLoopNode.explodeLoop(); - for (int iteration = 0; iteration < length; iteration++) { - if (i < end) { - Object a = RawLoadNode.load(nonNullSrc, arrayBaseOffset + i + (long) srcPos * scale, elementKind, arrayLocation); - RawStoreNode.storeObject(nonNullDest, arrayBaseOffset + i + (long) destPos * scale, a, elementKind, arrayLocation, false); - i += scale; - } - } - } - } - static class Counters { final SnippetCounter checkSuccessCounter; final SnippetCounter checkAIOOBECounter; - final SnippetCounter objectCheckcastCounter; - final SnippetCounter objectCheckcastSameTypeCounter; - final SnippetCounter predictedObjectArrayCopySlowPathCounter; - final SnippetCounter predictedObjectArrayCopyFastPathCounter; - - final SnippetCounter genericPrimitiveCallCounter; - final SnippetCounter genericObjectExactCallCounter; - final SnippetCounter systemArraycopyCounter; - final SnippetCounter zeroLengthStaticCounter; - final SnippetCounter zeroLengthDynamicCounter; - final SnippetCounter nonZeroLengthDynamicCounter; + final SnippetIntegerHistogram lengthHistogram; - final SnippetCounter nonZeroLengthDynamicCopiedCounter; - final SnippetCounter genericPrimitiveCallCopiedCounter; - final SnippetCounter genericObjectExactCallCopiedCounter; + final SnippetCounter systemArraycopyCounter; final SnippetCounter systemArraycopyCopiedCounter; - final SnippetCounter objectCheckcastCopiedCounter; + final SnippetCounter genericArraycopyDifferentTypeCopiedCounter; + final SnippetCounter genericArraycopyDifferentTypeCounter; + final SnippetCounter objectCheckcastSameTypeCopiedCounter; - final SnippetCounter predictedObjectArrayCopySlowPathCopiedCounter; - final SnippetCounter predictedObjectArrayCopyFastPathCopiedCounter; + final SnippetCounter objectCheckcastSameTypeCounter; + final SnippetCounter objectCheckcastDifferentTypeCopiedCounter; + final SnippetCounter objectCheckcastDifferentTypeCounter; final EnumMap arraycopyCallCounters = new EnumMap<>(JavaKind.class); - final EnumMap arraycopyCounters = new EnumMap<>(JavaKind.class); - final EnumMap arraycopyCallCopiedCounters = new EnumMap<>(JavaKind.class); - final EnumMap arraycopyCopiedCounters = new EnumMap<>(JavaKind.class); Counters(SnippetCounter.Group.Factory factory) { final Group checkCounters = factory.createSnippetCounterGroup("System.arraycopy checkInputs"); - final Group counters = factory.createSnippetCounterGroup("System.arraycopy"); - final Group copiedCounters = factory.createSnippetCounterGroup("System.arraycopy copied elements"); - final Group lengthCounters = factory.createSnippetCounterGroup("System.arraycopy 0-length checks"); + final Group callCounters = factory.createSnippetCounterGroup("System.arraycopy calls"); + final Group copiedElementsCounters = factory.createSnippetCounterGroup("System.arraycopy copied elements"); + final Group lengthCounters = factory.createSnippetCounterGroup("System.arraycopy with 0-length"); checkSuccessCounter = new SnippetCounter(checkCounters, "checkSuccess", "checkSuccess"); checkAIOOBECounter = new SnippetCounter(checkCounters, "checkAIOOBE", "checkAIOOBE"); - objectCheckcastCounter = new SnippetCounter(counters, "Object[]{non-exact}", "arraycopy for non-exact Object[] arrays"); - objectCheckcastSameTypeCounter = new SnippetCounter(counters, "Object[]{same-type}", "arraycopy call for src.klass == dest.klass Object[] arrays"); - predictedObjectArrayCopySlowPathCounter = new SnippetCounter(counters, "Object[]{slow-path}", "used System.arraycopy slow path for predicted Object[] arrays"); - predictedObjectArrayCopyFastPathCounter = new SnippetCounter(counters, "Object[]{fast-path}", "used oop_arraycopy for predicted Object[] arrays"); - genericPrimitiveCallCounter = new SnippetCounter(counters, "genericPrimitive", "generic arraycopy snippet for primitive arrays"); - genericObjectExactCallCounter = new SnippetCounter(counters, "genericObjectExact", "generic arraycopy snippet for special object arrays"); - systemArraycopyCounter = new SnippetCounter(counters, "genericObject", "call to System.arraycopy"); + zeroLengthStaticCounter = new SnippetCounter(lengthCounters, "0-length copy static", "calls where the length is statically 0"); + lengthHistogram = new SnippetIntegerHistogram(lengthCounters, 2, "length", "length"); - zeroLengthStaticCounter = new SnippetCounter(lengthCounters, "0-lengthcopy static", "arraycopy where the length is statically 0"); - zeroLengthDynamicCounter = new SnippetCounter(lengthCounters, "0-lengthcopy dynamically", "arraycopy where the length is dynamically 0"); - nonZeroLengthDynamicCounter = new SnippetCounter(lengthCounters, "non-0-lengthcopy dynamically", "arraycopy where the length is dynamically not zero"); + systemArraycopyCounter = new SnippetCounter(callCounters, "native System.arraycopy", "JNI-based System.arraycopy call"); + systemArraycopyCopiedCounter = new SnippetCounter(copiedElementsCounters, "native System.arraycopy", "JNI-based System.arraycopy call"); - nonZeroLengthDynamicCopiedCounter = new SnippetCounter(copiedCounters, "non-0-lengthcopy dynamically", "arraycopy where the length is dynamically not zero"); - genericPrimitiveCallCopiedCounter = new SnippetCounter(copiedCounters, "genericPrimitive", "generic arraycopy snippet for primitive arrays"); - genericObjectExactCallCopiedCounter = new SnippetCounter(copiedCounters, "genericObjectExact", "generic arraycopy snippet for special object arrays"); - systemArraycopyCopiedCounter = new SnippetCounter(copiedCounters, "genericObject", "call to System.arraycopy"); + genericArraycopyDifferentTypeCounter = new SnippetCounter(callCounters, "generic[] stub", "generic arraycopy stub"); + genericArraycopyDifferentTypeCopiedCounter = new SnippetCounter(copiedElementsCounters, "generic[] stub", "generic arraycopy stub"); - objectCheckcastCopiedCounter = new SnippetCounter(copiedCounters, "Object[]{non-exact}", "arraycopy for non-exact Object[] arrays"); - objectCheckcastSameTypeCopiedCounter = new SnippetCounter(copiedCounters, "Object[]{same-type}", "arraycopy call for src.klass == dest.klass Object[] arrays"); - predictedObjectArrayCopySlowPathCopiedCounter = new SnippetCounter(copiedCounters, "Object[]{slow-path}", - "used System.arraycopy slow path for predicted Object[] arrays"); - predictedObjectArrayCopyFastPathCopiedCounter = new SnippetCounter(copiedCounters, "Object[]{fast-path}", "used oop_arraycopy for predicted Object[] arrays"); - createArraycopyCounter(JavaKind.Byte, counters, copiedCounters); - createArraycopyCounter(JavaKind.Boolean, counters, copiedCounters); - createArraycopyCounter(JavaKind.Char, counters, copiedCounters); - createArraycopyCounter(JavaKind.Short, counters, copiedCounters); - createArraycopyCounter(JavaKind.Int, counters, copiedCounters); - createArraycopyCounter(JavaKind.Long, counters, copiedCounters); - createArraycopyCounter(JavaKind.Float, counters, copiedCounters); - createArraycopyCounter(JavaKind.Double, counters, copiedCounters); - createArraycopyCounter(JavaKind.Object, counters, copiedCounters); + objectCheckcastSameTypeCounter = new SnippetCounter(callCounters, "checkcast object[] (same-type)", "checkcast object[] stub but src.klass == dest.klass Object[] arrays"); + objectCheckcastSameTypeCopiedCounter = new SnippetCounter(copiedElementsCounters, "checkcast object[] (same-type)", "checkcast object[] stub but src.klass == dest.klass Object[] arrays"); + objectCheckcastDifferentTypeCounter = new SnippetCounter(callCounters, "checkcast object[] (store-check)", "checkcast object[] stub with store check"); + objectCheckcastDifferentTypeCopiedCounter = new SnippetCounter(copiedElementsCounters, "checkcast object[] (store-check)", "checkcast object[] stub with store check"); + + createArraycopyCounter(JavaKind.Byte, callCounters, copiedElementsCounters); + createArraycopyCounter(JavaKind.Boolean, callCounters, copiedElementsCounters); + createArraycopyCounter(JavaKind.Char, callCounters, copiedElementsCounters); + createArraycopyCounter(JavaKind.Short, callCounters, copiedElementsCounters); + createArraycopyCounter(JavaKind.Int, callCounters, copiedElementsCounters); + createArraycopyCounter(JavaKind.Long, callCounters, copiedElementsCounters); + createArraycopyCounter(JavaKind.Float, callCounters, copiedElementsCounters); + createArraycopyCounter(JavaKind.Double, callCounters, copiedElementsCounters); + createArraycopyCounter(JavaKind.Object, callCounters, copiedElementsCounters); } void createArraycopyCounter(JavaKind kind, Group counters, Group copiedCounters) { - arraycopyCallCounters.put(kind, new SnippetCounter(counters, kind + "[]{stub}", "arraycopy call for " + kind + "[] arrays")); - arraycopyCounters.put(kind, new SnippetCounter(counters, kind + "[]{inline}", "inline arraycopy for " + kind + "[] arrays")); - - arraycopyCallCopiedCounters.put(kind, new SnippetCounter(copiedCounters, kind + "[]{stub}", "arraycopy call for " + kind + "[] arrays")); - arraycopyCopiedCounters.put(kind, new SnippetCounter(copiedCounters, kind + "[]{inline}", "inline arraycopy for " + kind + "[] arrays")); + arraycopyCallCounters.put(kind, new SnippetCounter(counters, kind + "[] stub", "arraycopy call for " + kind + "[] arrays")); + arraycopyCallCopiedCounters.put(kind, new SnippetCounter(copiedCounters, kind + "[] stub", "arraycopy call for " + kind + "[] arrays")); } } public static class Templates extends SnippetTemplate.AbstractTemplates { + private final SnippetInfo arraycopyGenericSnippet = snippet("arraycopyGenericSnippet"); + private final SnippetInfo arraycopyUnrolledSnippet = snippet("arraycopyUnrolledSnippet"); + private final SnippetInfo arraycopyExactSnippet = snippet("arraycopyExactSnippet"); + private final SnippetInfo arraycopyZeroLengthSnippet = snippet("arraycopyZeroLengthSnippet"); + private final SnippetInfo arraycopyCheckcastSnippet = snippet("arraycopyCheckcastSnippet"); + private final SnippetInfo arraycopyNativeSnippet = snippet("arraycopyNativeSnippet"); + + private final SnippetInfo checkcastArraycopyWithSlowPathWork = snippet("checkcastArraycopyWithSlowPathWork"); + private final SnippetInfo genericArraycopyWithSlowPathWork = snippet("genericArraycopyWithSlowPathWork"); + + private ResolvedJavaMethod originalArraycopy; + private final Counters counters; public Templates(OptionValues options, Iterable factories, SnippetCounter.Group.Factory factory, HotSpotProviders providers, TargetDescription target) { super(options, factories, providers, providers.getSnippetReflection(), target); this.counters = new Counters(factory); } - private ResolvedJavaMethod originalArraycopy() throws GraalError { - if (originalArraycopy == null) { - Method method; - try { - method = System.class.getDeclaredMethod("arraycopy", Object.class, int.class, Object.class, int.class, int.class); - } catch (NoSuchMethodException | SecurityException e) { - throw new GraalError(e); - } - originalArraycopy = providers.getMetaAccess().lookupJavaMethod(method); - } - return originalArraycopy; - } - - private ResolvedJavaMethod originalArraycopy; - - private final SnippetInfo checkcastArraycopyWorkSnippet = snippet("checkcastArraycopyWork"); - private final SnippetInfo arraycopyGenericSnippet = snippet("arraycopyGeneric"); - - private final SnippetInfo arraycopySlowPathIntrinsicSnippet = snippet("arraycopySlowPathIntrinsic"); - private final SnippetInfo arraycopyUnrolledIntrinsicSnippet = snippet("arraycopyUnrolledIntrinsic"); - private final SnippetInfo arraycopyExactIntrinsicSnippet = snippet("arraycopyExactIntrinsic"); - private final SnippetInfo arraycopyZeroLengthIntrinsicSnippet = snippet("arraycopyZeroLengthIntrinsic"); - private final SnippetInfo arraycopyPredictedExactIntrinsicSnippet = snippet("arraycopyPredictedExactIntrinsic"); - private final SnippetInfo arraycopyPredictedObjectWorkSnippet = snippet("arraycopyPredictedObjectWork"); - - private final SnippetInfo arraycopyUnrolledWorkSnippet = snippet("arraycopyUnrolledWork"); - - private final Counters counters; - protected SnippetInfo snippet(String methodName) { SnippetInfo info = snippet(ArrayCopySnippets.class, methodName, LocationIdentity.any()); info.setOriginalMethod(originalArraycopy()); return info; } - public static JavaKind selectComponentKind(BasicArrayCopyNode arraycopy) { - return selectComponentKind(arraycopy, true); - } + public void lower(ArrayCopyNode arraycopy, LoweringTool tool) { + JavaKind elementKind = selectComponentKind(arraycopy); + SnippetInfo snippetInfo; + ArrayCopyTypeCheck arrayTypeCheck; - public static JavaKind selectComponentKind(BasicArrayCopyNode arraycopy, boolean exact) { ResolvedJavaType srcType = StampTool.typeOrNull(arraycopy.getSource().stamp()); ResolvedJavaType destType = StampTool.typeOrNull(arraycopy.getDestination().stamp()); - - if (srcType == null || !srcType.isArray() || destType == null || !destType.isArray()) { - if (!exact) { - JavaKind component = getComponentKind(srcType); - if (component != null) { - return component; - } - return getComponentKind(destType); - } - return null; - } - if (exact) { - if (!destType.getComponentType().isAssignableFrom(srcType.getComponentType())) { - return null; - } - if (!arraycopy.isExact()) { - return null; - } - } - return srcType.getComponentType().getJavaKind(); - } - - private static JavaKind getComponentKind(ResolvedJavaType type) { - if (type != null && type.isArray()) { - return type.getComponentType().getJavaKind(); - } - return null; - } - - private static boolean shouldUnroll(ValueNode length) { - return length.isConstant() && length.asJavaConstant().asInt() <= 8 && length.asJavaConstant().asInt() != 0; - } - - public void lower(ArrayCopyNode arraycopy, LoweringTool tool) { - JavaKind componentKind = selectComponentKind(arraycopy); - SnippetInfo snippetInfo = null; - SnippetInfo slowPathSnippetInfo = null; - Object slowPathArgument = null; - - if (arraycopy.getLength().isConstant() && arraycopy.getLength().asJavaConstant().asLong() == 0) { - snippetInfo = arraycopyZeroLengthIntrinsicSnippet; - } else if (arraycopy.isExact()) { - snippetInfo = arraycopyExactIntrinsicSnippet; - if (shouldUnroll(arraycopy.getLength())) { - snippetInfo = arraycopyUnrolledIntrinsicSnippet; - } + if (!canBeArray(srcType) || !canBeArray(destType)) { + // at least one of the objects is definitely not an array - use the native call + // right away as the copying will fail anyways + snippetInfo = arraycopyNativeSnippet; + arrayTypeCheck = ArrayCopyTypeCheck.UNDEFINED_ARRAY_TYPE_CHECK; } else { - if (componentKind == JavaKind.Object) { - ResolvedJavaType srcType = StampTool.typeOrNull(arraycopy.getSource().stamp()); - ResolvedJavaType destType = StampTool.typeOrNull(arraycopy.getDestination().stamp()); - ResolvedJavaType srcComponentType = srcType == null ? null : srcType.getComponentType(); - ResolvedJavaType destComponentType = destType == null ? null : destType.getComponentType(); - if (srcComponentType != null && destComponentType != null && !srcComponentType.isPrimitive() && !destComponentType.isPrimitive()) { - snippetInfo = arraycopySlowPathIntrinsicSnippet; - slowPathSnippetInfo = checkcastArraycopyWorkSnippet; - slowPathArgument = LocationIdentity.any(); - /* - * Because this snippet has to use Sysytem.arraycopy as a slow path, we must - * pretend to kill any() so clear the componentKind. - */ - componentKind = null; - } - } - if (componentKind == null && snippetInfo == null) { - JavaKind predictedKind = selectComponentKind(arraycopy, false); - if (predictedKind != null) { - /* - * At least one array is of a known type requiring no store checks, so - * assume the other is of the same type. Generally this is working around - * deficiencies in our propagation of type information. - */ - componentKind = predictedKind; - if (predictedKind == JavaKind.Object) { - snippetInfo = arraycopySlowPathIntrinsicSnippet; - slowPathSnippetInfo = arraycopyPredictedObjectWorkSnippet; - slowPathArgument = predictedKind; - componentKind = null; - } else { - snippetInfo = arraycopyPredictedExactIntrinsicSnippet; - } - } - } - if (snippetInfo == null) { + ResolvedJavaType srcComponentType = srcType == null ? null : srcType.getComponentType(); + ResolvedJavaType destComponentType = destType == null ? null : destType.getComponentType(); + + if (arraycopy.isExact()) { + // there is a sufficient type match - we don't need any additional type checks + snippetInfo = arraycopyExactSnippet; + arrayTypeCheck = ArrayCopyTypeCheck.NO_ARRAY_TYPE_CHECK; + } else if (srcComponentType == null && destComponentType == null) { + // we don't know anything about the types - use the generic copying snippetInfo = arraycopyGenericSnippet; + arrayTypeCheck = ArrayCopyTypeCheck.LAYOUT_HELPER_BASED_ARRAY_TYPE_CHECK; + } else if (srcComponentType != null && destComponentType != null) { + if (!srcComponentType.isPrimitive() && !destComponentType.isPrimitive()) { + // it depends on the array content if the copy succeeds - we need + // a type check for every store + snippetInfo = arraycopyCheckcastSnippet; + arrayTypeCheck = ArrayCopyTypeCheck.NO_ARRAY_TYPE_CHECK; + } else { + // one object is an object array, the other one is a primitive array. + // this copy will always fail - use the native call right away + assert !srcComponentType.equals(destComponentType) : "must be handled by arraycopy.isExact()"; + snippetInfo = arraycopyNativeSnippet; + arrayTypeCheck = ArrayCopyTypeCheck.UNDEFINED_ARRAY_TYPE_CHECK; + } + } else { + ResolvedJavaType nonNullComponentType = srcComponentType != null ? srcComponentType : destComponentType; + if (nonNullComponentType.isPrimitive()) { + // one involved object is a primitive array - we can safely assume that we + // are copying primitive arrays + snippetInfo = arraycopyExactSnippet; + arrayTypeCheck = ArrayCopyTypeCheck.HUB_BASED_ARRAY_TYPE_CHECK; + elementKind = nonNullComponentType.getJavaKind(); + } else { + // one involved object is an object array - we can safely assume that we are + // copying object arrays that might require a store check + snippetInfo = arraycopyCheckcastSnippet; + arrayTypeCheck = ArrayCopyTypeCheck.LAYOUT_HELPER_BASED_ARRAY_TYPE_CHECK; + } } } + + // a few special cases that are easier to handle when all other variables already have a + // value + if (arraycopy.getLength().isConstant() && arraycopy.getLength().asJavaConstant().asLong() == 0) { + snippetInfo = arraycopyZeroLengthSnippet; + } else if (snippetInfo == arraycopyExactSnippet && shouldUnroll(arraycopy.getLength())) { + snippetInfo = arraycopyUnrolledSnippet; + } + + // create the snippet Arguments args = new Arguments(snippetInfo, arraycopy.graph().getGuardsStage(), tool.getLoweringStage()); args.add("src", arraycopy.getSource()); args.add("srcPos", arraycopy.getSourcePosition()); args.add("dest", arraycopy.getDestination()); args.add("destPos", arraycopy.getDestinationPosition()); args.add("length", arraycopy.getLength()); - if (snippetInfo == arraycopyUnrolledIntrinsicSnippet) { - args.addConst("unrolledLength", arraycopy.getLength().asJavaConstant().asInt()); - args.addConst("elementKind", componentKind != null ? componentKind : JavaKind.Illegal); - } else if (snippetInfo == arraycopySlowPathIntrinsicSnippet) { - ValueNode predictedKlass = null; - if (slowPathArgument == arraycopyPredictedObjectWorkSnippet) { - HotSpotResolvedObjectType arrayClass = (HotSpotResolvedObjectType) tool.getMetaAccess().lookupJavaType(Object[].class); - predictedKlass = ConstantNode.forConstant(KlassPointerStamp.klassNonNull(), arrayClass.klass(), tool.getMetaAccess(), arraycopy.graph()); - } else { - predictedKlass = ConstantNode.forConstant(KlassPointerStamp.klassAlwaysNull(), JavaConstant.NULL_POINTER, tool.getMetaAccess(), arraycopy.graph()); - } - args.add("predictedKlass", predictedKlass); - args.addConst("elementKind", componentKind != null ? componentKind : JavaKind.Illegal); - args.addConst("slowPath", slowPathSnippetInfo); - assert slowPathArgument != null; - args.addConst("slowPathArgument", slowPathArgument); - } else if (snippetInfo == arraycopyExactIntrinsicSnippet || snippetInfo == arraycopyPredictedExactIntrinsicSnippet) { - assert componentKind != null; - args.addConst("elementKind", componentKind); - args.addConst("counter", counters.arraycopyCallCounters.get(componentKind)); - args.addConst("copiedCounter", counters.arraycopyCallCopiedCounters.get(componentKind)); + if (snippetInfo != arraycopyNativeSnippet) { + assert arrayTypeCheck != ArrayCopyTypeCheck.UNDEFINED_ARRAY_TYPE_CHECK; + args.addConst("arrayTypeCheck", arrayTypeCheck); } + if (snippetInfo == arraycopyUnrolledSnippet) { + args.addConst("elementKind", elementKind != null ? elementKind : JavaKind.Illegal); + args.addConst("unrolledLength", arraycopy.getLength().asJavaConstant().asInt()); + } + if (snippetInfo == arraycopyExactSnippet) { + assert elementKind != null; + args.addConst("elementKind", elementKind); + args.addConst("elementKindCounter", counters.arraycopyCallCounters.get(elementKind)); + args.addConst("elementKindCopiedCounter", counters.arraycopyCallCopiedCounters.get(elementKind)); + } + args.addConst("counters", counters); + if (snippetInfo == arraycopyCheckcastSnippet) { + args.addConst("workSnippet", checkcastArraycopyWithSlowPathWork); + args.addConst("elementKind", JavaKind.Illegal); + } + if (snippetInfo == arraycopyGenericSnippet) { + args.addConst("workSnippet", genericArraycopyWithSlowPathWork); + args.addConst("elementKind", JavaKind.Illegal); + } + + instantiate(args, arraycopy); + } + + public void lower(ArrayCopyWithSlowPathNode arraycopy, LoweringTool tool) { + StructuredGraph graph = arraycopy.graph(); + if (!graph.getGuardsStage().areFrameStatesAtDeopts()) { + // if an arraycopy contains a slow path, we can't lower it right away + return; + } + + SnippetInfo snippetInfo = arraycopy.getSnippet(); + Arguments args = new Arguments(snippetInfo, graph.getGuardsStage(), tool.getLoweringStage()); + args.add("src", arraycopy.getSource()); + args.add("srcPos", arraycopy.getSourcePosition()); + args.add("dest", arraycopy.getDestination()); + args.add("destPos", arraycopy.getDestinationPosition()); + args.add("length", arraycopy.getLength()); args.addConst("counters", counters); instantiate(args, arraycopy); } - public void lower(ArrayCopySlowPathNode arraycopy, LoweringTool tool) { - StructuredGraph graph = arraycopy.graph(); - if (!graph.getGuardsStage().areFrameStatesAtDeopts()) { - // Can't be lowered yet - return; - } - SnippetInfo snippetInfo = arraycopy.getSnippet(); - Arguments args = new Arguments(snippetInfo, graph.getGuardsStage(), tool.getLoweringStage()); - args.add("nonNullSrc", arraycopy.getSource()); - args.add("srcPos", arraycopy.getSourcePosition()); - args.add("nonNullDest", arraycopy.getDestination()); - args.add("destPos", arraycopy.getDestinationPosition()); - if (snippetInfo == arraycopyUnrolledWorkSnippet) { - args.addConst("length", ((Integer) arraycopy.getArgument()).intValue()); - args.addConst("elementKind", arraycopy.getElementKind()); - } else { - args.add("length", arraycopy.getLength()); - } - if (snippetInfo == arraycopyPredictedObjectWorkSnippet) { - args.add("objectArrayKlass", arraycopy.getPredictedKlass()); - args.addConst("counter", counters.arraycopyCallCounters.get(JavaKind.Object)); - args.addConst("copiedCounter", counters.arraycopyCallCopiedCounters.get(JavaKind.Object)); - args.addConst("counters", counters); - } - instantiate(args, arraycopy); + private static boolean canBeArray(ResolvedJavaType type) { + return type == null || type.isJavaLangObject() || type.isArray(); } - public void lower(ArrayCopyUnrollNode arraycopy, LoweringTool tool) { - StructuredGraph graph = arraycopy.graph(); - if (!graph.getGuardsStage().areFrameStatesAtDeopts()) { - // Can't be lowered yet - return; + public static JavaKind selectComponentKind(BasicArrayCopyNode arraycopy) { + ResolvedJavaType srcType = StampTool.typeOrNull(arraycopy.getSource().stamp()); + ResolvedJavaType destType = StampTool.typeOrNull(arraycopy.getDestination().stamp()); + + if (srcType == null || !srcType.isArray() || destType == null || !destType.isArray()) { + return null; } - SnippetInfo snippetInfo = arraycopyUnrolledWorkSnippet; - Arguments args = new Arguments(snippetInfo, graph.getGuardsStage(), tool.getLoweringStage()); - args.add("nonNullSrc", arraycopy.getSource()); - args.add("srcPos", arraycopy.getSourcePosition()); - args.add("nonNullDest", arraycopy.getDestination()); - args.add("destPos", arraycopy.getDestinationPosition()); - args.addConst("length", arraycopy.getUnrollLength()); - args.addConst("elementKind", arraycopy.getElementKind()); - template(graph.getDebug(), args).instantiate(providers.getMetaAccess(), arraycopy, SnippetTemplate.DEFAULT_REPLACER, args); + if (!destType.getComponentType().isAssignableFrom(srcType.getComponentType())) { + return null; + } + if (!arraycopy.isExact()) { + return null; + } + return srcType.getComponentType().getJavaKind(); + } + + private static boolean shouldUnroll(ValueNode length) { + return length.isConstant() && length.asJavaConstant().asInt() <= 8 && length.asJavaConstant().asInt() != 0; } /** @@ -650,8 +543,8 @@ public class ArrayCopySnippets implements Snippets { newInvoke.setStateAfter(arraycopy.stateAfter()); } graph.replaceFixedWithFixed((InvokeNode) invoke.asNode(), newInvoke); - } else if (originalNode instanceof ArrayCopySlowPathNode) { - ArrayCopySlowPathNode slowPath = (ArrayCopySlowPathNode) replacements.get(originalNode); + } else if (originalNode instanceof ArrayCopyWithSlowPathNode) { + ArrayCopyWithSlowPathNode slowPath = (ArrayCopyWithSlowPathNode) replacements.get(originalNode); assert arraycopy.stateAfter() != null : arraycopy; assert slowPath.stateAfter() == arraycopy.stateAfter(); slowPath.setBci(arraycopy.getBci()); @@ -659,5 +552,18 @@ public class ArrayCopySnippets implements Snippets { } GraphUtil.killCFG(arraycopy); } + + private ResolvedJavaMethod originalArraycopy() throws GraalError { + if (originalArraycopy == null) { + Method method; + try { + method = System.class.getDeclaredMethod("arraycopy", Object.class, int.class, Object.class, int.class, int.class); + } catch (NoSuchMethodException | SecurityException e) { + throw new GraalError(e); + } + originalArraycopy = providers.getMetaAccess().lookupJavaMethod(method); + } + return originalArraycopy; + } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopyUnrollNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopyUnrollNode.java deleted file mode 100644 index 7bd998024d3..00000000000 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopyUnrollNode.java +++ /dev/null @@ -1,148 +0,0 @@ -/* - * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ -package org.graalvm.compiler.hotspot.replacements.arraycopy; - -import jdk.vm.ci.meta.JavaKind; - -import static org.graalvm.word.LocationIdentity.any; - -import org.graalvm.compiler.core.common.type.StampFactory; -import org.graalvm.compiler.graph.NodeClass; -import org.graalvm.compiler.nodeinfo.InputType; -import org.graalvm.compiler.nodeinfo.NodeInfo; -import org.graalvm.compiler.nodes.NamedLocationIdentity; -import org.graalvm.compiler.nodes.ValueNode; -import org.graalvm.compiler.nodes.extended.ArrayRangeWriteNode; -import org.graalvm.compiler.nodes.memory.MemoryAccess; -import org.graalvm.compiler.nodes.memory.MemoryCheckpoint; -import org.graalvm.compiler.nodes.memory.MemoryNode; -import org.graalvm.compiler.nodes.spi.Lowerable; -import org.graalvm.compiler.nodes.spi.LoweringTool; -import org.graalvm.word.LocationIdentity; - -@NodeInfo(allowedUsageTypes = InputType.Memory) -public class ArrayCopyUnrollNode extends ArrayRangeWriteNode implements MemoryCheckpoint.Single, Lowerable, MemoryAccess { - - public static final NodeClass TYPE = NodeClass.create(ArrayCopyUnrollNode.class); - - @Input protected ValueNode src; - @Input protected ValueNode srcPos; - @Input protected ValueNode dest; - @Input protected ValueNode destPos; - @Input protected ValueNode length; - - private JavaKind elementKind; - - private int unrolledLength; - - @OptionalInput(InputType.Memory) private MemoryNode lastLocationAccess; - - public ArrayCopyUnrollNode(ValueNode src, ValueNode srcPos, ValueNode dest, ValueNode destPos, ValueNode length, int unrolledLength, JavaKind elementKind) { - super(TYPE, StampFactory.forKind(JavaKind.Void)); - this.src = src; - this.srcPos = srcPos; - this.dest = dest; - this.destPos = destPos; - this.length = length; - this.unrolledLength = unrolledLength; - assert elementKind != null && elementKind != JavaKind.Illegal; - this.elementKind = elementKind; - } - - public ValueNode getSource() { - return src; - } - - public ValueNode getSourcePosition() { - return srcPos; - } - - public ValueNode getDestination() { - return dest; - } - - public ValueNode getDestinationPosition() { - return destPos; - } - - @Override - public ValueNode getLength() { - return length; - } - - @Override - public ValueNode getArray() { - return dest; - } - - @Override - public ValueNode getIndex() { - return destPos; - } - - @Override - public boolean isObjectArray() { - return elementKind == JavaKind.Object; - } - - @Override - public boolean isInitialization() { - return false; - } - - @NodeIntrinsic - public static native void arraycopy(Object nonNullSrc, int srcPos, Object nonNullDest, int destPos, int length, @ConstantNodeParameter int unrolledLength, - @ConstantNodeParameter JavaKind elementKind); - - public int getUnrollLength() { - return unrolledLength; - } - - public JavaKind getElementKind() { - return elementKind; - } - - @Override - public LocationIdentity getLocationIdentity() { - if (elementKind != null) { - return NamedLocationIdentity.getArrayLocation(elementKind); - } - return any(); - } - - @Override - public void lower(LoweringTool tool) { - tool.getLowerer().lower(this, tool); - } - - @Override - public MemoryNode getLastLocationAccess() { - return lastLocationAccess; - } - - @Override - public void setLastLocationAccess(MemoryNode lla) { - updateUsagesInterface(lastLocationAccess, lla); - lastLocationAccess = lla; - } -} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopySlowPathNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopyWithSlowPathNode.java similarity index 54% rename from src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopySlowPathNode.java rename to src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopyWithSlowPathNode.java index 99591564e3d..0afcdee6c1c 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopySlowPathNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/ArrayCopyWithSlowPathNode.java @@ -22,76 +22,39 @@ */ package org.graalvm.compiler.hotspot.replacements.arraycopy; -import jdk.vm.ci.code.BytecodeFrame; -import jdk.vm.ci.meta.JavaKind; - -import static org.graalvm.word.LocationIdentity.any; - import org.graalvm.compiler.graph.NodeClass; -import org.graalvm.compiler.hotspot.word.KlassPointer; import org.graalvm.compiler.nodeinfo.InputType; import org.graalvm.compiler.nodeinfo.NodeInfo; -import org.graalvm.compiler.nodes.NamedLocationIdentity; import org.graalvm.compiler.nodes.ValueNode; import org.graalvm.compiler.nodes.type.StampTool; import org.graalvm.compiler.replacements.SnippetTemplate; import org.graalvm.compiler.replacements.nodes.BasicArrayCopyNode; -import org.graalvm.word.LocationIdentity; + +import jdk.vm.ci.code.BytecodeFrame; +import jdk.vm.ci.meta.JavaKind; @NodeInfo(allowedUsageTypes = InputType.Memory) -public final class ArrayCopySlowPathNode extends BasicArrayCopyNode { +public final class ArrayCopyWithSlowPathNode extends BasicArrayCopyNode { - public static final NodeClass TYPE = NodeClass.create(ArrayCopySlowPathNode.class); + public static final NodeClass TYPE = NodeClass.create(ArrayCopyWithSlowPathNode.class); private final SnippetTemplate.SnippetInfo snippet; - /** - * Extra context for the slow path snippet. - */ - private final Object argument; - - /** - * AOT compilation requires klass constants to be exposed after the first lowering to be handled - * automatically. Lowering for {@link ArrayCopySlowPathNode}, with snippet == - * {@link ArrayCopySnippets#arraycopyPredictedObjectWork}, requires a klass of Object[]. For - * other snippets {@link #predictedKlass} is a null constant. - */ - @Input protected ValueNode predictedKlass; - - public ArrayCopySlowPathNode(ValueNode src, ValueNode srcPos, ValueNode dest, ValueNode destPos, ValueNode length, ValueNode predictedKlass, JavaKind elementKind, - SnippetTemplate.SnippetInfo snippet, Object argument) { + public ArrayCopyWithSlowPathNode(ValueNode src, ValueNode srcPos, ValueNode dest, ValueNode destPos, ValueNode length, SnippetTemplate.SnippetInfo snippet, JavaKind elementKind) { super(TYPE, src, srcPos, dest, destPos, length, elementKind, BytecodeFrame.INVALID_FRAMESTATE_BCI); assert StampTool.isPointerNonNull(src) && StampTool.isPointerNonNull(dest) : "must have been null checked"; this.snippet = snippet; - this.argument = argument; - this.predictedKlass = predictedKlass; } @NodeIntrinsic - public static native void arraycopy(Object nonNullSrc, int srcPos, Object nonNullDest, int destPos, int length, KlassPointer predictedKlass, - @ConstantNodeParameter JavaKind elementKind, @ConstantNodeParameter SnippetTemplate.SnippetInfo snippet, @ConstantNodeParameter Object argument); + public static native void arraycopy(Object nonNullSrc, int srcPos, Object nonNullDest, int destPos, int length, @ConstantNodeParameter SnippetTemplate.SnippetInfo snippet, + @ConstantNodeParameter JavaKind elementKind); public SnippetTemplate.SnippetInfo getSnippet() { return snippet; } - public Object getArgument() { - return argument; - } - - @Override - public LocationIdentity getLocationIdentity() { - if (elementKind != null) { - return NamedLocationIdentity.getArrayLocation(elementKind); - } - return any(); - } - public void setBci(int bci) { this.bci = bci; } - - public ValueNode getPredictedKlass() { - return predictedKlass; - } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/CheckcastArrayCopyCallNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/CheckcastArrayCopyCallNode.java index 8853e1a3b66..e90d486a98d 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/CheckcastArrayCopyCallNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/CheckcastArrayCopyCallNode.java @@ -23,12 +23,13 @@ //JaCoCo Exclude package org.graalvm.compiler.hotspot.replacements.arraycopy; -import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_UNKNOWN; -import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_UNKNOWN; import static jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider.getArrayBaseOffset; import static jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider.getArrayIndexScale; +import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_UNKNOWN; +import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_UNKNOWN; import org.graalvm.compiler.core.common.spi.ForeignCallDescriptor; +import org.graalvm.compiler.core.common.type.PrimitiveStamp; import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.graph.NodeClass; import org.graalvm.compiler.hotspot.HotSpotGraalRuntimeProvider; @@ -114,8 +115,9 @@ public final class CheckcastArrayCopyCallNode extends AbstractMemoryCheckpoint i graph().addBeforeFixed(this, basePtr); int shift = CodeUtil.log2(getArrayIndexScale(JavaKind.Object)); - ValueNode scaledIndex = graph().unique(new LeftShiftNode(pos, ConstantNode.forInt(shift, graph()))); - ValueNode offset = graph().unique(new AddNode(scaledIndex, ConstantNode.forInt(getArrayBaseOffset(JavaKind.Object), graph()))); + ValueNode extendedPos = IntegerConvertNode.convert(pos, StampFactory.forKind(runtime.getTarget().wordJavaKind), graph()); + ValueNode scaledIndex = graph().unique(new LeftShiftNode(extendedPos, ConstantNode.forInt(shift, graph()))); + ValueNode offset = graph().unique(new AddNode(scaledIndex, ConstantNode.forIntegerBits(PrimitiveStamp.getBits(scaledIndex.stamp()), getArrayBaseOffset(JavaKind.Object), graph()))); return graph().unique(new OffsetAddressNode(basePtr, offset)); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/GenericArrayCopyCallNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/GenericArrayCopyCallNode.java new file mode 100644 index 00000000000..1017b07c2b2 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/GenericArrayCopyCallNode.java @@ -0,0 +1,122 @@ +/* + * Copyright (c) 2014, 2016, 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. + */ +//JaCoCo Exclude +package org.graalvm.compiler.hotspot.replacements.arraycopy; + +import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_UNKNOWN; +import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_UNKNOWN; + +import org.graalvm.compiler.core.common.type.StampFactory; +import org.graalvm.compiler.graph.NodeClass; +import org.graalvm.compiler.hotspot.HotSpotBackend; +import org.graalvm.compiler.hotspot.HotSpotGraalRuntimeProvider; +import org.graalvm.compiler.hotspot.nodes.GetObjectAddressNode; +import org.graalvm.compiler.nodeinfo.InputType; +import org.graalvm.compiler.nodeinfo.NodeInfo; +import org.graalvm.compiler.nodes.StructuredGraph; +import org.graalvm.compiler.nodes.ValueNode; +import org.graalvm.compiler.nodes.calc.IntegerConvertNode; +import org.graalvm.compiler.nodes.extended.ForeignCallNode; +import org.graalvm.compiler.nodes.memory.AbstractMemoryCheckpoint; +import org.graalvm.compiler.nodes.memory.MemoryCheckpoint; +import org.graalvm.compiler.nodes.spi.Lowerable; +import org.graalvm.compiler.nodes.spi.LoweringTool; +import org.graalvm.word.LocationIdentity; + +import jdk.vm.ci.meta.JavaKind; + +@NodeInfo(allowedUsageTypes = {InputType.Memory, InputType.Value}, cycles = CYCLES_UNKNOWN, size = SIZE_UNKNOWN) +public final class GenericArrayCopyCallNode extends AbstractMemoryCheckpoint implements Lowerable, MemoryCheckpoint.Single { + + public static final NodeClass TYPE = NodeClass.create(GenericArrayCopyCallNode.class); + @Input ValueNode src; + @Input ValueNode srcPos; + @Input ValueNode dest; + @Input ValueNode destPos; + @Input ValueNode length; + + protected final HotSpotGraalRuntimeProvider runtime; + + protected GenericArrayCopyCallNode(@InjectedNodeParameter HotSpotGraalRuntimeProvider runtime, ValueNode src, ValueNode srcPos, ValueNode dest, ValueNode destPos, ValueNode length) { + super(TYPE, StampFactory.forKind(JavaKind.Int)); + this.src = src; + this.srcPos = srcPos; + this.dest = dest; + this.destPos = destPos; + this.length = length; + this.runtime = runtime; + } + + public ValueNode getSource() { + return src; + } + + public ValueNode getSourcePosition() { + return srcPos; + } + + public ValueNode getDestination() { + return dest; + } + + public ValueNode getDestinationPosition() { + return destPos; + } + + public ValueNode getLength() { + return length; + } + + @Override + public void lower(LoweringTool tool) { + if (graph().getGuardsStage().areFrameStatesAtDeopts()) { + StructuredGraph graph = graph(); + ValueNode srcAddr = objectAddress(getSource()); + ValueNode destAddr = objectAddress(getDestination()); + ForeignCallNode call = graph.add(new ForeignCallNode(runtime.getHostBackend().getForeignCalls(), HotSpotBackend.GENERIC_ARRAYCOPY, srcAddr, srcPos, destAddr, destPos, length)); + call.setStateAfter(stateAfter()); + graph.replaceFixedWithFixed(this, call); + } + } + + private ValueNode objectAddress(ValueNode obj) { + GetObjectAddressNode result = graph().add(new GetObjectAddressNode(obj)); + graph().addBeforeFixed(this, result); + return result; + } + + private ValueNode wordValue(ValueNode value) { + if (value.stamp().getStackKind() != runtime.getTarget().wordJavaKind) { + return IntegerConvertNode.convert(value, StampFactory.forKind(runtime.getTarget().wordJavaKind), graph()); + } + return value; + } + + @Override + public LocationIdentity getLocationIdentity() { + return LocationIdentity.any(); + } + + @NodeIntrinsic + public static native int genericArraycopy(Object src, int srcPos, Object dest, int destPos, int length); +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/UnsafeArrayCopyNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/UnsafeArrayCopyNode.java deleted file mode 100644 index 83c7c4f9f10..00000000000 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/UnsafeArrayCopyNode.java +++ /dev/null @@ -1,153 +0,0 @@ -/* - * Copyright (c) 2013, 2016, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ -package org.graalvm.compiler.hotspot.replacements.arraycopy; - -import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_256; -import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_64; -import static org.graalvm.word.LocationIdentity.any; - -import org.graalvm.compiler.core.common.type.StampFactory; -import org.graalvm.compiler.graph.NodeClass; -import org.graalvm.compiler.nodeinfo.InputType; -import org.graalvm.compiler.nodeinfo.NodeInfo; -import org.graalvm.compiler.nodes.NamedLocationIdentity; -import org.graalvm.compiler.nodes.ValueNode; -import org.graalvm.compiler.nodes.extended.ArrayRangeWriteNode; -import org.graalvm.compiler.nodes.memory.MemoryAccess; -import org.graalvm.compiler.nodes.memory.MemoryCheckpoint; -import org.graalvm.compiler.nodes.memory.MemoryNode; -import org.graalvm.compiler.nodes.spi.Lowerable; -import org.graalvm.compiler.nodes.spi.LoweringTool; -import org.graalvm.compiler.replacements.SnippetTemplate.Arguments; -import org.graalvm.word.LocationIdentity; - -import jdk.vm.ci.meta.JavaKind; - -@NodeInfo(allowedUsageTypes = {InputType.Memory}, cycles = CYCLES_256, size = SIZE_64) -public final class UnsafeArrayCopyNode extends ArrayRangeWriteNode implements Lowerable, MemoryCheckpoint.Single, MemoryAccess { - - public static final NodeClass TYPE = NodeClass.create(UnsafeArrayCopyNode.class); - @Input ValueNode src; - @Input ValueNode srcPos; - @Input ValueNode dest; - @Input ValueNode destPos; - @Input ValueNode length; - @OptionalInput ValueNode layoutHelper; - - @OptionalInput(InputType.Memory) MemoryNode lastLocationAccess; - - protected JavaKind elementKind; - - public UnsafeArrayCopyNode(ValueNode src, ValueNode srcPos, ValueNode dest, ValueNode destPos, ValueNode length, ValueNode layoutHelper, JavaKind elementKind) { - super(TYPE, StampFactory.forVoid()); - assert layoutHelper == null || elementKind == null; - this.src = src; - this.srcPos = srcPos; - this.dest = dest; - this.destPos = destPos; - this.length = length; - this.layoutHelper = layoutHelper; - this.elementKind = elementKind; - } - - public UnsafeArrayCopyNode(ValueNode src, ValueNode srcPos, ValueNode dest, ValueNode destPos, ValueNode length, JavaKind elementKind) { - this(src, srcPos, dest, destPos, length, null, elementKind); - } - - public UnsafeArrayCopyNode(ValueNode src, ValueNode srcPos, ValueNode dest, ValueNode destPos, ValueNode length, ValueNode layoutHelper) { - this(src, srcPos, dest, destPos, length, layoutHelper, null); - } - - @Override - public ValueNode getArray() { - return dest; - } - - @Override - public ValueNode getIndex() { - return destPos; - } - - @Override - public ValueNode getLength() { - return length; - } - - @Override - public boolean isObjectArray() { - return elementKind == JavaKind.Object; - } - - @Override - public boolean isInitialization() { - return false; - } - - public JavaKind getElementKind() { - return elementKind; - } - - @Override - public void lower(LoweringTool tool) { - if (graph().getGuardsStage().areFrameStatesAtDeopts()) { - UnsafeArrayCopySnippets.Templates templates = tool.getReplacements().getSnippetTemplateCache(UnsafeArrayCopySnippets.Templates.class); - templates.lower(this, tool); - } - } - - public void addSnippetArguments(Arguments args) { - args.add("src", src); - args.add("srcPos", srcPos); - args.add("dest", dest); - args.add("destPos", destPos); - args.add("length", length); - if (layoutHelper != null) { - args.add("layoutHelper", layoutHelper); - } - } - - @Override - public LocationIdentity getLocationIdentity() { - if (elementKind != null) { - return NamedLocationIdentity.getArrayLocation(elementKind); - } - return any(); - } - - @Override - public MemoryNode getLastLocationAccess() { - return lastLocationAccess; - } - - @Override - public void setLastLocationAccess(MemoryNode lla) { - updateUsagesInterface(lastLocationAccess, lla); - lastLocationAccess = lla; - } - - @NodeIntrinsic - public static native void arraycopy(Object src, int srcPos, Object dest, int destPos, int length, @ConstantNodeParameter JavaKind elementKind); - - @NodeIntrinsic - public static native void arraycopyPrimitive(Object src, int srcPos, Object dest, int destPos, int length, int layoutHelper); -} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/UnsafeArrayCopySnippets.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/UnsafeArrayCopySnippets.java deleted file mode 100644 index 71d63706deb..00000000000 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/arraycopy/UnsafeArrayCopySnippets.java +++ /dev/null @@ -1,324 +0,0 @@ -/* - * Copyright (c) 2013, 2016, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ -package org.graalvm.compiler.hotspot.replacements.arraycopy; - -import static jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider.getArrayIndexScale; -import static org.graalvm.compiler.hotspot.GraalHotSpotVMConfig.INJECTED_VMCONFIG; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.arrayBaseOffset; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.arrayIndexScale; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.layoutHelperHeaderSizeMask; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.layoutHelperHeaderSizeShift; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.layoutHelperLog2ElementSizeMask; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.layoutHelperLog2ElementSizeShift; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.runtime; -import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.wordSize; -import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.NOT_FREQUENT_PROBABILITY; -import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.probability; -import static org.graalvm.compiler.replacements.SnippetTemplate.DEFAULT_REPLACER; -import static org.graalvm.word.LocationIdentity.any; - -import org.graalvm.compiler.api.replacements.Fold; -import org.graalvm.compiler.api.replacements.Snippet; -import org.graalvm.compiler.core.common.NumUtil; -import org.graalvm.compiler.debug.DebugHandlersFactory; -import org.graalvm.compiler.hotspot.meta.HotSpotProviders; -import org.graalvm.compiler.hotspot.phases.WriteBarrierAdditionPhase; -import org.graalvm.compiler.nodes.NamedLocationIdentity; -import org.graalvm.compiler.nodes.extended.RawLoadNode; -import org.graalvm.compiler.nodes.extended.RawStoreNode; -import org.graalvm.compiler.nodes.extended.UnsafeCopyNode; -import org.graalvm.compiler.nodes.spi.LoweringTool; -import org.graalvm.compiler.options.OptionValues; -import org.graalvm.compiler.replacements.SnippetTemplate; -import org.graalvm.compiler.replacements.SnippetTemplate.AbstractTemplates; -import org.graalvm.compiler.replacements.SnippetTemplate.Arguments; -import org.graalvm.compiler.replacements.SnippetTemplate.SnippetInfo; -import org.graalvm.compiler.replacements.Snippets; -import org.graalvm.compiler.word.ObjectAccess; -import org.graalvm.word.LocationIdentity; -import org.graalvm.word.UnsignedWord; -import org.graalvm.word.WordFactory; - -import jdk.vm.ci.code.TargetDescription; -import jdk.vm.ci.meta.JavaKind; - -/** - * As opposed to {@link ArrayCopySnippets}, these Snippets do not perform store checks. - */ -public class UnsafeArrayCopySnippets implements Snippets { - - private static final boolean supportsUnalignedMemoryAccess = runtime().getHostJVMCIBackend().getTarget().arch.supportsUnalignedMemoryAccess(); - - private static final JavaKind VECTOR_KIND = JavaKind.Long; - private static final long VECTOR_SIZE = getArrayIndexScale(VECTOR_KIND); - - private static void vectorizedCopy(Object src, int srcPos, Object dest, int destPos, int length, JavaKind baseKind, LocationIdentity locationIdentity) { - int arrayBaseOffset = arrayBaseOffset(baseKind); - int elementSize = arrayIndexScale(baseKind); - long byteLength = (long) length * elementSize; - long srcOffset = (long) srcPos * elementSize; - long destOffset = (long) destPos * elementSize; - - long preLoopBytes; - long mainLoopBytes; - long postLoopBytes; - - // We can easily vectorize the loop if both offsets have the same alignment. - if (byteLength >= VECTOR_SIZE && (srcOffset % VECTOR_SIZE) == (destOffset % VECTOR_SIZE)) { - preLoopBytes = NumUtil.roundUp(arrayBaseOffset + srcOffset, VECTOR_SIZE) - (arrayBaseOffset + srcOffset); - postLoopBytes = (byteLength - preLoopBytes) % VECTOR_SIZE; - mainLoopBytes = byteLength - preLoopBytes - postLoopBytes; - } else { - // Does the architecture support unaligned memory accesses? - if (supportsUnalignedMemoryAccess) { - preLoopBytes = byteLength % VECTOR_SIZE; - mainLoopBytes = byteLength - preLoopBytes; - postLoopBytes = 0; - } else { - // No. Let's do element-wise copying. - preLoopBytes = byteLength; - mainLoopBytes = 0; - postLoopBytes = 0; - } - } - - if (probability(NOT_FREQUENT_PROBABILITY, src == dest) && probability(NOT_FREQUENT_PROBABILITY, srcPos < destPos)) { - // bad aliased case - srcOffset += byteLength; - destOffset += byteLength; - - // Post-loop - for (long i = 0; i < postLoopBytes; i += elementSize) { - srcOffset -= elementSize; - destOffset -= elementSize; - UnsafeCopyNode.copy(src, arrayBaseOffset + srcOffset, dest, arrayBaseOffset + destOffset, baseKind, locationIdentity); - } - // Main-loop - for (long i = 0; i < mainLoopBytes; i += VECTOR_SIZE) { - srcOffset -= VECTOR_SIZE; - destOffset -= VECTOR_SIZE; - UnsafeCopyNode.copy(src, arrayBaseOffset + srcOffset, dest, arrayBaseOffset + destOffset, VECTOR_KIND, locationIdentity); - } - // Pre-loop - for (long i = 0; i < preLoopBytes; i += elementSize) { - srcOffset -= elementSize; - destOffset -= elementSize; - UnsafeCopyNode.copy(src, arrayBaseOffset + srcOffset, dest, arrayBaseOffset + destOffset, baseKind, locationIdentity); - } - } else { - // Pre-loop - for (long i = 0; i < preLoopBytes; i += elementSize) { - UnsafeCopyNode.copy(src, arrayBaseOffset + srcOffset, dest, arrayBaseOffset + destOffset, baseKind, locationIdentity); - srcOffset += elementSize; - destOffset += elementSize; - } - // Main-loop - for (long i = 0; i < mainLoopBytes; i += VECTOR_SIZE) { - UnsafeCopyNode.copy(src, arrayBaseOffset + srcOffset, dest, arrayBaseOffset + destOffset, VECTOR_KIND, locationIdentity); - srcOffset += VECTOR_SIZE; - destOffset += VECTOR_SIZE; - } - // Post-loop - for (long i = 0; i < postLoopBytes; i += elementSize) { - UnsafeCopyNode.copy(src, arrayBaseOffset + srcOffset, dest, arrayBaseOffset + destOffset, baseKind, locationIdentity); - srcOffset += elementSize; - destOffset += elementSize; - } - } - } - - @Fold - static LocationIdentity getArrayLocation(JavaKind kind) { - return NamedLocationIdentity.getArrayLocation(kind); - } - - @Snippet - public static void arraycopyByte(byte[] src, int srcPos, byte[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Byte; - vectorizedCopy(src, srcPos, dest, destPos, length, kind, getArrayLocation(kind)); - } - - @Snippet - public static void arraycopyBoolean(boolean[] src, int srcPos, boolean[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Boolean; - vectorizedCopy(src, srcPos, dest, destPos, length, kind, getArrayLocation(kind)); - } - - @Snippet - public static void arraycopyChar(char[] src, int srcPos, char[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Char; - vectorizedCopy(src, srcPos, dest, destPos, length, kind, getArrayLocation(kind)); - } - - @Snippet - public static void arraycopyShort(short[] src, int srcPos, short[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Short; - vectorizedCopy(src, srcPos, dest, destPos, length, kind, getArrayLocation(kind)); - } - - @Snippet - public static void arraycopyInt(int[] src, int srcPos, int[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Int; - vectorizedCopy(src, srcPos, dest, destPos, length, kind, getArrayLocation(kind)); - } - - @Snippet - public static void arraycopyFloat(float[] src, int srcPos, float[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Float; - vectorizedCopy(src, srcPos, dest, destPos, length, kind, getArrayLocation(kind)); - } - - @Snippet - public static void arraycopyLong(long[] src, int srcPos, long[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Long; - vectorizedCopy(src, srcPos, dest, destPos, length, kind, getArrayLocation(kind)); - } - - @Snippet - public static void arraycopyDouble(double[] src, int srcPos, double[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Double; - /* - * TODO atomicity problem on 32-bit architectures: The JVM spec requires double values to be - * copied atomically, but not long values. For example, on Intel 32-bit this code is not - * atomic as long as the vector kind remains Kind.Long. - */ - vectorizedCopy(src, srcPos, dest, destPos, length, kind, getArrayLocation(kind)); - } - - /** - * For this kind, Object, we want to avoid write barriers between writes, but instead have them - * at the end of the snippet. This is done by using {@link RawStoreNode}, and rely on - * {@link WriteBarrierAdditionPhase} to put write barriers after the {@link UnsafeArrayCopyNode} - * with kind Object. - */ - @Snippet - public static void arraycopyObject(Object[] src, int srcPos, Object[] dest, int destPos, int length) { - JavaKind kind = JavaKind.Object; - final int scale = arrayIndexScale(kind); - int arrayBaseOffset = arrayBaseOffset(kind); - LocationIdentity arrayLocation = getArrayLocation(kind); - if (src == dest && srcPos < destPos) { // bad aliased case - long start = (long) (length - 1) * scale; - for (long i = start; i >= 0; i -= scale) { - Object a = RawLoadNode.load(src, arrayBaseOffset + i + (long) srcPos * scale, kind, arrayLocation); - RawStoreNode.storeObject(dest, arrayBaseOffset + i + (long) destPos * scale, a, kind, getArrayLocation(kind), false); - } - } else { - long end = (long) length * scale; - for (long i = 0; i < end; i += scale) { - Object a = RawLoadNode.load(src, arrayBaseOffset + i + (long) srcPos * scale, kind, arrayLocation); - RawStoreNode.storeObject(dest, arrayBaseOffset + i + (long) destPos * scale, a, kind, getArrayLocation(kind), false); - } - } - } - - @Snippet - public static void arraycopyPrimitive(Object src, int srcPos, Object dest, int destPos, int length, int layoutHelper) { - int log2ElementSize = (layoutHelper >> layoutHelperLog2ElementSizeShift(INJECTED_VMCONFIG)) & layoutHelperLog2ElementSizeMask(INJECTED_VMCONFIG); - int headerSize = (layoutHelper >> layoutHelperHeaderSizeShift(INJECTED_VMCONFIG)) & layoutHelperHeaderSizeMask(INJECTED_VMCONFIG); - - UnsignedWord vectorSize = WordFactory.unsigned(VECTOR_SIZE); - UnsignedWord srcOffset = WordFactory.unsigned(srcPos).shiftLeft(log2ElementSize).add(headerSize); - UnsignedWord destOffset = WordFactory.unsigned(destPos).shiftLeft(log2ElementSize).add(headerSize); - UnsignedWord destStart = destOffset; - UnsignedWord destEnd = destOffset.add(WordFactory.unsigned(length).shiftLeft(log2ElementSize)); - - UnsignedWord destVectorEnd = null; - UnsignedWord nonVectorBytes = null; - UnsignedWord sizeInBytes = WordFactory.unsigned(length).shiftLeft(log2ElementSize); - if (supportsUnalignedMemoryAccess) { - nonVectorBytes = sizeInBytes.unsignedRemainder(vectorSize); - destVectorEnd = destEnd; - } else { - boolean inPhase = srcOffset.and((int) VECTOR_SIZE - 1).equal(destOffset.and((int) VECTOR_SIZE - 1)); - boolean hasAtLeastOneVector = sizeInBytes.aboveOrEqual(vectorSize); - // We must have at least one full vector, otherwise we must copy each byte separately - if (hasAtLeastOneVector && inPhase) { // If in phase, we can vectorize - nonVectorBytes = vectorSize.subtract(destStart.unsignedRemainder(vectorSize)); - } else { // fallback is byte-wise - nonVectorBytes = sizeInBytes; - } - destVectorEnd = destEnd.subtract(destEnd.unsignedRemainder(vectorSize)); - } - - UnsignedWord destNonVectorEnd = destStart.add(nonVectorBytes); - while (destOffset.belowThan(destNonVectorEnd)) { - ObjectAccess.writeByte(dest, destOffset, ObjectAccess.readByte(src, srcOffset, any()), any()); - destOffset = destOffset.add(1); - srcOffset = srcOffset.add(1); - } - // Unsigned destVectorEnd = destEnd.subtract(destEnd.unsignedRemainder(8)); - while (destOffset.belowThan(destVectorEnd)) { - ObjectAccess.writeWord(dest, destOffset, ObjectAccess.readWord(src, srcOffset, any()), any()); - destOffset = destOffset.add(wordSize()); - srcOffset = srcOffset.add(wordSize()); - } - // Do the last bytes each when it is required to have absolute alignment. - while (!supportsUnalignedMemoryAccess && destOffset.belowThan(destEnd)) { - ObjectAccess.writeByte(dest, destOffset, ObjectAccess.readByte(src, srcOffset, any()), any()); - destOffset = destOffset.add(1); - srcOffset = srcOffset.add(1); - } - } - - public static class Templates extends AbstractTemplates { - - private final SnippetInfo[] arraycopySnippets; - private final SnippetInfo genericPrimitiveSnippet; - - public Templates(OptionValues options, Iterable factories, HotSpotProviders providers, TargetDescription target) { - super(options, factories, providers, providers.getSnippetReflection(), target); - - arraycopySnippets = new SnippetInfo[JavaKind.values().length]; - arraycopySnippets[JavaKind.Boolean.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyBoolean"); - arraycopySnippets[JavaKind.Byte.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyByte"); - arraycopySnippets[JavaKind.Short.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyShort"); - arraycopySnippets[JavaKind.Char.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyChar"); - arraycopySnippets[JavaKind.Int.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyInt"); - arraycopySnippets[JavaKind.Long.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyLong"); - arraycopySnippets[JavaKind.Float.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyFloat"); - arraycopySnippets[JavaKind.Double.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyDouble"); - arraycopySnippets[JavaKind.Object.ordinal()] = snippet(UnsafeArrayCopySnippets.class, "arraycopyObject"); - - genericPrimitiveSnippet = snippet(UnsafeArrayCopySnippets.class, "arraycopyPrimitive"); - } - - public void lower(UnsafeArrayCopyNode node, LoweringTool tool) { - JavaKind elementKind = node.getElementKind(); - SnippetInfo snippet; - if (elementKind == null) { - // primitive array of unknown kind - snippet = genericPrimitiveSnippet; - } else { - snippet = arraycopySnippets[elementKind.ordinal()]; - assert snippet != null : "arraycopy snippet for " + elementKind.name() + " not found"; - } - - Arguments args = new Arguments(snippet, node.graph().getGuardsStage(), tool.getLoweringStage()); - node.addSnippetArguments(args); - - SnippetTemplate template = template(node.getDebug(), args); - template.instantiate(providers.getMetaAccess(), node, DEFAULT_REPLACER, args); - } - } -} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/profiling/ProbabilisticProfileSnippets.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/profiling/ProbabilisticProfileSnippets.java index a287f15dfaf..a13ef3c69f3 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/profiling/ProbabilisticProfileSnippets.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/profiling/ProbabilisticProfileSnippets.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2017, 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 @@ -54,6 +54,7 @@ import org.graalvm.compiler.replacements.SnippetTemplate.Arguments; import org.graalvm.compiler.replacements.SnippetTemplate.SnippetInfo; import org.graalvm.compiler.replacements.Snippets; +import jdk.vm.ci.code.CodeUtil; import jdk.vm.ci.code.TargetDescription; public class ProbabilisticProfileSnippets implements Snippets { @@ -64,22 +65,22 @@ public class ProbabilisticProfileSnippets implements Snippets { } @Snippet - public static int notificationMask(int freqLog, int probLog) { - int probabilityMask = (1 << probLog) - 1; + public static int notificationMask(int freqLog, int probLog, int stepLog) { int frequencyMask = (1 << freqLog) - 1; - return frequencyMask & ~probabilityMask; + int stepMask = (1 << (stepLog + probLog)) - 1; + return frequencyMask & ~stepMask; } @NodeIntrinsic(ForeignCallNode.class) public static native void methodInvocationEvent(@ConstantNodeParameter ForeignCallDescriptor descriptor, MethodCountersPointer counters); @Snippet - public static void profileMethodEntryWithProbability(MethodCountersPointer counters, int random, @ConstantParameter int freqLog, @ConstantParameter int probLog) { + public static void profileMethodEntryWithProbability(MethodCountersPointer counters, int random, int step, int stepLog, @ConstantParameter int freqLog, @ConstantParameter int probLog) { if (probability(1.0 / (1 << probLog), shouldProfile(probLog, random))) { - int counterValue = counters.readInt(config(INJECTED_VMCONFIG).invocationCounterOffset) + (config(INJECTED_VMCONFIG).invocationCounterIncrement << probLog); + int counterValue = counters.readInt(config(INJECTED_VMCONFIG).invocationCounterOffset) + ((config(INJECTED_VMCONFIG).invocationCounterIncrement * step) << probLog); counters.writeInt(config(INJECTED_VMCONFIG).invocationCounterOffset, counterValue); if (freqLog >= 0) { - int mask = notificationMask(freqLog, probLog); + int mask = notificationMask(freqLog, probLog, stepLog); if (probability(SLOW_PATH_PROBABILITY, (counterValue & (mask << config(INJECTED_VMCONFIG).invocationCounterShift)) == 0)) { methodInvocationEvent(HotSpotBackend.INVOCATION_EVENT, counters); } @@ -91,11 +92,12 @@ public class ProbabilisticProfileSnippets implements Snippets { public static native void methodBackedgeEvent(@ConstantNodeParameter ForeignCallDescriptor descriptor, MethodCountersPointer counters, int bci, int targetBci); @Snippet - public static void profileBackedgeWithProbability(MethodCountersPointer counters, int random, @ConstantParameter int freqLog, @ConstantParameter int probLog, int bci, int targetBci) { + public static void profileBackedgeWithProbability(MethodCountersPointer counters, int random, int step, int stepLog, @ConstantParameter int freqLog, @ConstantParameter int probLog, int bci, + int targetBci) { if (probability(1.0 / (1 << probLog), shouldProfile(probLog, random))) { - int counterValue = counters.readInt(config(INJECTED_VMCONFIG).backedgeCounterOffset) + (config(INJECTED_VMCONFIG).invocationCounterIncrement << probLog); + int counterValue = counters.readInt(config(INJECTED_VMCONFIG).backedgeCounterOffset) + ((config(INJECTED_VMCONFIG).invocationCounterIncrement * step) << probLog); counters.writeInt(config(INJECTED_VMCONFIG).backedgeCounterOffset, counterValue); - int mask = notificationMask(freqLog, probLog); + int mask = notificationMask(freqLog, probLog, stepLog); if (probability(SLOW_PATH_PROBABILITY, (counterValue & (mask << config(INJECTED_VMCONFIG).invocationCounterShift)) == 0)) { methodBackedgeEvent(HotSpotBackend.BACKEDGE_EVENT, counters, bci, targetBci); } @@ -103,10 +105,11 @@ public class ProbabilisticProfileSnippets implements Snippets { } @Snippet - public static void profileConditionalBackedgeWithProbability(MethodCountersPointer counters, int random, @ConstantParameter int freqLog, @ConstantParameter int probLog, boolean branchCondition, + public static void profileConditionalBackedgeWithProbability(MethodCountersPointer counters, int random, int step, int stepLog, @ConstantParameter int freqLog, + @ConstantParameter int probLog, boolean branchCondition, int bci, int targetBci) { if (branchCondition) { - profileBackedgeWithProbability(counters, random, freqLog, probLog, bci, targetBci); + profileBackedgeWithProbability(counters, random, step, stepLog, freqLog, probLog, bci, targetBci); } } @@ -124,6 +127,8 @@ public class ProbabilisticProfileSnippets implements Snippets { StructuredGraph graph = profileNode.graph(); LoadMethodCountersNode counters = graph.unique(new LoadMethodCountersNode(profileNode.getProfiledMethod())); + ConstantNode step = ConstantNode.forInt(profileNode.getStep(), graph); + ConstantNode stepLog = ConstantNode.forInt(CodeUtil.log2(profileNode.getStep()), graph); if (profileNode instanceof ProfileBranchNode) { // Backedge event @@ -132,8 +137,11 @@ public class ProbabilisticProfileSnippets implements Snippets { Arguments args = new Arguments(snippet, graph.getGuardsStage(), tool.getLoweringStage()); ConstantNode bci = ConstantNode.forInt(profileBranchNode.bci(), graph); ConstantNode targetBci = ConstantNode.forInt(profileBranchNode.targetBci(), graph); + args.add("counters", counters); args.add("random", profileBranchNode.getRandom()); + args.add("step", step); + args.add("stepLog", stepLog); args.addConst("freqLog", profileBranchNode.getNotificationFreqLog()); args.addConst("probLog", profileBranchNode.getProbabilityLog()); if (profileBranchNode.hasCondition()) { @@ -148,8 +156,11 @@ public class ProbabilisticProfileSnippets implements Snippets { ProfileInvokeNode profileInvokeNode = (ProfileInvokeNode) profileNode; // Method invocation event Arguments args = new Arguments(profileMethodEntryWithProbability, graph.getGuardsStage(), tool.getLoweringStage()); + args.add("counters", counters); args.add("random", profileInvokeNode.getRandom()); + args.add("step", step); + args.add("stepLog", stepLog); args.addConst("freqLog", profileInvokeNode.getNotificationFreqLog()); args.addConst("probLog", profileInvokeNode.getProbabilityLog()); SnippetTemplate template = template(graph.getDebug(), args); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/profiling/ProfileSnippets.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/profiling/ProfileSnippets.java index 1c5f05e1a13..79d21ab15ee 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/profiling/ProfileSnippets.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/profiling/ProfileSnippets.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2017, 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 @@ -54,6 +54,7 @@ import org.graalvm.compiler.replacements.SnippetTemplate.Arguments; import org.graalvm.compiler.replacements.SnippetTemplate.SnippetInfo; import org.graalvm.compiler.replacements.Snippets; +import jdk.vm.ci.code.CodeUtil; import jdk.vm.ci.code.TargetDescription; public class ProfileSnippets implements Snippets { @@ -61,12 +62,19 @@ public class ProfileSnippets implements Snippets { public static native void methodInvocationEvent(@ConstantNodeParameter ForeignCallDescriptor descriptor, MethodCountersPointer counters); @Snippet - public static void profileMethodEntry(MethodCountersPointer counters, @ConstantParameter int freqLog) { - int counterValue = counters.readInt(config(INJECTED_VMCONFIG).invocationCounterOffset) + config(INJECTED_VMCONFIG).invocationCounterIncrement; + protected static int notificationMask(int freqLog, int stepLog) { + int stepMask = (1 << stepLog) - 1; + int frequencyMask = (1 << freqLog) - 1; + return frequencyMask & ~stepMask; + } + + @Snippet + public static void profileMethodEntry(MethodCountersPointer counters, int step, int stepLog, @ConstantParameter int freqLog) { + int counterValue = counters.readInt(config(INJECTED_VMCONFIG).invocationCounterOffset) + config(INJECTED_VMCONFIG).invocationCounterIncrement * step; counters.writeInt(config(INJECTED_VMCONFIG).invocationCounterOffset, counterValue); if (freqLog >= 0) { - final int frequencyMask = (1 << freqLog) - 1; - if (probability(SLOW_PATH_PROBABILITY, (counterValue & (frequencyMask << config(INJECTED_VMCONFIG).invocationCounterShift)) == 0)) { + final int mask = notificationMask(freqLog, stepLog); + if (probability(SLOW_PATH_PROBABILITY, (counterValue & (mask << config(INJECTED_VMCONFIG).invocationCounterShift)) == 0)) { methodInvocationEvent(HotSpotBackend.INVOCATION_EVENT, counters); } } @@ -76,19 +84,19 @@ public class ProfileSnippets implements Snippets { public static native void methodBackedgeEvent(@ConstantNodeParameter ForeignCallDescriptor descriptor, MethodCountersPointer counters, int bci, int targetBci); @Snippet - public static void profileBackedge(MethodCountersPointer counters, @ConstantParameter int freqLog, int bci, int targetBci) { - int counterValue = counters.readInt(config(INJECTED_VMCONFIG).backedgeCounterOffset) + config(INJECTED_VMCONFIG).invocationCounterIncrement; + public static void profileBackedge(MethodCountersPointer counters, int step, int stepLog, @ConstantParameter int freqLog, int bci, int targetBci) { + int counterValue = counters.readInt(config(INJECTED_VMCONFIG).backedgeCounterOffset) + config(INJECTED_VMCONFIG).invocationCounterIncrement * step; counters.writeInt(config(INJECTED_VMCONFIG).backedgeCounterOffset, counterValue); - final int frequencyMask = (1 << freqLog) - 1; - if (probability(SLOW_PATH_PROBABILITY, (counterValue & (frequencyMask << config(INJECTED_VMCONFIG).invocationCounterShift)) == 0)) { + final int mask = notificationMask(freqLog, stepLog); + if (probability(SLOW_PATH_PROBABILITY, (counterValue & (mask << config(INJECTED_VMCONFIG).invocationCounterShift)) == 0)) { methodBackedgeEvent(HotSpotBackend.BACKEDGE_EVENT, counters, bci, targetBci); } } @Snippet - public static void profileConditionalBackedge(MethodCountersPointer counters, @ConstantParameter int freqLog, boolean branchCondition, int bci, int targetBci) { + public static void profileConditionalBackedge(MethodCountersPointer counters, int step, int stepLog, @ConstantParameter int freqLog, boolean branchCondition, int bci, int targetBci) { if (branchCondition) { - profileBackedge(counters, freqLog, bci, targetBci); + profileBackedge(counters, step, stepLog, freqLog, bci, targetBci); } } @@ -104,6 +112,8 @@ public class ProfileSnippets implements Snippets { public void lower(ProfileNode profileNode, LoweringTool tool) { StructuredGraph graph = profileNode.graph(); LoadMethodCountersNode counters = graph.unique(new LoadMethodCountersNode(profileNode.getProfiledMethod())); + ConstantNode step = ConstantNode.forInt(profileNode.getStep(), graph); + ConstantNode stepLog = ConstantNode.forInt(CodeUtil.log2(profileNode.getStep()), graph); if (profileNode instanceof ProfileBranchNode) { // Backedge event @@ -113,6 +123,8 @@ public class ProfileSnippets implements Snippets { ConstantNode bci = ConstantNode.forInt(profileBranchNode.bci(), graph); ConstantNode targetBci = ConstantNode.forInt(profileBranchNode.targetBci(), graph); args.add("counters", counters); + args.add("step", step); + args.add("stepLog", stepLog); args.addConst("freqLog", profileBranchNode.getNotificationFreqLog()); if (profileBranchNode.hasCondition()) { args.add("branchCondition", profileBranchNode.branchCondition()); @@ -127,6 +139,8 @@ public class ProfileSnippets implements Snippets { // Method invocation event Arguments args = new Arguments(profileMethodEntry, graph.getGuardsStage(), tool.getLoweringStage()); args.add("counters", counters); + args.add("step", step); + args.add("stepLog", stepLog); args.addConst("freqLog", profileInvokeNode.getNotificationFreqLog()); SnippetTemplate template = template(graph.getDebug(), args); template.instantiate(providers.getMetaAccess(), profileNode, DEFAULT_REPLACER, args); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/stubs/NewArrayStub.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/stubs/NewArrayStub.java index d7018a1ef75..b5cbb0e5585 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/stubs/NewArrayStub.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/stubs/NewArrayStub.java @@ -33,6 +33,7 @@ import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil. import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.layoutHelperLog2ElementSizeShift; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.readLayoutHelper; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.registerAsWord; +import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.useCMSIncrementalMode; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.wordSize; import static org.graalvm.compiler.hotspot.replacements.NewObjectSnippets.MAX_ARRAY_FAST_PATH_ALLOCATION_LENGTH; import static org.graalvm.compiler.hotspot.replacements.NewObjectSnippets.formatArray; @@ -122,7 +123,7 @@ public class NewArrayStub extends SnippetStub { // check that array length is small enough for fast path. Word thread = registerAsWord(threadRegister); boolean inlineContiguousAllocationSupported = GraalHotSpotVMConfigNode.inlineContiguousAllocationSupported(); - if (inlineContiguousAllocationSupported && length >= 0 && length <= MAX_ARRAY_FAST_PATH_ALLOCATION_LENGTH) { + if (inlineContiguousAllocationSupported && !useCMSIncrementalMode(INJECTED_VMCONFIG) && length >= 0 && length <= MAX_ARRAY_FAST_PATH_ALLOCATION_LENGTH) { Word memory = refillAllocate(thread, intArrayHub, sizeInBytes, logging(options)); if (memory.notEqual(0)) { if (logging(options)) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/stubs/NewInstanceStub.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/stubs/NewInstanceStub.java index 91a5acab1f1..9ebcee8bce7 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/stubs/NewInstanceStub.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/stubs/NewInstanceStub.java @@ -53,6 +53,7 @@ import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil. import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.tlabRefillWasteLimitOffset; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.tlabSlowAllocationsOffset; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.tlabStats; +import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.useCMSIncrementalMode; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.useG1GC; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.useTLAB; import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.wordSize; @@ -147,7 +148,7 @@ public class NewInstanceStub extends SnippetStub { */ Word thread = registerAsWord(threadRegister); boolean inlineContiguousAllocationSupported = GraalHotSpotVMConfigNode.inlineContiguousAllocationSupported(); - if (!forceSlowPath(options) && inlineContiguousAllocationSupported) { + if (!forceSlowPath(options) && inlineContiguousAllocationSupported && !useCMSIncrementalMode(INJECTED_VMCONFIG)) { if (isInstanceKlassFullyInitialized(hub)) { int sizeInBytes = readLayoutHelper(hub); Word memory = refillAllocate(thread, intArrayHub, sizeInBytes, logging(options)); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/BytecodeParser.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/BytecodeParser.java index 4c23ddbc6ea..8a8299098fe 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/BytecodeParser.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/BytecodeParser.java @@ -342,7 +342,7 @@ import org.graalvm.compiler.nodes.calc.AddNode; import org.graalvm.compiler.nodes.calc.AndNode; import org.graalvm.compiler.nodes.calc.CompareNode; import org.graalvm.compiler.nodes.calc.ConditionalNode; -import org.graalvm.compiler.nodes.calc.DivNode; +import org.graalvm.compiler.nodes.calc.FloatDivNode; import org.graalvm.compiler.nodes.calc.FloatConvertNode; import org.graalvm.compiler.nodes.calc.IntegerBelowNode; import org.graalvm.compiler.nodes.calc.IntegerEqualsNode; @@ -374,7 +374,6 @@ import org.graalvm.compiler.nodes.extended.MembarNode; import org.graalvm.compiler.nodes.extended.StateSplitProxyNode; import org.graalvm.compiler.nodes.extended.ValueAnchorNode; import org.graalvm.compiler.nodes.graphbuilderconf.ClassInitializationPlugin; -import org.graalvm.compiler.nodes.graphbuilderconf.InvokeDynamicPlugin; import org.graalvm.compiler.nodes.graphbuilderconf.GraphBuilderConfiguration; import org.graalvm.compiler.nodes.graphbuilderconf.GraphBuilderConfiguration.BytecodeExceptionMode; import org.graalvm.compiler.nodes.graphbuilderconf.GraphBuilderContext; @@ -383,6 +382,7 @@ import org.graalvm.compiler.nodes.graphbuilderconf.InlineInvokePlugin.InlineInfo import org.graalvm.compiler.nodes.graphbuilderconf.IntrinsicContext; import org.graalvm.compiler.nodes.graphbuilderconf.InvocationPlugin; import org.graalvm.compiler.nodes.graphbuilderconf.InvocationPlugins.InvocationPluginReceiver; +import org.graalvm.compiler.nodes.graphbuilderconf.InvokeDynamicPlugin; import org.graalvm.compiler.nodes.graphbuilderconf.NodePlugin; import org.graalvm.compiler.nodes.graphbuilderconf.ProfilingPlugin; import org.graalvm.compiler.nodes.java.ArrayLengthNode; @@ -435,6 +435,7 @@ import jdk.vm.ci.meta.ResolvedJavaMethod; import jdk.vm.ci.meta.ResolvedJavaType; import jdk.vm.ci.meta.Signature; import jdk.vm.ci.meta.TriState; +import org.graalvm.compiler.core.common.type.IntegerStamp; /** * The {@code GraphBuilder} class parses the bytecode of a method and builds the IR graph. @@ -1036,7 +1037,7 @@ public class BytecodeParser implements GraphBuilderContext { deopt.updateNodeSourcePosition(() -> createBytecodePosition()); } - private AbstractBeginNode handleException(ValueNode exceptionObject, int bci) { + private AbstractBeginNode handleException(ValueNode exceptionObject, int bci, boolean deoptimizeOnly) { assert bci == BytecodeFrame.BEFORE_BCI || bci == bci() : "invalid bci"; debug.log("Creating exception dispatch edges at %d, exception object=%s, exception seen=%s", bci, exceptionObject, (profilingInfo == null ? "" : profilingInfo.getExceptionSeen(bci))); @@ -1058,8 +1059,12 @@ public class BytecodeParser implements GraphBuilderContext { this.controlFlowSplit = true; FixedWithNextNode finishedDispatch = finishInstruction(dispatchBegin, dispatchState); - createHandleExceptionTarget(finishedDispatch, bci, dispatchState); - + if (deoptimizeOnly) { + DeoptimizeNode deoptimizeNode = graph.add(new DeoptimizeNode(DeoptimizationAction.None, DeoptimizationReason.TransferToInterpreter)); + dispatchBegin.setNext(BeginNode.begin(deoptimizeNode)); + } else { + createHandleExceptionTarget(finishedDispatch, bci, dispatchState); + } return dispatchBegin; } @@ -1111,7 +1116,7 @@ public class BytecodeParser implements GraphBuilderContext { } protected ValueNode genFloatDiv(ValueNode x, ValueNode y) { - return DivNode.create(x, y); + return FloatDivNode.create(x, y); } protected ValueNode genFloatRem(ValueNode x, ValueNode y) { @@ -1215,7 +1220,7 @@ public class BytecodeParser implements GraphBuilderContext { ValueNode exception = frameState.pop(JavaKind.Object); FixedGuardNode nullCheck = append(new FixedGuardNode(graph.addOrUniqueWithInputs(IsNullNode.create(exception)), NullCheckException, InvalidateReprofile, true)); ValueNode nonNullException = graph.maybeAddOrUnique(PiNode.create(exception, exception.stamp().join(objectNonNull()), nullCheck)); - lastInstr.setNext(handleException(nonNullException, bci())); + lastInstr.setNext(handleException(nonNullException, bci(), false)); } protected LogicNode createInstanceOf(TypeReference type, ValueNode object) { @@ -1275,12 +1280,12 @@ public class BytecodeParser implements GraphBuilderContext { } BytecodeExceptionNode exception = graph.add(new BytecodeExceptionNode(metaAccess, NullPointerException.class)); AbstractBeginNode falseSucc = graph.add(new BeginNode()); - ValueNode nonNullReceiver = graph.addOrUnique(PiNode.create(receiver, objectNonNull(), falseSucc)); + ValueNode nonNullReceiver = graph.addOrUniqueWithInputs(PiNode.create(receiver, objectNonNull(), falseSucc)); append(new IfNode(graph.addOrUniqueWithInputs(IsNullNode.create(receiver)), exception, falseSucc, SLOW_PATH_PROBABILITY)); lastInstr = falseSucc; exception.setStateAfter(createFrameState(bci(), exception)); - exception.setNext(handleException(exception, bci())); + exception.setNext(handleException(exception, bci(), false)); EXPLICIT_EXCEPTIONS.increment(debug); return nonNullReceiver; } @@ -1292,7 +1297,7 @@ public class BytecodeParser implements GraphBuilderContext { lastInstr = trueSucc; exception.setStateAfter(createFrameState(bci(), exception)); - exception.setNext(handleException(exception, bci())); + exception.setNext(handleException(exception, bci(), false)); } protected ValueNode genArrayLength(ValueNode x) { @@ -1532,8 +1537,8 @@ public class BytecodeParser implements GraphBuilderContext { @Override public void handleReplacedInvoke(CallTargetNode callTarget, JavaKind resultType) { BytecodeParser intrinsicCallSiteParser = getNonIntrinsicAncestor(); - boolean withExceptionEdge = intrinsicCallSiteParser == null ? !omitInvokeExceptionEdge(null) : !intrinsicCallSiteParser.omitInvokeExceptionEdge(null); - createNonInlinedInvoke(withExceptionEdge, bci(), callTarget, resultType); + ExceptionEdgeAction exceptionEdgeAction = intrinsicCallSiteParser == null ? getActionForInvokeExceptionEdge(null) : intrinsicCallSiteParser.getActionForInvokeExceptionEdge(null); + createNonInlinedInvoke(exceptionEdgeAction, bci(), callTarget, resultType); } protected Invoke appendInvoke(InvokeKind initialInvokeKind, ResolvedJavaMethod initialTargetMethod, ValueNode[] args) { @@ -1603,7 +1608,7 @@ public class BytecodeParser implements GraphBuilderContext { int invokeBci = bci(); JavaTypeProfile profile = getProfileForInvoke(invokeKind); - boolean withExceptionEdge = !omitInvokeExceptionEdge(inlineInfo); + ExceptionEdgeAction edgeAction = getActionForInvokeExceptionEdge(inlineInfo); boolean partialIntrinsicExit = false; if (intrinsicContext != null && intrinsicContext.isCallToOriginal(targetMethod)) { partialIntrinsicExit = true; @@ -1614,7 +1619,7 @@ public class BytecodeParser implements GraphBuilderContext { // must use the same context as the call to the intrinsic. invokeBci = intrinsicCallSiteParser.bci(); profile = intrinsicCallSiteParser.getProfileForInvoke(invokeKind); - withExceptionEdge = !intrinsicCallSiteParser.omitInvokeExceptionEdge(inlineInfo); + edgeAction = intrinsicCallSiteParser.getActionForInvokeExceptionEdge(inlineInfo); } else { // We are parsing the intrinsic for the root compilation or for inlining, // This call is a partial intrinsic exit, and we do not have profile information @@ -1624,7 +1629,7 @@ public class BytecodeParser implements GraphBuilderContext { assert intrinsicContext.isPostParseInlined(); invokeBci = BytecodeFrame.UNKNOWN_BCI; profile = null; - withExceptionEdge = graph.method().getAnnotation(Snippet.class) == null; + edgeAction = graph.method().getAnnotation(Snippet.class) == null ? ExceptionEdgeAction.INCLUDE_AND_HANDLE : ExceptionEdgeAction.OMIT; } if (originalMethod.isStatic()) { @@ -1637,10 +1642,10 @@ public class BytecodeParser implements GraphBuilderContext { Signature sig = originalMethod.getSignature(); returnType = sig.getReturnType(method.getDeclaringClass()); resultType = sig.getReturnKind(); - assert checkPartialIntrinsicExit(intrinsicCallSiteParser == null ? null : intrinsicCallSiteParser.currentInvoke.args, args); + assert intrinsicContext.allowPartialIntrinsicArgumentMismatch() || checkPartialIntrinsicExit(intrinsicCallSiteParser == null ? null : intrinsicCallSiteParser.currentInvoke.args, args); targetMethod = originalMethod; } - Invoke invoke = createNonInlinedInvoke(withExceptionEdge, invokeBci, args, targetMethod, invokeKind, resultType, returnType, profile); + Invoke invoke = createNonInlinedInvoke(edgeAction, invokeBci, args, targetMethod, invokeKind, resultType, returnType, profile); if (partialIntrinsicExit) { // This invoke must never be later inlined as it might select the intrinsic graph. // Until there is a mechanism to guarantee that any late inlining will not select @@ -1698,14 +1703,14 @@ public class BytecodeParser implements GraphBuilderContext { } else { for (int i = 0; i < recursiveArgs.length; i++) { ValueNode arg = GraphUtil.unproxify(recursiveArgs[i]); - assert arg instanceof ParameterNode && ((ParameterNode) arg).index() == i : String.format("argument %d of call denoting partial intrinsic exit should be a %s with index %d, not %s", i, - ParameterNode.class.getSimpleName(), i, arg); + assert arg instanceof ParameterNode && ((ParameterNode) arg).index() == i : String.format("argument %d of call denoting partial intrinsic exit should be a %s with index %d, not %s", + i, ParameterNode.class.getSimpleName(), i, arg); } } return true; } - protected Invoke createNonInlinedInvoke(boolean withExceptionEdge, int invokeBci, ValueNode[] invokeArgs, ResolvedJavaMethod targetMethod, + protected Invoke createNonInlinedInvoke(ExceptionEdgeAction exceptionEdge, int invokeBci, ValueNode[] invokeArgs, ResolvedJavaMethod targetMethod, InvokeKind invokeKind, JavaKind resultType, JavaType returnType, JavaTypeProfile profile) { StampPair returnStamp = graphBuilderConfig.getPlugins().getOverridingStamp(this, returnType, false); @@ -1714,7 +1719,7 @@ public class BytecodeParser implements GraphBuilderContext { } MethodCallTargetNode callTarget = graph.add(createMethodCallTarget(invokeKind, targetMethod, invokeArgs, returnStamp, profile)); - Invoke invoke = createNonInlinedInvoke(withExceptionEdge, invokeBci, callTarget, resultType); + Invoke invoke = createNonInlinedInvoke(exceptionEdge, invokeBci, callTarget, resultType); for (InlineInvokePlugin plugin : graphBuilderConfig.getPlugins().getInlineInvokePlugins()) { plugin.notifyNotInlined(this, targetMethod, invoke); @@ -1723,11 +1728,11 @@ public class BytecodeParser implements GraphBuilderContext { return invoke; } - protected Invoke createNonInlinedInvoke(boolean withExceptionEdge, int invokeBci, CallTargetNode callTarget, JavaKind resultType) { - if (!withExceptionEdge) { + protected Invoke createNonInlinedInvoke(ExceptionEdgeAction exceptionEdge, int invokeBci, CallTargetNode callTarget, JavaKind resultType) { + if (exceptionEdge == ExceptionEdgeAction.OMIT) { return createInvoke(invokeBci, callTarget, resultType); } else { - Invoke invoke = createInvokeWithException(invokeBci, callTarget, resultType); + Invoke invoke = createInvokeWithException(invokeBci, callTarget, resultType, exceptionEdge); AbstractBeginNode beginNode = graph.add(KillingBeginNode.create(LocationIdentity.any())); invoke.setNext(beginNode); lastInstr = beginNode; @@ -1736,20 +1741,29 @@ public class BytecodeParser implements GraphBuilderContext { } /** - * If the method returns true, the invocation of the given {@link MethodCallTargetNode call - * target} does not need an exception edge. + * Describes what should be done with the exception edge of an invocation. The edge can be + * omitted or included. An included edge can handle the exception or transfer execution to the + * interpreter for handling (deoptimize). */ - protected boolean omitInvokeExceptionEdge(InlineInfo lastInlineInfo) { + protected enum ExceptionEdgeAction { + OMIT, + INCLUDE_AND_HANDLE, + INCLUDE_AND_DEOPTIMIZE + } + + protected ExceptionEdgeAction getActionForInvokeExceptionEdge(InlineInfo lastInlineInfo) { if (lastInlineInfo == InlineInfo.DO_NOT_INLINE_WITH_EXCEPTION) { - return false; + return ExceptionEdgeAction.INCLUDE_AND_HANDLE; } else if (lastInlineInfo == InlineInfo.DO_NOT_INLINE_NO_EXCEPTION) { - return true; + return ExceptionEdgeAction.OMIT; + } else if (lastInlineInfo == InlineInfo.DO_NOT_INLINE_DEOPTIMIZE_ON_EXCEPTION) { + return ExceptionEdgeAction.INCLUDE_AND_DEOPTIMIZE; } else if (graphBuilderConfig.getBytecodeExceptionMode() == BytecodeExceptionMode.CheckAll) { - return false; + return ExceptionEdgeAction.INCLUDE_AND_HANDLE; } else if (graphBuilderConfig.getBytecodeExceptionMode() == BytecodeExceptionMode.ExplicitOnly) { - return false; + return ExceptionEdgeAction.INCLUDE_AND_HANDLE; } else if (graphBuilderConfig.getBytecodeExceptionMode() == BytecodeExceptionMode.OmitAll) { - return true; + return ExceptionEdgeAction.OMIT; } else { assert graphBuilderConfig.getBytecodeExceptionMode() == BytecodeExceptionMode.Profile; // be conservative if information was not recorded (could result in endless @@ -1759,12 +1773,12 @@ public class BytecodeParser implements GraphBuilderContext { if (profilingInfo != null) { TriState exceptionSeen = profilingInfo.getExceptionSeen(bci()); if (exceptionSeen == TriState.FALSE) { - return true; + return ExceptionEdgeAction.OMIT; } } } } - return false; + return ExceptionEdgeAction.INCLUDE_AND_HANDLE; } } @@ -1887,7 +1901,7 @@ public class BytecodeParser implements GraphBuilderContext { if (newProfile != profile) { if (newProfile.getTypes().length == 0) { // All profiled types select the intrinsic so - // emit a fixed guard instead of a if-then-else. + // emit a fixed guard instead of an if-then-else. lastInstr = append(new FixedGuardNode(compare, TypeCheckedInliningViolated, InvalidateReprofile, false)); return new IntrinsicGuard(currentLastInstr, intrinsicReceiver, mark, null, null); } @@ -1966,7 +1980,7 @@ public class BytecodeParser implements GraphBuilderContext { } lastInstr = intrinsicGuard.nonIntrinsicBranch; - createNonInlinedInvoke(omitInvokeExceptionEdge(null), bci(), args, targetMethod, invokeKind, resultType, returnType, intrinsicGuard.profile); + createNonInlinedInvoke(getActionForInvokeExceptionEdge(null), bci(), args, targetMethod, invokeKind, resultType, returnType, intrinsicGuard.profile); EndNode nonIntrinsicEnd = append(new EndNode()); AbstractMergeNode mergeNode = graph.add(new MergeNode()); @@ -2303,7 +2317,7 @@ public class BytecodeParser implements GraphBuilderContext { if (calleeBeforeUnwindNode != null) { ValueNode calleeUnwindValue = parser.getUnwindValue(); assert calleeUnwindValue != null; - calleeBeforeUnwindNode.setNext(handleException(calleeUnwindValue, bci())); + calleeBeforeUnwindNode.setNext(handleException(calleeUnwindValue, bci(), false)); } } } @@ -2319,7 +2333,7 @@ public class BytecodeParser implements GraphBuilderContext { return invoke; } - protected InvokeWithExceptionNode createInvokeWithException(int invokeBci, CallTargetNode callTarget, JavaKind resultType) { + protected InvokeWithExceptionNode createInvokeWithException(int invokeBci, CallTargetNode callTarget, JavaKind resultType, ExceptionEdgeAction exceptionEdgeAction) { if (currentBlock != null && stream.nextBCI() > currentBlock.endBci) { /* * Clear non-live locals early so that the exception handler entry gets the cleared @@ -2328,7 +2342,7 @@ public class BytecodeParser implements GraphBuilderContext { frameState.clearNonLiveLocals(currentBlock, liveness, false); } - AbstractBeginNode exceptionEdge = handleException(null, bci()); + AbstractBeginNode exceptionEdge = handleException(null, bci(), exceptionEdgeAction == ExceptionEdgeAction.INCLUDE_AND_DEOPTIMIZE); InvokeWithExceptionNode invoke = append(new InvokeWithExceptionNode(callTarget, exceptionEdge, invokeBci)); frameState.pushReturn(resultType, invoke); invoke.setStateAfter(createFrameState(stream.nextBCI(), invoke)); @@ -2359,20 +2373,43 @@ public class BytecodeParser implements GraphBuilderContext { } } + ValueNode realReturnVal = processReturnValue(returnVal, returnKind); + frameState.setRethrowException(false); frameState.clearStack(); - beforeReturn(returnVal, returnKind); + beforeReturn(realReturnVal, returnKind); if (parent == null) { - append(new ReturnNode(returnVal)); + append(new ReturnNode(realReturnVal)); } else { if (returnDataList == null) { returnDataList = new ArrayList<>(); } - returnDataList.add(new ReturnToCallerData(returnVal, lastInstr)); + returnDataList.add(new ReturnToCallerData(realReturnVal, lastInstr)); lastInstr = null; } } + private ValueNode processReturnValue(ValueNode value, JavaKind kind) { + JavaKind returnKind = method.getSignature().getReturnKind(); + if (kind != returnKind) { + // sub-word integer + assert returnKind.isNumericInteger() && returnKind.getStackKind() == JavaKind.Int; + IntegerStamp stamp = (IntegerStamp) value.stamp(); + + // the bytecode verifier doesn't check that the value is in the correct range + if (stamp.lowerBound() < returnKind.getMinValue() || returnKind.getMaxValue() < stamp.upperBound()) { + ValueNode narrow = append(genNarrow(value, returnKind.getBitCount())); + if (returnKind.isUnsigned()) { + return append(genZeroExtend(narrow, 32)); + } else { + return append(genSignExtend(narrow, 32)); + } + } + } + + return value; + } + private void beforeReturn(ValueNode x, JavaKind kind) { if (graph.method() != null && graph.method().isJavaLangObjectInit()) { /* @@ -2794,6 +2831,8 @@ public class BytecodeParser implements GraphBuilderContext { } private void createExceptionDispatch(ExceptionDispatchBlock block) { + lastInstr = finishInstruction(lastInstr, frameState); + assert frameState.stackSize() == 1 : frameState; if (block.handler.isCatchAll()) { assert block.getSuccessorCount() == 1; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/FrameStateBuilder.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/FrameStateBuilder.java index f045d25a6bd..b924c25f85c 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/FrameStateBuilder.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/FrameStateBuilder.java @@ -207,7 +207,7 @@ public final class FrameStateBuilder implements SideEffectsState { receiver = new ParameterNode(javaIndex, receiverStamp); } - locals[javaIndex] = graph.addOrUnique(receiver); + locals[javaIndex] = graph.addOrUniqueWithInputs(receiver); javaIndex = 1; index = 1; } @@ -241,7 +241,7 @@ public final class FrameStateBuilder implements SideEffectsState { param = new ParameterNode(index, stamp); } - locals[javaIndex] = graph.addOrUnique(param); + locals[javaIndex] = graph.addOrUniqueWithInputs(param); javaIndex++; if (kind.needsTwoSlots()) { locals[javaIndex] = TWO_SLOT_MARKER; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/ConditionalElimination02.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/ConditionalElimination02.java index e975bd8d3de..3d709ee1792 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/ConditionalElimination02.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/ConditionalElimination02.java @@ -29,6 +29,9 @@ import jdk.vm.ci.meta.DeoptimizationReason; import org.junit.Test; import org.graalvm.compiler.jtt.JTTTest; +import org.graalvm.compiler.phases.OptimisticOptimizations; +import org.graalvm.compiler.phases.OptimisticOptimizations.Optimization; +import org.graalvm.compiler.phases.tiers.HighTierContext; public class ConditionalElimination02 extends JTTTest { @@ -59,6 +62,14 @@ public class ConditionalElimination02 extends JTTTest { return -1; } + /** + * These tests assume all code paths are reachable so disable profile based dead code removal. + */ + @Override + protected HighTierContext getDefaultHighTierContext() { + return new HighTierContext(getProviders(), getDefaultGraphBuilderSuite(), OptimisticOptimizations.ALL.remove(Optimization.RemoveNeverExecutedCode)); + } + @Test public void run0() throws Throwable { runTest(EnumSet.of(DeoptimizationReason.NullCheckException), "test", new A(5), false, false); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/Fold_Double04.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/Fold_Double04.java new file mode 100644 index 00000000000..f2a73a25357 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/Fold_Double04.java @@ -0,0 +1,66 @@ +/* + * Copyright (c) 2017, 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.jtt.optimize; + +import org.junit.Test; + +import org.graalvm.compiler.jtt.JTTTest; + +/* + * Tests constant folding of double operations. + */ +public class Fold_Double04 extends JTTTest { + + // Contrived check whether both arguments are the same kind of zero + public static boolean test(double x, double y) { + if (x == 0) { + if (1 / x == Double.NEGATIVE_INFINITY) { + return 1 / y == Double.NEGATIVE_INFINITY; + } else { + return 1 / y == Double.POSITIVE_INFINITY; + } + } + return false; + } + + @Test + public void run0() throws Throwable { + runTest("test", -0d, -0d); + } + + @Test + public void run1() throws Throwable { + runTest("test", -0d, 0d); + } + + @Test + public void run2() throws Throwable { + runTest("test", 0d, -0d); + } + + @Test + public void run3() throws Throwable { + runTest("test", 0d, 0d); + } + +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/Fold_Float03.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/Fold_Float03.java new file mode 100644 index 00000000000..74e8fe8d434 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.jtt/src/org/graalvm/compiler/jtt/optimize/Fold_Float03.java @@ -0,0 +1,66 @@ +/* + * Copyright (c) 2017, 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.jtt.optimize; + +import org.junit.Test; + +import org.graalvm.compiler.jtt.JTTTest; + +/* + * Tests constant folding of float operations. + */ +public class Fold_Float03 extends JTTTest { + + // Contrived check whether both arguments are the same kind of zero + public static boolean test(float x, float y) { + if (x == 0) { + if (1 / x == Float.NEGATIVE_INFINITY) { + return 1 / y == Float.NEGATIVE_INFINITY; + } else { + return 1 / y == Float.POSITIVE_INFINITY; + } + } + return false; + } + + @Test + public void run0() throws Throwable { + runTest("test", -0f, -0f); + } + + @Test + public void run1() throws Throwable { + runTest("test", -0f, 0f); + } + + @Test + public void run2() throws Throwable { + runTest("test", 0f, -0f); + } + + @Test + public void run3() throws Throwable { + runTest("test", 0f, 0f); + } + +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64ArrayEqualsOp.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64ArrayEqualsOp.java index a9f8e9221cd..8ba05753b81 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64ArrayEqualsOp.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64ArrayEqualsOp.java @@ -37,6 +37,7 @@ import org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize; import org.graalvm.compiler.asm.amd64.AMD64Assembler.SSEOp; import org.graalvm.compiler.asm.amd64.AMD64MacroAssembler; import org.graalvm.compiler.core.common.LIRKind; +import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.lir.LIRInstructionClass; import org.graalvm.compiler.lir.Opcode; import org.graalvm.compiler.lir.asm.CompilationResultBuilder; @@ -132,12 +133,19 @@ public final class AMD64ArrayEqualsOp extends AMD64LIRInstruction { masm.leaq(array2, new AMD64Address(asRegister(array2Value), arrayBaseOffset)); // Get array length in bytes. - masm.imull(length, asRegister(lengthValue), arrayIndexScale); + masm.movl(length, asRegister(lengthValue)); + + if (arrayIndexScale > 1) { + masm.shll(length, NumUtil.log2Ceil(arrayIndexScale)); // scale length + } + masm.movl(result, length); // copy if (supportsAVX2(crb.target)) { emitAVXCompare(crb, masm, result, array1, array2, length, trueLabel, falseLabel); } else if (supportsSSE41(crb.target)) { + // this code is used for AVX as well because our backend correctly ensures that + // VEX-prefixed instructions are emitted if AVX is supported emitSSE41Compare(crb, masm, result, array1, array2, length, trueLabel, falseLabel); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64ControlFlow.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64ControlFlow.java index 41cf3c9a6f3..c3c7c84aef0 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64ControlFlow.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64ControlFlow.java @@ -22,21 +22,21 @@ */ package org.graalvm.compiler.lir.amd64; +import static jdk.vm.ci.code.ValueUtil.asRegister; +import static jdk.vm.ci.code.ValueUtil.isRegister; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.CONST; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.HINT; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.ILLEGAL; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.REG; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.STACK; -import static jdk.vm.ci.code.ValueUtil.asRegister; -import static jdk.vm.ci.code.ValueUtil.isRegister; import org.graalvm.compiler.asm.Label; -import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.asm.amd64.AMD64Address; import org.graalvm.compiler.asm.amd64.AMD64Address.Scale; import org.graalvm.compiler.asm.amd64.AMD64Assembler.ConditionFlag; import org.graalvm.compiler.asm.amd64.AMD64MacroAssembler; import org.graalvm.compiler.code.CompilationResult.JumpTable; +import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.core.common.calc.Condition; import org.graalvm.compiler.debug.GraalError; import org.graalvm.compiler.lir.LIRInstructionClass; @@ -312,6 +312,42 @@ public class AMD64ControlFlow { } } + @Opcode("SETcc") + public static final class CondSetOp extends AMD64LIRInstruction { + public static final LIRInstructionClass TYPE = LIRInstructionClass.create(CondSetOp.class); + @Def({REG, HINT}) protected Value result; + private final ConditionFlag condition; + + public CondSetOp(Variable result, Condition condition) { + super(TYPE); + this.result = result; + this.condition = intCond(condition); + } + + @Override + public void emitCode(CompilationResultBuilder crb, AMD64MacroAssembler masm) { + setcc(masm, result, condition); + } + } + + @Opcode("SETcc") + public static final class FloatCondSetOp extends AMD64LIRInstruction { + public static final LIRInstructionClass TYPE = LIRInstructionClass.create(FloatCondSetOp.class); + @Def({REG, HINT}) protected Value result; + private final ConditionFlag condition; + + public FloatCondSetOp(Variable result, Condition condition) { + super(TYPE); + this.result = result; + this.condition = floatCond(condition); + } + + @Override + public void emitCode(CompilationResultBuilder crb, AMD64MacroAssembler masm) { + setcc(masm, result, condition); + } + } + @Opcode("CMOVE") public static final class CondMoveOp extends AMD64LIRInstruction { public static final LIRInstructionClass TYPE = LIRInstructionClass.create(CondMoveOp.class); @@ -418,6 +454,21 @@ public class AMD64ControlFlow { } } + private static void setcc(AMD64MacroAssembler masm, Value result, ConditionFlag cond) { + switch ((AMD64Kind) result.getPlatformKind()) { + case BYTE: + case WORD: + case DWORD: + masm.setl(cond, asRegister(result)); + break; + case QWORD: + masm.setq(cond, asRegister(result)); + break; + default: + throw GraalError.shouldNotReachHere(); + } + } + private static ConditionFlag intCond(Condition cond) { switch (cond) { case EQ: @@ -464,6 +515,10 @@ public class AMD64ControlFlow { } } + public static boolean trueOnUnordered(Condition condition) { + return trueOnUnordered(floatCond(condition)); + } + private static boolean trueOnUnordered(ConditionFlag condition) { switch (condition) { case AboveEqual: diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64Move.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64Move.java index 8f5785159a1..38fdd263a6c 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64Move.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.amd64/src/org/graalvm/compiler/lir/amd64/AMD64Move.java @@ -22,8 +22,11 @@ */ package org.graalvm.compiler.lir.amd64; +import static org.graalvm.compiler.asm.amd64.AMD64Assembler.ConditionFlag.Equal; +import static org.graalvm.compiler.core.common.GraalOptions.GeneratePIC; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.COMPOSITE; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.HINT; +import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.ILLEGAL; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.REG; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.STACK; import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.UNINITIALIZED; @@ -35,12 +38,16 @@ import static jdk.vm.ci.code.ValueUtil.asRegister; import static jdk.vm.ci.code.ValueUtil.isRegister; import static jdk.vm.ci.code.ValueUtil.isStackSlot; +import org.graalvm.compiler.asm.Label; +import org.graalvm.compiler.core.common.CompressEncoding; +import org.graalvm.compiler.core.common.LIRKind; import org.graalvm.compiler.core.common.NumUtil; import org.graalvm.compiler.asm.amd64.AMD64Address; import org.graalvm.compiler.asm.amd64.AMD64Assembler.AMD64MIOp; import org.graalvm.compiler.asm.amd64.AMD64Assembler.AMD64MOp; import org.graalvm.compiler.asm.amd64.AMD64Assembler.OperandSize; import org.graalvm.compiler.asm.amd64.AMD64MacroAssembler; +import org.graalvm.compiler.core.common.spi.LIRKindTool; import org.graalvm.compiler.core.common.type.DataPointerConstant; import org.graalvm.compiler.debug.GraalError; import org.graalvm.compiler.lir.LIRFrameState; @@ -740,4 +747,110 @@ public class AMD64Move { throw GraalError.shouldNotReachHere("Unknown result Kind: " + result.getPlatformKind()); } } + + public abstract static class Pointer extends AMD64LIRInstruction { + protected final LIRKindTool lirKindTool; + protected final CompressEncoding encoding; + protected final boolean nonNull; + + @Def({REG, HINT}) private AllocatableValue result; + @Use({REG}) private AllocatableValue input; + @Alive({REG, ILLEGAL}) private AllocatableValue baseRegister; + + protected Pointer(LIRInstructionClass type, AllocatableValue result, AllocatableValue input, AllocatableValue baseRegister, CompressEncoding encoding, boolean nonNull, + LIRKindTool lirKindTool) { + super(type); + this.result = result; + this.input = input; + this.baseRegister = baseRegister; + this.encoding = encoding; + this.nonNull = nonNull; + this.lirKindTool = lirKindTool; + } + + protected boolean hasBase(CompilationResultBuilder crb) { + return GeneratePIC.getValue(crb.getOptions()) || encoding.hasBase(); + } + + protected final Register getResultRegister() { + return asRegister(result); + } + + protected final Register getBaseRegister() { + return asRegister(baseRegister); + } + + protected final int getShift() { + return encoding.getShift(); + } + + protected final void move(LIRKind kind, CompilationResultBuilder crb, AMD64MacroAssembler masm) { + AMD64Move.move((AMD64Kind) kind.getPlatformKind(), crb, masm, result, input); + } + } + + public static final class CompressPointer extends Pointer { + public static final LIRInstructionClass TYPE = LIRInstructionClass.create(CompressPointer.class); + + public CompressPointer(AllocatableValue result, AllocatableValue input, AllocatableValue baseRegister, CompressEncoding encoding, boolean nonNull, LIRKindTool lirKindTool) { + super(TYPE, result, input, baseRegister, encoding, nonNull, lirKindTool); + } + + @Override + public void emitCode(CompilationResultBuilder crb, AMD64MacroAssembler masm) { + move(lirKindTool.getObjectKind(), crb, masm); + + Register resReg = getResultRegister(); + if (hasBase(crb)) { + Register baseReg = getBaseRegister(); + if (!nonNull) { + masm.testq(resReg, resReg); + masm.cmovq(Equal, resReg, baseReg); + } + masm.subq(resReg, baseReg); + } + + int shift = getShift(); + if (shift != 0) { + masm.shrq(resReg, shift); + } + } + } + + public static final class UncompressPointer extends Pointer { + public static final LIRInstructionClass TYPE = LIRInstructionClass.create(UncompressPointer.class); + + public UncompressPointer(AllocatableValue result, AllocatableValue input, AllocatableValue baseRegister, CompressEncoding encoding, boolean nonNull, LIRKindTool lirKindTool) { + super(TYPE, result, input, baseRegister, encoding, nonNull, lirKindTool); + } + + @Override + public void emitCode(CompilationResultBuilder crb, AMD64MacroAssembler masm) { + move(lirKindTool.getNarrowOopKind(), crb, masm); + + Register resReg = getResultRegister(); + int shift = getShift(); + if (shift != 0) { + masm.shlq(resReg, shift); + } + + if (hasBase(crb)) { + Register baseReg = getBaseRegister(); + if (nonNull) { + masm.addq(resReg, baseReg); + return; + } + + if (shift == 0) { + // if encoding.shift != 0, the flags are already set by the shlq + masm.testq(resReg, resReg); + } + + Label done = new Label(); + masm.jccb(Equal, done); + masm.addq(resReg, baseReg); + masm.bind(done); + } + } + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.sparc/src/org/graalvm/compiler/lir/sparc/SPARCLoadConstantTableBaseOp.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.sparc/src/org/graalvm/compiler/lir/sparc/SPARCLoadConstantTableBaseOp.java index bb47d833eae..0eaedd1da7c 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.sparc/src/org/graalvm/compiler/lir/sparc/SPARCLoadConstantTableBaseOp.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.sparc/src/org/graalvm/compiler/lir/sparc/SPARCLoadConstantTableBaseOp.java @@ -55,8 +55,8 @@ import jdk.vm.ci.meta.AllocatableValue; * this case absolute addressing (without using the base pointer is used). See also: * CodeInstaller::pd_patch_DataSectionReference * - * @see SPARCMove#loadFromConstantTable(CompilationResultBuilder, SPARCMacroAssembler, int, - * Register, jdk.vm.ci.meta.Constant, Register, SPARCDelayedControlTransfer) + * @see SPARCMove#loadFromConstantTable(CompilationResultBuilder, SPARCMacroAssembler, Register, + * jdk.vm.ci.meta.Constant, Register, SPARCDelayedControlTransfer) */ public class SPARCLoadConstantTableBaseOp extends SPARCLIRInstruction { public static final LIRInstructionClass TYPE = LIRInstructionClass.create(SPARCLoadConstantTableBaseOp.class); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.sparc/src/org/graalvm/compiler/lir/sparc/SPARCMove.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.sparc/src/org/graalvm/compiler/lir/sparc/SPARCMove.java index b35e9b55750..22dedeaefe0 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.sparc/src/org/graalvm/compiler/lir/sparc/SPARCMove.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir.sparc/src/org/graalvm/compiler/lir/sparc/SPARCMove.java @@ -53,6 +53,7 @@ import org.graalvm.compiler.asm.sparc.SPARCAddress; import org.graalvm.compiler.asm.sparc.SPARCAssembler; import org.graalvm.compiler.asm.sparc.SPARCMacroAssembler; import org.graalvm.compiler.asm.sparc.SPARCMacroAssembler.ScratchRegister; +import org.graalvm.compiler.code.DataSection.Data; import org.graalvm.compiler.core.common.LIRKind; import org.graalvm.compiler.core.common.type.DataPointerConstant; import org.graalvm.compiler.debug.GraalError; @@ -116,11 +117,11 @@ public class SPARCMove { public static final LIRInstructionClass TYPE = LIRInstructionClass.create(LoadConstantFromTable.class); public static final SizeEstimate SIZE = SizeEstimate.create(1, 8); - private Constant constant; + private JavaConstant constant; @Def({REG, STACK}) AllocatableValue result; @Use({REG}) private AllocatableValue constantTableBase; - public LoadConstantFromTable(Constant constant, AllocatableValue constantTableBase, AllocatableValue result) { + public LoadConstantFromTable(JavaConstant constant, AllocatableValue constantTableBase, AllocatableValue result) { super(TYPE, SIZE); this.constant = constant; this.result = result; @@ -134,11 +135,11 @@ public class SPARCMove { Register baseRegister = asRegister(constantTableBase); if (isRegister(result)) { Register resultRegister = asRegister(result); - loadFromConstantTable(crb, masm, byteCount, baseRegister, constant, resultRegister, getDelayedControlTransfer()); + loadFromConstantTable(crb, masm, baseRegister, constant, resultRegister, getDelayedControlTransfer()); } else if (isStackSlot(result)) { try (ScratchRegister scratch = masm.getScratchRegister()) { Register scratchRegister = scratch.getRegister(); - loadFromConstantTable(crb, masm, byteCount, baseRegister, constant, scratchRegister, getDelayedControlTransfer()); + loadFromConstantTable(crb, masm, baseRegister, constant, scratchRegister, getDelayedControlTransfer()); StackSlot slot = asStackSlot(result); reg2stack(crb, masm, slot, scratchRegister.asValue(), getDelayedControlTransfer()); } @@ -642,7 +643,6 @@ public class SPARCMove { boolean hasVIS1 = cpuFeatures.contains(CPUFeature.VIS1); boolean hasVIS3 = cpuFeatures.contains(CPUFeature.VIS3); Register resultRegister = asRegister(result); - int byteCount = result.getPlatformKind().getSizeInBytes(); switch (input.getJavaKind().getStackKind()) { case Int: if (input.isDefaultForKind()) { @@ -655,7 +655,7 @@ public class SPARCMove { if (constantTableBase.equals(g0)) { throw GraalError.shouldNotReachHere(); } else { - loadFromConstantTable(crb, masm, byteCount, constantTableBase, input, resultRegister, delaySlotLir); + loadFromConstantTable(crb, masm, constantTableBase, input, resultRegister, delaySlotLir); } } break; @@ -667,7 +667,7 @@ public class SPARCMove { delaySlotLir.emitControlTransfer(crb, masm); masm.or(g0, (int) input.asLong(), resultRegister); } else { - loadFromConstantTable(crb, masm, byteCount, constantTableBase, input, resultRegister, delaySlotLir); + loadFromConstantTable(crb, masm, constantTableBase, input, resultRegister, delaySlotLir); } break; case Float: { @@ -683,7 +683,7 @@ public class SPARCMove { masm.movwtos(scratch, resultRegister); } else { // First load the address into the scratch register - loadFromConstantTable(crb, masm, byteCount, constantTableBase, input, resultRegister, delaySlotLir); + loadFromConstantTable(crb, masm, constantTableBase, input, resultRegister, delaySlotLir); } } break; @@ -700,7 +700,7 @@ public class SPARCMove { delaySlotLir.emitControlTransfer(crb, masm); masm.movxtod(scratch, resultRegister); } else { - loadFromConstantTable(crb, masm, byteCount, constantTableBase, input, resultRegister, delaySlotLir); + loadFromConstantTable(crb, masm, constantTableBase, input, resultRegister, delaySlotLir); } } break; @@ -710,7 +710,7 @@ public class SPARCMove { delaySlotLir.emitControlTransfer(crb, masm); masm.clr(resultRegister); } else { - loadFromConstantTable(crb, masm, byteCount, constantTableBase, input, resultRegister, delaySlotLir); + loadFromConstantTable(crb, masm, constantTableBase, input, resultRegister, delaySlotLir); } break; default: @@ -768,25 +768,30 @@ public class SPARCMove { * patterns used for small constant sections (<8k) and large constant sections (>=8k). The * generated patterns by this method must be understood by * CodeInstaller::pd_patch_DataSectionReference (jvmciCodeInstaller_sparc.cpp). + * + * @return the number of bytes loaded from the constant table */ - public static void loadFromConstantTable(CompilationResultBuilder crb, SPARCMacroAssembler masm, int byteCount, Register constantTableBase, Constant input, Register dest, + public static int loadFromConstantTable(CompilationResultBuilder crb, SPARCMacroAssembler masm, Register constantTableBase, Constant input, Register dest, SPARCDelayedControlTransfer delaySlotInstruction) { SPARCAddress address; ScratchRegister scratch = null; try { + Data data = crb.createDataItem(input); + int size = data.getSize(); if (masm.isImmediateConstantLoad()) { address = new SPARCAddress(constantTableBase, 0); // Make delayed only, when using immediate constant load. delaySlotInstruction.emitControlTransfer(crb, masm); - crb.recordDataReferenceInCode(input, byteCount); + crb.recordDataReferenceInCode(data, size); } else { scratch = masm.getScratchRegister(); Register sr = scratch.getRegister(); - crb.recordDataReferenceInCode(input, byteCount); + crb.recordDataReferenceInCode(data, size); masm.sethix(0, sr, true); address = new SPARCAddress(sr, 0); } - masm.ld(address, dest, byteCount, false); + masm.ld(address, dest, size, false); + return size; } finally { if (scratch != null) { scratch.close(); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/LIRValueUtil.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/LIRValueUtil.java index f0f9e6e9826..f0ec11125fe 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/LIRValueUtil.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/LIRValueUtil.java @@ -69,6 +69,16 @@ public final class LIRValueUtil { return asConstantValue(value).getJavaConstant(); } + public static boolean isIntConstant(Value value, long expected) { + if (isJavaConstant(value)) { + JavaConstant javaConstant = asJavaConstant(value); + if (javaConstant != null && javaConstant.getJavaKind().isNumericInteger()) { + return javaConstant.asLong() == expected; + } + } + return false; + } + public static boolean isStackSlotValue(Value value) { assert value != null; return value instanceof StackSlot || value instanceof VirtualStackSlot; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/RedundantMoveElimination.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/RedundantMoveElimination.java index 26434f7e526..69fbbf4eebf 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/RedundantMoveElimination.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/RedundantMoveElimination.java @@ -30,6 +30,7 @@ import java.util.Arrays; import java.util.Collections; import java.util.EnumSet; +import jdk.vm.ci.code.RegisterConfig; import org.graalvm.compiler.core.common.LIRKind; import org.graalvm.compiler.core.common.cfg.AbstractBlockBase; import org.graalvm.compiler.debug.CounterKey; @@ -138,8 +139,8 @@ public final class RedundantMoveElimination extends PostAllocationOptimizationPh private void doOptimize(LIR lir) { DebugContext debug = lir.getDebug(); try (Indent indent = debug.logAndIndent("eliminate redundant moves")) { - - callerSaveRegs = frameMap.getRegisterConfig().getCallerSaveRegisters(); + RegisterConfig registerConfig = frameMap.getRegisterConfig(); + callerSaveRegs = registerConfig.getCallerSaveRegisters(); initBlockData(lir); @@ -147,7 +148,7 @@ public final class RedundantMoveElimination extends PostAllocationOptimizationPh // Unallocatable registers should never be optimized. eligibleRegs = new int[numRegs]; Arrays.fill(eligibleRegs, -1); - for (Register reg : frameMap.getRegisterConfig().getAllocatableRegisters()) { + for (Register reg : registerConfig.getAllocatableRegisters()) { if (reg.number < numRegs) { eligibleRegs[reg.number] = reg.number; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/alloc/SaveCalleeSaveRegisters.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/alloc/SaveCalleeSaveRegisters.java index 976b618f8d9..cd0a15dff48 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/alloc/SaveCalleeSaveRegisters.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/alloc/SaveCalleeSaveRegisters.java @@ -31,7 +31,6 @@ import org.graalvm.compiler.lir.LIRInsertionBuffer; import org.graalvm.compiler.lir.LIRInstruction; import org.graalvm.compiler.lir.StandardOp; import org.graalvm.compiler.lir.Variable; -import org.graalvm.compiler.lir.framemap.FrameMapBuilder; import org.graalvm.compiler.lir.gen.LIRGenerationResult; import org.graalvm.compiler.lir.gen.LIRGeneratorTool; import org.graalvm.compiler.lir.phases.PreAllocationOptimizationPhase; @@ -48,8 +47,7 @@ public class SaveCalleeSaveRegisters extends PreAllocationOptimizationPhase { @Override protected void run(TargetDescription target, LIRGenerationResult lirGenRes, PreAllocationOptimizationContext context) { - FrameMapBuilder frameMapBuilder = lirGenRes.getFrameMapBuilder(); - RegisterArray calleeSaveRegisters = frameMapBuilder.getCodeCache().getRegisterConfig().getCalleeSaveRegisters(); + RegisterArray calleeSaveRegisters = lirGenRes.getRegisterConfig().getCalleeSaveRegisters(); if (calleeSaveRegisters == null || calleeSaveRegisters.size() == 0) { return; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/asm/CompilationResultBuilder.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/asm/CompilationResultBuilder.java index 36dd827fc3b..fec7aa0466a 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/asm/CompilationResultBuilder.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/asm/CompilationResultBuilder.java @@ -80,10 +80,10 @@ import jdk.vm.ci.meta.Value; */ public class CompilationResultBuilder { - // @formatter:off - @Option(help = "Include the LIR as comments with the final assembly.", type = OptionType.Debug) - public static final OptionKey PrintLIRWithAssembly = new OptionKey<>(false); - // @formatter:on + public static class Options { + @Option(help = "Include the LIR as comments with the final assembly.", type = OptionType.Debug) // + public static final OptionKey PrintLIRWithAssembly = new OptionKey<>(false); + } private static class ExceptionInfo { @@ -295,13 +295,24 @@ public class CompilationResultBuilder { public AbstractAddress recordDataReferenceInCode(Constant constant, int alignment) { assert constant != null; debug.log("Constant reference in code: pos = %d, data = %s", asm.position(), constant); + Data data = createDataItem(constant); + data.updateAlignment(alignment); + return recordDataSectionReference(data); + } + + public AbstractAddress recordDataReferenceInCode(Data data, int alignment) { + assert data != null; + data.updateAlignment(alignment); + return recordDataSectionReference(data); + } + + public Data createDataItem(Constant constant) { Data data = dataCache.get(constant); if (data == null) { data = dataBuilder.createDataItem(constant); dataCache.put(constant, data); } - data.updateAlignment(alignment); - return recordDataSectionReference(data); + return data; } public AbstractAddress recordDataReferenceInCode(byte[] data, int alignment) { @@ -472,7 +483,7 @@ public class CompilationResultBuilder { if (block == null) { return; } - boolean emitComment = debug.isDumpEnabled(DebugContext.BASIC_LEVEL) || PrintLIRWithAssembly.getValue(getOptions()); + boolean emitComment = debug.isDumpEnabled(DebugContext.BASIC_LEVEL) || Options.PrintLIRWithAssembly.getValue(getOptions()); if (emitComment) { blockComment(String.format("block B%d %s", block.getId(), block.getLoop())); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/ArithmeticLIRGenerator.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/ArithmeticLIRGenerator.java index 4a80de0705c..28df230967b 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/ArithmeticLIRGenerator.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/ArithmeticLIRGenerator.java @@ -50,8 +50,19 @@ public abstract class ArithmeticLIRGenerator implements ArithmeticLIRGeneratorTo protected abstract Variable emitAdd(LIRKind resultKind, Value a, Value b, boolean setFlags); + protected abstract Variable emitSub(LIRKind resultKind, Value a, Value b, boolean setFlags); + @Override public final Variable emitAdd(Value aVal, Value bVal, boolean setFlags) { + return emitAddOrSub(aVal, bVal, setFlags, true); + } + + @Override + public final Variable emitSub(Value aVal, Value bVal, boolean setFlags) { + return emitAddOrSub(aVal, bVal, setFlags, false); + } + + private Variable emitAddOrSub(Value aVal, Value bVal, boolean setFlags, boolean isAdd) { LIRKind resultKind; Value a = aVal; Value b = bVal; @@ -90,47 +101,7 @@ public abstract class ArithmeticLIRGenerator implements ArithmeticLIRGeneratorTo resultKind = LIRKind.combine(a, b); } - return emitAdd(resultKind, a, b, setFlags); + return isAdd ? emitAdd(resultKind, a, b, setFlags) : emitSub(resultKind, a, b, setFlags); } - protected abstract Variable emitSub(LIRKind resultKind, Value a, Value b, boolean setFlags); - - @Override - public final Variable emitSub(Value aVal, Value bVal, boolean setFlags) { - LIRKind resultKind; - Value a = aVal; - Value b = bVal; - - if (isNumericInteger(a.getPlatformKind())) { - LIRKind aKind = a.getValueKind(LIRKind.class); - LIRKind bKind = b.getValueKind(LIRKind.class); - assert a.getPlatformKind() == b.getPlatformKind(); - - if (aKind.isUnknownReference()) { - resultKind = aKind; - } else if (bKind.isUnknownReference()) { - resultKind = bKind; - } - - if (aKind.isValue() && bKind.isValue()) { - resultKind = aKind; - } else if (bKind.isValue()) { - if (aKind.isDerivedReference()) { - resultKind = aKind; - } else { - AllocatableValue allocatable = getLIRGen().asAllocatable(a); - resultKind = aKind.makeDerivedReference(allocatable); - a = allocatable; - } - } else if (aKind.isDerivedReference() && bKind.isDerivedReference() && aKind.getDerivedReferenceBase().equals(bKind.getDerivedReferenceBase())) { - resultKind = LIRKind.value(a.getPlatformKind()); - } else { - resultKind = aKind.makeUnknownReference(); - } - } else { - resultKind = LIRKind.combine(a, b); - } - - return emitSub(resultKind, a, b, setFlags); - } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGenerationResult.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGenerationResult.java index e4eb673740c..7d6922adaab 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGenerationResult.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGenerationResult.java @@ -22,6 +22,7 @@ */ package org.graalvm.compiler.lir.gen; +import jdk.vm.ci.code.RegisterConfig; import org.graalvm.compiler.core.common.CompilationIdentifier; import org.graalvm.compiler.core.common.CompilationIdentifier.Verbosity; import org.graalvm.compiler.debug.DebugContext; @@ -123,6 +124,10 @@ public class LIRGenerationResult { return frameMap; } + public final RegisterConfig getRegisterConfig() { + return frameMapBuilder.getRegisterConfig(); + } + public LIR getLIR() { return lir; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGenerator.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGenerator.java index 0c92168d8fe..383927856ee 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGenerator.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGenerator.java @@ -34,6 +34,7 @@ import static org.graalvm.compiler.lir.LIRValueUtil.isVirtualStackSlot; import java.util.ArrayList; import java.util.List; +import jdk.vm.ci.code.RegisterConfig; import org.graalvm.compiler.asm.Label; import org.graalvm.compiler.core.common.LIRKind; import org.graalvm.compiler.core.common.calc.Condition; @@ -200,9 +201,14 @@ public abstract class LIRGenerator implements LIRGeneratorTool { return new Variable(valueKind, ((VariableProvider) res.getLIR()).nextVariable()); } + @Override + public RegisterConfig getRegisterConfig() { + return res.getRegisterConfig(); + } + @Override public RegisterAttributes attributes(Register register) { - return res.getFrameMapBuilder().getRegisterConfig().getAttributesMap()[register.number]; + return getRegisterConfig().getAttributesMap()[register.number]; } @Override @@ -228,7 +234,7 @@ public abstract class LIRGenerator implements LIRGeneratorTool { if (moveFactory.canInlineConstant(constant)) { return new ConstantValue(toRegisterKind(kind), constant); } else { - return emitLoadConstant(kind, constant); + return emitLoadConstant(toRegisterKind(kind), constant); } } @@ -289,7 +295,7 @@ public abstract class LIRGenerator implements LIRGeneratorTool { */ @Override public AllocatableValue resultOperandFor(JavaKind javaKind, ValueKind valueKind) { - Register reg = res.getFrameMapBuilder().getRegisterConfig().getReturnRegister(javaKind); + Register reg = getRegisterConfig().getReturnRegister(javaKind); assert target().arch.canStoreValue(reg.getRegisterCategory(), valueKind.getPlatformKind()) : reg.getRegisterCategory() + " " + valueKind.getPlatformKind(); return reg.asValue(valueKind); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGeneratorTool.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGeneratorTool.java index 6533ba63180..c82c7e7ec33 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGeneratorTool.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/gen/LIRGeneratorTool.java @@ -22,6 +22,7 @@ */ package org.graalvm.compiler.lir.gen; +import jdk.vm.ci.code.RegisterConfig; import org.graalvm.compiler.core.common.CompressEncoding; import org.graalvm.compiler.core.common.LIRKind; import org.graalvm.compiler.core.common.calc.Condition; @@ -109,6 +110,8 @@ public interface LIRGeneratorTool extends DiagnosticLIRGeneratorTool, ValueKindF LIRGenerationResult getResult(); + RegisterConfig getRegisterConfig(); + boolean hasBlockEnd(AbstractBlockBase block); MoveFactory getMoveFactory(); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop.test/src/org/graalvm/compiler/loop/test/LoopPartialUnrollTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop.test/src/org/graalvm/compiler/loop/test/LoopPartialUnrollTest.java index 8a93d77bf93..4b411ca7aa3 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop.test/src/org/graalvm/compiler/loop/test/LoopPartialUnrollTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop.test/src/org/graalvm/compiler/loop/test/LoopPartialUnrollTest.java @@ -214,7 +214,7 @@ public class LoopPartialUnrollTest extends GraalCompilerTest { } }; ResolvedJavaMethod method = getResolvedJavaMethod(name); - OptionValues options = new OptionValues(getInitialOptions(), DefaultLoopPolicies.UnrollMaxIterations, 2); + OptionValues options = new OptionValues(getInitialOptions(), DefaultLoopPolicies.Options.UnrollMaxIterations, 2); StructuredGraph graph = parse(builder(method, StructuredGraph.AllowAssumptions.YES, id, options), getEagerGraphBuilderSuite()); try (DebugContext.Scope buildScope = graph.getDebug().scope(name, method, graph)) { MidTierContext context = new MidTierContext(getProviders(), getTargetProvider(), OptimisticOptimizations.ALL, null); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop/src/org/graalvm/compiler/loop/CountedLoopInfo.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop/src/org/graalvm/compiler/loop/CountedLoopInfo.java index 4e4c12b7e64..ae6bb412278 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop/src/org/graalvm/compiler/loop/CountedLoopInfo.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop/src/org/graalvm/compiler/loop/CountedLoopInfo.java @@ -83,7 +83,11 @@ public class CountedLoopInfo { range = add(graph, range, oneDirection); } // round-away-from-zero divison: (range + stride -/+ 1) / stride - ValueNode denominator = add(graph, sub(graph, range, oneDirection), iv.strideNode()); + ValueNode denominator = range; + if (!oneDirection.stamp().equals(iv.strideNode().stamp())) { + ValueNode subedRanged = sub(graph, range, oneDirection); + denominator = add(graph, subedRanged, iv.strideNode()); + } ValueNode div = divBefore(graph, loop.entryPoint(), denominator, iv.strideNode()); if (assumePositive) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop/src/org/graalvm/compiler/loop/DefaultLoopPolicies.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop/src/org/graalvm/compiler/loop/DefaultLoopPolicies.java index e25c955027b..ada82be03ef 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop/src/org/graalvm/compiler/loop/DefaultLoopPolicies.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.loop/src/org/graalvm/compiler/loop/DefaultLoopPolicies.java @@ -55,16 +55,19 @@ import org.graalvm.compiler.options.OptionValues; import jdk.vm.ci.meta.MetaAccessProvider; public class DefaultLoopPolicies implements LoopPolicies { - @Option(help = "", type = OptionType.Expert) public static final OptionKey LoopUnswitchMaxIncrease = new OptionKey<>(500); - @Option(help = "", type = OptionType.Expert) public static final OptionKey LoopUnswitchTrivial = new OptionKey<>(10); - @Option(help = "", type = OptionType.Expert) public static final OptionKey LoopUnswitchFrequencyBoost = new OptionKey<>(10.0); - @Option(help = "", type = OptionType.Expert) public static final OptionKey FullUnrollMaxNodes = new OptionKey<>(300); - @Option(help = "", type = OptionType.Expert) public static final OptionKey FullUnrollMaxIterations = new OptionKey<>(600); - @Option(help = "", type = OptionType.Expert) public static final OptionKey ExactFullUnrollMaxNodes = new OptionKey<>(1200); - @Option(help = "", type = OptionType.Expert) public static final OptionKey ExactPartialUnrollMaxNodes = new OptionKey<>(200); + public static class Options { + @Option(help = "", type = OptionType.Expert) public static final OptionKey LoopUnswitchMaxIncrease = new OptionKey<>(500); + @Option(help = "", type = OptionType.Expert) public static final OptionKey LoopUnswitchTrivial = new OptionKey<>(10); + @Option(help = "", type = OptionType.Expert) public static final OptionKey LoopUnswitchFrequencyBoost = new OptionKey<>(10.0); - @Option(help = "", type = OptionType.Expert) public static final OptionKey UnrollMaxIterations = new OptionKey<>(16); + @Option(help = "", type = OptionType.Expert) public static final OptionKey FullUnrollMaxNodes = new OptionKey<>(300); + @Option(help = "", type = OptionType.Expert) public static final OptionKey FullUnrollMaxIterations = new OptionKey<>(600); + @Option(help = "", type = OptionType.Expert) public static final OptionKey ExactFullUnrollMaxNodes = new OptionKey<>(1200); + @Option(help = "", type = OptionType.Expert) public static final OptionKey ExactPartialUnrollMaxNodes = new OptionKey<>(200); + + @Option(help = "", type = OptionType.Expert) public static final OptionKey UnrollMaxIterations = new OptionKey<>(16); + } @Override public boolean shouldPeel(LoopEx loop, ControlFlowGraph cfg, MetaAccessProvider metaAccess) { @@ -87,10 +90,10 @@ public class DefaultLoopPolicies implements LoopPolicies { OptionValues options = loop.entryPoint().getOptions(); CountedLoopInfo counted = loop.counted(); long maxTrips = counted.constantMaxTripCount(); - int maxNodes = (counted.isExactTripCount() && counted.isConstantExactTripCount()) ? ExactFullUnrollMaxNodes.getValue(options) : FullUnrollMaxNodes.getValue(options); + int maxNodes = (counted.isExactTripCount() && counted.isConstantExactTripCount()) ? Options.ExactFullUnrollMaxNodes.getValue(options) : Options.FullUnrollMaxNodes.getValue(options); maxNodes = Math.min(maxNodes, Math.max(0, MaximumDesiredSize.getValue(options) - loop.loopBegin().graph().getNodeCount())); int size = Math.max(1, loop.size() - 1 - loop.loopBegin().phis().count()); - if (maxTrips <= FullUnrollMaxIterations.getValue(options) && size * (maxTrips - 1) <= maxNodes) { + if (maxTrips <= Options.FullUnrollMaxIterations.getValue(options) && size * (maxTrips - 1) <= maxNodes) { // check whether we're allowed to unroll this loop return loop.canDuplicateLoop(); } else { @@ -106,7 +109,7 @@ public class DefaultLoopPolicies implements LoopPolicies { return false; } OptionValues options = loop.entryPoint().getOptions(); - int maxNodes = ExactPartialUnrollMaxNodes.getValue(options); + int maxNodes = Options.ExactPartialUnrollMaxNodes.getValue(options); maxNodes = Math.min(maxNodes, Math.max(0, MaximumDesiredSize.getValue(options) - loop.loopBegin().graph().getNodeCount())); int size = Math.max(1, loop.size() - 1 - loop.loopBegin().phis().count()); int unrollFactor = loopBegin.getUnrollFactor(); @@ -118,7 +121,7 @@ public class DefaultLoopPolicies implements LoopPolicies { } loopBegin.setLoopOrigFrequency(loopFrequency); } - int maxUnroll = UnrollMaxIterations.getValue(options); + int maxUnroll = Options.UnrollMaxIterations.getValue(options); // Now correct size for the next unroll. UnrollMaxIterations == 1 means perform the // pre/main/post transformation but don't actually unroll the main loop. size += size; @@ -190,9 +193,9 @@ public class DefaultLoopPolicies implements LoopPolicies { CountingClosure stateNodesCount = new CountingClosure(); double loopFrequency = loop.loopBegin().loopFrequency(); OptionValues options = loop.loopBegin().getOptions(); - int maxDiff = LoopUnswitchTrivial.getValue(options) + (int) (LoopUnswitchFrequencyBoost.getValue(options) * (loopFrequency - 1.0 + phis)); + int maxDiff = Options.LoopUnswitchTrivial.getValue(options) + (int) (Options.LoopUnswitchFrequencyBoost.getValue(options) * (loopFrequency - 1.0 + phis)); - maxDiff = Math.min(maxDiff, LoopUnswitchMaxIncrease.getValue(options)); + maxDiff = Math.min(maxDiff, Options.LoopUnswitchMaxIncrease.getValue(options)); int remainingGraphSpace = MaximumDesiredSize.getValue(options) - graph.getNodeCount(); maxDiff = Math.min(maxDiff, remainingGraphSpace); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodeinfo/src/org/graalvm/compiler/nodeinfo/NodeCycles.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodeinfo/src/org/graalvm/compiler/nodeinfo/NodeCycles.java index 13857f3f4e3..34057c7631e 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodeinfo/src/org/graalvm/compiler/nodeinfo/NodeCycles.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodeinfo/src/org/graalvm/compiler/nodeinfo/NodeCycles.java @@ -68,6 +68,10 @@ public enum NodeCycles { this.value = value; } + public boolean isValueKnown() { + return this != NodeCycles.CYCLES_UNKNOWN && this != NodeCycles.CYCLES_UNSET; + } + public static final int IGNORE_CYCLES_CONTRACT_FACTOR = 0xFFFF; public static NodeCycles compute(NodeCycles base, int opCount) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes.test/src/org/graalvm/compiler/nodes/test/IntegerStampTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes.test/src/org/graalvm/compiler/nodes/test/IntegerStampTest.java index bf22e803e0e..04832ecd1a3 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes.test/src/org/graalvm/compiler/nodes/test/IntegerStampTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes.test/src/org/graalvm/compiler/nodes/test/IntegerStampTest.java @@ -552,4 +552,21 @@ public class IntegerStampTest extends GraphTest { return result.longValue(); } } + + @Test + public void testDiv() { + testDiv(32, Integer.MIN_VALUE, Integer.MAX_VALUE); + testDiv(64, Long.MIN_VALUE, Long.MAX_VALUE); + } + + private static void testDiv(int bits, long min, long max) { + BinaryOp div = IntegerStamp.OPS.getDiv(); + assertEquals(IntegerStamp.create(bits, -50, 50), div.foldStamp(IntegerStamp.create(bits, -100, 100), IntegerStamp.create(bits, 2, 5))); + assertEquals(IntegerStamp.create(bits, 20, 500), div.foldStamp(IntegerStamp.create(bits, 100, 1000), IntegerStamp.create(bits, 2, 5))); + assertEquals(IntegerStamp.create(bits, -500, -20), div.foldStamp(IntegerStamp.create(bits, -1000, -100), IntegerStamp.create(bits, 2, 5))); + assertEquals(IntegerStamp.create(bits, min, max), div.foldStamp(IntegerStamp.create(bits, min, max), IntegerStamp.create(bits, 1, max))); + assertEquals(IntegerStamp.create(bits, -100, 100), div.foldStamp(IntegerStamp.create(bits, -100, 100), IntegerStamp.create(bits, 1, max))); + assertEquals(IntegerStamp.create(bits, 0, 1000), div.foldStamp(IntegerStamp.create(bits, 100, 1000), IntegerStamp.create(bits, 1, max))); + assertEquals(IntegerStamp.create(bits, -1000, 0), div.foldStamp(IntegerStamp.create(bits, -1000, -100), IntegerStamp.create(bits, 1, max))); + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes.test/src/org/graalvm/compiler/nodes/test/PrimitiveStampBoundaryTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes.test/src/org/graalvm/compiler/nodes/test/PrimitiveStampBoundaryTest.java new file mode 100644 index 00000000000..c8ed8a5816d --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes.test/src/org/graalvm/compiler/nodes/test/PrimitiveStampBoundaryTest.java @@ -0,0 +1,326 @@ +/* + * Copyright (c) 2012, 2015, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.nodes.test; + +import java.util.EnumSet; +import java.util.HashSet; + +import org.graalvm.compiler.core.common.calc.FloatConvert; +import org.graalvm.compiler.core.common.calc.FloatConvertCategory; +import org.graalvm.compiler.core.common.type.ArithmeticOpTable; +import org.graalvm.compiler.core.common.type.ArithmeticOpTable.BinaryOp; +import org.graalvm.compiler.core.common.type.ArithmeticOpTable.IntegerConvertOp; +import org.graalvm.compiler.core.common.type.ArithmeticOpTable.ShiftOp; +import org.graalvm.compiler.core.common.type.FloatStamp; +import org.graalvm.compiler.core.common.type.IntegerStamp; +import org.graalvm.compiler.core.common.type.PrimitiveStamp; +import org.graalvm.compiler.core.common.type.Stamp; +import org.graalvm.compiler.test.GraalTest; +import org.junit.Test; + +import jdk.vm.ci.meta.Constant; +import jdk.vm.ci.meta.JavaConstant; +import jdk.vm.ci.meta.JavaKind; + +/** + * Exercise the various stamp folding operations by generating ranges from a set of boundary values + * and then ensuring that the values that produced those ranges are in the resulting stamp. + */ +public class PrimitiveStampBoundaryTest extends GraalTest { + + static long[] longBoundaryValues = {Long.MIN_VALUE, Long.MIN_VALUE + 1, Integer.MIN_VALUE, Integer.MIN_VALUE + 1, -1, 0, 1, Integer.MAX_VALUE - 1, Integer.MAX_VALUE, Long.MAX_VALUE - 1, + Long.MAX_VALUE}; + + static int[] shiftBoundaryValues = {-128, -1, 0, 1, 4, 8, 16, 31, 63, 128}; + + static HashSet shiftStamps; + static HashSet integerTestStamps; + static HashSet floatTestStamps; + + static { + shiftStamps = new HashSet<>(); + for (long v1 : shiftBoundaryValues) { + for (long v2 : shiftBoundaryValues) { + shiftStamps.add(IntegerStamp.create(32, Math.min(v1, v2), Math.max(v1, v2))); + } + } + + integerTestStamps = new HashSet<>(); + for (long v1 : longBoundaryValues) { + for (long v2 : longBoundaryValues) { + if (v2 == (int) v2 && v1 == (int) v1) { + integerTestStamps.add(IntegerStamp.create(32, Math.min(v1, v2), Math.max(v1, v2))); + } + integerTestStamps.add(IntegerStamp.create(64, Math.min(v1, v2), Math.max(v1, v2))); + } + } + } + + static double[] doubleBoundaryValues = {Double.NEGATIVE_INFINITY, Double.MIN_VALUE, Float.NEGATIVE_INFINITY, Float.MIN_VALUE, + Long.MIN_VALUE, Long.MIN_VALUE + 1, Integer.MIN_VALUE, Integer.MIN_VALUE + 1, -1, 0, 1, + Integer.MAX_VALUE - 1, Integer.MAX_VALUE, Long.MAX_VALUE - 1, Long.MAX_VALUE, + Float.MAX_VALUE, Float.POSITIVE_INFINITY, Double.MAX_VALUE, Double.POSITIVE_INFINITY}; + + static double[] doubleSpecialValues = {Double.NaN, -0.0, -0.0F, Float.NaN}; + + static { + floatTestStamps = new HashSet<>(); + + for (double d1 : doubleBoundaryValues) { + for (double d2 : doubleBoundaryValues) { + float f1 = (float) d2; + float f2 = (float) d1; + if (d2 == f1 && d1 == f2) { + generateFloatingStamps(new FloatStamp(32, Math.min(f2, f1), Math.max(f2, f1), true)); + generateFloatingStamps(new FloatStamp(32, Math.min(f2, f1), Math.max(f2, f1), false)); + } + generateFloatingStamps(new FloatStamp(64, Math.min(d1, d2), Math.max(d1, d2), true)); + generateFloatingStamps(new FloatStamp(64, Math.min(d1, d2), Math.max(d1, d2), false)); + } + } + } + + private static void generateFloatingStamps(FloatStamp floatStamp) { + floatTestStamps.add(floatStamp); + for (double d : doubleSpecialValues) { + FloatStamp newStamp = (FloatStamp) floatStamp.meet(floatStampForConstant(d, floatStamp.getBits())); + if (!newStamp.isUnrestricted()) { + floatTestStamps.add(newStamp); + } + } + } + + @Test + public void testConvertBoundaryValues() { + testConvertBoundaryValues(IntegerStamp.OPS.getSignExtend(), 32, 64, integerTestStamps); + testConvertBoundaryValues(IntegerStamp.OPS.getZeroExtend(), 32, 64, integerTestStamps); + testConvertBoundaryValues(IntegerStamp.OPS.getNarrow(), 64, 32, integerTestStamps); + } + + private static void testConvertBoundaryValues(IntegerConvertOp op, int inputBits, int resultBits, HashSet stamps) { + for (PrimitiveStamp stamp : stamps) { + if (inputBits == stamp.getBits()) { + Stamp lower = boundaryStamp(stamp, false); + Stamp upper = boundaryStamp(stamp, true); + checkConvertOperation(op, inputBits, resultBits, op.foldStamp(inputBits, resultBits, stamp), lower); + checkConvertOperation(op, inputBits, resultBits, op.foldStamp(inputBits, resultBits, stamp), upper); + } + } + } + + private static void checkConvertOperation(IntegerConvertOp op, int inputBits, int resultBits, Stamp result, Stamp v1stamp) { + Stamp folded = op.foldStamp(inputBits, resultBits, v1stamp); + assertTrue(folded.asConstant() != null, "should constant fold %s %s %s", op, v1stamp, folded); + assertTrue(result.meet(folded).equals(result), "result out of range %s %s %s %s %s", op, v1stamp, folded, result, result.meet(folded)); + } + + @Test + public void testFloatConvertBoundaryValues() { + for (FloatConvert op : EnumSet.allOf(FloatConvert.class)) { + ArithmeticOpTable.FloatConvertOp floatConvert = IntegerStamp.OPS.getFloatConvert(op); + if (floatConvert == null) { + continue; + } + assert op.getCategory() == FloatConvertCategory.IntegerToFloatingPoint : op; + testConvertBoundaryValues(floatConvert, op.getInputBits(), integerTestStamps); + } + for (FloatConvert op : EnumSet.allOf(FloatConvert.class)) { + ArithmeticOpTable.FloatConvertOp floatConvert = FloatStamp.OPS.getFloatConvert(op); + if (floatConvert == null) { + continue; + } + assert op.getCategory() == FloatConvertCategory.FloatingPointToInteger || op.getCategory() == FloatConvertCategory.FloatingPointToFloatingPoint : op; + testConvertBoundaryValues(floatConvert, op.getInputBits(), floatTestStamps); + } + } + + private static void testConvertBoundaryValues(ArithmeticOpTable.FloatConvertOp op, int bits, HashSet stamps) { + for (PrimitiveStamp stamp : stamps) { + if (bits == stamp.getBits()) { + Stamp lower = boundaryStamp(stamp, false); + Stamp upper = boundaryStamp(stamp, true); + checkConvertOperation(op, op.foldStamp(stamp), lower); + checkConvertOperation(op, op.foldStamp(stamp), upper); + } + } + } + + private static void checkConvertOperation(ArithmeticOpTable.FloatConvertOp op, Stamp result, Stamp v1stamp) { + Stamp folded = op.foldStamp(v1stamp); + assertTrue(folded.asConstant() != null, "should constant fold %s %s %s", op, v1stamp, folded); + assertTrue(result.meet(folded).equals(result), "result out of range %s %s %s %s %s", op, v1stamp, folded, result, result.meet(folded)); + } + + @Test + public void testShiftBoundaryValues() { + for (ShiftOp op : IntegerStamp.OPS.getShiftOps()) { + testShiftBoundaryValues(op, integerTestStamps, shiftStamps); + } + } + + private static void testShiftBoundaryValues(ShiftOp shiftOp, HashSet stamps, HashSet shifts) { + for (PrimitiveStamp testStamp : stamps) { + if (testStamp instanceof IntegerStamp) { + IntegerStamp stamp = (IntegerStamp) testStamp; + for (IntegerStamp shiftStamp : shifts) { + IntegerStamp foldedStamp = (IntegerStamp) shiftOp.foldStamp(stamp, shiftStamp); + checkShiftOperation(stamp.getBits(), shiftOp, foldedStamp, stamp.lowerBound(), shiftStamp.lowerBound()); + checkShiftOperation(stamp.getBits(), shiftOp, foldedStamp, stamp.lowerBound(), shiftStamp.upperBound()); + checkShiftOperation(stamp.getBits(), shiftOp, foldedStamp, stamp.upperBound(), shiftStamp.lowerBound()); + checkShiftOperation(stamp.getBits(), shiftOp, foldedStamp, stamp.upperBound(), shiftStamp.upperBound()); + } + } + } + } + + private static void checkShiftOperation(int bits, ShiftOp op, IntegerStamp result, long v1, long v2) { + IntegerStamp v1stamp = IntegerStamp.create(bits, v1, v1); + IntegerStamp v2stamp = IntegerStamp.create(32, v2, v2); + IntegerStamp folded = (IntegerStamp) op.foldStamp(v1stamp, v2stamp); + Constant constant = op.foldConstant(JavaConstant.forPrimitiveInt(bits, v1), (int) v2); + assertTrue(constant != null); + assertTrue(folded.asConstant() != null, "should constant fold %s %s %s %s", op, v1stamp, v2stamp, folded); + assertTrue(result.meet(folded).equals(result), "result out of range %s %s %s %s %s %s", op, v1stamp, v2stamp, folded, result, result.meet(folded)); + } + + private static void checkBinaryOperation(ArithmeticOpTable.BinaryOp op, Stamp result, Stamp v1stamp, Stamp v2stamp) { + Stamp folded = op.foldStamp(v1stamp, v2stamp); + Constant constant = op.foldConstant(v1stamp.asConstant(), v2stamp.asConstant()); + if (constant != null) { + Constant constant2 = folded.asConstant(); + if (constant2 == null && v1stamp instanceof FloatStamp) { + JavaConstant c = (JavaConstant) constant; + assertTrue((c.getJavaKind() == JavaKind.Double && Double.isNaN(c.asDouble())) || + (c.getJavaKind() == JavaKind.Float && Float.isNaN(c.asFloat()))); + } else { + assertTrue(constant2 != null, "should constant fold %s %s %s %s", op, v1stamp, v2stamp, folded); + if (!constant.equals(constant2)) { + op.foldConstant(v1stamp.asConstant(), v2stamp.asConstant()); + op.foldStamp(v1stamp, v2stamp); + } + assertTrue(constant.equals(constant2), "should produce same constant %s %s %s %s %s", op, v1stamp, v2stamp, constant, constant2); + } + assertTrue(result.meet(folded).equals(result), "result out of range %s %s %s %s %s %s", op, v1stamp, v2stamp, folded, result, result.meet(folded)); + } + } + + @Test + public void testBinaryBoundaryValues() { + for (BinaryOp op : IntegerStamp.OPS.getBinaryOps()) { + if (op != null) { + testBinaryBoundaryValues(op, integerTestStamps); + } + } + for (BinaryOp op : FloatStamp.OPS.getBinaryOps()) { + if (op != null) { + testBinaryBoundaryValues(op, floatTestStamps); + } + } + } + + private static Stamp boundaryStamp(Stamp v1, boolean upper) { + if (v1 instanceof IntegerStamp) { + IntegerStamp istamp = (IntegerStamp) v1; + long bound = upper ? istamp.upperBound() : istamp.lowerBound(); + return IntegerStamp.create(istamp.getBits(), bound, bound); + } else if (v1 instanceof FloatStamp) { + FloatStamp floatStamp = (FloatStamp) v1; + double bound = upper ? floatStamp.upperBound() : floatStamp.lowerBound(); + int bits = floatStamp.getBits(); + return floatStampForConstant(bound, bits); + } else { + throw new InternalError("unexpected stamp type " + v1); + } + } + + private static FloatStamp floatStampForConstant(double bound, int bits) { + if (bits == 32) { + float fbound = (float) bound; + return new FloatStamp(bits, fbound, fbound, !Float.isNaN(fbound)); + } else { + return new FloatStamp(bits, bound, bound, !Double.isNaN(bound)); + } + } + + private static void testBinaryBoundaryValues(ArithmeticOpTable.BinaryOp op, HashSet stamps) { + for (PrimitiveStamp v1 : stamps) { + for (PrimitiveStamp v2 : stamps) { + if (v1.getBits() == v2.getBits() && v1.getClass() == v2.getClass()) { + Stamp result = op.foldStamp(v1, v2); + Stamp v1lower = boundaryStamp(v1, false); + Stamp v1upper = boundaryStamp(v1, true); + Stamp v2lower = boundaryStamp(v2, false); + Stamp v2upper = boundaryStamp(v2, true); + checkBinaryOperation(op, result, v1lower, v2lower); + checkBinaryOperation(op, result, v1lower, v2upper); + checkBinaryOperation(op, result, v1upper, v2lower); + checkBinaryOperation(op, result, v1upper, v2upper); + } + } + } + } + + @Test + public void testUnaryBoundaryValues() { + for (ArithmeticOpTable.UnaryOp op : IntegerStamp.OPS.getUnaryOps()) { + if (op != null) { + testUnaryBoundaryValues(op, integerTestStamps); + } + } + for (ArithmeticOpTable.UnaryOp op : FloatStamp.OPS.getUnaryOps()) { + if (op != null) { + testUnaryBoundaryValues(op, floatTestStamps); + } + } + } + + private static void testUnaryBoundaryValues(ArithmeticOpTable.UnaryOp op, HashSet stamps) { + for (PrimitiveStamp v1 : stamps) { + Stamp result = op.foldStamp(v1); + checkUnaryOperation(op, result, boundaryStamp(v1, false)); + checkUnaryOperation(op, result, boundaryStamp(v1, true)); + } + } + + private static void checkUnaryOperation(ArithmeticOpTable.UnaryOp op, Stamp result, Stamp v1stamp) { + Stamp folded = op.foldStamp(v1stamp); + Constant v1constant = v1stamp.asConstant(); + if (v1constant != null) { + Constant constant = op.foldConstant(v1constant); + if (constant != null) { + Constant constant2 = folded.asConstant(); + if (constant2 == null && v1stamp instanceof FloatStamp) { + JavaConstant c = (JavaConstant) constant; + assertTrue((c.getJavaKind() == JavaKind.Double && Double.isNaN(c.asDouble())) || + (c.getJavaKind() == JavaKind.Float && Float.isNaN(c.asFloat()))); + } else { + assertTrue(constant2 != null, "should constant fold %s %s %s", op, v1stamp, folded); + assertTrue(constant.equals(constant2), "should produce same constant %s %s %s %s", op, v1stamp, constant, constant2); + } + } + } else { + assert v1stamp instanceof FloatStamp; + } + assertTrue(result.meet(folded).equals(result), "result out of range %s %s %s %s %s", op, v1stamp, folded, result, result.meet(folded)); + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/CompressionNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/CompressionNode.java index f5fb7981032..6f9a420af59 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/CompressionNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/CompressionNode.java @@ -122,9 +122,9 @@ public abstract class CompressionNode extends UnaryNode implements ConvertNode, // We always want uncompressed constants return this; } - int stableDimension = ((ConstantNode) forValue).getStableDimension(); - boolean isDefaultStable = ((ConstantNode) forValue).isDefaultStable(); - return ConstantNode.forConstant(stamp(), convert(forValue.asConstant(), tool.getConstantReflection()), stableDimension, isDefaultStable, tool.getMetaAccess()); + + ConstantNode constant = (ConstantNode) forValue; + return ConstantNode.forConstant(stamp(), convert(constant.getValue(), tool.getConstantReflection()), constant.getStableDimension(), constant.isDefaultStable(), tool.getMetaAccess()); } else if (forValue instanceof CompressionNode) { CompressionNode other = (CompressionNode) forValue; if (op != other.op && encoding.equals(other.encoding)) { @@ -136,22 +136,22 @@ public abstract class CompressionNode extends UnaryNode implements ConvertNode, @Override public void generate(NodeLIRBuilderTool gen) { - LIRGeneratorTool hsGen = gen.getLIRGeneratorTool(); boolean nonNull; - if (getValue().stamp() instanceof AbstractObjectStamp) { - nonNull = StampTool.isPointerNonNull(getValue().stamp()); + if (value.stamp() instanceof AbstractObjectStamp) { + nonNull = StampTool.isPointerNonNull(value.stamp()); } else { // metaspace pointers are never null nonNull = true; } + LIRGeneratorTool tool = gen.getLIRGeneratorTool(); Value result; switch (op) { case Compress: - result = hsGen.emitCompress(gen.operand(getValue()), encoding, nonNull); + result = tool.emitCompress(gen.operand(value), encoding, nonNull); break; case Uncompress: - result = hsGen.emitUncompress(gen.operand(getValue()), encoding, nonNull); + result = tool.emitUncompress(gen.operand(value), encoding, nonNull); break; default: throw GraalError.shouldNotReachHere(); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/ConstantNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/ConstantNode.java index 73c807f7495..4dd797fe32c 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/ConstantNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/ConstantNode.java @@ -54,7 +54,7 @@ import jdk.vm.ci.meta.PrimitiveConstant; /** * The {@code ConstantNode} represents a {@link Constant constant}. */ -@NodeInfo(nameTemplate = "C({p#rawvalue})", cycles = CYCLES_0, size = SIZE_1) +@NodeInfo(nameTemplate = "C({p#rawvalue}) {p#stampKind}", cycles = CYCLES_0, size = SIZE_1) public final class ConstantNode extends FloatingNode implements LIRLowerable { public static final NodeClass TYPE = NodeClass.create(ConstantNode.class); @@ -518,13 +518,14 @@ public final class ConstantNode extends FloatingNode implements LIRLowerable { public Map getDebugProperties(Map map) { Map properties = super.getDebugProperties(map); properties.put("rawvalue", value.toValueString()); + properties.put("stampKind", stamp.unrestricted().toString()); return properties; } @Override public String toString(Verbosity verbosity) { if (verbosity == Verbosity.Name) { - return super.toString(Verbosity.Name) + "(" + value.toValueString() + ")"; + return super.toString(Verbosity.Name) + "(" + value.toValueString() + ", " + stamp().unrestricted().toString() + ")"; } else { return super.toString(verbosity); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/IfNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/IfNode.java index dba8f7295b2..616b1de0ac1 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/IfNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/IfNode.java @@ -22,7 +22,7 @@ */ package org.graalvm.compiler.nodes; -import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_2; +import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_1; import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_2; import java.util.ArrayList; @@ -74,7 +74,7 @@ import jdk.vm.ci.meta.PrimitiveConstant; * The {@code IfNode} represents a branch that can go one of two directions depending on the outcome * of a comparison. */ -@NodeInfo(cycles = CYCLES_2, size = SIZE_2, sizeRationale = "2 jmps") +@NodeInfo(cycles = CYCLES_1, size = SIZE_2, sizeRationale = "2 jmps") public final class IfNode extends ControlSplitNode implements Simplifiable, LIRLowerable { public static final NodeClass TYPE = NodeClass.create(IfNode.class); @@ -375,7 +375,7 @@ public final class IfNode extends ControlSplitNode implements Simplifiable, LIRL } // Falsify the reference check. - setCondition(graph().addOrUnique(LogicConstantNode.contradiction())); + setCondition(graph().addOrUniqueWithInputs(LogicConstantNode.contradiction())); return true; } @@ -726,10 +726,11 @@ public final class IfNode extends ControlSplitNode implements Simplifiable, LIRL protected void removeThroughFalseBranch(SimplifierTool tool, AbstractMergeNode merge) { AbstractBeginNode trueBegin = trueSuccessor(); + LogicNode conditionNode = condition(); graph().removeSplitPropagate(this, trueBegin); tool.addToWorkList(trueBegin); - if (condition() != null) { - GraphUtil.tryKillUnused(condition()); + if (conditionNode != null) { + GraphUtil.tryKillUnused(conditionNode); } if (merge.isAlive() && merge.forwardEndCount() > 1) { for (FixedNode end : merge.forwardEnds()) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/PhiNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/PhiNode.java index 8ee5706389f..2d517d17ba5 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/PhiNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/PhiNode.java @@ -132,12 +132,24 @@ public abstract class PhiNode extends FloatingNode implements Canonicalizable { } str.append(valueAt(i) == null ? "-" : valueAt(i).toString(Verbosity.Id)); } + String description = valueDescription(); + if (description.length() > 0) { + str.append(", ").append(description); + } return super.toString(Verbosity.Name) + "(" + str + ")"; } else { return super.toString(verbosity); } } + /** + * String describing the kind of value this Phi merges. Used by {@link #toString(Verbosity)} and + * dumping. + */ + protected String valueDescription() { + return ""; + } + public void addInput(ValueNode x) { assert !(x instanceof ValuePhiNode) || ((ValuePhiNode) x).merge() instanceof LoopBeginNode || ((ValuePhiNode) x).merge() != this.merge(); assert !(this instanceof ValuePhiNode) || x.stamp().isCompatible(stamp()); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/PiNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/PiNode.java index 55a2cd06297..2b0207c53b5 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/PiNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/PiNode.java @@ -59,14 +59,14 @@ import jdk.vm.ci.meta.ResolvedJavaType; * * In contrast to a {@link GuardedValueNode}, a {@link PiNode} is useless as soon as the type of its * input is as narrow or narrower than the {@link PiNode}'s type. The {@link PiNode}, and therefore - * also the scheduling restriction enforced by the anchor, will go away. + * also the scheduling restriction enforced by the guard, will go away. */ @NodeInfo(cycles = CYCLES_0, size = SIZE_0) public class PiNode extends FloatingGuardedNode implements LIRLowerable, Virtualizable, IterableNodeType, Canonicalizable, ValueProxy { public static final NodeClass TYPE = NodeClass.create(PiNode.class); @Input ValueNode object; - protected final Stamp piStamp; + protected Stamp piStamp; public ValueNode object() { return object; @@ -84,12 +84,12 @@ public class PiNode extends FloatingGuardedNode implements LIRLowerable, Virtual this(object, stamp, null); } - public PiNode(ValueNode object, Stamp stamp, ValueNode anchor) { - this(TYPE, object, stamp, (GuardingNode) anchor); + public PiNode(ValueNode object, Stamp stamp, ValueNode guard) { + this(TYPE, object, stamp, (GuardingNode) guard); } - public PiNode(ValueNode object, ValueNode anchor) { - this(object, AbstractPointerStamp.pointerNonNull(object.stamp()), anchor); + public PiNode(ValueNode object, ValueNode guard) { + this(object, AbstractPointerStamp.pointerNonNull(object.stamp()), guard); } public PiNode(ValueNode object, ResolvedJavaType toType, boolean exactType, boolean nonNull) { @@ -104,29 +104,29 @@ public class PiNode extends FloatingGuardedNode implements LIRLowerable, Virtual return new PiNode(object, stamp); } - public static ValueNode create(ValueNode object, Stamp stamp, ValueNode anchor) { - ValueNode value = canonical(object, stamp, (GuardingNode) anchor); + public static ValueNode create(ValueNode object, Stamp stamp, ValueNode guard) { + ValueNode value = canonical(object, stamp, (GuardingNode) guard); if (value != null) { return value; } - return new PiNode(object, stamp, anchor); + return new PiNode(object, stamp, guard); } - public static ValueNode create(ValueNode object, ValueNode anchor) { + public static ValueNode create(ValueNode object, ValueNode guard) { Stamp stamp = AbstractPointerStamp.pointerNonNull(object.stamp()); - ValueNode value = canonical(object, stamp, (GuardingNode) anchor); + ValueNode value = canonical(object, stamp, (GuardingNode) guard); if (value != null) { return value; } - return new PiNode(object, stamp, anchor); + return new PiNode(object, stamp, guard); } @SuppressWarnings("unused") - public static boolean intrinsify(GraphBuilderContext b, ResolvedJavaMethod method, ValueNode object, ValueNode anchor) { + public static boolean intrinsify(GraphBuilderContext b, ResolvedJavaMethod method, ValueNode object, ValueNode guard) { Stamp stamp = AbstractPointerStamp.pointerNonNull(object.stamp()); - ValueNode value = canonical(object, stamp, (GuardingNode) anchor); + ValueNode value = canonical(object, stamp, (GuardingNode) guard); if (value == null) { - value = new PiNode(object, stamp, anchor); + value = new PiNode(object, stamp, guard); } b.push(JavaKind.Object, b.append(value)); return true; @@ -147,6 +147,11 @@ public class PiNode extends FloatingGuardedNode implements LIRLowerable, Virtual return piStamp; } + public void strengthenPiStamp(Stamp newPiStamp) { + assert this.piStamp.join(newPiStamp).equals(newPiStamp) : "stamp can only improve"; + this.piStamp = newPiStamp; + } + @Override public void generate(NodeLIRBuilderTool generator) { if (generator.hasOperand(object)) { @@ -256,17 +261,17 @@ public class PiNode extends FloatingGuardedNode implements LIRLowerable, Virtual /** * Changes the stamp of an object and ensures the newly stamped value is non-null and does not - * float above a given anchor. + * float above a given guard. */ @NodeIntrinsic - public static native Object piCastNonNull(Object object, GuardingNode anchor); + public static native Object piCastNonNull(Object object, GuardingNode guard); /** * Changes the stamp of an object and ensures the newly stamped value is non-null and does not - * float above a given anchor. + * float above a given guard. */ @NodeIntrinsic - public static native Class piCastNonNullClass(Class type, GuardingNode anchor); + public static native Class piCastNonNullClass(Class type, GuardingNode guard); /** * Changes the stamp of an object to represent a given type and to indicate that the object is diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/ValuePhiNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/ValuePhiNode.java index 71f1357a493..8a60c072cdd 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/ValuePhiNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/ValuePhiNode.java @@ -22,6 +22,8 @@ */ package org.graalvm.compiler.nodes; +import java.util.Map; + import org.graalvm.compiler.core.common.type.Stamp; import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.graph.NodeClass; @@ -35,7 +37,7 @@ import org.graalvm.util.CollectionsUtil; /** * Value {@link PhiNode}s merge data flow values at control flow merges. */ -@NodeInfo(nameTemplate = "Phi({i#values})") +@NodeInfo(nameTemplate = "Phi({i#values}, {p#valueDescription})") public class ValuePhiNode extends PhiNode implements ArrayLengthProvider { public static final NodeClass TYPE = NodeClass.create(ValuePhiNode.class); @@ -113,4 +115,16 @@ public class ValuePhiNode extends PhiNode implements ArrayLengthProvider { } return super.verify(); } + + @Override + protected String valueDescription() { + return stamp().unrestricted().toString(); + } + + @Override + public Map getDebugProperties(Map map) { + Map properties = super.getDebugProperties(map); + properties.put("valueDescription", valueDescription()); + return properties; + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/BinaryArithmeticNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/BinaryArithmeticNode.java index 87fb4e0ca69..7593daecd74 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/BinaryArithmeticNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/BinaryArithmeticNode.java @@ -91,7 +91,9 @@ public abstract class BinaryArithmeticNode extends BinaryNode implements Ari public static ConstantNode tryConstantFold(BinaryOp op, ValueNode forX, ValueNode forY, Stamp stamp) { if (forX.isConstant() && forY.isConstant()) { Constant ret = op.foldConstant(forX.asConstant(), forY.asConstant()); - return ConstantNode.forPrimitive(stamp, ret); + if (ret != null) { + return ConstantNode.forPrimitive(stamp, ret); + } } return null; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/ConditionalNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/ConditionalNode.java index e1161d63417..c06eea37c9b 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/ConditionalNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/ConditionalNode.java @@ -22,7 +22,7 @@ */ package org.graalvm.compiler.nodes.calc; -import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_0; +import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_1; import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_2; import static org.graalvm.compiler.nodes.calc.CompareNode.createCompareNode; @@ -47,10 +47,10 @@ import org.graalvm.compiler.nodes.spi.NodeLIRBuilderTool; import jdk.vm.ci.meta.JavaConstant; /** - * The {@code ConditionalNode} class represents a comparison that yields one of two values. Note - * that these nodes are not built directly from the bytecode but are introduced by canonicalization. + * The {@code ConditionalNode} class represents a comparison that yields one of two (eagerly + * evaluated) values. */ -@NodeInfo(cycles = CYCLES_0, size = SIZE_2) +@NodeInfo(cycles = CYCLES_1, size = SIZE_2) public final class ConditionalNode extends FloatingNode implements Canonicalizable, LIRLowerable { public static final NodeClass TYPE = NodeClass.create(ConditionalNode.class); @@ -116,7 +116,6 @@ public final class ConditionalNode extends FloatingNode implements Canonicalizab valueStamp = valueStamp.join(bounds); } } - } return updateStamp(valueStamp); } @@ -145,49 +144,10 @@ public final class ConditionalNode extends FloatingNode implements Canonicalizab } public static ValueNode canonicalizeConditional(LogicNode condition, ValueNode trueValue, ValueNode falseValue, Stamp stamp) { - // this optimizes the case where a value from the range 0 - 1 is mapped to the range 0 - 1 - if (trueValue.isConstant() && falseValue.isConstant() && trueValue.stamp() instanceof IntegerStamp && falseValue.stamp() instanceof IntegerStamp) { - long constTrueValue = trueValue.asJavaConstant().asLong(); - long constFalseValue = falseValue.asJavaConstant().asLong(); - if (condition instanceof IntegerEqualsNode) { - IntegerEqualsNode equals = (IntegerEqualsNode) condition; - if (equals.getY().isConstant() && equals.getX().stamp() instanceof IntegerStamp) { - IntegerStamp equalsXStamp = (IntegerStamp) equals.getX().stamp(); - if (equalsXStamp.upMask() == 1) { - long equalsY = equals.getY().asJavaConstant().asLong(); - if (equalsY == 0) { - if (constTrueValue == 0 && constFalseValue == 1) { - // return x when: x == 0 ? 0 : 1; - return IntegerConvertNode.convertUnsigned(equals.getX(), stamp); - } else if (constTrueValue == 1 && constFalseValue == 0) { - // negate a boolean value via xor - return IntegerConvertNode.convertUnsigned(XorNode.create(equals.getX(), ConstantNode.forIntegerStamp(equals.getX().stamp(), 1)), stamp); - } - } else if (equalsY == 1) { - if (constTrueValue == 1 && constFalseValue == 0) { - // return x when: x == 1 ? 1 : 0; - return IntegerConvertNode.convertUnsigned(equals.getX(), stamp); - } else if (constTrueValue == 0 && constFalseValue == 1) { - // negate a boolean value via xor - return IntegerConvertNode.convertUnsigned(XorNode.create(equals.getX(), ConstantNode.forIntegerStamp(equals.getX().stamp(), 1)), stamp); - } - } - } - } - } else if (condition instanceof IntegerTestNode) { - // replace IntegerTestNode with AndNode for the following patterns: - // (value & 1) == 0 ? 0 : 1 - // (value & 1) == 1 ? 1 : 0 - IntegerTestNode integerTestNode = (IntegerTestNode) condition; - if (integerTestNode.getY().isConstant()) { - assert integerTestNode.getX().stamp() instanceof IntegerStamp; - long testY = integerTestNode.getY().asJavaConstant().asLong(); - if (testY == 1 && constTrueValue == 0 && constFalseValue == 1) { - return IntegerConvertNode.convertUnsigned(AndNode.create(integerTestNode.getX(), integerTestNode.getY()), stamp); - } - } - } + if (trueValue == falseValue) { + return trueValue; } + if (condition instanceof CompareNode && ((CompareNode) condition).isIdentityComparison()) { // optimize the pattern (x == y) ? x : y CompareNode compare = (CompareNode) condition; @@ -195,25 +155,87 @@ public final class ConditionalNode extends FloatingNode implements Canonicalizab return falseValue; } } - if (trueValue == falseValue) { - return trueValue; - } - if (condition instanceof IntegerLessThanNode && trueValue.stamp() instanceof IntegerStamp) { - /* - * Convert a conditional add ((x < 0) ? (x + y) : x) into (x + (y & (x >> (bits - 1)))) - * to avoid the test. - */ - IntegerLessThanNode lt = (IntegerLessThanNode) condition; - if (lt.getY().isConstant() && lt.getY().asConstant().isDefaultForKind()) { - if (falseValue == lt.getX()) { - if (trueValue instanceof AddNode) { - AddNode add = (AddNode) trueValue; - if (add.getX() == falseValue) { - int bits = ((IntegerStamp) trueValue.stamp()).getBits(); - ValueNode shift = new RightShiftNode(lt.getX(), ConstantNode.forIntegerBits(32, bits - 1)); - ValueNode and = new AndNode(shift, add.getY()); - return new AddNode(add.getX(), and); + if (trueValue.stamp() instanceof IntegerStamp) { + // check if the conditional is redundant + if (condition instanceof IntegerLessThanNode) { + IntegerLessThanNode lessThan = (IntegerLessThanNode) condition; + IntegerStamp falseValueStamp = (IntegerStamp) falseValue.stamp(); + IntegerStamp trueValueStamp = (IntegerStamp) trueValue.stamp(); + if (lessThan.getX() == trueValue && lessThan.getY() == falseValue) { + // return "x" for "x < y ? x : y" in case that we know "x <= y" + if (trueValueStamp.upperBound() <= falseValueStamp.lowerBound()) { + return trueValue; + } + } else if (lessThan.getX() == falseValue && lessThan.getY() == trueValue) { + // return "x" for "x < y ? y : x" in case that we know "x <= y" + if (falseValueStamp.upperBound() <= trueValueStamp.lowerBound()) { + return falseValue; + } + } + } + + // this optimizes the case where a value from the range 0 - 1 is mapped to the + // range 0 - 1 + if (trueValue.isConstant() && falseValue.isConstant()) { + long constTrueValue = trueValue.asJavaConstant().asLong(); + long constFalseValue = falseValue.asJavaConstant().asLong(); + if (condition instanceof IntegerEqualsNode) { + IntegerEqualsNode equals = (IntegerEqualsNode) condition; + if (equals.getY().isConstant() && equals.getX().stamp() instanceof IntegerStamp) { + IntegerStamp equalsXStamp = (IntegerStamp) equals.getX().stamp(); + if (equalsXStamp.upMask() == 1) { + long equalsY = equals.getY().asJavaConstant().asLong(); + if (equalsY == 0) { + if (constTrueValue == 0 && constFalseValue == 1) { + // return x when: x == 0 ? 0 : 1; + return IntegerConvertNode.convertUnsigned(equals.getX(), stamp); + } else if (constTrueValue == 1 && constFalseValue == 0) { + // negate a boolean value via xor + return IntegerConvertNode.convertUnsigned(XorNode.create(equals.getX(), ConstantNode.forIntegerStamp(equals.getX().stamp(), 1)), stamp); + } + } else if (equalsY == 1) { + if (constTrueValue == 1 && constFalseValue == 0) { + // return x when: x == 1 ? 1 : 0; + return IntegerConvertNode.convertUnsigned(equals.getX(), stamp); + } else if (constTrueValue == 0 && constFalseValue == 1) { + // negate a boolean value via xor + return IntegerConvertNode.convertUnsigned(XorNode.create(equals.getX(), ConstantNode.forIntegerStamp(equals.getX().stamp(), 1)), stamp); + } + } + } + } + } else if (condition instanceof IntegerTestNode) { + // replace IntegerTestNode with AndNode for the following patterns: + // (value & 1) == 0 ? 0 : 1 + // (value & 1) == 1 ? 1 : 0 + IntegerTestNode integerTestNode = (IntegerTestNode) condition; + if (integerTestNode.getY().isConstant()) { + assert integerTestNode.getX().stamp() instanceof IntegerStamp; + long testY = integerTestNode.getY().asJavaConstant().asLong(); + if (testY == 1 && constTrueValue == 0 && constFalseValue == 1) { + return IntegerConvertNode.convertUnsigned(AndNode.create(integerTestNode.getX(), integerTestNode.getY()), stamp); + } + } + } + } + + if (condition instanceof IntegerLessThanNode) { + /* + * Convert a conditional add ((x < 0) ? (x + y) : x) into (x + (y & (x >> (bits - + * 1)))) to avoid the test. + */ + IntegerLessThanNode lt = (IntegerLessThanNode) condition; + if (lt.getY().isConstant() && lt.getY().asConstant().isDefaultForKind()) { + if (falseValue == lt.getX()) { + if (trueValue instanceof AddNode) { + AddNode add = (AddNode) trueValue; + if (add.getX() == falseValue) { + int bits = ((IntegerStamp) trueValue.stamp()).getBits(); + ValueNode shift = new RightShiftNode(lt.getX(), ConstantNode.forIntegerBits(32, bits - 1)); + ValueNode and = new AndNode(shift, add.getY()); + return new AddNode(add.getX(), and); + } } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/DivNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/FloatDivNode.java similarity index 69% rename from src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/DivNode.java rename to src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/FloatDivNode.java index e7b0fa20ca3..195241a9c2a 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/DivNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/FloatDivNode.java @@ -25,6 +25,7 @@ package org.graalvm.compiler.nodes.calc; import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_32; import org.graalvm.compiler.core.common.type.ArithmeticOpTable; +import org.graalvm.compiler.core.common.type.FloatStamp; import org.graalvm.compiler.core.common.type.ArithmeticOpTable.BinaryOp; import org.graalvm.compiler.core.common.type.ArithmeticOpTable.BinaryOp.Div; import org.graalvm.compiler.core.common.type.Stamp; @@ -36,21 +37,20 @@ import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.ValueNode; import org.graalvm.compiler.nodes.spi.NodeLIRBuilderTool; -import jdk.vm.ci.code.CodeUtil; import jdk.vm.ci.meta.Constant; -import jdk.vm.ci.meta.PrimitiveConstant; @NodeInfo(shortName = "/", cycles = CYCLES_32) -public class DivNode extends BinaryArithmeticNode

{ +public class FloatDivNode extends BinaryArithmeticNode
{ - public static final NodeClass TYPE = NodeClass.create(DivNode.class); + public static final NodeClass TYPE = NodeClass.create(FloatDivNode.class); - public DivNode(ValueNode x, ValueNode y) { - super(TYPE, ArithmeticOpTable::getDiv, x, y); + public FloatDivNode(ValueNode x, ValueNode y) { + this(TYPE, x, y); } - protected DivNode(NodeClass c, ValueNode x, ValueNode y) { + protected FloatDivNode(NodeClass c, ValueNode x, ValueNode y) { super(c, ArithmeticOpTable::getDiv, x, y); + assert stamp instanceof FloatStamp; } public static ValueNode create(ValueNode x, ValueNode y) { @@ -73,33 +73,14 @@ public class DivNode extends BinaryArithmeticNode
{ return canonical(this, getOp(forX, forY), forX, forY); } - private static ValueNode canonical(DivNode self, BinaryOp
op, ValueNode forX, ValueNode forY) { + private static ValueNode canonical(FloatDivNode self, BinaryOp
op, ValueNode forX, ValueNode forY) { if (forY.isConstant()) { Constant c = forY.asConstant(); if (op.isNeutral(c)) { return forX; } - if (c instanceof PrimitiveConstant && ((PrimitiveConstant) c).getJavaKind().isNumericInteger()) { - long i = ((PrimitiveConstant) c).asLong(); - boolean signFlip = false; - if (i < 0) { - i = -i; - signFlip = true; - } - ValueNode divResult = null; - if (CodeUtil.isPowerOf2(i)) { - divResult = new RightShiftNode(forX, ConstantNode.forInt(CodeUtil.log2(i))); - } - if (divResult != null) { - if (signFlip) { - return NegateNode.create(divResult); - } else { - return divResult; - } - } - } } - return self != null ? self : new DivNode(forX, forY); + return self != null ? self : new FloatDivNode(forX, forY); } @Override diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IntegerConvertNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IntegerConvertNode.java index cf9e8df7aa1..90bfdfde901 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IntegerConvertNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IntegerConvertNode.java @@ -129,7 +129,7 @@ public abstract class IntegerConvertNode extends UnaryNode implements A ValueNode convert = convert(input, stamp, false); if (!convert.isAlive()) { assert !convert.isDeleted(); - convert = graph.addOrUnique(convert); + convert = graph.addOrUniqueWithInputs(convert); } return convert; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IntegerLessThanNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IntegerLessThanNode.java index 3c43b3ee629..bce2a2c39f3 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IntegerLessThanNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IntegerLessThanNode.java @@ -52,7 +52,7 @@ import org.graalvm.compiler.options.OptionValues; @NodeInfo(shortName = "<") public final class IntegerLessThanNode extends IntegerLowerThanNode { public static final NodeClass TYPE = NodeClass.create(IntegerLessThanNode.class); - public static final LessThanOp OP = new LessThanOp(); + private static final LessThanOp OP = new LessThanOp(); public IntegerLessThanNode(ValueNode x, ValueNode y) { super(TYPE, x, y, OP); @@ -203,49 +203,52 @@ public final class IntegerLessThanNode extends IntegerLowerThanNode { } } - int bits = ((IntegerStamp) forX.stamp()).getBits(); - assert ((IntegerStamp) forY.stamp()).getBits() == bits; - long min = OP.minValue(bits); - long xResidue = 0; - ValueNode left = null; - JavaConstant leftCst = null; - if (forX instanceof AddNode) { - AddNode xAdd = (AddNode) forX; - if (xAdd.getY().isJavaConstant()) { - long xCst = xAdd.getY().asJavaConstant().asLong(); - xResidue = xCst - min; - left = xAdd.getX(); - } - } else if (forX.isJavaConstant()) { - leftCst = forX.asJavaConstant(); - } - if (left != null || leftCst != null) { - long yResidue = 0; - ValueNode right = null; - JavaConstant rightCst = null; - if (forY instanceof AddNode) { - AddNode yAdd = (AddNode) forY; - if (yAdd.getY().isJavaConstant()) { - long yCst = yAdd.getY().asJavaConstant().asLong(); - yResidue = yCst - min; - right = yAdd.getX(); + if (forX.stamp() instanceof IntegerStamp) { + assert forY.stamp() instanceof IntegerStamp; + int bits = ((IntegerStamp) forX.stamp()).getBits(); + assert ((IntegerStamp) forY.stamp()).getBits() == bits; + long min = OP.minValue(bits); + long xResidue = 0; + ValueNode left = null; + JavaConstant leftCst = null; + if (forX instanceof AddNode) { + AddNode xAdd = (AddNode) forX; + if (xAdd.getY().isJavaConstant()) { + long xCst = xAdd.getY().asJavaConstant().asLong(); + xResidue = xCst - min; + left = xAdd.getX(); } - } else if (forY.isJavaConstant()) { - rightCst = forY.asJavaConstant(); + } else if (forX.isJavaConstant()) { + leftCst = forX.asJavaConstant(); } - if (right != null || rightCst != null) { - if ((xResidue == 0 && left != null) || (yResidue == 0 && right != null)) { - if (left == null) { - left = ConstantNode.forIntegerBits(bits, leftCst.asLong() - min); - } else if (xResidue != 0) { - left = AddNode.create(left, ConstantNode.forIntegerBits(bits, xResidue)); + if (left != null || leftCst != null) { + long yResidue = 0; + ValueNode right = null; + JavaConstant rightCst = null; + if (forY instanceof AddNode) { + AddNode yAdd = (AddNode) forY; + if (yAdd.getY().isJavaConstant()) { + long yCst = yAdd.getY().asJavaConstant().asLong(); + yResidue = yCst - min; + right = yAdd.getX(); } - if (right == null) { - right = ConstantNode.forIntegerBits(bits, rightCst.asLong() - min); - } else if (yResidue != 0) { - right = AddNode.create(right, ConstantNode.forIntegerBits(bits, yResidue)); + } else if (forY.isJavaConstant()) { + rightCst = forY.asJavaConstant(); + } + if (right != null || rightCst != null) { + if ((xResidue == 0 && left != null) || (yResidue == 0 && right != null)) { + if (left == null) { + left = ConstantNode.forIntegerBits(bits, leftCst.asLong() - min); + } else if (xResidue != 0) { + left = AddNode.create(left, ConstantNode.forIntegerBits(bits, xResidue)); + } + if (right == null) { + right = ConstantNode.forIntegerBits(bits, rightCst.asLong() - min); + } else if (yResidue != 0) { + right = AddNode.create(right, ConstantNode.forIntegerBits(bits, yResidue)); + } + return new IntegerBelowNode(left, right); } - return new IntegerBelowNode(left, right); } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IsNullNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IsNullNode.java index 7524ef0dfb1..68b5700559b 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IsNullNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/IsNullNode.java @@ -121,7 +121,8 @@ public final class IsNullNode extends UnaryOpLogicNode implements LIRLowerable, @Override public Stamp getSucceedingStampForValue(boolean negated) { - AbstractPointerStamp pointerStamp = (AbstractPointerStamp) getValue().stamp(); + // Ignore any more precise input stamp since canonicalization will skip through PiNodes + AbstractPointerStamp pointerStamp = (AbstractPointerStamp) getValue().stamp().unrestricted(); return negated ? pointerStamp.asNonNull() : pointerStamp.asAlwaysNull(); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/SignedDivNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/SignedDivNode.java index 79fc9beffd1..c8ba664abad 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/SignedDivNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/SignedDivNode.java @@ -63,29 +63,9 @@ public class SignedDivNode extends IntegerDivRemNode implements LIRLowerable { return ConstantNode.forIntegerStamp(stamp(), forX.asJavaConstant().asLong() / y); } else if (forY.isConstant()) { long c = forY.asJavaConstant().asLong(); - if (c == 1) { - return forX; - } - if (c == -1) { - return NegateNode.create(forX); - } - long abs = Math.abs(c); - if (CodeUtil.isPowerOf2(abs) && forX.stamp() instanceof IntegerStamp) { - ValueNode dividend = forX; - IntegerStamp stampX = (IntegerStamp) forX.stamp(); - int log2 = CodeUtil.log2(abs); - // no rounding if dividend is positive or if its low bits are always 0 - if (stampX.canBeNegative() || (stampX.upMask() & (abs - 1)) != 0) { - int bits = PrimitiveStamp.getBits(stamp()); - RightShiftNode sign = new RightShiftNode(forX, ConstantNode.forInt(bits - 1)); - UnsignedRightShiftNode round = new UnsignedRightShiftNode(sign, ConstantNode.forInt(bits - log2)); - dividend = BinaryArithmeticNode.add(dividend, round); - } - RightShiftNode shift = new RightShiftNode(dividend, ConstantNode.forInt(log2)); - if (c < 0) { - return NegateNode.create(shift); - } - return shift; + ValueNode v = canonical(forX, c); + if (v != null) { + return v; } } @@ -113,6 +93,34 @@ public class SignedDivNode extends IntegerDivRemNode implements LIRLowerable { return this; } + public static ValueNode canonical(ValueNode forX, long c) { + if (c == 1) { + return forX; + } + if (c == -1) { + return NegateNode.create(forX); + } + long abs = Math.abs(c); + if (CodeUtil.isPowerOf2(abs) && forX.stamp() instanceof IntegerStamp) { + ValueNode dividend = forX; + IntegerStamp stampX = (IntegerStamp) forX.stamp(); + int log2 = CodeUtil.log2(abs); + // no rounding if dividend is positive or if its low bits are always 0 + if (stampX.canBeNegative() || (stampX.upMask() & (abs - 1)) != 0) { + int bits = PrimitiveStamp.getBits(forX.stamp()); + RightShiftNode sign = new RightShiftNode(forX, ConstantNode.forInt(bits - 1)); + UnsignedRightShiftNode round = new UnsignedRightShiftNode(sign, ConstantNode.forInt(bits - log2)); + dividend = BinaryArithmeticNode.add(dividend, round); + } + RightShiftNode shift = new RightShiftNode(dividend, ConstantNode.forInt(log2)); + if (c < 0) { + return NegateNode.create(shift); + } + return shift; + } + return null; + } + @Override public void generate(NodeLIRBuilderTool gen) { gen.setResult(this, gen.getLIRGeneratorTool().getArithmetic().emitDiv(gen.operand(getX()), gen.operand(getY()), gen.state(this))); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/SignedRemNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/SignedRemNode.java index deb5c18d3e5..4dbfd845b54 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/SignedRemNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/SignedRemNode.java @@ -72,13 +72,14 @@ public class SignedRemNode extends IntegerDivRemNode implements LIRLowerable { return ConstantNode.forIntegerStamp(stamp(), 0); } else if (CodeUtil.isPowerOf2(constY)) { if (xStamp.isPositive()) { + // x & (y - 1) return new AndNode(forX, ConstantNode.forIntegerStamp(stamp(), constY - 1)); } else if (xStamp.isNegative()) { + // -((-x) & (y - 1)) return new NegateNode(new AndNode(new NegateNode(forX), ConstantNode.forIntegerStamp(stamp(), constY - 1))); } else { - return new ConditionalNode(IntegerLessThanNode.create(forX, ConstantNode.forIntegerStamp(forX.stamp(), 0)), - new NegateNode(new AndNode(new NegateNode(forX), ConstantNode.forIntegerStamp(stamp(), constY - 1))), - new AndNode(forX, ConstantNode.forIntegerStamp(stamp(), constY - 1))); + // x - ((x / y) << log2(y)) + return SubNode.create(forX, LeftShiftNode.create(SignedDivNode.canonical(forX, constY), ConstantNode.forInt(CodeUtil.log2(constY)))); } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/UnaryNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/UnaryNode.java index 542dfaa234a..66a8ee32342 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/UnaryNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/UnaryNode.java @@ -45,6 +45,11 @@ public abstract class UnaryNode extends FloatingNode implements Canonicalizable. return value; } + public void setValue(ValueNode value) { + updateUsages(this.value, value); + this.value = value; + } + /** * Creates a new UnaryNode instance. * diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/UnpackEndianHalfNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/UnpackEndianHalfNode.java new file mode 100644 index 00000000000..ad7cfdc9932 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/calc/UnpackEndianHalfNode.java @@ -0,0 +1,90 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.nodes.calc; + +import java.nio.ByteOrder; + +import org.graalvm.compiler.core.common.type.StampFactory; +import org.graalvm.compiler.graph.Node; +import org.graalvm.compiler.graph.NodeClass; +import org.graalvm.compiler.graph.spi.CanonicalizerTool; +import org.graalvm.compiler.nodeinfo.NodeCycles; +import org.graalvm.compiler.nodeinfo.NodeInfo; +import org.graalvm.compiler.nodes.ConstantNode; +import org.graalvm.compiler.nodes.ValueNode; +import org.graalvm.compiler.nodes.spi.Lowerable; +import org.graalvm.compiler.nodes.spi.LoweringTool; + +import jdk.vm.ci.meta.JavaKind; + +/** + * Produces the platform dependent first or second half of a long or double value as an int. + */ +@NodeInfo(cycles = NodeCycles.CYCLES_2) +public final class UnpackEndianHalfNode extends UnaryNode implements Lowerable { + public static final NodeClass TYPE = NodeClass.create(UnpackEndianHalfNode.class); + + private final boolean firstHalf; + + protected UnpackEndianHalfNode(ValueNode value, boolean firstHalf) { + super(TYPE, StampFactory.forKind(JavaKind.Int), value); + assert value.getStackKind() == JavaKind.Double || value.getStackKind() == JavaKind.Long : "unexpected kind " + value.getStackKind(); + this.firstHalf = firstHalf; + } + + public static ValueNode create(ValueNode value, boolean firstHalf) { + if (value.isConstant() && value.asConstant().isDefaultForKind()) { + return ConstantNode.defaultForKind(JavaKind.Int); + } + return new UnpackEndianHalfNode(value, firstHalf); + } + + public boolean isFirstHalf() { + return firstHalf; + } + + @Override + public Node canonical(CanonicalizerTool tool, ValueNode forValue) { + if (forValue.isConstant() && forValue.asConstant().isDefaultForKind()) { + return ConstantNode.defaultForKind(stamp.getStackKind()); + } + return this; + } + + @Override + public void lower(LoweringTool tool) { + tool.getLowerer().lower(this, tool); + } + + public void lower(ByteOrder byteOrder) { + ValueNode result = value; + if (value.getStackKind() == JavaKind.Double) { + result = graph().unique(new ReinterpretNode(JavaKind.Long, value)); + } + if ((byteOrder == ByteOrder.BIG_ENDIAN) == firstHalf) { + result = graph().unique(new UnsignedRightShiftNode(result, ConstantNode.forInt(32, graph()))); + } + result = IntegerConvertNode.convert(result, StampFactory.forKind(JavaKind.Int), graph()); + replaceAtUsagesAndDelete(result); + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/cfg/ControlFlowGraph.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/cfg/ControlFlowGraph.java index 7f268788351..bb4918a96c4 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/cfg/ControlFlowGraph.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/cfg/ControlFlowGraph.java @@ -592,7 +592,11 @@ public final class ControlFlowGraph implements AbstractControlFlowGraph { for (Block block : reversePostOrder) { AbstractBeginNode beginNode = block.getBeginNode(); if (beginNode instanceof LoopBeginNode) { - Loop loop = new HIRLoop(block.getLoop(), loops.size(), block); + Loop parent = block.getLoop(); + Loop loop = new HIRLoop(parent, loops.size(), block); + if (parent != null) { + parent.getChildren().add(loop); + } loops.add(loop); block.setLoop(loop); loop.getBlocks().add(block); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/BranchProbabilityNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/BranchProbabilityNode.java index ac74a474147..cc3e63d14f8 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/BranchProbabilityNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/BranchProbabilityNode.java @@ -82,6 +82,9 @@ public final class BranchProbabilityNode extends FloatingNode implements Simplif @Override public void simplify(SimplifierTool tool) { + if (!hasUsages()) { + return; + } if (probability.isConstant()) { double probabilityValue = probability.asJavaConstant().asDouble(); if (probabilityValue < 0.0) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/RawLoadNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/RawLoadNode.java index 63e6dfcffe2..de715c7e2c3 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/RawLoadNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/RawLoadNode.java @@ -79,7 +79,7 @@ public class RawLoadNode extends UnsafeAccessNode implements Lowerable, Virtuali super(TYPE, stamp, object, offset, accessKind, locationIdentity, false); } - public RawLoadNode(NodeClass c, ValueNode object, ValueNode offset, JavaKind accessKind, LocationIdentity locationIdentity) { + protected RawLoadNode(NodeClass c, ValueNode object, ValueNode offset, JavaKind accessKind, LocationIdentity locationIdentity) { super(c, StampFactory.forKind(accessKind.getStackKind()), object, offset, accessKind, locationIdentity, false); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/RawStoreNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/RawStoreNode.java index b9d3f71dc07..720654f5a82 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/RawStoreNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/extended/RawStoreNode.java @@ -29,7 +29,6 @@ import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_1; import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.graph.NodeClass; import org.graalvm.compiler.nodeinfo.NodeInfo; -import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.FrameState; import org.graalvm.compiler.nodes.StateSplit; import org.graalvm.compiler.nodes.ValueNode; @@ -39,12 +38,10 @@ import org.graalvm.compiler.nodes.spi.Lowerable; import org.graalvm.compiler.nodes.spi.LoweringTool; import org.graalvm.compiler.nodes.spi.Virtualizable; import org.graalvm.compiler.nodes.spi.VirtualizerTool; -import org.graalvm.compiler.nodes.type.StampTool; import org.graalvm.compiler.nodes.virtual.VirtualObjectNode; import org.graalvm.word.LocationIdentity; import jdk.vm.ci.meta.Assumptions; -import jdk.vm.ci.meta.JavaConstant; import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.ResolvedJavaField; @@ -123,41 +120,8 @@ public final class RawStoreNode extends UnsafeAccessNode implements StateSplit, if (indexValue.isConstant()) { long off = indexValue.asJavaConstant().asLong(); int entryIndex = virtual.entryIndexForOffset(off, accessKind()); - if (entryIndex != -1) { - JavaKind entryKind = virtual.entryKind(entryIndex); - boolean canVirtualize = entryKind == accessKind() || (entryKind == accessKind().getStackKind() && !StampTool.typeOrNull(object()).isArray()); - if (!canVirtualize) { - /* - * Special case: If the entryKind is long, allow arbitrary kinds as long as - * a value of the same kind is already there. This can only happen if some - * other node initialized the entry with a value of a different kind. One - * example where this happens is the Truffle NewFrameNode. - */ - ValueNode entry = tool.getEntry(virtual, entryIndex); - if (entryKind == JavaKind.Long && entry.getStackKind() == value.getStackKind()) { - canVirtualize = true; - } - } - if (canVirtualize) { - tool.setVirtualEntry(virtual, entryIndex, value(), true); - tool.delete(); - } else { - /* - * Special case: Allow storing a single long or double value into two - * consecutive int slots. - */ - if ((accessKind() == JavaKind.Long || accessKind() == JavaKind.Double) && entryKind == JavaKind.Int) { - int nextIndex = virtual.entryIndexForOffset(off + 4, entryKind); - if (nextIndex != -1) { - JavaKind nextKind = virtual.entryKind(nextIndex); - if (nextKind == JavaKind.Int) { - tool.setVirtualEntry(virtual, entryIndex, value(), true); - tool.setVirtualEntry(virtual, nextIndex, ConstantNode.forConstant(JavaConstant.forIllegal(), tool.getMetaAccessProvider(), graph()), true); - tool.delete(); - } - } - } - } + if (entryIndex != -1 && tool.setVirtualEntry(virtual, entryIndex, value(), accessKind(), off)) { + tool.delete(); } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/GraphBuilderContext.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/GraphBuilderContext.java index 3d5578eaa50..09b4cc72c89 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/GraphBuilderContext.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/GraphBuilderContext.java @@ -280,7 +280,7 @@ public interface GraphBuilderContext extends GraphBuilderTool { ObjectStamp receiverStamp = (ObjectStamp) value.stamp(); Stamp stamp = receiverStamp.join(objectNonNull()); FixedGuardNode fixedGuard = append(new FixedGuardNode(condition, NullCheckException, action, true)); - ValueNode nonNullReceiver = getGraph().addOrUnique(PiNode.create(value, stamp, fixedGuard)); + ValueNode nonNullReceiver = getGraph().addOrUniqueWithInputs(PiNode.create(value, stamp, fixedGuard)); // TODO: Propogating the non-null into the frame state would // remove subsequent null-checks on the same value. However, // it currently causes an assertion failure when merging states. diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/InlineInvokePlugin.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/InlineInvokePlugin.java index 73ac712a8f8..a85d3589ea0 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/InlineInvokePlugin.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/InlineInvokePlugin.java @@ -53,6 +53,12 @@ public interface InlineInvokePlugin extends GraphBuilderPlugin { */ public static final InlineInfo DO_NOT_INLINE_NO_EXCEPTION = new InlineInfo(null, null); + /** + * Denotes a call site must not be inlined and the execution should be transferred to + * interpreter in case of an exception. + */ + public static final InlineInfo DO_NOT_INLINE_DEOPTIMIZE_ON_EXCEPTION = new InlineInfo(null, null); + private final ResolvedJavaMethod methodToInline; private final BytecodeProvider intrinsicBytecodeProvider; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/IntrinsicContext.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/IntrinsicContext.java index dedabd6099a..f06676378ef 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/IntrinsicContext.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/graphbuilderconf/IntrinsicContext.java @@ -66,6 +66,34 @@ public class IntrinsicContext { */ final BytecodeProvider bytecodeProvider; + final CompilationContext compilationContext; + + final boolean allowPartialIntrinsicArgumentMismatch; + + public IntrinsicContext(ResolvedJavaMethod method, ResolvedJavaMethod intrinsic, BytecodeProvider bytecodeProvider, CompilationContext compilationContext) { + this(method, intrinsic, bytecodeProvider, compilationContext, false); + } + + public IntrinsicContext(ResolvedJavaMethod method, ResolvedJavaMethod intrinsic, BytecodeProvider bytecodeProvider, CompilationContext compilationContext, + boolean allowPartialIntrinsicArgumentMismatch) { + this.method = method; + this.intrinsic = intrinsic; + this.bytecodeProvider = bytecodeProvider; + assert bytecodeProvider != null; + this.compilationContext = compilationContext; + this.allowPartialIntrinsicArgumentMismatch = allowPartialIntrinsicArgumentMismatch; + assert !isCompilationRoot() || method.hasBytecodes() : "Cannot root compile intrinsic for native or abstract method " + method.format("%H.%n(%p)"); + } + + /** + * A partial intrinsic exits by (effectively) calling the intrinsified method. Normally, this + * call must use exactly the same arguments as the call that is being intrinsified. This allows + * to override this behavior. + */ + public boolean allowPartialIntrinsicArgumentMismatch() { + return allowPartialIntrinsicArgumentMismatch; + } + /** * Gets the method being intrinsified. */ @@ -96,17 +124,6 @@ public class IntrinsicContext { return method.equals(targetMethod) || intrinsic.equals(targetMethod); } - final CompilationContext compilationContext; - - public IntrinsicContext(ResolvedJavaMethod method, ResolvedJavaMethod intrinsic, BytecodeProvider bytecodeProvider, CompilationContext compilationContext) { - this.method = method; - this.intrinsic = intrinsic; - this.bytecodeProvider = bytecodeProvider; - assert bytecodeProvider != null; - this.compilationContext = compilationContext; - assert !isCompilationRoot() || method.hasBytecodes() : "Cannot root compile intrinsic for native or abstract method " + method.format("%H.%n(%p)"); - } - public boolean isPostParseInlined() { return compilationContext.equals(INLINE_AFTER_PARSING); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/InstanceOfNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/InstanceOfNode.java index b9aa170b065..e63cffefe88 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/InstanceOfNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/InstanceOfNode.java @@ -49,6 +49,8 @@ import org.graalvm.compiler.nodes.type.StampTool; import jdk.vm.ci.meta.JavaTypeProfile; import jdk.vm.ci.meta.TriState; +import java.util.Objects; + /** * The {@code InstanceOfNode} represents an instanceof test. */ @@ -56,7 +58,7 @@ import jdk.vm.ci.meta.TriState; public class InstanceOfNode extends UnaryOpLogicNode implements Lowerable, Virtualizable { public static final NodeClass TYPE = NodeClass.create(InstanceOfNode.class); - protected final ObjectStamp checkedStamp; + private ObjectStamp checkedStamp; private JavaTypeProfile profile; @OptionalInput(Anchor) protected AnchoringNode anchor; @@ -126,7 +128,9 @@ public class InstanceOfNode extends UnaryOpLogicNode implements Lowerable, Virtu // The check will always succeed, the union of the two stamps is equal to the // checked stamp. return LogicConstantNode.tautology(); - } else if (checkedStamp.type().equals(meetStamp.type()) && checkedStamp.isExactType() == meetStamp.isExactType() && checkedStamp.alwaysNull() == meetStamp.alwaysNull()) { + } else if (checkedStamp.alwaysNull()) { + return IsNullNode.create(object); + } else if (Objects.equals(checkedStamp.type(), meetStamp.type()) && checkedStamp.isExactType() == meetStamp.isExactType() && checkedStamp.alwaysNull() == meetStamp.alwaysNull()) { assert checkedStamp.nonNull() != inputStamp.nonNull(); // The only difference makes the null-ness of the value => simplify the check. if (checkedStamp.nonNull()) { @@ -135,8 +139,8 @@ public class InstanceOfNode extends UnaryOpLogicNode implements Lowerable, Virtu return IsNullNode.create(object); } } + assert checkedStamp.type() != null; } - return null; } @@ -204,4 +208,13 @@ public class InstanceOfNode extends UnaryOpLogicNode implements Lowerable, Virtu public AnchoringNode getAnchor() { return anchor; } + + public ObjectStamp getCheckedStamp() { + return checkedStamp; + } + + public void strengthenCheckedStamp(ObjectStamp newCheckedStamp) { + assert this.checkedStamp.join(newCheckedStamp).equals(newCheckedStamp) : "stamp can only improve"; + this.checkedStamp = newCheckedStamp; + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/LoadFieldNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/LoadFieldNode.java index 0373f894197..5ff55d2a65f 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/LoadFieldNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/LoadFieldNode.java @@ -25,7 +25,6 @@ package org.graalvm.compiler.nodes.java; import static org.graalvm.compiler.graph.iterators.NodePredicates.isNotA; import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_2; -import jdk.vm.ci.meta.ConstantReflectionProvider; import org.graalvm.compiler.core.common.spi.ConstantFieldProvider; import org.graalvm.compiler.core.common.type.Stamp; import org.graalvm.compiler.core.common.type.StampFactory; @@ -47,15 +46,16 @@ import org.graalvm.compiler.nodes.type.StampTool; import org.graalvm.compiler.nodes.util.ConstantFoldUtil; import org.graalvm.compiler.nodes.virtual.VirtualInstanceNode; import org.graalvm.compiler.nodes.virtual.VirtualObjectNode; +import org.graalvm.compiler.options.OptionValues; import jdk.vm.ci.meta.Assumptions; +import jdk.vm.ci.meta.ConstantReflectionProvider; import jdk.vm.ci.meta.DeoptimizationAction; import jdk.vm.ci.meta.DeoptimizationReason; import jdk.vm.ci.meta.JavaConstant; import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.MetaAccessProvider; import jdk.vm.ci.meta.ResolvedJavaField; -import org.graalvm.compiler.options.OptionValues; /** * The {@code LoadFieldNode} represents a read of a static or instance field. @@ -150,7 +150,7 @@ public final class LoadFieldNode extends AccessFieldNode implements Canonicaliza } public ConstantNode asConstant(CanonicalizerTool tool, JavaConstant constant) { - return ConstantFoldUtil.tryConstantFold(tool.getConstantFieldProvider(), tool.getConstantReflection(), tool.getMetaAccess(), field(), constant, getOptions()); + return ConstantFoldUtil.tryConstantFold(tool.getConstantFieldProvider(), tool.getConstantReflection(), tool.getMetaAccess(), field(), constant, tool.getOptions()); } private static PhiNode asPhi(ConstantFieldProvider constantFields, ConstantReflectionProvider constantReflection, diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/StoreFieldNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/StoreFieldNode.java index 343d7d3c614..84c26fa1b9e 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/StoreFieldNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/StoreFieldNode.java @@ -29,7 +29,6 @@ import org.graalvm.compiler.graph.NodeClass; import org.graalvm.compiler.nodeinfo.InputType; import org.graalvm.compiler.nodeinfo.NodeCycles; import org.graalvm.compiler.nodeinfo.NodeInfo; -import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.FrameState; import org.graalvm.compiler.nodes.StateSplit; import org.graalvm.compiler.nodes.ValueNode; @@ -38,7 +37,6 @@ import org.graalvm.compiler.nodes.spi.VirtualizerTool; import org.graalvm.compiler.nodes.virtual.VirtualInstanceNode; import org.graalvm.compiler.nodes.virtual.VirtualObjectNode; -import jdk.vm.ci.meta.JavaConstant; import jdk.vm.ci.meta.ResolvedJavaField; /** @@ -90,15 +88,8 @@ public final class StoreFieldNode extends AccessFieldNode implements StateSplit, VirtualInstanceNode virtual = (VirtualInstanceNode) alias; int fieldIndex = virtual.fieldIndex(field()); if (fieldIndex != -1) { - ValueNode existing = tool.getEntry((VirtualObjectNode) alias, fieldIndex); - if (existing.stamp().isCompatible(value().stamp())) { - tool.setVirtualEntry(virtual, fieldIndex, value(), false); - tool.delete(); - } else { - // stamp of the value is not compatible with the value in the virtualizer - // can only happen with unsafe two slot writes on one slot fields - assert existing instanceof ConstantNode && ((ConstantNode) existing).asConstant().equals(JavaConstant.forIllegal()) : value.stamp() + " vs " + existing.stamp(); - } + tool.setVirtualEntry(virtual, fieldIndex, value()); + tool.delete(); } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/StoreIndexedNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/StoreIndexedNode.java index 04e6fb4aeaf..b5e55e7bf47 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/StoreIndexedNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/java/StoreIndexedNode.java @@ -89,7 +89,7 @@ public final class StoreIndexedNode extends AccessIndexedNode implements StateSp ResolvedJavaType componentType = virtual.type().getComponentType(); if (componentType.isPrimitive() || StampTool.isPointerAlwaysNull(value) || componentType.getSuperclass() == null || (StampTool.typeReferenceOrNull(value) != null && componentType.isAssignableFrom(StampTool.typeOrNull(value)))) { - tool.setVirtualEntry(virtual, idx, value(), false); + tool.setVirtualEntry(virtual, idx, value()); tool.delete(); } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/memory/MemoryPhiNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/memory/MemoryPhiNode.java index fadb2cad9a9..5e7f2a30819 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/memory/MemoryPhiNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/memory/MemoryPhiNode.java @@ -62,4 +62,9 @@ public final class MemoryPhiNode extends PhiNode implements MemoryNode { public NodeInputList values() { return values; } + + @Override + protected String valueDescription() { + return locationIdentity.toString(); + } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/memory/address/OffsetAddressNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/memory/address/OffsetAddressNode.java index f0be982e4d0..345bef6cd99 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/memory/address/OffsetAddressNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/memory/address/OffsetAddressNode.java @@ -22,6 +22,7 @@ */ package org.graalvm.compiler.nodes.memory.address; +import org.graalvm.compiler.core.common.type.AbstractPointerStamp; import org.graalvm.compiler.core.common.type.IntegerStamp; import org.graalvm.compiler.core.common.type.Stamp; import org.graalvm.compiler.graph.Node; @@ -52,6 +53,9 @@ public class OffsetAddressNode extends AddressNode implements Canonicalizable { super(TYPE); this.base = base; this.offset = offset; + + assert base != null && (base.stamp() instanceof AbstractPointerStamp || IntegerStamp.getBits(base.stamp()) == 64) && + offset != null && IntegerStamp.getBits(offset.stamp()) == 64 : "both values must have 64 bits"; } @Override @@ -62,6 +66,7 @@ public class OffsetAddressNode extends AddressNode implements Canonicalizable { public void setBase(ValueNode base) { updateUsages(this.base, base); this.base = base; + assert base != null && (base.stamp() instanceof AbstractPointerStamp || IntegerStamp.getBits(base.stamp()) == 64); } public ValueNode getOffset() { @@ -71,6 +76,7 @@ public class OffsetAddressNode extends AddressNode implements Canonicalizable { public void setOffset(ValueNode offset) { updateUsages(this.offset, offset); this.offset = offset; + assert offset != null && IntegerStamp.getBits(offset.stamp()) == 64; } @Override diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/spi/VirtualizerTool.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/spi/VirtualizerTool.java index e7acfc7bec4..785675adde5 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/spi/VirtualizerTool.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/spi/VirtualizerTool.java @@ -25,6 +25,7 @@ package org.graalvm.compiler.nodes.spi; import java.util.List; import org.graalvm.compiler.debug.DebugContext; +import org.graalvm.compiler.debug.GraalError; import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.nodes.ValueNode; import org.graalvm.compiler.nodes.java.MonitorIdNode; @@ -91,9 +92,17 @@ public interface VirtualizerTool { * * @param index the index to be set. * @param value the new value for the given index. - * @param unsafe if true, then mismatching value {@link JavaKind}s will be accepted. + * @param accessKind the kind of the store which might be different than + * {@link VirtualObjectNode#entryKind(int)}. + * @return true if the operation was permitted */ - void setVirtualEntry(VirtualObjectNode virtualObject, int index, ValueNode value, boolean unsafe); + boolean setVirtualEntry(VirtualObjectNode virtualObject, int index, ValueNode value, JavaKind accessKind, long offset); + + default void setVirtualEntry(VirtualObjectNode virtualObject, int index, ValueNode value) { + if (!setVirtualEntry(virtualObject, index, value, null, 0)) { + throw new GraalError("unexpected failure when updating virtual entry"); + } + } ValueNode getEntry(VirtualObjectNode virtualObject, int index); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/GraphUtil.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/GraphUtil.java index 083fa733f84..4990cc79e70 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/GraphUtil.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/GraphUtil.java @@ -242,7 +242,7 @@ public class GraphUtil { EconomicSet collectedUnusedNodes = unusedNodes = EconomicSet.create(Equivalence.IDENTITY); nodeEventScope = node.graph().trackNodeEvents(new Graph.NodeEventListener() { @Override - public void event(Graph.NodeEvent e, Node n) { + public void changed(Graph.NodeEvent e, Node n) { if (e == Graph.NodeEvent.ZERO_USAGES && isFloatingNode(n) && !(n instanceof GuardNode)) { collectedUnusedNodes.add(n); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/JavaConstantFormattable.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/JavaConstantFormattable.java new file mode 100644 index 00000000000..61f44e8d08f --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/JavaConstantFormattable.java @@ -0,0 +1,31 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.nodes.util; + +/** + * Performs special formatting of values involving {@link jdk.vm.ci.meta.JavaConstant JavaConstants} + * when they are being dumped. + */ +public interface JavaConstantFormattable { + String format(JavaConstantFormatter formatter); +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/JavaConstantFormatter.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/JavaConstantFormatter.java new file mode 100644 index 00000000000..5f475d8aacf --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/util/JavaConstantFormatter.java @@ -0,0 +1,29 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.nodes.util; + +import jdk.vm.ci.meta.JavaConstant; + +public interface JavaConstantFormatter { + String format(JavaConstant constant); +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/virtual/VirtualArrayNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/virtual/VirtualArrayNode.java index 19b9c608fc1..65f644b73dc 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/virtual/VirtualArrayNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/virtual/VirtualArrayNode.java @@ -37,7 +37,7 @@ import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.ResolvedJavaType; import sun.misc.Unsafe; -@NodeInfo(nameTemplate = "VirtualArray {p#componentType/s}[{p#length}]") +@NodeInfo(nameTemplate = "VirtualArray({p#objectId}) {p#componentType/s}[{p#length}]") public class VirtualArrayNode extends VirtualObjectNode implements ArrayLengthProvider { public static final NodeClass TYPE = NodeClass.create(VirtualArrayNode.class); @@ -76,7 +76,7 @@ public class VirtualArrayNode extends VirtualObjectNode implements ArrayLengthPr @Override public String toString(Verbosity verbosity) { if (verbosity == Verbosity.Name) { - return super.toString(Verbosity.Name) + " " + componentType.getName() + "[" + length + "]"; + return super.toString(Verbosity.Name) + "(" + getObjectId() + ") " + componentType.getName() + "[" + length + "]"; } else { return super.toString(verbosity); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/virtual/VirtualInstanceNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/virtual/VirtualInstanceNode.java index 0591a2d6b54..bee87d12029 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/virtual/VirtualInstanceNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.nodes/src/org/graalvm/compiler/nodes/virtual/VirtualInstanceNode.java @@ -32,7 +32,7 @@ import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.ResolvedJavaField; import jdk.vm.ci.meta.ResolvedJavaType; -@NodeInfo(nameTemplate = "VirtualInstance {p#type/s}") +@NodeInfo(nameTemplate = "VirtualInstance({p#objectId}) {p#type/s}") public class VirtualInstanceNode extends VirtualObjectNode { public static final NodeClass TYPE = NodeClass.create(VirtualInstanceNode.class); @@ -78,7 +78,7 @@ public class VirtualInstanceNode extends VirtualObjectNode { @Override public String toString(Verbosity verbosity) { if (verbosity == Verbosity.Name) { - return super.toString(Verbosity.Name) + " " + type.toJavaName(false); + return super.toString(Verbosity.Name) + "(" + getObjectId() + ") " + type.toJavaName(false); } else { return super.toString(verbosity); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.options.processor/src/org/graalvm/compiler/options/processor/OptionProcessor.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.options.processor/src/org/graalvm/compiler/options/processor/OptionProcessor.java index 7749a77cb39..797a127bbf6 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.options.processor/src/org/graalvm/compiler/options/processor/OptionProcessor.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.options.processor/src/org/graalvm/compiler/options/processor/OptionProcessor.java @@ -394,6 +394,9 @@ public class OptionProcessor extends AbstractProcessor { options = new OptionsInfo(topDeclaringType); map.put(topDeclaringType, options); } + if (!element.getEnclosingElement().getSimpleName().toString().endsWith("Options")) { + processingEnv.getMessager().printMessage(Kind.ERROR, "Option declaring classes must have a name that ends with 'Options'", element.getEnclosingElement()); + } processElement(element, options); } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/ConditionalEliminationPhase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/ConditionalEliminationPhase.java index 276643ae150..3530b938204 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/ConditionalEliminationPhase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/ConditionalEliminationPhase.java @@ -248,7 +248,7 @@ public class ConditionalEliminationPhase extends BasePhase { /** * Tests which may be eliminated because post dominating tests to prove a broader condition. */ - private Deque pendingTests; + private Deque pendingTests; public Instance(StructuredGraph graph, BlockMap> blockToNodes, PhaseContext context) { this.graph = graph; @@ -555,7 +555,8 @@ public class ConditionalEliminationPhase extends BasePhase { } } if (guard instanceof DeoptimizingGuard) { - pendingTests.push(new PendingTest(condition, (DeoptimizingGuard) guard)); + assert ((DeoptimizingGuard) guard).getCondition() == condition; + pendingTests.push((DeoptimizingGuard) guard); } registerCondition(condition, negated, guard); } @@ -628,15 +629,16 @@ public class ConditionalEliminationPhase extends BasePhase { } protected boolean foldPendingTest(DeoptimizingGuard thisGuard, ValueNode original, Stamp newStamp, GuardRewirer rewireGuardFunction) { - for (PendingTest pending : pendingTests) { + for (DeoptimizingGuard pendingGuard : pendingTests) { + LogicNode pendingCondition = pendingGuard.getCondition(); TriState result = TriState.UNKNOWN; - if (pending.condition instanceof UnaryOpLogicNode) { - UnaryOpLogicNode unaryLogicNode = (UnaryOpLogicNode) pending.condition; + if (pendingCondition instanceof UnaryOpLogicNode) { + UnaryOpLogicNode unaryLogicNode = (UnaryOpLogicNode) pendingCondition; if (unaryLogicNode.getValue() == original) { result = unaryLogicNode.tryFold(newStamp); } - } else if (pending.condition instanceof BinaryOpLogicNode) { - BinaryOpLogicNode binaryOpLogicNode = (BinaryOpLogicNode) pending.condition; + } else if (pendingCondition instanceof BinaryOpLogicNode) { + BinaryOpLogicNode binaryOpLogicNode = (BinaryOpLogicNode) pendingCondition; ValueNode x = binaryOpLogicNode.getX(); ValueNode y = binaryOpLogicNode.getY(); if (x == original) { @@ -659,7 +661,7 @@ public class ConditionalEliminationPhase extends BasePhase { */ InputFilter v = new InputFilter(original); thisGuard.getCondition().applyInputs(v); - if (v.ok && foldGuard(thisGuard, pending.guard, newStamp, rewireGuardFunction)) { + if (v.ok && foldGuard(thisGuard, pendingGuard, newStamp, rewireGuardFunction)) { return true; } } @@ -1026,16 +1028,6 @@ public class ConditionalEliminationPhase extends BasePhase { boolean rewire(GuardingNode guard, boolean result, Stamp guardedValueStamp, ValueNode newInput); } - protected static class PendingTest { - private final LogicNode condition; - private final DeoptimizingGuard guard; - - public PendingTest(LogicNode condition, DeoptimizingGuard guard) { - this.condition = condition; - this.guard = guard; - } - } - protected static final class InfoElement { private final Stamp stamp; private final GuardingNode guard; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/FixReadsPhase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/FixReadsPhase.java index a83fc06422d..b1379d9b0ca 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/FixReadsPhase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/FixReadsPhase.java @@ -221,7 +221,12 @@ public class FixReadsPhase extends BasePhase { EconomicMap endMap = endMaps.get(node); MapCursor entries = endMap.getEntries(); while (entries.advance()) { - if (registerNewValueStamp(entries.getKey(), entries.getValue())) { + ValueNode value = entries.getKey(); + if (value.isDeleted()) { + // nodes from this map can be deleted when a loop dies + continue; + } + if (registerNewValueStamp(value, entries.getValue())) { counterBetterMergedStamps.increment(debug); } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/ProfileCompiledMethodsPhase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/ProfileCompiledMethodsPhase.java index 6e8cd9de1b9..c395bcca3fc 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/ProfileCompiledMethodsPhase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/ProfileCompiledMethodsPhase.java @@ -48,7 +48,7 @@ import org.graalvm.compiler.nodes.UnwindNode; import org.graalvm.compiler.nodes.VirtualState; import org.graalvm.compiler.nodes.calc.BinaryNode; import org.graalvm.compiler.nodes.calc.ConvertNode; -import org.graalvm.compiler.nodes.calc.DivNode; +import org.graalvm.compiler.nodes.calc.FloatDivNode; import org.graalvm.compiler.nodes.calc.IntegerDivRemNode; import org.graalvm.compiler.nodes.calc.MulNode; import org.graalvm.compiler.nodes.calc.NotNode; @@ -162,7 +162,7 @@ public class ProfileCompiledMethodsPhase extends Phase { return 2; } else if (node instanceof LogicNode || node instanceof ConvertNode || node instanceof BinaryNode || node instanceof NotNode) { return 1; - } else if (node instanceof IntegerDivRemNode || node instanceof DivNode || node instanceof RemNode) { + } else if (node instanceof IntegerDivRemNode || node instanceof FloatDivNode || node instanceof RemNode) { return 10; } else if (node instanceof MulNode) { return 3; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/inlining/info/elem/InlineableGraph.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/inlining/info/elem/InlineableGraph.java index d7334e35baa..35d076543a6 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/inlining/info/elem/InlineableGraph.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/inlining/info/elem/InlineableGraph.java @@ -43,7 +43,6 @@ import org.graalvm.compiler.phases.common.inlining.InliningUtil; import org.graalvm.compiler.phases.graph.FixedNodeProbabilityCache; import org.graalvm.compiler.phases.tiers.HighTierContext; -import jdk.vm.ci.meta.Constant; import jdk.vm.ci.meta.ResolvedJavaMethod; /** @@ -159,11 +158,11 @@ public class InlineableGraph implements Inlineable { if (param.usages().isNotEmpty()) { ValueNode arg = args.get(param.index()); if (arg.isConstant()) { - Constant constant = arg.asConstant(); + ConstantNode constant = (ConstantNode) arg; parameterUsages = trackParameterUsages(param, parameterUsages); // collect param usages before replacing the param param.replaceAtUsagesAndDelete(graph.unique( - ConstantNode.forConstant(arg.stamp(), constant, ((ConstantNode) arg).getStableDimension(), ((ConstantNode) arg).isDefaultStable(), context.getMetaAccess()))); + ConstantNode.forConstant(arg.stamp(), constant.getValue(), constant.getStableDimension(), constant.isDefaultStable(), context.getMetaAccess()))); // param-node gone, leaving a gap in the sequence given by param.index() } else { Stamp impro = improvedStamp(arg, param); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/util/HashSetNodeEventListener.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/util/HashSetNodeEventListener.java index 57c13252ff1..eef9f035006 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/util/HashSetNodeEventListener.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases.common/src/org/graalvm/compiler/phases/common/util/HashSetNodeEventListener.java @@ -37,7 +37,7 @@ import org.graalvm.util.EconomicSet; * A simple {@link NodeEventListener} implementation that accumulates event nodes in a * {@link HashSet}. */ -public class HashSetNodeEventListener implements NodeEventListener { +public class HashSetNodeEventListener extends NodeEventListener { private final EconomicSet nodes; private final Set filter; @@ -68,7 +68,7 @@ public class HashSetNodeEventListener implements NodeEventListener { } @Override - public void event(NodeEvent e, Node node) { + public void changed(NodeEvent e, Node node) { if (filter.contains(e)) { nodes.add(node); if (node instanceof IndirectCanonicalization) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/BasePhase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/BasePhase.java index 7f2574950b9..927b94ede62 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/BasePhase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/BasePhase.java @@ -258,7 +258,7 @@ public abstract class BasePhase implements PhaseSizeContract { return false; } - private final class GraphChangeListener implements NodeEventListener { + private final class GraphChangeListener extends NodeEventListener { boolean changed; private StructuredGraph graph; private Mark mark; @@ -269,7 +269,7 @@ public abstract class BasePhase implements PhaseSizeContract { } @Override - public void event(NodeEvent e, Node node) { + public void changed(NodeEvent e, Node node) { if (!graph.isNew(mark, node) && node.isAlive()) { if (e == NodeEvent.INPUT_CHANGED || e == NodeEvent.ZERO_USAGES) { changed = true; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/schedule/SchedulePhase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/schedule/SchedulePhase.java index c2b10b5a349..1b896801e58 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/schedule/SchedulePhase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/schedule/SchedulePhase.java @@ -109,7 +109,7 @@ public final class SchedulePhase extends Phase { if (immutableGraph && Assertions.assertionsEnabled()) { return graph.trackNodeEvents(new NodeEventListener() { @Override - public void event(NodeEvent e, Node node) { + public void changed(NodeEvent e, Node node) { assert false : "graph changed: " + e + " on node " + node; } }); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/verify/VerifyGetOptionsUsage.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/verify/VerifyGetOptionsUsage.java new file mode 100644 index 00000000000..9018846e367 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/verify/VerifyGetOptionsUsage.java @@ -0,0 +1,86 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.phases.verify; + +import java.lang.reflect.MalformedParametersException; +import java.lang.reflect.Method; + +import org.graalvm.compiler.graph.Node; +import org.graalvm.compiler.graph.spi.CanonicalizerTool; +import org.graalvm.compiler.nodes.StructuredGraph; +import org.graalvm.compiler.nodes.java.MethodCallTargetNode; +import org.graalvm.compiler.phases.VerifyPhase; +import org.graalvm.compiler.phases.tiers.PhaseContext; + +import jdk.vm.ci.meta.MetaAccessProvider; +import jdk.vm.ci.meta.ResolvedJavaMethod; +import jdk.vm.ci.meta.ResolvedJavaType; + +/** + * {@link Node#getOptions()} is unsafe for use during canonicalization so try to verify that it + * isn't used when a {@link CanonicalizerTool} is available in the arguments. This is slightly more + * general but since there are several canonical methods with varying signatures this covers more + * cases. + */ +public class VerifyGetOptionsUsage extends VerifyPhase { + static Method lookupMethod(Class klass, String name) { + for (Method m : klass.getDeclaredMethods()) { + if (m.getName().equals(name)) { + return m; + } + } + throw new InternalError(); + } + + @Override + protected boolean verify(StructuredGraph graph, PhaseContext context) { + MetaAccessProvider metaAccess = context.getMetaAccess(); + ResolvedJavaType canonicalizerToolClass = metaAccess.lookupJavaType(CanonicalizerTool.class); + boolean hasTool = false; + try { + for (ResolvedJavaMethod.Parameter parameter : graph.method().getParameters()) { + if (parameter.getType().getName().equals(canonicalizerToolClass.getName())) { + hasTool = true; + break; + } + } + } catch (MalformedParametersException e) { + // Lambdas sometimes have malformed parameters so ignore this. + } + if (hasTool) { + ResolvedJavaMethod getOptionsMethod = metaAccess.lookupJavaMethod(lookupMethod(Node.class, "getOptions")); + for (MethodCallTargetNode t : graph.getNodes(MethodCallTargetNode.TYPE)) { + ResolvedJavaMethod callee = t.targetMethod(); + if (callee.equals(getOptionsMethod)) { + if (hasTool) { + throw new VerificationError("Must use CanonicalizerTool.getOptions() instead of Node.getOptions() in method '%s' of class '%s'.", + graph.method().getName(), graph.method().getDeclaringClass().getName()); + } + } + } + } + + return true; + } + +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/verify/VerifyGraphAddUsage.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/verify/VerifyGraphAddUsage.java new file mode 100644 index 00000000000..17d7dbaaaf6 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.phases/src/org/graalvm/compiler/phases/verify/VerifyGraphAddUsage.java @@ -0,0 +1,117 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.phases.verify; + +import jdk.vm.ci.meta.ResolvedJavaMethod; +import jdk.vm.ci.meta.ResolvedJavaType; +import org.graalvm.compiler.debug.GraalError; +import org.graalvm.compiler.graph.Graph; +import org.graalvm.compiler.graph.Node; +import org.graalvm.compiler.nodes.ConstantNode; +import org.graalvm.compiler.nodes.Invoke; +import org.graalvm.compiler.nodes.ParameterNode; +import org.graalvm.compiler.nodes.PiNode; +import org.graalvm.compiler.nodes.StructuredGraph; +import org.graalvm.compiler.nodes.ValueNode; +import org.graalvm.compiler.nodes.ValuePhiNode; +import org.graalvm.compiler.nodes.ValueProxyNode; +import org.graalvm.compiler.nodes.java.MethodCallTargetNode; +import org.graalvm.compiler.nodes.java.NewInstanceNode; +import org.graalvm.compiler.nodes.spi.LoweringProvider; +import org.graalvm.compiler.phases.VerifyPhase; +import org.graalvm.compiler.phases.tiers.PhaseContext; +import org.graalvm.util.EconomicSet; + +import java.lang.reflect.Constructor; +import java.lang.reflect.Method; + +public class VerifyGraphAddUsage extends VerifyPhase { + private static final Method ADD_OR_UNIQUE; + private static final Method CONSTRUCTOR_NEW_INSTANCE; + private static final EconomicSet> ALLOWED_CLASSES = EconomicSet.create(); + + static { + try { + ADD_OR_UNIQUE = Graph.class.getDeclaredMethod("addOrUnique", Node.class); + CONSTRUCTOR_NEW_INSTANCE = Constructor.class.getDeclaredMethod("newInstance", Object[].class); + } catch (NoSuchMethodException e) { + throw new GraalError(e); + } + + ALLOWED_CLASSES.add(Graph.class); + ALLOWED_CLASSES.add(LoweringProvider.class); + } + + @Override + protected boolean verify(StructuredGraph graph, PhaseContext context) { + boolean allowed = false; + for (Class cls : ALLOWED_CLASSES) { + ResolvedJavaType declaringClass = graph.method().getDeclaringClass(); + if (context.getMetaAccess().lookupJavaType(cls).isAssignableFrom(declaringClass)) { + allowed = true; + } + } + if (!allowed) { + ResolvedJavaMethod addOrUniqueMethod = context.getMetaAccess().lookupJavaMethod(ADD_OR_UNIQUE); + for (MethodCallTargetNode t : graph.getNodes(MethodCallTargetNode.TYPE)) { + ResolvedJavaMethod callee = t.targetMethod(); + if (callee.equals(addOrUniqueMethod)) { + ValueNode nodeArgument = t.arguments().get(1); + EconomicSet seen = EconomicSet.create(); + checkNonFactory(graph, seen, context, nodeArgument); + } + } + } + + return true; + } + + private void checkNonFactory(StructuredGraph graph, EconomicSet seen, PhaseContext context, ValueNode node) { + if (seen.contains(node)) { + return; + } + seen.add(node); + + // Check where the value came from recursively, or if it is allowed. + if (node instanceof ValuePhiNode) { + for (ValueNode input : ((ValuePhiNode) node).values()) { + checkNonFactory(graph, seen, context, input); + } + } else if (node instanceof PiNode) { + checkNonFactory(graph, seen, context, ((PiNode) node).object()); + } else if (node instanceof ParameterNode) { + return; + } else if (node instanceof ConstantNode) { + return; + } else if (node instanceof ValueProxyNode) { + checkNonFactory(graph, seen, context, ((ValueProxyNode) node).value()); + } else if (node instanceof Invoke && ((Invoke) node).callTarget().targetMethod().equals(context.getMetaAccess().lookupJavaMethod(CONSTRUCTOR_NEW_INSTANCE))) { + return; + } else if (!(node instanceof NewInstanceNode)) { + // In all other cases, the argument must be a new instance. + throw new VerificationError("Must add node '%s' with inputs in method '%s' of class '%s'.", + node, graph.method().getName(), graph.method().getDeclaringClass().getName()); + } + } + +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/BasicIdealGraphPrinter.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/BasicIdealGraphPrinter.java deleted file mode 100644 index 1e853dfe31c..00000000000 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/BasicIdealGraphPrinter.java +++ /dev/null @@ -1,366 +0,0 @@ -/* - * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ -package org.graalvm.compiler.printer; - -import java.io.BufferedOutputStream; -import java.io.OutputStream; -import java.io.PrintStream; -import java.io.UnsupportedEncodingException; -import java.nio.charset.Charset; -import java.util.Map; -import java.util.Map.Entry; - -/** - * Elementary, generic generator of Ideal Graph Visualizer input for use in printers for specific - * data structures. - */ -class BasicIdealGraphPrinter { - - /** - * Edge between two nodes. - */ - protected static class Edge { - - final String from; - final int fromIndex; - final String to; - final int toIndex; - final String label; - - public Edge(String from, int fromIndex, String to, int toIndex, String label) { - assert (from != null && to != null); - this.from = from; - this.fromIndex = fromIndex; - this.to = to; - this.toIndex = toIndex; - this.label = label; - } - - @Override - public int hashCode() { - int h = from.hashCode() ^ to.hashCode(); - h = 3 * h + fromIndex; - h = 5 * h + toIndex; - if (label != null) { - h ^= label.hashCode(); - } - return h; - } - - @Override - public boolean equals(Object obj) { - if (obj == this) { - return true; - } - if (obj instanceof Edge) { - Edge other = (Edge) obj; - return from.equals(other.from) && fromIndex == other.fromIndex && to.equals(other.to) && toIndex == other.toIndex && - (label == other.label || (label != null && label.equals(other.label))); - } - return false; - } - } - - private final PrintStream stream; - - /** - * Creates a new {@link IdealGraphPrinter} that writes to the specified output stream. - */ - protected BasicIdealGraphPrinter(OutputStream stream) { - try { - OutputStream buffered; - if (stream instanceof BufferedOutputStream) { - buffered = stream; - } else { - buffered = new BufferedOutputStream(stream, 256 * 1024); - } - this.stream = new PrintStream(buffered, false, Charset.defaultCharset().name()); - } catch (UnsupportedEncodingException e) { - throw new RuntimeException(e); - } - } - - /** - * Flushes any buffered output. - */ - protected void flush() { - stream.flush(); - } - - /** - * Starts a new graph document. - */ - protected void begin() { - stream.println(""); - } - - protected void beginGroup() { - stream.println(""); - } - - protected void beginMethod(String name, String shortName, int bci) { - stream.printf(" %n", escape(name), escape(shortName), bci); - } - - protected void beginBytecodes() { - stream.println(" \n"); - } - - protected void printBytecodes(String disassembly) { - beginBytecodes(); - stream.println(disassembly); - endBytecodes(); - } - - protected void endMethod() { - stream.println(" "); - } - - protected void beginGraph(String title) { - stream.printf(" %n", escape(title)); - } - - protected void beginProperties() { - stream.print(""); - } - - protected void printProperty(String name, String value) { - stream.printf("

%s

", escape(name), escape(value)); - } - - protected void endProperties() { - stream.print("
"); - } - - protected void printProperties(Map properties) { - beginProperties(); - for (Entry entry : properties.entrySet()) { - printProperty(entry.getKey(), entry.getValue()); - } - endProperties(); - } - - protected void beginNodes() { - stream.println(" "); - } - - protected void beginNode(String id) { - stream.printf(" ", escape(id)); - } - - protected void endNode() { - stream.println(" "); - } - - protected void printNode(String id, Map properties) { - beginNode(id); - if (properties != null) { - printProperties(properties); - } - endNode(); - } - - protected void endNodes() { - stream.println(" "); - } - - protected void beginEdges() { - stream.println(" "); - } - - protected void printEdge(Edge edge) { - stream.printf(" %n", escape(edge.from), edge.fromIndex, escape(edge.to), edge.toIndex, escape(edge.label)); - } - - protected void endEdges() { - stream.println(" "); - } - - protected void beginControlFlow() { - stream.println(" "); - } - - protected void beginBlock(String name) { - stream.printf(" %n", escape(name)); - } - - protected void beginSuccessors() { - stream.println(" "); - } - - protected void printSuccessor(String name) { - stream.printf(" %n", escape(name)); - } - - protected void endSuccessors() { - stream.println(" "); - } - - protected void beginBlockNodes() { - stream.println(" "); - } - - protected void printBlockNode(String nodeId) { - stream.printf(" %n", escape(nodeId)); - } - - protected void endBlockNodes() { - stream.println(" "); - } - - protected void endBlock() { - stream.println(" "); - } - - protected void endControlFlow() { - stream.println(" "); - } - - protected void endGraph() { - stream.println("
"); - } - - /** - * Ends the current group. - */ - public void endGroup() { - stream.println("
"); - } - - /** - * Finishes the graph document and flushes the output stream. - */ - protected void end() { - stream.println("
"); - flush(); - } - - public void close() { - end(); - stream.close(); - } - - public boolean isValid() { - return !stream.checkError(); - } - - private static String escape(String s) { - StringBuilder str = null; - for (int i = 0; i < s.length(); i++) { - char c = s.charAt(i); - switch (c) { - case '&': - case '<': - case '>': - case '"': - case '\'': - if (str == null) { - str = new StringBuilder(); - str.append(s, 0, i); - } - switch (c) { - case '&': - str.append("&"); - break; - case '<': - str.append("<"); - break; - case '>': - str.append(">"); - break; - case '"': - str.append("""); - break; - case '\'': - str.append("'"); - break; - default: - assert false; - } - break; - case '\u0000': - case '\u0001': - case '\u0002': - case '\u0003': - case '\u0004': - case '\u0005': - case '\u0006': - case '\u0007': - case '\u0008': - case '\u000b': - case '\u000c': - case '\u000e': - case '\u000f': - case '\u0010': - case '\u0011': - case '\u0012': - case '\u0013': - case '\u0014': - case '\u0015': - case '\u0016': - case '\u0017': - case '\u0018': - case '\u0019': - case '\u001a': - case '\u001b': - case '\u001c': - case '\u001d': - case '\u001e': - case '\u001f': - if (str == null) { - str = new StringBuilder(); - str.append(s, 0, i); - } - str.append("'0x").append(Integer.toHexString(c)); - break; - default: - if (str != null) { - str.append(c); - } - break; - } - } - if (str == null) { - return s; - } else { - return str.toString(); - } - } -} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/BinaryGraphPrinter.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/BinaryGraphPrinter.java index 03ebcce69d6..f40d78c6fc5 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/BinaryGraphPrinter.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/BinaryGraphPrinter.java @@ -26,18 +26,17 @@ import static org.graalvm.compiler.graph.Edges.Type.Inputs; import static org.graalvm.compiler.graph.Edges.Type.Successors; import java.io.IOException; -import java.nio.channels.WritableByteChannel; import java.util.Arrays; import java.util.Collection; import java.util.Collections; import java.util.LinkedList; import java.util.List; import java.util.Map; -import jdk.vm.ci.meta.ResolvedJavaField; import org.graalvm.compiler.api.replacements.SnippetReflectionProvider; import org.graalvm.compiler.bytecode.Bytecode; import org.graalvm.compiler.core.common.cfg.BlockMap; +import org.graalvm.compiler.debug.DebugContext; import org.graalvm.compiler.debug.DebugOptions; import org.graalvm.compiler.graph.CachedGraph; import org.graalvm.compiler.graph.Edges; @@ -46,6 +45,7 @@ import org.graalvm.compiler.graph.InputEdges; import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.graph.NodeClass; import org.graalvm.compiler.graph.NodeMap; +import org.graalvm.compiler.graph.NodeSourcePosition; import org.graalvm.compiler.nodes.AbstractBeginNode; import org.graalvm.compiler.nodes.AbstractEndNode; import org.graalvm.compiler.nodes.AbstractMergeNode; @@ -55,15 +55,11 @@ import org.graalvm.compiler.nodes.ControlSplitNode; import org.graalvm.compiler.nodes.FixedNode; import org.graalvm.compiler.nodes.PhiNode; import org.graalvm.compiler.nodes.ProxyNode; +import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.VirtualState; import org.graalvm.compiler.nodes.cfg.Block; import org.graalvm.compiler.nodes.cfg.ControlFlowGraph; -import jdk.vm.ci.meta.ResolvedJavaMethod; -import jdk.vm.ci.meta.ResolvedJavaType; -import jdk.vm.ci.meta.Signature; -import org.graalvm.compiler.debug.DebugContext; -import org.graalvm.compiler.graph.NodeSourcePosition; -import org.graalvm.compiler.nodes.StructuredGraph; +import org.graalvm.compiler.nodes.util.JavaConstantFormattable; import org.graalvm.compiler.phases.schedule.SchedulePhase; import org.graalvm.graphio.GraphBlocks; import org.graalvm.graphio.GraphElements; @@ -71,6 +67,11 @@ import org.graalvm.graphio.GraphOutput; import org.graalvm.graphio.GraphStructure; import org.graalvm.graphio.GraphTypes; +import jdk.vm.ci.meta.ResolvedJavaField; +import jdk.vm.ci.meta.ResolvedJavaMethod; +import jdk.vm.ci.meta.ResolvedJavaType; +import jdk.vm.ci.meta.Signature; + public class BinaryGraphPrinter implements GraphStructure, Edges>, GraphBlocks, @@ -79,8 +80,8 @@ public class BinaryGraphPrinter implements private final SnippetReflectionProvider snippetReflection; private final GraphOutput output; - public BinaryGraphPrinter(WritableByteChannel channel, SnippetReflectionProvider snippetReflection) throws IOException { - this.output = GraphOutput.newBuilder(this).blocks(this).elements(this).types(this).build(channel); + public BinaryGraphPrinter(DebugContext ctx, SnippetReflectionProvider snippetReflection) throws IOException { + this.output = ctx.buildOutput(GraphOutput.newBuilder(this).protocolVersion(5, 0).blocks(this).elements(this).types(this)); this.snippetReflection = snippetReflection; } @@ -115,17 +116,24 @@ public class BinaryGraphPrinter implements } } + @Override + public Node node(Object obj) { + return obj instanceof Node ? (Node) obj : null; + } + @Override public NodeClass nodeClass(Object obj) { if (obj instanceof NodeClass) { return (NodeClass) obj; } - if (obj instanceof Node) { - return ((Node) obj).getNodeClass(); - } return null; } + @Override + public NodeClass classForNode(Node node) { + return node.getNodeClass(); + } + @Override public Object nodeClassType(NodeClass node) { return node.getJavaClass(); @@ -257,6 +265,13 @@ public class BinaryGraphPrinter implements } props.put("category", "floating"); } + if (getSnippetReflectionProvider() != null) { + for (Map.Entry prop : props.entrySet()) { + if (prop.getValue() instanceof JavaConstantFormattable) { + props.put(prop.getKey(), ((JavaConstantFormattable) prop.getValue()).format(this)); + } + } + } } private Object getBlockForNode(Node node, NodeMap nodeToBlocks) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/CFGPrinterObserver.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/CFGPrinterObserver.java index 6b91eb09e17..ac408784cfd 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/CFGPrinterObserver.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/CFGPrinterObserver.java @@ -23,13 +23,13 @@ package org.graalvm.compiler.printer; import static org.graalvm.compiler.debug.DebugOptions.PrintCFG; -import static org.graalvm.compiler.printer.GraalDebugHandlersFactory.createDumpPath; import java.io.BufferedOutputStream; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStream; +import java.nio.file.Path; import java.util.ArrayList; import java.util.Collections; import java.util.List; @@ -153,8 +153,8 @@ public class CFGPrinterObserver implements DebugDumpHandler { if (cfgPrinter == null) { try { - Graph graph = debug.contextLookupTopdown(Graph.class); - cfgFile = createDumpPath(options, graph, "cfg", false).toFile(); + Path dumpFile = debug.getDumpPath(".cfg", false); + cfgFile = dumpFile.toFile(); OutputStream out = new BufferedOutputStream(new FileOutputStream(cfgFile)); cfgPrinter = new CFGPrinter(out); } catch (IOException e) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/CanonicalStringGraphPrinter.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/CanonicalStringGraphPrinter.java index f062850d20d..147dfd8b44a 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/CanonicalStringGraphPrinter.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/CanonicalStringGraphPrinter.java @@ -26,7 +26,6 @@ import static org.graalvm.compiler.debug.DebugOptions.CanonicalGraphStringsCheck import static org.graalvm.compiler.debug.DebugOptions.CanonicalGraphStringsExcludeVirtuals; import static org.graalvm.compiler.debug.DebugOptions.CanonicalGraphStringsRemoveIdentities; import static org.graalvm.compiler.debug.DebugOptions.PrintCanonicalGraphStringFlavor; -import static org.graalvm.compiler.printer.GraalDebugHandlersFactory.sanitizedFileName; import java.io.BufferedWriter; import java.io.FileWriter; @@ -44,6 +43,7 @@ import java.util.regex.Pattern; import org.graalvm.compiler.api.replacements.SnippetReflectionProvider; import org.graalvm.compiler.core.common.Fields; import org.graalvm.compiler.debug.DebugContext; +import org.graalvm.compiler.debug.PathUtilities; import org.graalvm.compiler.graph.Graph; import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.graph.NodeMap; @@ -260,11 +260,11 @@ public class CanonicalStringGraphPrinter implements GraphPrinter { private StructuredGraph currentGraph; private Path currentDirectory; - private Path getDirectory(StructuredGraph graph) throws IOException { + private Path getDirectory(DebugContext debug, StructuredGraph graph) { if (graph == currentGraph) { return currentDirectory; } - currentDirectory = GraalDebugHandlersFactory.createDumpPath(graph.getOptions(), graph, "graph-strings", true); + currentDirectory = debug.getDumpPath(".graph-strings", true); currentGraph = graph; return currentDirectory; } @@ -274,9 +274,9 @@ public class CanonicalStringGraphPrinter implements GraphPrinter { if (graph instanceof StructuredGraph) { OptionValues options = graph.getOptions(); StructuredGraph structuredGraph = (StructuredGraph) graph; - Path outDirectory = getDirectory(structuredGraph); + Path outDirectory = getDirectory(debug, structuredGraph); String title = String.format("%03d-%s.txt", id, String.format(format, simplifyClassArgs(args))); - Path filePath = outDirectory.resolve(sanitizedFileName(title)); + Path filePath = outDirectory.resolve(PathUtilities.sanitizeFileName(title)); try (PrintWriter writer = new PrintWriter(new BufferedWriter(new FileWriter(filePath.toFile())))) { switch (PrintCanonicalGraphStringFlavor.getValue(options)) { case 1: diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraalDebugHandlersFactory.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraalDebugHandlersFactory.java index 01ade572f42..b788848ef22 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraalDebugHandlersFactory.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraalDebugHandlersFactory.java @@ -22,44 +22,18 @@ */ package org.graalvm.compiler.printer; -import static org.graalvm.compiler.debug.DebugOptions.PrintBinaryGraphPort; -import static org.graalvm.compiler.debug.DebugOptions.PrintBinaryGraphs; -import static org.graalvm.compiler.debug.DebugOptions.PrintGraphHost; -import static org.graalvm.compiler.debug.DebugOptions.PrintXmlGraphPort; -import static org.graalvm.compiler.debug.DebugOptions.ShowDumpFiles; - -import java.io.File; -import java.io.IOException; -import java.io.InterruptedIOException; -import java.net.InetSocketAddress; -import java.net.Socket; -import java.nio.channels.ClosedByInterruptException; -import java.nio.channels.FileChannel; -import java.nio.channels.SocketChannel; -import java.nio.file.FileAlreadyExistsException; -import java.nio.file.Files; -import java.nio.file.InvalidPathException; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.nio.file.StandardOpenOption; import java.util.ArrayList; import java.util.List; -import java.util.concurrent.atomic.AtomicInteger; import org.graalvm.compiler.api.replacements.SnippetReflectionProvider; -import org.graalvm.compiler.core.common.CompilationIdentifier; -import org.graalvm.compiler.debug.Assertions; import org.graalvm.compiler.debug.DebugContext; import org.graalvm.compiler.debug.DebugDumpHandler; import org.graalvm.compiler.debug.DebugHandler; import org.graalvm.compiler.debug.DebugHandlersFactory; import org.graalvm.compiler.debug.DebugOptions; import org.graalvm.compiler.debug.TTY; -import org.graalvm.compiler.debug.PathUtilities; -import org.graalvm.compiler.graph.Graph; import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.nodeinfo.Verbosity; -import org.graalvm.compiler.nodes.StructuredGraph; import org.graalvm.compiler.nodes.util.GraphUtil; import org.graalvm.compiler.options.OptionValues; import org.graalvm.compiler.serviceprovider.ServiceProvider; @@ -80,17 +54,13 @@ public class GraalDebugHandlersFactory implements DebugHandlersFactory { @Override public List createHandlers(OptionValues options) { List handlers = new ArrayList<>(); - if (DebugOptions.PrintGraphFile.getValue(options)) { - handlers.add(new GraphPrinterDumpHandler((graph) -> createFilePrinter(graph, options, snippetReflection))); - } else { - handlers.add(new GraphPrinterDumpHandler((graph) -> createNetworkPrinter(graph, options, snippetReflection))); - } + handlers.add(new GraphPrinterDumpHandler((debug, graph) -> new BinaryGraphPrinter(debug, snippetReflection))); if (DebugOptions.PrintCanonicalGraphStrings.getValue(options)) { - handlers.add(new GraphPrinterDumpHandler((graph) -> createStringPrinter(snippetReflection))); + handlers.add(new GraphPrinterDumpHandler((debug, graph) -> createStringPrinter(snippetReflection))); } handlers.add(new NodeDumper()); if (DebugOptions.PrintCFG.getValue(options) || DebugOptions.PrintBackendCFG.getValue(options)) { - if (DebugOptions.PrintBinaryGraphs.getValue(options) && DebugOptions.PrintCFG.getValue(options)) { + if (DebugOptions.PrintCFG.getValue(options)) { TTY.out.println("Complete C1Visualizer dumping slows down PrintBinaryGraphs: use -Dgraal.PrintCFG=false to disable it"); } handlers.add(new CFGPrinterObserver()); @@ -119,162 +89,4 @@ public class GraalDebugHandlersFactory implements DebugHandlersFactory { return new CanonicalStringGraphPrinter(snippetReflection); } - public static String sanitizedFileName(String n) { - /* - * First ensure that the name does not contain the directory separator (which would be - * considered a valid path). - */ - String name = n.replace(File.separatorChar, '_'); - - try { - Paths.get(name); - return name; - } catch (InvalidPathException e) { - // fall through - } - StringBuilder buf = new StringBuilder(name.length()); - for (int i = 0; i < name.length(); i++) { - char c = name.charAt(i); - try { - Paths.get(String.valueOf(c)); - } catch (InvalidPathException e) { - buf.append('_'); - } - buf.append(c); - } - return buf.toString(); - } - - private static GraphPrinter createNetworkPrinter(Graph graph, OptionValues options, SnippetReflectionProvider snippetReflection) throws IOException { - String host = PrintGraphHost.getValue(options); - int port = PrintBinaryGraphs.getValue(options) ? PrintBinaryGraphPort.getValue(options) : PrintXmlGraphPort.getValue(options); - try { - GraphPrinter printer; - if (DebugOptions.PrintBinaryGraphs.getValue(options)) { - printer = new BinaryGraphPrinter(SocketChannel.open(new InetSocketAddress(host, port)), snippetReflection); - } else { - printer = new IdealGraphPrinter(new Socket(host, port).getOutputStream(), true, snippetReflection); - } - TTY.println("Connected to the IGV on %s:%d", host, port); - return printer; - } catch (ClosedByInterruptException | InterruptedIOException e) { - /* - * Interrupts should not count as errors because they may be caused by a cancelled Graal - * compilation. ClosedByInterruptException occurs if the SocketChannel could not be - * opened. InterruptedIOException occurs if new Socket(..) was interrupted. - */ - return null; - } catch (IOException e) { - if (!DebugOptions.PrintGraphFile.hasBeenSet(options)) { - return createFilePrinter(graph, options, snippetReflection); - } else { - throw new IOException(String.format("Could not connect to the IGV on %s:%d", host, port), e); - } - } - } - - private static final AtomicInteger unknownCompilationId = new AtomicInteger(); - - /** - * Creates a new file or directory for dumping based on a given graph and a file extension. - * - * @param graph a base path name is derived from {@code graph} - * @param extension a suffix which if non-null and non-empty added to the end of the returned - * path separated by a {@code "."} - * @param createDirectory specifies if this is a request to create a directory instead of a file - * @return the created directory or file - * @throws IOException if there was an error creating the directory or file - */ - static Path createDumpPath(OptionValues options, Graph graph, String extension, boolean createDirectory) throws IOException { - CompilationIdentifier compilationId = CompilationIdentifier.INVALID_COMPILATION_ID; - String id = null; - String label = null; - if (graph instanceof StructuredGraph) { - StructuredGraph sgraph = (StructuredGraph) graph; - label = getGraphName(sgraph); - compilationId = sgraph.compilationId(); - if (compilationId == CompilationIdentifier.INVALID_COMPILATION_ID) { - id = graph.getClass().getSimpleName() + "-" + sgraph.graphId(); - } else { - id = compilationId.toString(CompilationIdentifier.Verbosity.ID); - } - } else { - label = graph == null ? null : graph.name != null ? graph.name : graph.toString(); - id = "UnknownCompilation-" + unknownCompilationId.incrementAndGet(); - } - String ext = PathUtilities.formatExtension(extension); - Path result = createUnique(DebugOptions.getDumpDirectory(options), id, label, ext, createDirectory); - if (ShowDumpFiles.getValue(options) || Assertions.assertionsEnabled()) { - TTY.println("Dumping debug output to %s", result.toAbsolutePath().toString()); - } - return result; - } - - /** - * A maximum file name length supported by most file systems. There is no platform independent - * way to get this in Java. - */ - private static final int MAX_FILE_NAME_LENGTH = 255; - - private static final String ELLIPSIS = "..."; - - private static Path createUnique(Path dumpDir, String id, String label, String ext, boolean createDirectory) throws IOException { - String timestamp = ""; - for (;;) { - int fileNameLengthWithoutLabel = timestamp.length() + ext.length() + id.length() + "[]".length(); - int labelLengthLimit = MAX_FILE_NAME_LENGTH - fileNameLengthWithoutLabel; - String fileName; - if (labelLengthLimit < ELLIPSIS.length()) { - // This means `id` is very long - String suffix = timestamp + ext; - int idLengthLimit = Math.min(MAX_FILE_NAME_LENGTH - suffix.length(), id.length()); - fileName = sanitizedFileName(id.substring(0, idLengthLimit) + suffix); - } else { - if (label == null) { - fileName = sanitizedFileName(id + timestamp + ext); - } else { - String adjustedLabel = label; - if (label.length() > labelLengthLimit) { - adjustedLabel = label.substring(0, labelLengthLimit - ELLIPSIS.length()) + ELLIPSIS; - } - fileName = sanitizedFileName(id + '[' + adjustedLabel + ']' + timestamp + ext); - } - } - Path result = dumpDir.resolve(fileName); - try { - if (createDirectory) { - return Files.createDirectory(result); - } else { - return Files.createFile(result); - } - } catch (FileAlreadyExistsException e) { - timestamp = "_" + Long.toString(System.currentTimeMillis()); - } - } - } - - private static String getGraphName(StructuredGraph graph) { - if (graph.name != null) { - return graph.name; - } else if (graph.method() != null) { - return graph.method().format("%h.%n(%p)").replace(" ", ""); - } else { - return graph.toString(); - } - } - - private static GraphPrinter createFilePrinter(Graph graph, OptionValues options, SnippetReflectionProvider snippetReflection) throws IOException { - Path path = createDumpPath(options, graph, PrintBinaryGraphs.getValue(options) ? "bgv" : "gv.xml", false); - try { - GraphPrinter printer; - if (DebugOptions.PrintBinaryGraphs.getValue(options)) { - printer = new BinaryGraphPrinter(FileChannel.open(path, StandardOpenOption.WRITE), snippetReflection); - } else { - printer = new IdealGraphPrinter(Files.newOutputStream(path), true, snippetReflection); - } - return printer; - } catch (IOException e) { - throw new IOException(String.format("Failed to open %s to dump IGV graphs", path), e); - } - } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraphPrinter.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraphPrinter.java index 194e86e53e6..979dd3e85ee 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraphPrinter.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraphPrinter.java @@ -35,6 +35,7 @@ import org.graalvm.compiler.debug.DebugContext.Scope; import org.graalvm.compiler.graph.Graph; import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.StructuredGraph; +import org.graalvm.compiler.nodes.util.JavaConstantFormatter; import org.graalvm.compiler.phases.schedule.SchedulePhase; import org.graalvm.compiler.serviceprovider.JDK9Method; @@ -46,7 +47,7 @@ import jdk.vm.ci.meta.ResolvedJavaMethod; import jdk.vm.ci.runtime.JVMCI; import jdk.vm.ci.services.Services; -interface GraphPrinter extends Closeable { +interface GraphPrinter extends Closeable, JavaConstantFormatter { /** * Starts a new group of graphs with the given name, short name and method byte code index (BCI) @@ -116,30 +117,44 @@ interface GraphPrinter extends Closeable { return true; } } + if (c.getClassLoader() == GraphPrinter.class.getClassLoader()) { + return true; + } return false; } + /** + * Use the real {@link Object#toString()} method for {@link JavaConstant JavaConstants} that are + * wrapping trusted types, other just return the results of {@link JavaConstant#toString()}. + */ + @Override + default String format(JavaConstant constant) { + SnippetReflectionProvider snippetReflection = getSnippetReflectionProvider(); + if (snippetReflection != null) { + if (constant.getJavaKind() == JavaKind.Object) { + Object obj = snippetReflection.asObject(Object.class, constant); + if (obj != null) { + return GraphPrinter.constantToString(obj); + } + } + } + return constant.toString(); + } + /** * Sets or updates the {@code "rawvalue"} and {@code "toString"} properties in {@code props} for * {@code cn} if it's a boxed Object value and {@code snippetReflection} can access the raw * value. */ default void updateStringPropertiesForConstant(Map props, ConstantNode cn) { - SnippetReflectionProvider snippetReflection = getSnippetReflectionProvider(); - if (snippetReflection != null && cn.getValue() instanceof JavaConstant) { - JavaConstant constant = (JavaConstant) cn.getValue(); - if (constant.getJavaKind() == JavaKind.Object) { - Object obj = snippetReflection.asObject(Object.class, constant); - if (obj != null) { - String toString = GraphPrinter.constantToString(obj); - String rawvalue = GraphPrinter.truncate(toString); - // Overwrite the value inserted by - // ConstantNode.getDebugProperties() - props.put("rawvalue", rawvalue); - if (!rawvalue.equals(toString)) { - props.put("toString", toString); - } - } + if (cn.isJavaConstant() && cn.getStackKind().isObject()) { + String toString = format(cn.asJavaConstant()); + String rawvalue = GraphPrinter.truncate(toString); + // Overwrite the value inserted by + // ConstantNode.getDebugProperties() + props.put("rawvalue", rawvalue); + if (!rawvalue.equals(toString)) { + props.put("toString", toString); } } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraphPrinterDumpHandler.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraphPrinterDumpHandler.java index e1bb20c91b4..a875ca30b01 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraphPrinterDumpHandler.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/GraphPrinterDumpHandler.java @@ -52,8 +52,8 @@ import jdk.vm.ci.meta.ResolvedJavaMethod; //JaCoCo Exclude /** - * Observes compilation events and uses {@link IdealGraphPrinter} to generate a graph representation - * that can be inspected with the Graph Visualizer. + * Observes compilation events and uses {@link BinaryGraphPrinter} to generate a graph + * representation that can be inspected with the Graph Visualizer. */ public class GraphPrinterDumpHandler implements DebugDumpHandler { @@ -69,7 +69,7 @@ public class GraphPrinterDumpHandler implements DebugDumpHandler { @FunctionalInterface public interface GraphPrinterSupplier { - GraphPrinter get(Graph graph) throws IOException; + GraphPrinter get(DebugContext ctx, Graph graph) throws IOException; } /** @@ -93,7 +93,7 @@ public class GraphPrinterDumpHandler implements DebugDumpHandler { } } - private void ensureInitialized(Graph graph) { + private void ensureInitialized(DebugContext ctx, Graph graph) { if (printer == null) { if (failuresCount >= FAILURE_LIMIT) { return; @@ -102,7 +102,7 @@ public class GraphPrinterDumpHandler implements DebugDumpHandler { inlineContextMap = new WeakHashMap<>(); DebugContext debug = graph.getDebug(); try { - printer = printerSupplier.get(graph); + printer = printerSupplier.get(ctx, graph); } catch (IOException e) { handleException(debug, e); } @@ -123,7 +123,7 @@ public class GraphPrinterDumpHandler implements DebugDumpHandler { OptionValues options = debug.getOptions(); if (object instanceof Graph && DebugOptions.PrintGraph.getValue(options)) { final Graph graph = (Graph) object; - ensureInitialized(graph); + ensureInitialized(debug, graph); if (printer == null) { return; } @@ -168,11 +168,13 @@ public class GraphPrinterDumpHandler implements DebugDumpHandler { // Save inline context for next dump. previousInlineContext = inlineContext; + // Capture before creating the sandbox + String currentScopeName = debug.getCurrentScopeName(); try (DebugContext.Scope s = debug.sandbox("PrintingGraph", null)) { // Finally, output the graph. Map properties = new HashMap<>(); properties.put("graph", graph.toString()); - properties.put("scope", debug.getCurrentScopeName()); + properties.put("scope", currentScopeName); if (graph instanceof StructuredGraph) { properties.put("compilationIdentifier", ((StructuredGraph) graph).compilationId()); try { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/IdealGraphPrinter.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/IdealGraphPrinter.java deleted file mode 100644 index 493e62df1a5..00000000000 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.printer/src/org/graalvm/compiler/printer/IdealGraphPrinter.java +++ /dev/null @@ -1,356 +0,0 @@ -/* - * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ -package org.graalvm.compiler.printer; - -import java.io.OutputStream; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Map; -import java.util.Map.Entry; -import java.util.Set; - -import org.graalvm.compiler.api.replacements.SnippetReflectionProvider; -import org.graalvm.compiler.bytecode.BytecodeDisassembler; -import org.graalvm.compiler.debug.DebugContext; -import org.graalvm.compiler.debug.DebugOptions; -import org.graalvm.compiler.graph.Graph; -import org.graalvm.compiler.graph.Node; -import org.graalvm.compiler.graph.NodeMap; -import org.graalvm.compiler.graph.Position; -import org.graalvm.compiler.nodeinfo.Verbosity; -import org.graalvm.compiler.nodes.AbstractMergeNode; -import org.graalvm.compiler.nodes.BeginNode; -import org.graalvm.compiler.nodes.ConstantNode; -import org.graalvm.compiler.nodes.EndNode; -import org.graalvm.compiler.nodes.ParameterNode; -import org.graalvm.compiler.nodes.PhiNode; -import org.graalvm.compiler.nodes.StateSplit; -import org.graalvm.compiler.nodes.StructuredGraph; -import org.graalvm.compiler.nodes.StructuredGraph.ScheduleResult; -import org.graalvm.compiler.nodes.cfg.Block; -import org.graalvm.compiler.nodes.cfg.ControlFlowGraph; -import org.graalvm.compiler.options.OptionValues; -import org.graalvm.compiler.phases.schedule.SchedulePhase; -import org.graalvm.util.EconomicSet; -import org.graalvm.util.Equivalence; - -import jdk.vm.ci.meta.ResolvedJavaMethod; - -/** - * Generates a representation of {@link Graph Graphs} that can be visualized and inspected with the - * Ideal Graph Visualizer. - */ -public class IdealGraphPrinter extends BasicIdealGraphPrinter implements GraphPrinter { - - private final boolean tryToSchedule; - private final SnippetReflectionProvider snippetReflection; - - /** - * Creates a new {@link IdealGraphPrinter} that writes to the specified output stream. - * - * @param tryToSchedule If false, no scheduling is done, which avoids exceptions for - * non-schedulable graphs. - */ - public IdealGraphPrinter(OutputStream stream, boolean tryToSchedule, SnippetReflectionProvider snippetReflection) { - super(stream); - this.snippetReflection = snippetReflection; - this.begin(); - this.tryToSchedule = tryToSchedule; - } - - @Override - public SnippetReflectionProvider getSnippetReflectionProvider() { - return snippetReflection; - } - - /** - * Starts a new group of graphs with the given name, short name and method byte code index (BCI) - * as properties. - */ - @Override - public void beginGroup(DebugContext debug, String name, String shortName, ResolvedJavaMethod method, int bci, Map properties) { - beginGroup(); - beginProperties(); - printProperty("name", name); - if (properties != null) { - for (Entry entry : properties.entrySet()) { - printProperty(entry.getKey().toString(), entry.getValue().toString()); - } - } - endProperties(); - beginMethod(name, shortName, bci); - if (method != null && method.getCode() != null) { - printBytecodes(new BytecodeDisassembler(false).disassemble(method)); - } - endMethod(); - } - - /** - * Prints an entire {@link Graph} with the specified title, optionally using short names for - * nodes. - */ - @Override - public void print(DebugContext debug, Graph graph, Map properties, int id, String format, Object... args) { - String title = id + ": " + String.format(format, simplifyClassArgs(args)); - beginGraph(title); - EconomicSet noBlockNodes = EconomicSet.create(Equivalence.IDENTITY); - ScheduleResult schedule = null; - if (graph instanceof StructuredGraph) { - StructuredGraph structuredGraph = (StructuredGraph) graph; - schedule = structuredGraph.getLastSchedule(); - if (schedule == null && tryToSchedule) { - OptionValues options = graph.getOptions(); - if (DebugOptions.PrintGraphWithSchedule.getValue(options)) { - try { - SchedulePhase schedulePhase = new SchedulePhase(options); - schedulePhase.apply(structuredGraph); - schedule = structuredGraph.getLastSchedule(); - } catch (Throwable t) { - } - } - } - } - ControlFlowGraph cfg = schedule == null ? null : schedule.getCFG(); - - if (properties != null) { - beginProperties(); - for (Entry entry : properties.entrySet()) { - printProperty(entry.getKey().toString(), entry.getValue().toString()); - } - endProperties(); - } - - beginNodes(); - List edges = printNodes(graph, cfg == null ? null : cfg.getNodeToBlock(), noBlockNodes); - endNodes(); - - beginEdges(); - for (Edge edge : edges) { - printEdge(edge); - } - endEdges(); - - if (cfg != null && cfg.getBlocks() != null) { - beginControlFlow(); - for (Block block : cfg.getBlocks()) { - printBlock(graph, block, cfg.getNodeToBlock()); - } - printNoBlock(noBlockNodes); - endControlFlow(); - } - - endGraph(); - flush(); - } - - private List printNodes(Graph graph, NodeMap nodeToBlock, EconomicSet noBlockNodes) { - ArrayList edges = new ArrayList<>(); - - NodeMap>> colors = graph.createNodeMap(); - NodeMap>> colorsToString = graph.createNodeMap(); - NodeMap> bits = graph.createNodeMap(); - - for (Node node : graph.getNodes()) { - - beginNode(node.toString(Verbosity.Id)); - beginProperties(); - printProperty("idx", node.toString(Verbosity.Id)); - - Map props = node.getDebugProperties(); - if (!props.containsKey("name") || props.get("name").toString().trim().length() == 0) { - String name = node.toString(Verbosity.Name); - printProperty("name", name); - } - printProperty("class", node.getClass().getSimpleName()); - - Block block = nodeToBlock == null || nodeToBlock.isNew(node) ? null : nodeToBlock.get(node); - if (block != null) { - printProperty("block", Integer.toString(block.getId())); - // if (!(node instanceof PhiNode || node instanceof FrameState || node instanceof - // ParameterNode) && !block.nodes().contains(node)) { - // printProperty("notInOwnBlock", "true"); - // } - } else { - printProperty("block", "noBlock"); - noBlockNodes.add(node); - } - - Set> nodeColors = colors.get(node); - if (nodeColors != null) { - for (Entry color : nodeColors) { - String name = color.getKey(); - Integer value = color.getValue(); - printProperty(name, Integer.toString(value)); - } - } - Set> nodeColorStrings = colorsToString.get(node); - if (nodeColorStrings != null) { - for (Entry color : nodeColorStrings) { - String name = color.getKey(); - String value = color.getValue(); - printProperty(name, value); - } - } - Set nodeBits = bits.get(node); - if (nodeBits != null) { - for (String bit : nodeBits) { - printProperty(bit, "true"); - } - } - if (node instanceof BeginNode) { - printProperty("shortName", "B"); - } else if (node.getClass() == EndNode.class) { - printProperty("shortName", "E"); - } else if (node instanceof ConstantNode) { - ConstantNode cn = (ConstantNode) node; - updateStringPropertiesForConstant(props, cn); - } - if (node.predecessor() != null) { - printProperty("hasPredecessor", "true"); - } - - try { - printProperty("NodeCost-Size", node.estimatedNodeSize().toString()); - printProperty("NodeCost-Cycles", node.estimatedNodeCycles().toString()); - } catch (Throwable t) { - props.put("node-cost-exception", t.getMessage()); - } - - for (Entry entry : props.entrySet()) { - String key = entry.getKey().toString(); - Object value = entry.getValue(); - String valueString; - if (value == null) { - valueString = "null"; - } else { - Class type = value.getClass(); - if (type.isArray()) { - if (!type.getComponentType().isPrimitive()) { - valueString = Arrays.toString((Object[]) value); - } else if (type.getComponentType() == Integer.TYPE) { - valueString = Arrays.toString((int[]) value); - } else if (type.getComponentType() == Double.TYPE) { - valueString = Arrays.toString((double[]) value); - } else { - valueString = toString(); - } - } else { - valueString = value.toString(); - } - } - printProperty(key, valueString); - } - - endProperties(); - endNode(); - - // successors - int fromIndex = 0; - for (Position position : node.successorPositions()) { - Node successor = position.get(node); - if (successor != null) { - edges.add(new Edge(node.toString(Verbosity.Id), fromIndex, successor.toString(Verbosity.Id), 0, position.getName())); - } - fromIndex++; - } - - // inputs - int toIndex = 1; - for (Position position : node.inputPositions()) { - Node input = position.get(node); - if (input != null) { - edges.add(new Edge(input.toString(Verbosity.Id), input.successors().count(), node.toString(Verbosity.Id), toIndex, position.getName())); - } - toIndex++; - } - } - - return edges; - } - - private void printBlock(Graph graph, Block block, NodeMap nodeToBlock) { - beginBlock(Integer.toString(block.getId())); - beginSuccessors(); - for (Block sux : block.getSuccessors()) { - if (sux != null) { - printSuccessor(Integer.toString(sux.getId())); - } - } - endSuccessors(); - beginBlockNodes(); - - EconomicSet nodes = EconomicSet.create(Equivalence.IDENTITY); - - if (nodeToBlock != null) { - for (Node n : graph.getNodes()) { - Block blk = nodeToBlock.isNew(n) ? null : nodeToBlock.get(n); - if (blk == block) { - nodes.add(n); - } - } - } - - if (nodes.size() > 0) { - // if this is the first block: add all locals to this block - if (block.getBeginNode() == ((StructuredGraph) graph).start()) { - for (Node node : graph.getNodes()) { - if (node instanceof ParameterNode) { - nodes.add(node); - } - } - } - - EconomicSet snapshot = EconomicSet.create(Equivalence.IDENTITY, nodes); - // add all framestates and phis to their blocks - for (Node node : snapshot) { - if (node instanceof StateSplit && ((StateSplit) node).stateAfter() != null) { - nodes.add(((StateSplit) node).stateAfter()); - } - if (node instanceof AbstractMergeNode) { - for (PhiNode phi : ((AbstractMergeNode) node).phis()) { - nodes.add(phi); - } - } - } - - for (Node node : nodes) { - printBlockNode(node.toString(Verbosity.Id)); - } - } - endBlockNodes(); - endBlock(); - } - - private void printNoBlock(EconomicSet noBlockNodes) { - if (!noBlockNodes.isEmpty()) { - beginBlock("noBlock"); - beginBlockNodes(); - for (Node node : noBlockNodes) { - printBlockNode(node.toString(Verbosity.Id)); - } - endBlockNodes(); - endBlock(); - } - } -} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.amd64/src/org/graalvm/compiler/replacements/amd64/AMD64StringIndexOfNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.amd64/src/org/graalvm/compiler/replacements/amd64/AMD64StringIndexOfNode.java index 87dedbd92e8..67e858a0cdb 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.amd64/src/org/graalvm/compiler/replacements/amd64/AMD64StringIndexOfNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements.amd64/src/org/graalvm/compiler/replacements/amd64/AMD64StringIndexOfNode.java @@ -22,13 +22,13 @@ */ package org.graalvm.compiler.replacements.amd64; -import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_256; import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_64; import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.graph.NodeClass; import org.graalvm.compiler.graph.NodeInputList; import org.graalvm.compiler.nodeinfo.InputType; +import org.graalvm.compiler.nodeinfo.NodeCycles; import org.graalvm.compiler.nodeinfo.NodeInfo; import org.graalvm.compiler.nodes.FixedWithNextNode; import org.graalvm.compiler.nodes.NamedLocationIdentity; @@ -44,7 +44,7 @@ import org.graalvm.word.Pointer; import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.Value; -@NodeInfo(size = SIZE_64, cycles = CYCLES_256) +@NodeInfo(size = SIZE_64, cycles = NodeCycles.CYCLES_UNKNOWN) public class AMD64StringIndexOfNode extends FixedWithNextNode implements LIRLowerable, MemoryAccess { public static final NodeClass TYPE = NodeClass.create(AMD64StringIndexOfNode.class); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/ConstantBindingParameterPlugin.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/ConstantBindingParameterPlugin.java index 9a1fecea546..09622018bed 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/ConstantBindingParameterPlugin.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/ConstantBindingParameterPlugin.java @@ -63,7 +63,7 @@ public class ConstantBindingParameterPlugin implements ParameterPlugin { * This is a node from another graph, so copy over extra state into a new * ConstantNode. */ - constantNode = ConstantNode.forConstant(stamp.getTrustedStamp(), otherCon.asConstant(), otherCon.getStableDimension(), otherCon.isDefaultStable(), metaAccess); + constantNode = ConstantNode.forConstant(stamp.getTrustedStamp(), otherCon.getValue(), otherCon.getStableDimension(), otherCon.isDefaultStable(), metaAccess); } else { constantNode = otherCon; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/DefaultJavaLoweringProvider.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/DefaultJavaLoweringProvider.java index 0c8f8ba2859..dd0746320f3 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/DefaultJavaLoweringProvider.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/DefaultJavaLoweringProvider.java @@ -33,6 +33,7 @@ import static org.graalvm.compiler.nodes.NamedLocationIdentity.ARRAY_LENGTH_LOCA import static org.graalvm.compiler.nodes.java.ArrayLengthNode.readArrayLength; import static org.graalvm.compiler.nodes.util.GraphUtil.skipPiWhileNonNull; +import java.nio.ByteOrder; import java.util.ArrayList; import java.util.BitSet; import java.util.List; @@ -43,6 +44,7 @@ import org.graalvm.compiler.api.replacements.SnippetReflectionProvider; import org.graalvm.compiler.core.common.spi.ForeignCallDescriptor; import org.graalvm.compiler.core.common.spi.ForeignCallsProvider; import org.graalvm.compiler.core.common.type.IntegerStamp; +import org.graalvm.compiler.core.common.type.ObjectStamp; import org.graalvm.compiler.core.common.type.Stamp; import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.core.common.type.TypeReference; @@ -50,6 +52,7 @@ import org.graalvm.compiler.debug.DebugHandlersFactory; import org.graalvm.compiler.debug.GraalError; import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.nodeinfo.InputType; +import org.graalvm.compiler.nodes.CompressionNode.CompressionOp; import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.FieldLocationIdentity; import org.graalvm.compiler.nodes.FixedNode; @@ -69,6 +72,7 @@ import org.graalvm.compiler.nodes.calc.NarrowNode; import org.graalvm.compiler.nodes.calc.RightShiftNode; import org.graalvm.compiler.nodes.calc.SignExtendNode; import org.graalvm.compiler.nodes.calc.SubNode; +import org.graalvm.compiler.nodes.calc.UnpackEndianHalfNode; import org.graalvm.compiler.nodes.calc.ZeroExtendNode; import org.graalvm.compiler.nodes.debug.VerifyHeapNode; import org.graalvm.compiler.nodes.extended.BoxNode; @@ -150,14 +154,16 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { protected final MetaAccessProvider metaAccess; protected final ForeignCallsProvider foreignCalls; protected final TargetDescription target; + private final boolean useCompressedOops; private BoxingSnippets.Templates boxingSnippets; private ConstantStringIndexOfSnippets.Templates indexOfSnippets; - public DefaultJavaLoweringProvider(MetaAccessProvider metaAccess, ForeignCallsProvider foreignCalls, TargetDescription target) { + public DefaultJavaLoweringProvider(MetaAccessProvider metaAccess, ForeignCallsProvider foreignCalls, TargetDescription target, boolean useCompressedOops) { this.metaAccess = metaAccess; this.foreignCalls = foreignCalls; this.target = target; + this.useCompressedOops = useCompressedOops; } public void initialize(OptionValues options, Iterable factories, SnippetCounter.Group.Factory factory, Providers providers, SnippetReflectionProvider snippetReflection) { @@ -218,11 +224,18 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { lowerBinaryMath((BinaryMathIntrinsicNode) n, tool); } else if (n instanceof StringIndexOfNode) { lowerIndexOf((StringIndexOfNode) n); + } else if (n instanceof UnpackEndianHalfNode) { + lowerSecondHalf((UnpackEndianHalfNode) n); } else { throw GraalError.shouldNotReachHere("Node implementing Lowerable not handled: " + n); } } + private void lowerSecondHalf(UnpackEndianHalfNode n) { + ByteOrder byteOrder = target.arch.getByteOrder(); + n.lower(byteOrder); + } + private void lowerIndexOf(StringIndexOfNode n) { if (n.getArgument(3).isConstant()) { SnippetLowering lowering = new SnippetLowering() { @@ -326,19 +339,21 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { } } + protected abstract JavaKind getStorageKind(ResolvedJavaField field); + protected void lowerLoadFieldNode(LoadFieldNode loadField, LoweringTool tool) { assert loadField.getStackKind() != JavaKind.Illegal; StructuredGraph graph = loadField.graph(); ResolvedJavaField field = loadField.field(); ValueNode object = loadField.isStatic() ? staticFieldBase(graph, field) : loadField.object(); object = createNullCheckedValue(object, loadField, tool); - Stamp loadStamp = loadStamp(loadField.stamp(), field.getJavaKind()); + Stamp loadStamp = loadStamp(loadField.stamp(), getStorageKind(field)); AddressNode address = createFieldAddress(graph, object, field); assert address != null : "Field that is loaded must not be eliminated: " + field.getDeclaringClass().toJavaName(true) + "." + field.getName(); ReadNode memoryRead = graph.add(new ReadNode(address, fieldLocationIdentity(field), loadStamp, fieldLoadBarrierType(field))); - ValueNode readValue = implicitLoadConvert(graph, field.getJavaKind(), memoryRead); + ValueNode readValue = implicitLoadConvert(graph, getStorageKind(field), memoryRead); loadField.replaceAtUsages(readValue); graph.replaceFixed(loadField, memoryRead); @@ -355,7 +370,7 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { ResolvedJavaField field = storeField.field(); ValueNode object = storeField.isStatic() ? staticFieldBase(graph, field) : storeField.object(); object = createNullCheckedValue(object, storeField, tool); - ValueNode value = implicitStoreConvert(graph, storeField.field().getJavaKind(), storeField.value()); + ValueNode value = implicitStoreConvert(graph, getStorageKind(storeField.field()), storeField.value()); AddressNode address = createFieldAddress(graph, object, field); assert address != null; @@ -651,9 +666,7 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { protected void lowerJavaWriteNode(JavaWriteNode write) { StructuredGraph graph = write.graph(); - JavaKind valueKind = write.getWriteKind(); - ValueNode value = implicitStoreConvert(graph, valueKind, write.value(), write.isCompressible()); - + ValueNode value = implicitStoreConvert(graph, write.getWriteKind(), write.value(), write.isCompressible()); WriteNode memoryWrite = graph.add(new WriteNode(write.getAddress(), write.getLocationIdentity(), value, write.getBarrierType())); memoryWrite.setStateAfter(write.stateAfter()); graph.replaceFixedWithFixed(write, memoryWrite); @@ -918,10 +931,20 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { return loadStamp(stamp, kind, true); } + private boolean useCompressedOops(JavaKind kind, boolean compressible) { + return kind == JavaKind.Object && compressible && useCompressedOops; + } + + protected abstract Stamp loadCompressedStamp(ObjectStamp stamp); + /** * @param compressible whether the stamp should be compressible */ protected Stamp loadStamp(Stamp stamp, JavaKind kind, boolean compressible) { + if (useCompressedOops(kind, compressible)) { + return loadCompressedStamp((ObjectStamp) stamp); + } + switch (kind) { case Boolean: case Byte: @@ -949,10 +972,16 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { return ret; } + protected abstract ValueNode newCompressionNode(CompressionOp op, ValueNode value); + /** - * @param compressible whether the covert should be compressible + * @param compressible whether the convert should be compressible */ protected ValueNode implicitLoadConvert(JavaKind kind, ValueNode value, boolean compressible) { + if (useCompressedOops(kind, compressible)) { + return newCompressionNode(CompressionOp.Uncompress, value); + } + switch (kind) { case Byte: case Short: @@ -984,6 +1013,10 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { * @param compressible whether the covert should be compressible */ protected ValueNode implicitStoreConvert(JavaKind kind, ValueNode value, boolean compressible) { + if (useCompressedOops(kind, compressible)) { + return newCompressionNode(CompressionOp.Compress, value); + } + switch (kind) { case Boolean: case Byte: @@ -1011,9 +1044,8 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { LogicNode boundsCheck = IntegerBelowNode.create(n.index(), arrayLength); if (boundsCheck.isTautology()) { return null; - } else { - return tool.createGuard(n, graph.addOrUniqueWithInputs(boundsCheck), BoundsCheckException, InvalidateReprofile); } + return tool.createGuard(n, graph.addOrUniqueWithInputs(boundsCheck), BoundsCheckException, InvalidateReprofile); } protected GuardingNode createNullCheck(ValueNode object, FixedNode before, LoweringTool tool) { @@ -1027,9 +1059,8 @@ public abstract class DefaultJavaLoweringProvider implements LoweringProvider { GuardingNode nullCheck = createNullCheck(object, before, tool); if (nullCheck == null) { return object; - } else { - return before.graph().maybeAddOrUnique(PiNode.create(object, (object.stamp()).join(StampFactory.objectNonNull()), (ValueNode) nullCheck)); } + return before.graph().maybeAddOrUnique(PiNode.create(object, (object.stamp()).join(StampFactory.objectNonNull()), (ValueNode) nullCheck)); } @Override diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/GraphKit.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/GraphKit.java index 5a2be59ea7f..99d6ff7d413 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/GraphKit.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/GraphKit.java @@ -75,7 +75,6 @@ import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaType; import jdk.vm.ci.meta.MetaAccessProvider; import jdk.vm.ci.meta.ResolvedJavaMethod; -import jdk.vm.ci.meta.ResolvedJavaType; import jdk.vm.ci.meta.Signature; /** @@ -252,6 +251,10 @@ public class GraphKit implements GraphBuilderTool { return new MethodCallTargetNode(invokeKind, targetMethod, args, returnStamp, null); } + protected final JavaKind asKind(JavaType type) { + return wordTypes != null ? wordTypes.asKind(type) : type.getJavaKind(); + } + /** * Determines if a given set of arguments is compatible with the signature of a given method. * @@ -267,14 +270,12 @@ public class GraphKit implements GraphBuilderTool { } int argIndex = 0; if (!isStatic) { - ResolvedJavaType expectedType = method.getDeclaringClass(); - JavaKind expected = wordTypes == null ? expectedType.getJavaKind() : wordTypes.asKind(expectedType); + JavaKind expected = asKind(method.getDeclaringClass()); JavaKind actual = args[argIndex++].stamp().getStackKind(); assert expected == actual : graph + ": wrong kind of value for receiver argument of call to " + method + " [" + actual + " != " + expected + "]"; } for (int i = 0; i != signature.getParameterCount(false); i++) { - JavaType expectedType = signature.getParameterType(i, method.getDeclaringClass()); - JavaKind expected = wordTypes == null ? expectedType.getJavaKind().getStackKind() : wordTypes.asKind(expectedType).getStackKind(); + JavaKind expected = asKind(signature.getParameterType(i, method.getDeclaringClass())).getStackKind(); JavaKind actual = args[argIndex++].stamp().getStackKind(); if (expected != actual) { throw new AssertionError(graph + ": wrong kind of value for argument " + i + " of call to " + method + " [" + actual + " != " + expected + "]"); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/ReplacementsImpl.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/ReplacementsImpl.java index e5173c70cc7..c569e541320 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/ReplacementsImpl.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/ReplacementsImpl.java @@ -433,13 +433,14 @@ public class ReplacementsImpl implements Replacements, InlineInvokePlugin { } IntrinsicContext initialIntrinsicContext = null; - if (method.getAnnotation(Snippet.class) == null) { + Snippet snippetAnnotation = method.getAnnotation(Snippet.class); + if (snippetAnnotation == null) { // Post-parse inlined intrinsic initialIntrinsicContext = new IntrinsicContext(substitutedMethod, method, bytecodeProvider, INLINE_AFTER_PARSING); } else { // Snippet ResolvedJavaMethod original = substitutedMethod != null ? substitutedMethod : method; - initialIntrinsicContext = new IntrinsicContext(original, method, bytecodeProvider, INLINE_AFTER_PARSING); + initialIntrinsicContext = new IntrinsicContext(original, method, bytecodeProvider, INLINE_AFTER_PARSING, snippetAnnotation.allowPartialIntrinsicArgumentMismatch()); } createGraphBuilder(metaAccess, replacements.providers.getStampProvider(), replacements.providers.getConstantReflection(), replacements.providers.getConstantFieldProvider(), config, diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/SnippetCounter.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/SnippetCounter.java index b52ad3881a7..8b3276fecaa 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/SnippetCounter.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/SnippetCounter.java @@ -93,12 +93,7 @@ public final class SnippetCounter implements Comparable { */ @Override public int compareTo(SnippetCounter o) { - if (value > o.value) { - return -1; - } else if (o.value < value) { - return 1; - } - return 0; + return Long.signum(o.value - value); } private final Group group; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/SnippetIntegerHistogram.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/SnippetIntegerHistogram.java new file mode 100644 index 00000000000..9a00d6157da --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/SnippetIntegerHistogram.java @@ -0,0 +1,153 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package org.graalvm.compiler.replacements; + +/** + * A histogram that can (only) be {@linkplain #inc(long) incremented} from within a snippet for + * gathering snippet specific metrics. + */ +public final class SnippetIntegerHistogram { + private final SnippetCounter.Group group; + private final String name; + + private final SnippetCounter counter0; + private final SnippetCounter counter1; + private final SnippetCounter counter2; + private final SnippetCounter counter3; + private final SnippetCounter counter4; + private final SnippetCounter counter5; + private final SnippetCounter counter6; + private final SnippetCounter counter7; + private final SnippetCounter counter8; + private final SnippetCounter counter9; + private final SnippetCounter counter10; + + private final int counter0UpperBound; + private final int counter1UpperBound; + private final int counter2UpperBound; + private final int counter3UpperBound; + private final int counter4UpperBound; + private final int counter5UpperBound; + private final int counter6UpperBound; + private final int counter7UpperBound; + private final int counter8UpperBound; + private final int counter9UpperBound; + + public SnippetIntegerHistogram(SnippetCounter.Group group, int log2StepLength, String name, String description) { + assert log2StepLength > 0; + + this.group = group; + this.name = name; + + int lowerBound = 0; + counter0UpperBound = 0; + counter0 = createCounter(group, name, description, lowerBound, counter0UpperBound); + + lowerBound = counter0UpperBound + 1; + counter1UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter1 = createCounter(group, name, description, lowerBound, counter1UpperBound); + + lowerBound = counter1UpperBound + 1; + counter2UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter2 = createCounter(group, name, description, lowerBound, counter2UpperBound); + + lowerBound = counter2UpperBound + 1; + counter3UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter3 = createCounter(group, name, description, lowerBound, counter3UpperBound); + + lowerBound = counter3UpperBound + 1; + counter4UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter4 = createCounter(group, name, description, lowerBound, counter4UpperBound); + + lowerBound = counter4UpperBound + 1; + counter5UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter5 = createCounter(group, name, description, lowerBound, counter5UpperBound); + + lowerBound = counter5UpperBound + 1; + counter6UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter6 = createCounter(group, name, description, lowerBound, counter6UpperBound); + + lowerBound = counter6UpperBound + 1; + counter7UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter7 = createCounter(group, name, description, lowerBound, counter7UpperBound); + + lowerBound = counter7UpperBound + 1; + counter8UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter8 = createCounter(group, name, description, lowerBound, counter8UpperBound); + + lowerBound = counter8UpperBound + 1; + counter9UpperBound = Math.max(1, lowerBound - 1) << log2StepLength; + counter9 = createCounter(group, name, description, lowerBound, counter9UpperBound); + + lowerBound = counter9UpperBound + 1; + counter10 = createCounter(group, name, description, lowerBound, Long.MAX_VALUE); + } + + private static SnippetCounter createCounter(SnippetCounter.Group group, String name, String description, long lowerBound, long upperBound) { + if (group != null) { + SnippetCounter snippetCounter = new SnippetCounter(group, name + "[" + lowerBound + ", " + upperBound + "]", description); + return snippetCounter; + } + return null; + } + + /** + * Increments the value of the matching histogram element. This method can only be used in a + * snippet on a compile-time constant {@link SnippetIntegerHistogram} object. + */ + public void inc(long value) { + if (group != null) { + if (value <= counter0UpperBound) { + counter0.inc(); + } else if (value <= counter1UpperBound) { + counter1.inc(); + } else if (value <= counter2UpperBound) { + counter2.inc(); + } else if (value <= counter3UpperBound) { + counter3.inc(); + } else if (value <= counter4UpperBound) { + counter4.inc(); + } else if (value <= counter5UpperBound) { + counter5.inc(); + } else if (value <= counter6UpperBound) { + counter6.inc(); + } else if (value <= counter7UpperBound) { + counter7.inc(); + } else if (value <= counter8UpperBound) { + counter8.inc(); + } else if (value <= counter9UpperBound) { + counter9.inc(); + } else { + counter10.inc(); + } + } + } + + @Override + public String toString() { + if (group != null) { + return "SnippetHistogram-" + group.name + ":" + name; + } + return super.toString(); + } +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/ArrayEqualsNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/ArrayEqualsNode.java index bf2edb8abaf..a08ff3182cc 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/ArrayEqualsNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/ArrayEqualsNode.java @@ -23,15 +23,15 @@ package org.graalvm.compiler.replacements.nodes; import static org.graalvm.compiler.nodeinfo.InputType.Memory; -import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_1024; -import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_1024; import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.graph.Node; import org.graalvm.compiler.graph.NodeClass; import org.graalvm.compiler.graph.spi.Canonicalizable; import org.graalvm.compiler.graph.spi.CanonicalizerTool; +import org.graalvm.compiler.nodeinfo.NodeCycles; import org.graalvm.compiler.nodeinfo.NodeInfo; +import org.graalvm.compiler.nodeinfo.NodeSize; import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.FixedWithNextNode; import org.graalvm.compiler.nodes.NamedLocationIdentity; @@ -57,7 +57,7 @@ import jdk.vm.ci.meta.Value; /** * Compares two arrays with the same length. */ -@NodeInfo(cycles = CYCLES_1024, size = SIZE_1024) +@NodeInfo(cycles = NodeCycles.CYCLES_UNKNOWN, size = NodeSize.SIZE_128) public final class ArrayEqualsNode extends FixedWithNextNode implements LIRLowerable, Canonicalizable, Virtualizable, MemoryAccess { public static final NodeClass TYPE = NodeClass.create(ArrayEqualsNode.class); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BasicArrayCopyNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BasicArrayCopyNode.java index 59750cc6454..0c5e6a17724 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BasicArrayCopyNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BasicArrayCopyNode.java @@ -24,7 +24,6 @@ package org.graalvm.compiler.replacements.nodes; import static org.graalvm.compiler.nodeinfo.InputType.Memory; import static org.graalvm.compiler.nodeinfo.InputType.State; -import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_256; import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_64; import static org.graalvm.word.LocationIdentity.any; @@ -32,6 +31,7 @@ import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.debug.DebugContext; import org.graalvm.compiler.graph.NodeClass; import org.graalvm.compiler.graph.NodeInputList; +import org.graalvm.compiler.nodeinfo.NodeCycles; import org.graalvm.compiler.nodeinfo.NodeInfo; import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.DeoptimizingNode; @@ -56,7 +56,7 @@ import jdk.vm.ci.code.BytecodeFrame; import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.ResolvedJavaType; -@NodeInfo(cycles = CYCLES_256, size = SIZE_64) +@NodeInfo(cycles = NodeCycles.CYCLES_UNKNOWN, size = SIZE_64) public class BasicArrayCopyNode extends AbstractMemoryCheckpoint implements Virtualizable, MemoryCheckpoint.Single, MemoryAccess, Lowerable, DeoptimizingNode.DeoptDuring { public static final NodeClass TYPE = NodeClass.create(BasicArrayCopyNode.class); @@ -215,7 +215,7 @@ public class BasicArrayCopyNode extends AbstractMemoryCheckpoint implements Virt return; } for (int i = 0; i < len; i++) { - tool.setVirtualEntry(destVirtual, destPosInt + i, tool.getEntry(srcVirtual, srcPosInt + i), false); + tool.setVirtualEntry(destVirtual, destPosInt + i, tool.getEntry(srcVirtual, srcPosInt + i)); } tool.delete(); DebugContext debug = getDebug(); @@ -235,7 +235,7 @@ public class BasicArrayCopyNode extends AbstractMemoryCheckpoint implements Virt for (int i = 0; i < len; i++) { LoadIndexedNode load = new LoadIndexedNode(graph().getAssumptions(), srcAlias, ConstantNode.forInt(i + srcPosInt, graph()), destComponentType.getJavaKind()); tool.addNode(load); - tool.setVirtualEntry(destVirtual, destPosInt + i, load, false); + tool.setVirtualEntry(destVirtual, destPosInt + i, load); } tool.delete(); } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BasicObjectCloneNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BasicObjectCloneNode.java index c1c49af5da7..9261266c0eb 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BasicObjectCloneNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BasicObjectCloneNode.java @@ -22,7 +22,6 @@ */ package org.graalvm.compiler.replacements.nodes; -import static org.graalvm.compiler.nodeinfo.NodeCycles.CYCLES_8; import static org.graalvm.compiler.nodeinfo.NodeSize.SIZE_8; import java.util.Collections; @@ -32,6 +31,7 @@ import org.graalvm.compiler.core.common.type.Stamp; import org.graalvm.compiler.core.common.type.StampFactory; import org.graalvm.compiler.core.common.type.StampPair; import org.graalvm.compiler.graph.NodeClass; +import org.graalvm.compiler.nodeinfo.NodeCycles; import org.graalvm.compiler.nodeinfo.NodeInfo; import org.graalvm.compiler.nodes.CallTargetNode.InvokeKind; import org.graalvm.compiler.nodes.ValueNode; @@ -49,7 +49,7 @@ import jdk.vm.ci.meta.ResolvedJavaField; import jdk.vm.ci.meta.ResolvedJavaMethod; import jdk.vm.ci.meta.ResolvedJavaType; -@NodeInfo(cycles = CYCLES_8, size = SIZE_8) +@NodeInfo(cycles = NodeCycles.CYCLES_UNKNOWN, size = SIZE_8) public abstract class BasicObjectCloneNode extends MacroStateSplitNode implements VirtualizableAllocation, ArrayLengthProvider { public static final NodeClass TYPE = NodeClass.create(BasicObjectCloneNode.class); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BinaryMathIntrinsicNode.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BinaryMathIntrinsicNode.java index bff7c320f02..2e68bc7e6b8 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BinaryMathIntrinsicNode.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.replacements/src/org/graalvm/compiler/replacements/nodes/BinaryMathIntrinsicNode.java @@ -37,7 +37,7 @@ import org.graalvm.compiler.nodeinfo.NodeInfo; import org.graalvm.compiler.nodes.ConstantNode; import org.graalvm.compiler.nodes.ValueNode; import org.graalvm.compiler.nodes.calc.BinaryNode; -import org.graalvm.compiler.nodes.calc.DivNode; +import org.graalvm.compiler.nodes.calc.FloatDivNode; import org.graalvm.compiler.nodes.calc.MulNode; import org.graalvm.compiler.nodes.calc.SqrtNode; import org.graalvm.compiler.nodes.spi.ArithmeticLIRLowerable; @@ -141,7 +141,7 @@ public final class BinaryMathIntrinsicNode extends BinaryNode implements Arithme // x**-1 = 1/x if (yValue == -1.0D) { - return new DivNode(ConstantNode.forDouble(1), x); + return new FloatDivNode(ConstantNode.forDouble(1), x); } // x**2 = x*x diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.test/src/org/graalvm/compiler/test/GraalTest.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.test/src/org/graalvm/compiler/test/GraalTest.java index 695d4d2ccbd..256e1ee6123 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.test/src/org/graalvm/compiler/test/GraalTest.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.test/src/org/graalvm/compiler/test/GraalTest.java @@ -36,6 +36,7 @@ import java.util.Collection; import java.util.Collections; import java.util.List; +import jdk.vm.ci.meta.ResolvedJavaMethod; import org.graalvm.compiler.debug.DebugHandlersFactory; import org.graalvm.compiler.debug.DebugContext; import org.graalvm.compiler.debug.DebugDumpHandler; @@ -392,6 +393,20 @@ public class GraalTest { * {@link DebugDumpHandler}s closed in {@link #afterTest()}. */ protected DebugContext getDebugContext(OptionValues options) { + return getDebugContext(options, null, null); + } + + /** + * Gets a {@link DebugContext} object corresponding to {@code options}, creating a new one if + * none currently exists.Debug contexts created by this method will have their + * {@link DebugDumpHandler}s closed in {@link #afterTest()}. + * + * @param options currently active options + * @param id identification of the compilation or {@code null} + * @param method method to use for a proper description of the context or {@code null} + * @return configured context for compilation + */ + protected DebugContext getDebugContext(OptionValues options, String id, ResolvedJavaMethod method) { List cached = cachedDebugs.get(); if (cached == null) { cached = new ArrayList<>(); @@ -402,7 +417,13 @@ public class GraalTest { return debug; } } - DebugContext debug = DebugContext.create(options, NO_DESCRIPTION, NO_GLOBAL_METRIC_VALUES, DEFAULT_LOG_STREAM, getDebugHandlersFactories()); + final DebugContext.Description descr; + if (method == null) { + descr = NO_DESCRIPTION; + } else { + descr = new DebugContext.Description(method, id == null ? method.getName() : id); + } + DebugContext debug = DebugContext.create(options, descr, NO_GLOBAL_METRIC_VALUES, DEFAULT_LOG_STREAM, getDebugHandlersFactories()); cached.add(debug); return debug; } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectList.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectList.java index 42d73c5715b..db38a0497c9 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectList.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectList.java @@ -171,9 +171,14 @@ public class EffectList implements Iterable { } public void apply(StructuredGraph graph, ArrayList obsoleteNodes, boolean cfgKills) { + boolean message = false; for (int i = 0; i < size(); i++) { Effect effect = effects[i]; if (effect.isCfgKill() == cfgKills) { + if (!message) { + message = true; + debug.log(cfgKills ? " ==== cfg kill effects" : " ==== effects"); + } try { effect.apply(graph, obsoleteNodes); } catch (Throwable t) { @@ -202,7 +207,7 @@ public class EffectList implements Iterable { // Inner classes could capture the EffectList itself. continue; } - str.append(first ? "" : ", ").append(format(object)); + str.append(first ? "" : ", ").append(field.getName()).append("=").append(format(object)); first = false; } catch (SecurityException | IllegalAccessException e) { throw new RuntimeException(e); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectsClosure.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectsClosure.java index c5336c81c7a..fcdc2185b32 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectsClosure.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectsClosure.java @@ -184,7 +184,6 @@ public abstract class EffectsClosure> e }; ReentrantBlockIterator.apply(closure, cfg.getStartBlock()); for (GraphEffectList effects : effectList) { - debug.log(" ==== effects"); effects.apply(graph, obsoleteNodes, false); } /* @@ -193,7 +192,6 @@ public abstract class EffectsClosure> e * indexes. */ for (GraphEffectList effects : effectList) { - debug.log(" ==== cfg kill effects"); effects.apply(graph, obsoleteNodes, true); } debug.dump(DebugContext.DETAILED_LEVEL, graph, "After applying effects"); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectsPhase.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectsPhase.java index a3df229bfeb..5a6d0937579 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectsPhase.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/EffectsPhase.java @@ -99,8 +99,8 @@ public abstract class EffectsPhase extends B closure.applyEffects(); } - if (debug.isDumpEnabled(DebugContext.INFO_LEVEL)) { - debug.dump(DebugContext.DETAILED_LEVEL, graph, "%s iteration", getName()); + if (debug.isDumpEnabled(DebugContext.VERBOSE_LEVEL)) { + debug.dump(DebugContext.VERBOSE_LEVEL, graph, "%s iteration", getName()); } new DeadCodeEliminationPhase(Required).apply(graph); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/ObjectState.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/ObjectState.java index 95fa92769b6..aed90d161cf 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/ObjectState.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/ObjectState.java @@ -68,6 +68,7 @@ public class ObjectState { } public ObjectState(ValueNode[] entries, LockState locks, boolean ensureVirtualized) { + assert checkIllegalValues(entries); this.entries = entries; this.locks = locks; this.ensureVirtualized = ensureVirtualized; @@ -92,6 +93,30 @@ public class ObjectState { return new ObjectState(this); } + /** + * Ensure that if an {@link JavaConstant#forIllegal() illegal value} is seen that the previous + * value is a double word value. + */ + public static boolean checkIllegalValues(ValueNode[] values) { + if (values != null) { + for (int v = 1; v < values.length; v++) { + checkIllegalValue(values, v); + } + } + return true; + } + + /** + * Ensure that if an {@link JavaConstant#forIllegal() illegal value} is seen that the previous + * value is a double word value. + */ + public static boolean checkIllegalValue(ValueNode[] values, int v) { + if (v > 0 && values[v].isConstant() && values[v].asConstant().equals(JavaConstant.forIllegal())) { + assert values[v - 1].getStackKind().needsTwoSlots(); + } + return true; + } + public EscapeObjectState createEscapeObjectState(DebugContext debug, VirtualObjectNode virtual) { GET_ESCAPED_OBJECT_STATE.increment(debug); if (cachedState == null) { diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/PartialEscapeClosure.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/PartialEscapeClosure.java index e00e407ff78..f88b86b763b 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/PartialEscapeClosure.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.virtual/src/org/graalvm/compiler/virtual/phases/ea/PartialEscapeClosure.java @@ -854,6 +854,7 @@ public abstract class PartialEscapeClosure closure, Assumptions assumptions, OptionValues options, DebugContext debug, LoweringProvider loweringProvider) { @@ -125,17 +130,81 @@ class VirtualizerToolImpl implements VirtualizerTool, CanonicalizerTool { } @Override - public void setVirtualEntry(VirtualObjectNode virtual, int index, ValueNode value, boolean unsafe) { + public boolean setVirtualEntry(VirtualObjectNode virtual, int index, ValueNode value, JavaKind theAccessKind, long offset) { ObjectState obj = state.getObjectState(virtual); assert obj.isVirtual() : "not virtual: " + obj; ValueNode newValue; + JavaKind entryKind = virtual.entryKind(index); + JavaKind accessKind = theAccessKind != null ? theAccessKind : entryKind; if (value == null) { newValue = null; } else { newValue = closure.getAliasAndResolve(state, value); - assert unsafe || obj.getEntry(index) == null || obj.getEntry(index).getStackKind() == newValue.getStackKind() || (isObjectEntry(obj.getEntry(index)) && isObjectEntry(newValue)); } - state.setEntry(virtual.getObjectId(), index, newValue); + getDebug().log(DebugContext.DETAILED_LEVEL, "Setting entry %d in virtual object %s %s results in %s", index, virtual.getObjectId(), virtual, state.getObjectState(virtual.getObjectId())); + ValueNode oldValue = getEntry(virtual, index); + boolean canVirtualize = entryKind == accessKind || (entryKind == accessKind.getStackKind() && virtual instanceof VirtualInstanceNode); + if (!canVirtualize) { + if (entryKind == JavaKind.Long && oldValue.getStackKind() == newValue.getStackKind() && oldValue.getStackKind().isPrimitive()) { + /* + * Special case: If the entryKind is long, allow arbitrary kinds as long as a value + * of the same kind is already there. This can only happen if some other node + * initialized the entry with a value of a different kind. One example where this + * happens is the Truffle NewFrameNode. + */ + getDebug().log(DebugContext.DETAILED_LEVEL, "virtualizing %s with primitive of kind %s in long entry ", current, oldValue.getStackKind()); + canVirtualize = true; + } else if (entryKind == JavaKind.Int && (accessKind == JavaKind.Long || accessKind == JavaKind.Double) && offset % 8 == 0) { + /* + * Special case: Allow storing a single long or double value into two consecutive + * int slots. + */ + int nextIndex = virtual.entryIndexForOffset(offset + 4, JavaKind.Int); + if (nextIndex != -1) { + canVirtualize = true; + assert nextIndex == index + 1 : "expected to be sequential"; + getDebug().log(DebugContext.DETAILED_LEVEL, "virtualizing %s for double word stored in two ints", current); + } + } + } + + if (canVirtualize) { + getDebug().log(DebugContext.DETAILED_LEVEL, "virtualizing %s for entryKind %s and access kind %s", current, entryKind, accessKind); + state.setEntry(virtual.getObjectId(), index, newValue); + if (entryKind == JavaKind.Int) { + if (accessKind.needsTwoSlots()) { + // Storing double word value two int slots + assert virtual.entryKind(index + 1) == JavaKind.Int; + state.setEntry(virtual.getObjectId(), index + 1, getIllegalConstant()); + } else if (oldValue.getStackKind() == JavaKind.Double || oldValue.getStackKind() == JavaKind.Long) { + // Splitting double word constant by storing over it with an int + getDebug().log(DebugContext.DETAILED_LEVEL, "virtualizing %s producing second half of double word value %s", current, oldValue); + ValueNode secondHalf = UnpackEndianHalfNode.create(oldValue, false); + addNode(secondHalf); + state.setEntry(virtual.getObjectId(), index + 1, secondHalf); + } + } + if (oldValue.isConstant() && oldValue.asConstant().equals(JavaConstant.forIllegal())) { + // Storing into second half of double, so replace previous value + ValueNode previous = getEntry(virtual, index - 1); + getDebug().log(DebugContext.DETAILED_LEVEL, "virtualizing %s producing first half of double word value %s", current, previous); + ValueNode firstHalf = UnpackEndianHalfNode.create(previous, true); + addNode(firstHalf); + state.setEntry(virtual.getObjectId(), index - 1, firstHalf); + } + return true; + } + // Should only occur if there are mismatches between the entry and access kind + assert entryKind != accessKind; + return false; + } + + private ValueNode getIllegalConstant() { + if (illegalConstant == null) { + illegalConstant = ConstantNode.forConstant(JavaConstant.forIllegal(), getMetaAccessProvider()); + addNode(illegalConstant); + } + return illegalConstant; } @Override @@ -149,10 +218,6 @@ class VirtualizerToolImpl implements VirtualizerTool, CanonicalizerTool { return state.getObjectState(virtualObject).getEnsureVirtualized(); } - private static boolean isObjectEntry(ValueNode value) { - return value.getStackKind() == JavaKind.Object || value instanceof VirtualObjectNode; - } - @Override public void replaceWithVirtual(VirtualObjectNode virtual) { closure.addVirtualAlias(virtual, current); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphOutput.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphOutput.java index 24b4cc10302..b08671ebd43 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphOutput.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphOutput.java @@ -113,11 +113,29 @@ public final class GraphOutput implements Closeable { private GraphElements elements = null; private GraphTypes types = DefaultGraphTypes.DEFAULT; private GraphBlocks blocks = DefaultGraphBlocks.empty(); + private int major = 4; + private int minor = 0; Builder(GraphStructure structure) { this.structure = structure; } + /** + * Chooses which version of the protocol to use. The default version is 4.0 + * (when the {@link GraphOutput} & co. classes were introduced). The default can be changed + * to other known versions manually by calling this method. + * + * @param majorVersion by default 4, newer version may be known + * @param minorVersion usually 0 + * @return this builder + * @since 0.28 + */ + public Builder protocolVersion(int majorVersion, int minorVersion) { + this.major = majorVersion; + this.minor = minorVersion; + return this; + } + /** * Associates different implementation of types. * @@ -161,7 +179,28 @@ public final class GraphOutput implements Closeable { * @throws IOException if something goes wrong when writing to the channel */ public GraphOutput build(WritableByteChannel channel) throws IOException { - ProtocolImpl p = new ProtocolImpl<>(structure, types, blocks, elements, channel); + ProtocolImpl p = new ProtocolImpl<>(major, minor, structure, types, blocks, elements, channel); + return new GraphOutput<>(p); + } + + /** + * Support for nesting heterogenous graphs. The newly created output uses all the interfaces + * currently associated with this builder, but shares with {@code parent} the output + * {@code channel}, internal constant pool and {@link #protocolVersion(int, int) protocol + * version}. + *

+ * Both GraphOutput (the {@code parent} and the returned one) has to be used in + * synchronization - e.g. only one + * {@link #beginGroup(java.lang.Object, java.lang.String, java.lang.String, java.lang.Object, int, java.util.Map) + * begin}, {@link #endGroup() end} of group or + * {@link #print(java.lang.Object, java.util.Map, int, java.lang.String, java.lang.Object...) + * printing} can be on at a given moment. + * + * @param parent the output to inherit {@code channel} and protocol version from + * @return new output sharing {@code channel} and other internals with {@code parent} + */ + public GraphOutput build(GraphOutput parent) { + ProtocolImpl p = new ProtocolImpl<>(parent.printer, structure, types, blocks, elements); return new GraphOutput<>(p); } } diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphProtocol.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphProtocol.java index fd63dd7b586..e0fb0a3f804 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphProtocol.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphProtocol.java @@ -52,6 +52,7 @@ abstract class GraphProtocol 4) { - throw new IllegalArgumentException(); - } - if (major == 4 && minor > 0) { - throw new IllegalArgumentException(); + GraphProtocol(WritableByteChannel channel, int major, int minor) throws IOException { + if (major > 5 || (major == 5 && minor > 0)) { + throw new IllegalArgumentException("Unrecognized version " + major + "." + minor); } this.versionMajor = major; this.versionMinor = minor; @@ -93,6 +87,14 @@ abstract class GraphProtocol parent) { + this.versionMajor = parent.versionMajor; + this.versionMinor = parent.versionMinor; + this.constantPool = parent.constantPool; + this.buffer = parent.buffer; + this.channel = parent.channel; + } + @SuppressWarnings("all") public final void print(Graph graph, Map properties, int id, String format, Object... args) throws IOException { writeByte(BEGIN_GRAPH); @@ -137,8 +139,32 @@ abstract class GraphProtocolnull if it is not a node object, non-null otherwise + */ + protected abstract Node findNode(Object obj); + + /** + * Determines whether the provided object is node class or not. + * + * @param obj object to check + * @return {@code null} if {@code obj} does not represent a NodeClass otherwise the NodeClass + * represented by {@code obj} + */ protected abstract NodeClass findNodeClass(Object obj); + /** + * Returns the NodeClass for a given Node {@code obj}. + * + * @param obj instance of node + * @return non-{@code null} instance of the node's class object + */ + protected abstract NodeClass findClassForNode(Node obj); + /** * Find a Java class. The returned object must be acceptable by * {@link #findJavaTypeName(java.lang.Object)} and return valid name for the class. @@ -239,7 +265,7 @@ abstract class GraphProtocol || findEnumOrdinal(object) >= 0) { - writeByte(POOL_ENUM); - } else if (object instanceof Class || findJavaTypeName(object) != null) { - writeByte(POOL_CLASS); - } else if (findJavaField(object) != null) { + if (findJavaField(object) != null) { writeByte(POOL_FIELD); } else if (findSignature(object) != null) { writeByte(POOL_SIGNATURE); } else if (versionMajor >= 4 && findNodeSourcePosition(object) != null) { writeByte(POOL_NODE_SOURCE_POSITION); } else { + final Node node = findNode(object); + if (versionMajor == 4 && node != null) { + object = classForNode(node); + } if (findNodeClass(object) != null) { writeByte(POOL_NODE_CLASS); + } else if (versionMajor >= 5 && node != null) { + writeByte(POOL_NODE); } else if (findMethod(object) != null) { writeByte(POOL_METHOD); } else { - writeByte(POOL_STRING); + if (object instanceof Enum || findEnumOrdinal(object) >= 0) { + writeByte(POOL_ENUM); + } else if (object instanceof Class || findJavaTypeName(object) != null) { + writeByte(POOL_CLASS); + } else { + writeByte(POOL_STRING); + } } } writeShort(id.charValue()); @@ -383,10 +418,7 @@ abstract class GraphProtocol= 0) { - writeByte(POOL_ENUM); - writePoolObject(findEnumClass(object)); - writeInt(enumOrdinal); - } else if ((field = findJavaField(object)) != null) { + if ((field = findJavaField(object)) != null) { writeByte(POOL_FIELD); writePoolObject(findFieldDeclaringClass(field)); writePoolObject(findFieldName(field)); @@ -535,6 +559,18 @@ abstract class GraphProtocol= 5) { + writeByte(POOL_NODE); + writeInt(findNodeId(node)); + writePoolObject(classForNode(node)); + return; + } + if (versionMajor == 4) { + object = classForNode(node); + } + } NodeClass nodeClass = findNodeClass(object); if (nodeClass != null) { writeByte(POOL_NODE_CLASS); @@ -553,8 +589,27 @@ abstract class GraphProtocol= 0) { + writeByte(POOL_ENUM); + writePoolObject(findEnumClass(object)); + writeInt(enumOrdinal); + } else { + writeByte(POOL_STRING); + writeString(object.toString()); + } return; } writeByte(POOL_METHOD); diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphSnippets.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphSnippets.java new file mode 100644 index 00000000000..3fb3c7f8e07 --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphSnippets.java @@ -0,0 +1,285 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package org.graalvm.graphio; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.nio.channels.FileChannel; +import java.nio.channels.WritableByteChannel; +import java.util.Collection; +import java.util.LinkedHashSet; +import java.util.Map; +import java.util.Set; + +final class GraphSnippets { + static GraphStructure acmeGraphStructure() { + // @formatter:off + // BEGIN: org.graalvm.graphio.GraphSnippets#acmeGraphStructure + class AcmeGraphStructure implements + GraphStructure { + + @Override + public AcmeGraph graph(AcmeGraph currentGraph, Object obj) { + return obj instanceof AcmeGraph ? (AcmeGraph) obj : null; + } + + @Override + public Iterable nodes(AcmeGraph graph) { + return graph.allNodes(); + } + + @Override + public int nodesCount(AcmeGraph graph) { + return graph.allNodes().size(); + } + + @Override + public int nodeId(AcmeNode node) { + return node.id; + } + + @Override + public boolean nodeHasPredecessor(AcmeNode node) { + return node.id > 0; + } + + @Override + public void nodeProperties( + AcmeGraph graph, AcmeNode node, Map properties + ) { + properties.put("id", node.id); + } + + @Override + public AcmeNodeType nodeClass(Object obj) { + return obj instanceof AcmeNodeType ? (AcmeNodeType) obj : null; + } + + @Override + public AcmeNode node(Object obj) { + return obj instanceof AcmeNode ? (AcmeNode) obj : null; + } + + @Override + public AcmeNodeType classForNode(AcmeNode node) { + // we have only one type of nodes + return AcmeNodeType.STANDARD; + } + + + @Override + public String nameTemplate(AcmeNodeType nodeClass) { + return "Acme ({p#id})"; + } + + @Override + public Object nodeClassType(AcmeNodeType nodeClass) { + return nodeClass.getClass(); + } + + @Override + public AcmePorts portInputs(AcmeNodeType nodeClass) { + return AcmePorts.INPUT; + } + + @Override + public AcmePorts portOutputs(AcmeNodeType nodeClass) { + return AcmePorts.OUTPUT; + } + + @Override + public int portSize(AcmePorts port) { + return port == AcmePorts.OUTPUT ? 1 : 0; + } + + @Override + public boolean edgeDirect(AcmePorts port, int index) { + return false; + } + + @Override + public String edgeName(AcmePorts port, int index) { + return port.name(); + } + + @Override + public Object edgeType(AcmePorts port, int index) { + return port; + } + + @Override + public Collection edgeNodes( + AcmeGraph graph, AcmeNode node, AcmePorts port, int index + ) { + if (port == AcmePorts.OUTPUT) { + return node.outgoing.targets; + } + return null; + } + } + + // END: org.graalvm.graphio.GraphSnippets#acmeGraphStructure + + return new AcmeGraphStructure(); + } + + // BEGIN: org.graalvm.graphio.GraphSnippets#buildOutput + static GraphOutput buildOutput(WritableByteChannel channel) + throws IOException { + return GraphOutput.newBuilder(acmeGraphStructure()). + // use the latest version; currently 5.0 + protocolVersion(5, 0). + build(channel); + } + // END: org.graalvm.graphio.GraphSnippets#buildOutput + + // BEGIN: org.graalvm.graphio.GraphSnippets#buildAll + static GraphOutput buildAll(WritableByteChannel channel) + throws IOException { + GraphBlocks graphBlocks = acmeBlocks(); + GraphElements graphElements = acmeElements(); + GraphTypes graphTypes = acmeTypes(); + + return GraphOutput.newBuilder(acmeGraphStructure()). + protocolVersion(5, 0). + blocks(graphBlocks). + elements(graphElements). + types(graphTypes). + build(channel); + } + // END: org.graalvm.graphio.GraphSnippets#buildAll + + private static GraphTypes acmeTypes() { + GraphTypes graphTypes = null; + // in real world don't return null + return graphTypes; + } + + private static GraphElements acmeElements() { + GraphElements graphElements = null; + // in real world don't return null + return graphElements; + } + + private static GraphBlocks acmeBlocks() { + GraphBlocks graphBlocks = null; + // in real world don't return null + return graphBlocks; + } + + private static class AcmeGraph { + final AcmeNode root; + + AcmeGraph(AcmeNode root) { + this.root = root; + } + + Set allNodes() { + return allNodes(root, new LinkedHashSet<>()); + } + + private static Set allNodes(AcmeNode node, Set collectTo) { + if (collectTo.add(node)) { + for (AcmeNode target : node.outgoing.targets) { + allNodes(target, collectTo); + } + } + return collectTo; + } + } + + private static class AcmeNode { + final int id; + final AcmeEdges outgoing; + + AcmeNode(int id) { + this.id = id; + this.outgoing = new AcmeEdges(); + } + + void linkTo(AcmeNode target) { + outgoing.targets.add(target); + } + } + + private enum AcmeNodeType { + STANDARD + } + + private enum AcmePorts { + INPUT, + OUTPUT; + } + + private static class AcmeEdges { + final Set targets; + + AcmeEdges() { + this.targets = new LinkedHashSet<>(); + } + } + + private static class AcmeBlocks { + } + + private static class AcmeMethod { + } + + private static class AcmeField { + } + + private static class AcmeSignature { + } + + private static class AcmeCodePosition { + } + + // BEGIN: org.graalvm.graphio.GraphSnippets#dump + static void dump(File toFile) throws IOException { + try ( + FileChannel ch = new FileOutputStream(toFile).getChannel(); + GraphOutput output = buildOutput(ch); + ) { + AcmeNode root = new AcmeNode(0); + AcmeNode n1 = new AcmeNode(1); + AcmeNode n2 = new AcmeNode(2); + AcmeNode n3 = new AcmeNode(3); + + root.linkTo(n1); + root.linkTo(n2); + n1.linkTo(n3); + n2.linkTo(n3); + + AcmeGraph diamondGraph = new AcmeGraph(root); + + output.beginGroup(diamondGraph, "Diamond", "dia", null, 0, null); + output.print(diamondGraph, null, 0, "Diamond graph #%d", 1); + output.endGroup(); + } + } + // END: org.graalvm.graphio.GraphSnippets#dump + +} diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphStructure.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphStructure.java index 7b950e42471..9c4ae934749 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphStructure.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/GraphStructure.java @@ -38,9 +38,9 @@ import java.util.Map; */ public interface GraphStructure { /** - * Casts the provided object to graph, if possible. If the given object obj can be - * seen as a graph or sub-graph of a graph, then return the properly typed instance. Otherwise - * return null + * Casts {@code obj} to graph, if possible. If the given object obj can be seen as + * a graph or sub-graph of a graph, then return the properly typed instance. Otherwise return + * null * * @param currentGraph the currently processed graph * @param obj an object to check and view as a graph @@ -69,8 +69,8 @@ public interface GraphStructure { int nodesCount(G graph); /** - * Id of a node. Each node in the graph is uniquely identified by a integer value. If two nodes - * have the same id, then they shall be == to each other. + * Id of {@code node}. Each node in the graph is uniquely identified by an integer value. If two + * nodes have the same id, then they shall be == to each other. * * @param node the node to query for an id * @return the id of the node @@ -96,15 +96,34 @@ public interface GraphStructure { void nodeProperties(G graph, N node, Map properties); /** - * Finds the node class for the provided object, if possible. If the given object - * obj can be seen as an instance of node class or it is a node in this graph, - * return the properly typed instance of the node class. Otherwise return null + * Finds a node for {@code obj}, if possible. If the given object obj can be seen + * as an instance of node return the properly typed instance of the node class. Otherwise return + * null. + * + * @param obj an object to find node for + * @return appropriate graph object or null if the object doesn't represent a node + */ + N node(Object obj); + + /** + * Finds a node class for {@code obj}, if possible. If the given object obj can be + * seen as an instance of node class return the properly typed instance of the node class. + * Otherwise return null. * * @param obj an object to find node class for - * @return appropriate graph object or null if the object doesn't represent a graph + * @return appropriate graph object or null if the object doesn't represent a node + * class */ C nodeClass(Object obj); + /** + * Finds a node class for {@code node}. + * + * @param node an instance of node in this graph + * @return the node's node class, never null + */ + C classForNode(N node); + /** * The template used to build the name of nodes of this class. The template may use references * to inputs ({i#inputName}) and its properties ({p#propertyName}). diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/ProtocolImpl.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/ProtocolImpl.java index a7d486af4dc..a6ef36838e1 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/ProtocolImpl.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.graphio/src/org/graalvm/graphio/ProtocolImpl.java @@ -34,9 +34,18 @@ final class ProtocolImpl blocks; private final GraphElements elements; - ProtocolImpl(GraphStructure structure, GraphTypes enums, GraphBlocks blocks, + ProtocolImpl(int major, int minor, GraphStructure structure, GraphTypes enums, GraphBlocks blocks, GraphElements elements, WritableByteChannel channel) throws IOException { - super(channel); + super(channel, major, minor); + this.structure = structure; + this.types = enums; + this.blocks = blocks; + this.elements = elements; + } + + ProtocolImpl(GraphProtocol parent, GraphStructure structure, GraphTypes enums, GraphBlocks blocks, + GraphElements elements) { + super(parent); this.structure = structure; this.types = enums; this.blocks = blocks; @@ -48,11 +57,21 @@ final class ProtocolImplG9W1f;vWTe|Of&iVc~)@3bY z*X(!RdEyt(5Gp4lij07d00028xR{Uv0Ko770MsBnH2944mp8@WKQQ(J;!5!F@QdrR zYv98-He%}b0D#i-_YYL!XA}YeAOXaM_?2AJk20J+FsJWdvQm#fYQ|f*QN*AjeVpil z#utdU5#~>W-?$1-{H$8p;a)z6)#Stvg{aK;3KKRbMC-@b1Rdyr%giGq$$IIBCkDN0 ziy;d!G3ec=!OxX7##yc-_D05|9@n*9Qo}}kBxvxk(7t?^aAzkdfbYK#yKu;s|9zUo z4Oy(2^uM>@2ehjbG~nERiJUs@TcXbk-{k-I(~W~wa~f&G*qY(X(r5q)c4z`ZlSsJ& zCDt+Vp=fiW$`r1WXWsjK;w6_w)82J{GS$f^{cO4U@OME=C+g31U(< zbd+IXX!?k8ADYtjOAYRtO2*6pImJ}~7>|9MO&s#?UW|v^G%s~I5+Dl8_vT}HD#pWH zwcW*9$pkMiNy+6%YAqgB*r~2_zdWRP7E{R%RehsROtk#S{w8=cZl%MixY)X=*!=6H zSQ{x?8=_`-TVAUjpNN~7Xwz~}XpX&qH6uOJVNNHo){|-kYyQpC09d7 zH?ufhUavgJ;;|jO>04LvO30A#LC<=_AodTsA^Hz`)^HyN@W!kL_iZjJeG07%F#fKL zJF;ZqK8XB7ke>WoH$fqn!UyM;(1jql@VY<}abF-=((s%?KJI%w7sv0^3!F<0%K)9B z?F7|ndZc(X0wNp?$Ghg?@RTC>RHcb_B52>ej1(FbK(6$s%Yvo_&C(+aVZQ=8h6ELr zIt8_vvYMU8Qed^+b^fqb!<@!rg~r+E<)y3ojMHqIGI4}geidVuGKHmk0-+-L64GHf;k}r` z^BPC`RYjAC`s1d|eLF@n%?wBejG7AQ8}vw%CCV8(=^PQ({!x4+&6wb}O-g-FgkWnT zMj7yvC*Ad0RADD81AJ!sT(Va)7WFRJ8Vd5Mgzx%`SydVmm6Y6^dcQ!1J@O&^l~6^n zMPw$e15v`1lmHqlo}o0GAu~SQH)y_Z-)`~B?`I`ZBb4qr%>DIJHjPKf`}*cGWayT$ zEkpRUt>A@5`?sBvRj22OWMbutCQF(;;_?Pd%{4TRaHH^kcB{K?*)@GOGb56r=%&|` zFjUZrm?bfFb&Ld0G9p}07r{oW(DZ8lo0(PB;zx9GOV3=Da*2@03Ud=P=G0GN=m@a3 zds;@VRg0<^vX7Vu==!<&aWhdB*ulBg1vAny`mvvQSR^Qu519{27C4d@ zY2JoIby`#**ly}E+z*2kASfso{;S|I-2)LR>ZG>Ck6DWKVxuT=#eBNA}NcD z^7bWz^kc0AN!$=WZYN4KanVrv<`r#Bgw;QWNfZmkO;<*P`4%Rbk8YcyK#KVWdEG5v zKxXo9pi3Dc7XAi&@;+&XbMuf>$*B)xmr_OSXJv-1vjLyB7&BC8&=CmIBCw|Go*wF{ z2*RglDA?b2|7Z^CL)=bG^;FPAW!0RB!w?qGv-C6&R+bjYQp%S5u0Rz@ZeL3Z4Wy+F zXPyH-<-JzS5T*&mS8OMCO@RXZF^X?noie{&9l`*Z(pfyJ79nF*uas403ku1_KgN|T zG*5(;TqckKHW)&do`M<*esX(-hG1bJ63kcn@$;9Yf$c`9#O z;nt-y1#x?&dH9r++6zC#7v^x_t@r2dc)Y} zDGFWJrcge7K9pOG$q#W&Kgh(PeblBnxHMGMEvps)5=O0llo5TOkA3xyhPD?^%6fxg z2O1;MI;BRI@4K(29g$@UC8-htz5pExbOb2AQkAOD^Xi*meKHY*6_>{@rC}Ld!;4cR z%lXtiFjqEfx{zUw)9Tc&q%s>siHVk5(d1DA6Y)h zITC6&{RvtfRasd{L9bN!Nv0Ib~G z3Y4Q0Z{dX+KA^v$49hW1(~c02o$B>?8>E-IZ_2hp?tmM_?wk##dm51lHD%@HH94k4 zK6-93JIew9fDIn-aToh2JFWztdA-JD96H#lC$~8q>n-0VLQJ%GW#pN~6)eM$AB$+6 zgayD8Yv6o}vL4Kv*a~3tBcV%Bota@3A$U)Z$B@8N7*ib;^+$%T8stmi;uv=cR++OZyJ7R` zpS^M3WthS8WV24`Y#$zEs%bW6QD2hqXKiBG z*`=|p#EnnQE!pZcfRBU&F;tB@Uc&qUUq9zPWm<(bWAz92N%Zei-S>SD!j>!QS zs3=9|;cL=)BAL?iKux~Rw>XBMXiEX4pLH6sYC>^j!gsp)G75QepD^KJnY)>_sFU-v z2Z~fE1EKjHwKJrs&S`oP+f=j_z$zTg~X2Y_3L1qRUwANG!y^|~dYdBW_ zN&!pfpXnmyX3S}926Wkdrg?~l+e_~iBNtwfG&-WQUe+g)`{#u`q_&ph?I$NDytwV^ zWz8P<*Xz?g%Ha)T6)&Z`{|{~ePR~cQ$*hwEGdc3 zphbTpEFZgmq{Rw|NraS^6mu#&WEPmlswJQilqU=son5*4x& zhUUwc=Y%VDH23Om1(|$-w_+&z1x7WhGef6k#5^Cgal8jq-tD5yG)ktmH_nn;A5iEe>=t$t^{VdCRZ$z*-1u$H(fwm)-t+H9GI4tgOsv`-3~?KgeHU zn_%a6sQBpRalCeRs7tz6r!+aIXc$ zez-3vbas{t8G-$ehg6_@Ej>&Y5b++MGv$g%z4EcI``A%=tH02rX-&k`+#+}}dh%h~ z6~@5L9jSjv3I?5%ZDz@L;z^7(NKe4d_F|azKp%j(O3T9 zPH=}!7LJxJHh(@W5QnKeTf3Fr~4}m3Yo3_JQx?{u#Zz4w7Wec>FnSMtwuyfzVZn~#PPZu7X{>0Rd1rb9}U2_ zKgzBd2949|#MTn1CBywa!?t8?_v@ZKZC10nwb=dg&uINd6jX-j299?X-0mDC%g?Q2 zn1#uxQZxFvMpZ3zQ}Y$A4v6CCDaTf&{_ydk3#`XnW@$tW3o5?yi$`p_=0quZ$nl|C zu#VX)*aXepPL7%H_OVSm9Zpx;w9*Rp+uU#Ll9GA*4z`zOeTI+tve84y=;%(tvx0Cq zQb8`3omaPab_TyO%35cjsOB;l^x5pog}0=iRaaL>5piM@5iL(mIox0F|M~N$v9WQp zKib&P(ACpJ8(-x0<7+w;7Q@BzAB4_C9|FE?5uWh2t{b@6yr%{hlb9u1GLT#m-& zh&Z&A%2c8ZaDk%j4gLWRmR0EqQAz0n>;^4r+?XQyprmO)DYLZR9*$+KsV1KG0E9dmI!Onxt7-S&a-0F+qlzrAO1gl$Ml)dAB{zR8&-)biIB@ z$mw+CkIJK_s)|Dv&uR1W{A9m7GCVW{VYlBI$>4NN;&j*><9)vMFWAo&jTR9SBINN% zDW4t77UVwUG5VcybvP@V#=JkmdAii$%?sJT*?NGRr!*coe1z>t=c)_sXI4M{5Gs-8-Oq(m5Hz`>&<-r*fh_O*(QP8 znR1x(n#?4;WFhL0WJ?kQ2Oj@KW;S%^K-dm^;x`K?hX4t?>eep{lZiW@BB@z9@o!w> z5?0-a=$OEywy((zE)-QG46x3Y@iUsH0aA_wL`bh0zkY_W{wSZx+yxWU>+<-rdD&^v zQp;^LlRGoXVx#S5WoUHN`{`;{Hk0e_{;;xAr!CE!uzh6J`{fB@-R?S5rhYhEc|&Cm ze!!qV>MD#B{_Wehl$4Z#fr0+7QF+`hO*}kWyS{#>I{-RjQE}d3D;h ze*eC9y=cvvDz)TGy6*a`+Hfccnc(`Q<+MVx;iSA_#cexL72445b^G;#xpl{rlY~Ur zyoFAK711aF=2#M)uFT|RmiIHe)ion=yM?I7+A`_Jv0}A7b$hO2_ps+ko0&7OHGGso8a>I01|e_Bdy59;Y-cR0Gr+c1p5>z-15Yl zHk0J&UH0{_E!(XVUZOJd4gT)&1De%f;7v5U8lL=&?A z`H4tJkHloTP&>-=@Vm5(#P;5BFoBYQ#R8YvWUN3Ma?<{A;`E{4mW!OcprpiXwn7^b zn|iLsSk}$$-UkYXgJpd1qUeKYRO{_VI8wJ%77sTsZ&qoUHM#TJ)xK|vHT zx7+&9KrVwZvpTk@+3D%T#6%j6T9cjMsZK4+Rt<+&Zk5r|L=wbYE;fRKf-&)@1mBD^ zK90)r($Ue~ZufiybFWL}wxQz*S=tBe<{=EZ)^~d~SCpPi!Z{In@UMUbmRhY(JP(J_ zt2C|7I8k_*#;RPlAA;jG`lfJ#!@dHHf7MC?z~gFK)vk%F@m=N!UQo6vem8w3?fX^O zcZ^I-(&Uq1c$zZfGchsU31ucF)f)j!>% z^0y4TWczNg^le+ZhreDfah9wsiqMa=;g-0C5LD{>>}asr&8G)wWj5z)J!Lp!6E}= z%pW@G;nuY3KE?=wA>Y`ShEFt;nchNu-&=x`jN6plOn)AzVQ!{07syd6Gf!3GAn^vG zu3)`Q{WNnZmH4sud&K0>7liXuV*-!xNy?Ap5Y2bwP(FEL+o6AqMhy`xM2VvGV1>hB zp*YzD{o1zQ03=gQR%^C}#uy(&BGjt%0~e23Ql+fhu99?lxiTi1Gcq#Tyq?_B3Ro={ zwltH($g0(-AH%p(bM&t%cKp8-lIvtdEr!7>|PINRV)PPEO7|^uMULr0XK8LW%0M&G{sPlRT62 zaD`7D0ZDZ-V`jE&P%qzg$%SN-grjoI=1zF3O2R({hMEqu_+p)NI$4Z zGyRV4H}mFPM5~CaHR>~cW?Q}8tirF<`7F!0IW5hi+BI%A+SMgPAw|BSA_MTfxJc(t z*X^1}`}Zc|RW{+4H&@Iz?V2Eq*Ij+RD^FISdc3me&dmJ+_8~7;>+p$RjqEmXb<8wW zM<%JJqvHpwqhCr4%k}bb6CPxy997D>dGDRzq&zR$`hLt2eL88qkjao?wO(nB>H9%0 zlm08edd{%qxZaB57u=*t^*HJkeA1GHKll@>BZCpv$?T> zO{4LRF*PSQ7l%&gH*e7I-%`m#UrfeGg7m6uYNW{(!N%^7%3WjHL9uEblVO+CdgQMw zqxEy&VP|)@zqgnB>2lJf6BFbL00|eD$Dgf%T#-mHOn~nS#qfT(Y&PGe$Tb_!x;Rf7 zHxDa2P`V%zKPho#g#;d(g=olh;_!0gd-pVMeu*&zRK2P6VihL$@3*NYx}jIZO))n_j*6HYj%%D24i<<6 z=8N3%QnWjovi4sj1*fJB17EGHUcGtiyZ1&algstt`gob>6$mICnEDSFy&qc#zAd!6 zUMR|P*V(N57s_TK4Zf#N^kK5FR^vDVj|o^SF3g-TVb*?!o@0LA`c!w_ ztWI-mQRep#vC5&aDG?vDssuGmGa6c2I3G|TgsbLnv*O09n59vah2l$*K;ApQ_}439 zf#o{>`SYh=2#XNiPsKvg|HHm=`6rP7F>`c&XIG30g{XKih(Q=Qy$3 ze#ow8x+j}4Sf$jEm0@>px?CUiWyO(WR>PAM;1F(mwc3IX8)s#6K98vST2)cI0(z|9 zcQ60)K(ayg^(s_F-Mtgw5avc_Qt>ZkgcquW`u|;HkdOT`N9@wllAoX7XSuOy1ZU{+ z)UDAn_M6wP&dX`C6>4N5=4hA7$Rl+&UM@4|8f;kbNiwaU14)*_WUBFcx^Hf7-rL(t zWinm?AsToq*jp9W(gwab!0~3kNslRhtD`>tE4SKU(;A%gf2FJ4kghFuc)#p|J@YryXkI}9eqm70~s>v?|(F8)WVQO#W_MJ6t<*W=^e#b`QvHw6d-5Q@7>ii`0$ zoyz1s_0WT7=tb9h1`Syq8$0{;Av)q6?{lmuv8jm(yb>o%6}XoD$-)}l=b)e<;c)Ej z?QL7Q;~B%DB${v>IvHu{S3ri$&6ptStI#f-j_0ypA0J|7yt<)XKQEJa0VxzFJwXYB1qTr`?~%kAl=~LpW}k9@X)`_fXT%?iq6 z;dt$q;|?r)&lg!QfKRV}D!rcHun~-p`$((wN`U?1G zTl;8y`a>fSUSB# z*TC(5jYQ0qB8%lgz}FG^`}I2y8&ID(U20OK#%^oIZVTU+7;cd5=nFHNb_gyP(uW-ri&3y1|$N;Z}2w+ z+nv5J;EFLIeYWjsaV_`uc6KgKLc6=Wt6-lLjV3NHE!C{j?_KNif9J9W1+aPCo=MWR z!2&k3m$Cnw1z~Wh$#_#^I#~dwXC}Bun#AB9#eNXg@i-R-Qg}S?3$wiN zd)5;JyxsQ#cMGv6+CGfE-Z~T_LF?)1QBqYsU8p;JemGrib9Y{C_pAg#Ld@sTJqCvv z?a)exLr_K<&lLq>+OKvluSXXR4SdK>yERhx8C}Xl<|Om5q19r&G;Q*Q^9DQL8vA;# zc7ntF&ZL#UVcO`a#~7eLF+o40DB~;H6h{B`^eT3ovlK7@W#GE#Gaa- z#=^#~P_Gdwoqh(B<)mk$0vI;3Z()b;N$feDovC5i?@X@f9wtEy(gQ2xY3l_F@Xg4m zWcmO^>Js@AV1d28b;(q2B;;7m))la{ds@lnGQGhbqs%KTJe{uz)=PPOd?ey@di*m) z=f#UDWSw1eBH`h^EtSFHxIdAPL#qY&AmY%rg0S3q*%9X7@kM}po-KT|} zHSmC1yGN56btaqbh9D~UC780WtBz6|-{{X*Cv(3n!LE~soJ1koBCp>OZB6zB-sd0M zF#YPH{|y$EcUV}x?dc#G#L@MrvRZCduQ56=^Rph>Y>35Uvmt@pZ71v6C1tZb5bZER zwBX$oH&4mny)wWAy9IeuDx2+lKq{ThcCp?vH8qv!B6{!5-$<~oTm;*N7VT`@N6C_s z5`6=M-C;=owFXq@&p^avT5Yq2MdQ)T;rUT8kiq^2%YOz|jnXCLf(pp7exd9uN{Q8@~pS88jP8?i+qlyX>Ay6PTCFNHQLOV_L zs>n`uFtHoh3}?zUSsf3QXoKV+Pbl`O(zLK`#pllGupKfJ6BA2TtPkRmV1(WM*#V7oc zjO^QZ&_tfMSfSn0%n2(*zi-A9`3XIvJ^Q6{wq-}|L;v!A$;Fe-hy(Qt3b+O!L-@<5`51W__V_y_@0SG z$X^X#d1Oe|VSU6YL`MSNei@qk9tH%N73x-l8B^d&-8dc)+cnvE*Pf60Z%wW=9Cn4z zdO92s_f7Lctc)k!E|so^3>SX;j8+Mgu#I64hma6b*#kaeV_o7~dn4g%Z{n;}-!L$M zjYWX2pt%`a8^pgkif9NwqKHB?NMU8ieD;<7R7&C277*d`1tnBfXLHPnK`@W-GwQlI zTLr!gW9rZ;(P!O{nAFeB693Xn ze~lkMb!XD>;CjTn=Sx){qK@h5X}31e$_o^D+f$aG|EmA|;NSp6V@{qGOVzs>yYI5n z-2G2Ab+HINj`C4gT|BH>WGaAH9T`iSYE!&4UM4DkYFqWF3=fx#!-wTHl)Ve`n%8DG z4eaiBP)h+KkTf zS@2WEq$ix-I)V8@-QC^*JLw&l$E{5Z3s^l71}G54<3w_+^74c#XS29m89#i;t(hW- z+BB<6BY_8g%v%`Muv1f0D^Ym3FYCA&PZz!Iro_+HsHBl9jehhPl4*bdfHO_T-PxTe z`R)<}6%3Fr!#b*qYSjQKFm2L0ao~8u7t2>@Sd83kTrGMOl2PQmA;#r?Z+zOI>$Vf! zDgn?H}&qVTY3EVG@@_0!-G4Buhbk8NZEPF3%w7*t`@Bbk_INT zf$qkzb>I_`S#z)!E~qu6eU$sWWy3a-$z3Y<2`rje2!UF(#<+bJESlvradB}q#-m2Y z#*+CHF@pyPSzXzt_ldnXuNiJ{G_xGO6U~?~^RHNfXxK+KEC-)}07O|+W?#hPG|I0p z_b2i2@VZ)3Z~Zsf77^h9q;3Zif%XaM!fA2}ik5-;RoDSaN-b8k^&LUPi7O1zZyxQt zdkLD=Z*X+?MJAc0Cy~0v$l~~7ZFq%lZ}UtCE$o5Ll*K~#Hp6;$leCRi$0LfsmVyGk z>5_w4-FI?opNPHC-8aiVS+m8q`N@LjlAW2>7O$l?b*jV_n~P(eZT!gLhMnbJ zL$IN0HfbF=TA;ai-0_~Q5E2j^J5mEKM0@Ja6||uH$dQrdb#FSH_NA__Zhd{d#%OqO zXb5rt>|eG=8-_gY7^C!>AfB&W&eqmeOG^vX#CI(v3n%v-n!%U_!3ri8mNtk9NE4^A zc>Dzw^TLswj7-9g!kpNYvM_}~9G{q!WLBqft)|kJ!ERq*z1p6UzrxjhD6lWWgv2kO znZ>}b0DP@$;hp!S=HM{9L>638f7xt!cqU|e4{>033HnLN)GzG19wU*>~S<;48=y+Km8Jure zAYc!yB&2QgI_`%iyx>wPoV0NS<%;p9qt%WV5Kw#Xq#7$IDn^QIg7j1%(JDj-f#1>Q zTKK!lFRKOuq#!=g-oC!)Q|}i7`mcOqhK_-wYjR}s8z@djpBTUG&s3beQO2!mE-x#S zDx7pB7L!jMYzH~Aoa*6m~rd4J}Kr=4KS*AM$+P;E?kK#lELTG(do**r5*L~6?-2?WfE4#eMtl9mhp7AlsEZs zm=A)htNR+8z0SE~NC(7YHt!g+J_ePjC4avIc_Yg^yecczcpqNrd@U_W#Gt=*$$7fv zwro+K^rz%tBd$6N^%$vm7P-^8A9mbOQuPhGdn-RK;!g^`rOAnm>KwJ@b;?i03 zre@G#-_xqMn6Gv@-vIj(Ft+aiqE$4=>3niuJ}M#G4^W?7SkM5qNstG(wze)UY0Urt z0s;a)K0XNO?4BR*YHMqavpiYezxVTW1|Nc+RkB2WH21ZdtLq(jY4a5JK_I?T)ierD zb4I_|VX`=HDIF#AqphVSYGvl>C|6y0h>``J;2PPElw? z-3*TSX|rTZ%vgmW5+CiZ7Xp2|X>hLckPM{CN7|##wOonHi9=I$bA{WUC;61TC}C?x z0JI1?$wxAthldKaD%hOwXB7*>FHTdCfyH*6_|)-W&!PP?IG{?cOlc6nt&p4PP#_Ht z4}bMlCWH1mtfRhS;B)3G%2C1)mR*zI@?dW90Gr1Y>B+dk9}5TS zpgrjh;#_SoHcBho2Z>rdAFe^os{^#BCJUquCPCRnaH5=ml-q6jkH~{MQ-9S%p?@YF zOr1a(I1U>~9zU?}z=$Lmm)EOr+oUCs9n&TpwBYwQi(_E*oL^6VBI6Ix$f-QZ{xk?) zRV`XS5~268BDuC~guX`KTtT>@rlUg)Hb^naDo<-rpOBQ4M4VW~L_eynl8fG0J>1c4 zzLG_%EUkd{A+RRWt4VwPjr{YLLI*LluU^XHG0GIozEew+fa_hJSfGUB`K)YYJ_Q3( zhYJZ&<`l2jnMoC}yVF?(UUcOaP_qksIejU2y_#l^9u9uu1v=B5Rm@*jUxo&U5C;-h zby&-))4h)GrtlP%8dhoHQBNo5Ozy513vhJgcpu7sa9;(jv#}ORA24)8N`6lKv+8}< zo}CSDz$6mS@rirk%9n>0OLMxB!~kgmUKoJIdnmb7yJ9Br!(zSH8{kf{;AQ~VYvUIW z-HSSm^?p1<&h${%yd6{v)nt&a~O) zhMtMp{qKjM=(Mwo73)imawT%mtZ#Syy$Y-y(wO5d&?Hi^xnCb-aahv2)cT}*&yYrq zw=?_&`*M1m-GC34npSf+%EiUS|2#%Krl7;Di>oKcPKJ}AO{Mq_%9mf2>->BUBWUN7%8?DS~US?WzI_qhk*uPKly5h#!?_lXSN zua&J;Y?4_x(1Y#}#7MTBL@0fu^^~$lZAHF}K6=&N9%DeX?6a(5rww7LrAbMN<)s@d4JkTch95CK{RKatjE4gqS0s0X6E{LHrC0xr zL#RvC#2$xg86YjQxRfYpYn}Zbo9pcoBJAjaQAL2bhRS-g93j<6 z6R3Q~L?`TjN*eM)wn^vwF-?)|uZK%+(#B)S;x5$KK47BnxVS^dn=BTvVAG~kXlZ7a z(QRx|vRbQmRf~7QYN>Xt9xEvuyNYY`mkP<9DRKEu$!76*%A~Wcb83b>oVc*KfnIb` zY3XNbYONZhNR`s3n~s-ZCiKxtowI^C+20{DUFH>wGs_Q!TgzL9HI6r4tCY6B8l);x zF=PWJpOLyhh3V^Vu#M0o{db9v6U8v*iH?bxWs#&TDBm8H}Iw)Om7s8Z_=a>hfZ7WRQ2**nP{P z!1tp8Q(Sy|lO#OfftUDld+S$aBatZzHylh_{vrwUn)g@fy}x(g72Ob zUZeVXhp14B{As7u(Hp`c`DsndZ&AA13-~%zKos}h)AZ>1y#q( zPz;Le)76)%eqsYA;rFq_Mh4FD_~1B0v*V%BXl5IzE*~8oSrxzaqaZt8PO-Rq-sS9H zJ-VBigy-&kJhIrq^MF8ICp!2|;BeL-D#;nEW)5F>;8$CM4gTp7ACI&kdMqD@vPwlFiB z*mr>Crw8vdNu4sQo41JCse07|YQIeml9EH<#2eS@@bA`dm9)FMd|;lU-_Kq}T?RFj|6?s^tOSfzG!^C@XzAq}c{oOBu9zlc?~x3Ay0 z%eZoD#2t=S#;c#j^Z0zlH`buPGzY!Q;6Ss8)#CVQz7E17wExB0qw34P?u*;z)ofO) zvBYr!^1^(Lpq;Y1@t_H9dhB13>Vm1JPAwS=ny7&9zpOvy8bWrq-=Hl=GKCLj`|QLYN4sf{KdR z;mcMP^}ex?iR*tBFdCm)lLnMdZru4czI5ETTD;Cj62T=#%(=E z8Mw>5TUsHJ57-^HUM0#xr=-$y8Hmhu_?ab}0qdhbI|`gV*Xwz8vh<`-SPbGV7myVqe z+3zNMLAz(>bTni z@whq7v&d;97$7e16*OtkI6`0o$ujWc;el@c&9g$NP^D_=^k4xw#@Dmyol&T6C_bM} z#?%khx!c6>RpISRRZ^~G{Sr9 zk0D>k#9;tq>w2{YuePWh)2}AkXgCN$0&fdc0^bPV&Q#9IV-_n(Oq=Dak=bHYkmsJ4 z4rd|43ZzQG1G<*aS|n(S37VgJ+73oC-Z6g-+j9_!2IrhEO)7uC{UBmv4qpHcfx2

ppzz}z7jNKPar|f%zlzR_#cjbY zBMjXU1e7SDC1kh8bmFam`EMtETRPAa5|3j{jX3TJi-=&xk0cY9 zAd5qx_Bk7DN_P)XqmGJ>2IsAAN>@O|$3S0Si3;n*R_$eC7$>Np#EbyMxgh4S5}&ht z+x)eK!2Rb5z7fH!RM?_2=UXk9V+FV>!-B0cgnR_pJ=2TY{_pU?xkXjuz zKRrC~KBqjCH+;r7M**dj%VWcjS)O6bXcm#^k}dYj@84hU7!e4^qXbKmp$AZ6*#(km zRc=}cz4w(OA2)Y2i$n+L3A^c~$jrF!1bor@ZA;5PzF6DWd)DZX^)TaOxtAp)2k9*~ zRA7Y*;YT9$pYv_Au;4urh8wU&nNLy85Cd%B%yrNE;ZaJ}G=oRh4|{bbW2D!@Z0J5g&Q{M>4X6 zH1*0su!b=I zecssC2F|W&$*gXJ&58vwIXTHWp*_Dc)gb2yS`+?VohRRZ(MKpzZ5NORZ%I_nMiVA6 zkxZJ^%pW;%3d;jusqq`Fhtmj_PmTgnk+kodOK*SPZNDSp(TscS!8qpVhfVQ$3o9#% zd&%K>$R+G6C#K!%_P9gv{sWigoP$B`dB0!flv$txDI)5u>)pD=T&+qj!nAk)5%;8p zCvxoD%xnIr?GaRbdhphP4CrO~ee5DEP&nar*pqW}bJN$)5qhsjA*rtUo1)a4UBh#& zcu7`wkx7;+CL}=r`8Y*4%Q05|`U5BC;yVX+vtvLd9GDb!R9S4?T36?BtpV05%3Woe z@c=Oji_B^Q0nf8N&EL&UAddmmPas8XS zi;Y_5{jqHMXfE{F)Oo9PwxCS``9nnT=HObfPYAR7K=)5NTDh~j&?v?qrfal4wBdyq z|IRu&AP}v@vC4hUn>GW@WeEuhnM|$$o+}910M7ww2hoG!mSb)^_j5Kb1~J>8i(Ask z+Xe$K&oFe6%-dr=KN)h&kTT?N>i%-9UNXjYk-qipWE}yoUbGHPyE}v`3(FXhFy6i$ zUs|$YfoRhvgSBD^)Jh9rU<%5(7fUBXThM-dD8$0T(kQPlE*|(w|07&V&~c<;*3x6e zlA&LgP1~yDnlRUzI3q6omo67LK=0IW?ljCn?((+*U2GaZRrr3zHR|+%;x4HaCPVK< z#m|$H07tftei|i8^j=H+35~Bg*Xo>TGIbKC4BCw7&;81|b5TlEx$kI;9BGKuO^-yB zEn8&%$R9!Sxp=y;7!9+<( zxY|R?9*?rJu}KhIP-x`7ro3$!RFK|YV@kxIFpQ(BbK6wIZ3?xAp4;b-`TcznUeEdg zM6T5|;HZ*~98p_2tm$sPf~m;gA2{*2Ba~~_C7bT%EZ$qESyhx-lvJTuR+EGD@#p{t z(TgiQr*7mK&%>~GUH)m9is`9HG&rA0*{5n!e!a@{XJE+yVxHyaunp0DC@kv;9@r_rru z0%8M3bF`XT9b z-$|cd(`RE%;FLibv~HQ<-6*O~%;K|N_M^ikP7tCN3YJS92bT(htODZx^B+IxS-1~( zf{m7!>V55lPfap;x~)>NoFo^2?O-)Zx~=4L=Ccv@bKhXEwbZ1bf(eyL$9b+4(yL!F z85XR`>-@1DB#Xo-K?doAdf($Galei&*oQ*V``DHgOQYMelae)0jtUsu+9Nz z0Qmgy6Cv!{nRFT(qB}7=hW5YTfN2*CnJ%;uCKtKdo}jd~93>E!wOVA9U#19_JY)=N z2rM;qKZ+o2wKNaH6{Jm$3ddVjQF;94WG(@G-IKfy*WC9uep*TmcMo!QW(&~~ z33gu|%fRWvct-sxaH(bV+w^4~OBHu#17p5X$XRp>1vIw-J|^hV@GPSkhZgMY)EUc) z7%M>sA6g=?t^!w}*XP1rnY&0TUftA`df2dcrsV%QdYq~_zE--OnT3s8_kG3NCE7RQ zaZ50T+_Uqb73yGZ5t13b8mx7m-B24JV)a^_RfuA8ZtAjQ6+t$u#2bmsv3-8AuJ!oG57Y^6RyUTf*h$si=hqLNy2=aga>n8E2A721$H@${n zI!pD_>!AloL^JPbvHcCS$M=60f& z*PGj9EAXBJZ9=3CD)46H`}_Ns*4EnjrIE=A(8M0m?UzA1$O5%GO7zeG0a8IGaAJVd zDfT1^1^hJ^63-V#CI^pMh$nwI#r;p2qy)DgvvgTzv0pKSNZ`FqjwR>;#tE`bo98V~ zPClW94E0Ve9oaR(`+e>IVUm{fNrD88#c9|{5$9k(J_k6N!pUtrsQ-(+w+^Z@Zreqd zg@6)L5&|OKNJ=9m-JODjAl(fLh;+BKbV*CAq;w-vBHi6}Zr*Qyd-mS5=gj&4tZ|fa zE!Oiqzq;$Xt{d^u07tAW8Q^1KWZICZ`nvPhCYi{teEYMDc0e%f9e#r7U#%S$t6jbC z9rV;CxuS9mTb>vYYHDT=&f!rL=*(%|LiT(5^rhjQ#AEU|kJN7!|77%Wl9t$-=bBx$ z85vh|GS>O@4p3vP=SRN?KBuH6Wsbf@d>@(es6G*%(|;w8n=*g8|8vo& z%84v~Vz>JGxOpp?o8wz=!J+~HV!)%Opa8VaVM+bgzrDzyHM6^<&nYF9sAbi< znKV75zDWl0wGcm3rZGet1dS2Xzc(5qg-s;|>p#VHWWVI*2vqL~)XL!fe!93yjn}w8 zSWwZ{_Cps2^TX3t!{=g(B>T6ta7P768@Nv9qy;Z}tOX>a?!F}asVW(1H|s3DzIpd$ z6a~SQt3>)3*86W=f9a>kwyx}F$>)p1-^8e!7>@NAi2HPrIszCKLuwqcWK6#4bW#zi*@zyk{_l2#jtgNVLD`aB1T=y2j z+UD;KxdOzBnv3i4O;N_>&wiA`XRA9to&iPgoUeZk^UBt~&v?rvTd$S2{jv@=)ccHE z=}iZi@`8wn$jIG;VR^uxC3^lkYPVelNWNSG%N#)Cbou~r<~AS71!OWOVC)sDmzoTp zA8(tY0n*@x;036;TKna((NWu1o&ZYans!ju)0>%Tbf4$}Xb}L}LV*zjh?Re&n^h07 z5xFf^R#Y;wG0SPa|LpQ*M?xw#((3h#yA2KK=xYTFO2+2C-rrk|zd&(4#opheURspz ze6adGEMqk9`h4xdv&a?Hn6_xu?Pgr#Pe%AAM+f=QdsgdPHYJz{7V9wwwVD)e_pRT* zhMOKQ4`S!`edfQ9uuHUJH*&1{6MY;KCL1yOb?o*N5K8hzEL8HQN!6 zP9)6Re!h2ufEJ%Ykd%4L=lz)G1~-@P_41Et1ZLuhoODNIVxp#=-d1wIWB28{{`H@x zM$7#=fSA>O{kj9N@FX@<^klEq?0BoXRVsuF;JK^;xYjRt?XlPu3rY7O_wB|Gk`MDI zO`hlI#5@O}BLs+YB*cuzes9^E4knGSuMgKo`Q{szc4x(woa%Mtv_$pxO1j1BWE{H3 zi}xp6l|Cy}`;=K$esWTl3*ddNo{FJN6_&UrL z=4scR8o}F|ay|8%eHW5NLM+LQ)N&~kb@T37`2>g_yzaAJ{rYL9)JJ+$!K9imo>q_^ z+wG58J8xY5NF%%R{bDQRPL#7N!;2ltp`6tk$N84TY>kblP{v$$Tly11{7WLFLdInz{EDt>-q#Lt$NG?#xU zoXtQ}1XvXV9o-tB9|0$8z=F>gzISdVC)Wp1vRd;|y69g(Q6vMfJvH?*fQ$i-87o6| zKHhxQRviGG$?KC2K|rQ0_as)phrjHDWonf!0Z!`Ps{8uuwCgSMnc?<*r1q1fL5f}l z9NFg&CE0qiF~-n{1Q(EKrkaDB9rn9__}-6a)K*he{2}|yB$3ydDQsRO0KH0sL8q>| zI?l5ORs*nTkbHogH0?4UI;9BchM%^ux#m7E!e%eCm|z{713I4zcZMGMU9HAtr}}kE zZ;ttR8J6anP>cMSS-? zo_1f-*&RBL>*#2|*dM$uEGl4REJZf**l5i+CiP&z3e0>)P9@*zzLoDSaJ{CKm7hqF zKl=dVesF2U>E8Ngz4u9=>~Qg=?95ksPw6hLLTt-VpV^t5uYWK_TCJMp8%DB3AQ47P z^eP-@^G-q3#kbjm8~sT^*KbvG?(pr2|bMpR7HBFOpBnQQ4&!mOsqueK=|b3 zvj(>vmm@D=Bktb*!CTt=Z7n!7aJNjaZZp}s`Fg3|N7pLy(cG@>b7p4G4q41L%IpAC zjBvu|m0yZHeoQ64_~?Bx@kjd3Yu*ronn@7pI=t$*PHW^1FQmPkM_f%+Ed-V0g=R+( zNJp_&m&s&@l(S3tQTQhP`0Z6LV~8|yELk1J@FoJs(1?Jo+=K<~d1hTC@P2&Yl4-Z|LV)G6@R@mNS`b->EQ*59xw`~(PCGW+h2WZzML*v?M))I)@Xdj zOHFMvZNe1Kwno3)Z|T0ex_X&>!OH&c#Cw&%A@7>c(`4dTrZaP4W5y%z)_3eVKbLKr zQ&t=4DGscD-l^3cYJi%6VFm2y`b`8B0;^eQ=3 zP3A}(IZ}2WKZUWU8#kc{D^k?S{X*Rrh3r)(Q+FIaB^FAsw3I+i*)zWjr_Jymm*+a( z6N?8|#w{Ja0`3kSS?=S=WDmxl(Bi5sNZwOzjwzTuzdW9`kKx}cI4S%Oy})8Zc&CuT zN6jpzPS#;VOyZP8IepP+89);XN^5O}2f*am##R=29+@Z*rLLl+lWw363|n+$WZ*|M z=(tKs1~xS{snAWAy$SfJ3m%1q#nI;E^5$lv_w|)t<4B4e80FQZ`xqo64cBL~r_Kgy z?e&pbo>|qFMNgEjPv2u2Mx{(-fB802q^p(H?iZ2y;bJK2SzC2zXi@=Bg6OjuBEskH zXDTGyS%N*({jHxvg4D3HCA;xf6E%MxM+fO;H*HOK2Vt(AHvj7C`w&HcL4I|FOnv{X zY2jQ~#m!_X)i*#%Ki6|8=TPd}-+QJmB{5bO`B1F6KfYKiIXQmWNYH4@BJFbS6}lg( zB?ac)VlA)rt;=O*y02<|ujWE+39N?77k@6&`wI`Y+P*nBJeavJ{zy1GC!1K{YQ44W z%@Q+O`g;8fZVjjMleyeOe=)Hzk^8;ZPD@cfqlRUEnC2UKk8=Ck{fPOQumU+6mMKBV@KeNe*LPjf3?o08r_7Jj6Y0DK2Jp@esmi+-gkLJ+n$N%89;8n@kB4h)<5E#tY73*7>*ZlESMsaEP5orF*g| zWnd7gK%-*kg1a>(|GReiyUg}4I7_mm-JG^Lhu!!3`0v9T0}Z5YYC-&kN>$!E-5UxN zaY05{RgKN9LzF$Qj72gOo<)X`%(1)qVK+Xv?4mHMmYGJjI>mKF*pwd852qExf-_z0@uzJInd&0ot(9 z4LI7g(^eDZi-4IQeG)tY9?T)oJyt9Q1qBZWZ<7&*7eu-US_kCO3_2t|VdPd7D9Uq~Girfrmz2R(ekpH!r5e}CQ!I~TpMV?Dc| zoQP@`89o<*7v5_M#?od_9WlhhC?s$CVo8pe% zlO$lgoO;I((`|n1vfSE%wa&{tzY=$NKQ85mH@_0MPke03o1wKV&C`R6>TQ!#Ok6IR ziWz-B)Cl6&la31wRP&bXkrFhq601auH#}7?jR_P>fkz?4Tt|OPz8j`$`o?!yUAJ#f zxyBaAeSApp3@h;B@-jL)T2@8|IPEXv&!MiR*YJXZ;tK%8fa4VTE*mBck6DioXi*fO~*72&#t4i zV=Q40&zwe=!-5`sWU^AdT#k2RG2Qb)qR)gSlE~+0yY!@$?D>H`3!%KxBNp>ByJe?id{5 zPw56C0Bw7cw52b#^5~m$ak;%+G9eLVvlN-%SB;{a_I3hIO}h-W?O&r_9`R{U6H+6)VMMpn%H`*@ZgHyRP)kYG@@AzrX! zRoSko39-QGLUdoKU;{EeHJ2d3Na**AlUfeT*Pg#S8zKI2zC9)X)X(DFb^Cyp+7fCO%DGaU^z~OD6!s{m9zR>*Wud0A{ggivP~J+?5KhG0guR zHTs24dQ)FcUX0KQNs&ifM`+Ms$q026iaDU_{U$Z8EomUro{?0FRfgY+? z40r%mhbv+WC$XFDb+6``+UB;j)JOzHd6=Afv{*|aKU$okQ>*{^3GZ-oG9J^=&=3m~ zGksG7;Sin`o1YcC7vI&ri-D?ERrn;IsIeq@$)y2_*ge#If(3eQ$i1y(r&|`7B*koF6-TP75uf4g;pyq`bb!jF^S`{ zr<^SqlPP|(N_tuFpLL^K?4K=cwygHhMe>YK^^-KtHHruFujD@mg$Q_Wj1MbQhs~b6 z@sVHcS}d$HJ|L-)Zhn9@LY|R*^^}D3%8q7dcCh-NdJq@#UQ0tJ27Gw zO)Jq5nbhN#OS#G`4oqFeubHqLt#gX@&dZGZW;_3}S_Cm*IWaL|tgaLsn-VRJobiRU zY3*NhFo(|h9Un5NjwzV|q#~p?I9BbX$bDO2Doc;}FOKyY zzV{E$FGSXk&eL||nL4tw8x-ty00v=@iR73nSs3^JD)`~Uu2d#X4V6ld!@5eYH6#b> z5C0yP8)2`PM-mO9yqk0iW%v3$Z}qAH+k{Z}+~slIn5Q;q`7f+wH+4d*o}X^4L7 zwou9(DNcji`|ywXlV=h0g^y1@H(&p$-VE>_*04a&Zz9UE1KAth=+*RpK|d6xX#g!C zSzq-!pRm2Nx7SPaj81!RaNJzWR?WbGWLlJlftHTx7xURy5^F`^)=TUH0aZ_B>M4(5N(6FR85L;=oHNl1t$sqrT(QmFF0aTs+~Lw;{!f*H1T3 zpqgh;K!t((7`O3+JT&cOkhJ!Lx3rYB)GF1pjFpM~Wy)+)r5Zp+M=EF}=Bq2rUR*Bx z8v35*kt0{WF`{VZ-%2g-dmNBYYJCTh*K4l3Minb%Do0YN(Wm6(Yq)&%W|RaoCb~eT z@Tp+UU#SwqTG3}2$vBU3DXuPQcDG+i9p8St;`k@oCvD96d}OF^EJr#wti4^_FQTYu zqdASQk-+=Bv?K5Y`yPU)m@~&p^Rz8!Q7Zntop6?C!Lne_8Oi3`)W;7upAV@Cau!T% zL?nSH_am#h+k*xdAymJhCn(SEUTC%{Jxzgy7i@8D8Do*L9TCld2V;1-_)oTmdGjIR z%JbN&M{lb3kDWA$mHO1R$C^YspIa!kE4NDwA`t0%fuSIhGsEsd`XuRP=3~=!*lz$*{Cq?}|<#OOwCb zo5IL*cCND8iGMujjJSR$(w5HQEo6 zJoawZZO?SRv~c@|c3p1r?-pXyu$&8RpHqoQnXAmN*Dw|*h_op*S6T|S%>r+IqFY7e zWsCX*rED_0>(O47*yNalF^BFY9)W@3nAe{R)#n{YZ7KXqd{#+kBEx|a=5|AYm)aT( zahp%6m9{IQ8#ZL_yv#yE`q1ik!ldTpK;|i%a2FlH?tHj3ZHB!t+s0tJJ%=qZjab9z zjpo$48CXNkU{#%`Jxp@cN`WtjLM5Xz)jI+%^OuiLzZ-CiJ!1>*tFda;tmH7Rn7|6G z^C!D%^0q(w$>A5_byV`>-Q&T0?Zm9N#gkegdk4G(b)j<8M(gM9^egdBzNx`%X(+o!P8^KXvEuIRgXG%7}4)@e*&3ENe$qHk+OAMGXnqF%x_VY0HA z3b9&zVPZEm@y3Jy@)60bhsh;383p~N?C)w-&FgG5d`>O1*X6iOuZWHJavx!vw5IT@ zC8QDIdq}B9_Eyu6^D$D@kAh$G~(7ISvtSn|_(SWYr~&gRh(-V-=X^vZX9d|wC8 zCr@`#XKd}>-n$%S9LmT~`Z)~(@iQL_EpC@;JR=z66MObZ8K=$dy6+(Zp;^W1_@+WC zNHim0d~#J0T^A`KeA4@9DOk;8I7FC?jG|D%Xy`1uaQFD5hgFt?M=5g)t8=c28CI&TZky@4$f zr#tu2Na@wd_!*6_lK6E<^kpM-NCaOW2Y##Q$b9yJzgf404oNfZXrqFMV$Y?s-KzQy zk7L!$g5dsn8v9ut-QHAzB!|zRlARApDf_0finDH)eRrg~+YHOVd79|?s{7zhTO;!o zJxHD!7O5d43PyF-a!e9+CPv-b44=n$Yc1{4lTn<;D_Zg#MjK7JN5vfL|9FtG(DFco zp;yqLLX+bZV@l9_I!vIBYrUBrNEG#PXF|Ep8SuI9+X3S%-fTu^RS?*~G+U719&jME1Cb_N|2fpG8S-u{>YMB(s?8_^hP$>Oy z?GhyREJK2ZWrT}WrQFI!{Jly7Ee_i@rSgyV_K%o9O)4g<-Ysu$&w?0S&^={;p#5-J z{I=O&k_{9c0x3WUyU>79jtTTS^5c*0+{S8%JZVs%84q|;YDCQuQ)hNXcP~(*$Dm9r zIRQ(U1UFE+A{CSW>RXAJm>6YqhnUy%^C!Bp`pvfj+LW53nlcoJr;0+_+k5xQ+0a6h z%tn5Cq|UrSgPC*w_pV!lU*;;U)M8Z zfv<0ncsZ~t2dw1+)xt2+(haVci@g9^Tj-1gLdBxZ3rO|Mfb3lg+MmDo%8*|HJUY@TJC?s_x;xn9@OL-U9 zk1xiIF8sK_Rpx!_-`S~=qV0jw6;m9e(47+#>j*(HMg14`W?nqx)V*Q98{u#yr%Fqx z_pVTo?Jb@1MBmi#Jnf0}LMM+^vq?O;IT)M8#1E>R7in}eG#n#-4n-}piF2@-vDsEP zFZ~c~I?k2#zHeYAY2a02G1LCAbCAfz=GIz6s^!7q;i0R;np*96{~d%17CkR@Jf`q3 zh6ubg9i8aXp!j(<>(d<4_@PN0G2Dga$9TApgb4}pkqJ4X+!xHZqEvqnsna$*aNV!-*yY6~ zg|a)UO$HuzyBz@JgYqEA8^J3zl|+dPF~EbDEMh*Fm#p$El345A&==TgvLh-bk?H2a zxU(T-e(YvrKYQPmEERv#(W??uv04`iQ@`RA9QgurB(+i7x#7xM%8Gh=sTmm=OC3PG zj83_iar%B{DF$<`#S*!fXvQ@)+tmeusCo%@@Jp2DEc2)nPL;}ctHt5J1zD<&>HF1n z{8U=UG@b*X?B#oX})x0mU43FB+7aFIMmL!M>rkX9DVSg&bD=hF&XK(X)gv83eB{$GqAa12W1zC_q{D>@Av!X{9xwHtt-v0wf|+09s-f# zJA!83_g|4XT4lWcmd4oZv9+B3w~v@Z{WBaco6mS0>Fsff9kqiMQ@yA~Y6kO@Ej-<7 zoBqG}CM#V1OD4uoW~!(*7XcX6L*1hz(~L4cJK1L2vj8^r#O;inuPI~pEGhPi%3`vh z=z0k0AyF|q%LHnCRv7E)@Fx_ue|;qMJ^g481+wYDJ~cOaUB%cwh?ew{4o$AfR%5oADupPxV zS8v=MZ!O4Ub#@Vbzq(Ar!tHz`^0nJ|{#W8Sez($#(=W#1Gt?MXrb*zq8Bd0V>?UE` zTQKWl;Xaz0b`tu&g{_mihpX`nV{^-ZOQB5IO{?pI!+oVeR`q}Q7~p+mJ^L`)`g9$P zj9P%G)*%7=>ZI}eAJp)qTEtlS!*3#wa&{|~C^CwCs=8S2KB|I1 zDRBx3g7BIW0s|h?dp>!Tw(tR(U|w2tZ#Wf6Mem1>Bj=s=kJ$m zx>3mBjJ(%F*j>5ye{&ODC7%2IuWrK9VnxA+C+r>!(>B@L0jga%Zxmwo{;jUi&I))s zY-8I?hcc_)V@!J9#Cpe?aEc{+CA8vkDaLiVpt9G|KSz`JvyZsHA846cTaywI6>icX z3Zgo`tePKv4A>lOpbx2giADg%gep87pO7OvItqMoaQ~8y)AqK9m#8ckP~|r!S zU+?6ry30vtlWs-+YKrrnh$~KyS>ot!U9dF&P$cy5YK95^5Lhe4d$aL4ba$P5x1YM# z^EDltQf8^j@5c(z4Jc4VvZg67mk#;o_UXqmK9-EpK~-I}n^t46OKSYjD>`}@itbq0 z*!07yloOMa@APWFef!tM#-1FygZ#tn#I-C)3d|+AJ&k#b(`OutVvBxT@0=0 z8oX{EiASiAmT5d8J`5fT*VxqN$0c+0O%e%j%rD(H1Krnfep%#G{-c9j{>5dx{3sYh zh(auy@u}3A@K9Vm_=7T;E(i-;KNHtEgu|Z4w4*$WKT4_3FEl$XSsKM(Tqmhn|9vQR zU@P2NWlyc@m`Nv*vy2`$%@N|^D6w!t@Zijg+t5+M2#o$x$xIdatHY^X)Iq8ls7BU= zoRQLlVXr!f!^5KKRnqDX*SUnYL<(pe$^BZ}+uK`P;lWKnG%9wZAj;saUM-YCkB*Lj zs3ju_UJVF)nZ16kaQeOkv`Pd#P9iWt%OW-*At5SC7}@pW#1>ei-9)CbS>b&P<_GOC zDpmMcLTyjs@zYOrYJQNnXE&ZXpN1%w>I>v<<6~9zi=DaGGwc(buD>t#4Db>Y%zFCH zuY=3o_X1qe;bHYXDTJ!HT+(H~?_{@Wh?LAXjv@QFq$K^?^9#rGvps?=De0j9GT}da z3*i%VOmu&3VwkZC=_Cxye!=@BQ*}qN?%aCBM<+9BanqLm*_8gLwY^%^CS9wyxF^u| zcKubWiZkqtt&`O8f@x(13JAqT_0-N3hrix_^5n_N6MsxYYin#<<+WOdcAr{Lkx>{jX?9Fb5?bFl0 zsc9HFHHo4M)v-gH;6gWPVtrZC=&82dSDU^18am_jgiZ4~*8Ea&lXh(irro(#k(u^` zYy*0z#mvOw8i|IU;_C70i?ebc``=CKd7^2?b~=*<{Yc?b$SeDF8Hq@|(@c zjzOIq`~pZ*+Z=D{zI(S0h^AyvNCKfCr%Zf#;GC!xX}mhv{R>6I>3ZkssVOWhtdAI^SAf=$NoBaM5qaJRASuv0dJIw%M*mdYZiYM-=*tV1 z>5P=w*?e2|4blf!RJ6*O++VB|L{@689MAnxrHW5X9+G{LtiZFhQYgCt7#g%uH&pNezt2OoOY;T#;r)!lN(?svyjNTU!E#t;lsd0K}qKFNsO9 zalJe{?2cz1p6tn@1BGs&@Ue1nk+K?f0wESwo1MM=n=*|?w?jjuQ&4m(1!zemCM!s_ zUY;Kh`-5aBkH^W^s;WbftnCPom(RhSpR;)ZYQv@a%>uBrK&YfYO>h(aFUc@d=xu)Ve=Reag z!L`!RlWIO!ohDG#u_%nV$<(evC%Ln9$Fau5`)RB*e7b_|lBlD4?8FxBYJ zMNeO-%|(gLgFC-El0OX8G@2kgM@J~tZonEd>L4_BZ*tu?3C3eYCWCnm|MW>ANnLd* zgA|Qik%D(de3nz}2#JH1nyaPx`qb+lP3+Y zf7}-Cgk9YM_XCh0kd58|`TMMQbc`yc+BH_dcEP5SdEvAHyh`BBjg{!GFD_>P7B%A= z8yidJvd&q@AQj-Fpzvdge`9J|13MqI*D-|!posxIk2k36CAFFv8;j`|LeS0XU%-@3 z714^K~+^R0jX3@t1+=J&VqabP3s>(7FifU@$<_!iKrY=fYkZj*D>xaXEnpPY}+2tk5E{Fc7ch02^WMS2rZm69ftF8mf&78e8)S8 z!?A!T<4(vNU(mx*;mpPs)~Yhko)aQQjF`Z(c0;SYMisH*u{``>S45XPytYF+va<5l zG9aA|zA<+b6CuE|wdZe01_7EvE;=Fz{OHzf_kVYTmL(@YcT;ZD^xnyo35Cxb@TzPso&xObnzWIE< z{Iu0|Eoq`~t6Z)VA3=PqHJWp$gNswl*|E_OC2l&Rl*#DnrK7b=R7d40IY2xQURoFi178gkvoRF!*rp0#;~saUc!iNS<6HNQ z4?m8J`$8%BK+u-%@bKOEBgZ4v3&-vfY$hWbZk9_J2S_$oP$jmd#5aHG%pAU@w(mv9z{OPqZcLAp zf4EpS4ZW_AL~DI7l>vn<3lxw4sF?eex|NFS>51I&L=i(H-4u&|NW>qDiRJ2HEpq2_ zs)4UJd?sHxcW6k-+Iky~Qx|vl__#Qyjo**V@DL-WH|r<%G(p(Oi8Hx3PAVJJ>0Qo` ziVZ!{NR>OmnleHbmQlO}Pyb5Np*UBJa*A6hQf;tawQ)U!?Y)IX&Sf-*K--vjOmC=F*MIMClf@~8ti;-F#6 ziorDSs@8d%MEOToSHY8I1w0Teg9A=jLQM^inDO%4J6t3Dab(S1ugTB|Yk@!c2G>D9 zSZrAnl=HBta&jaC!$UU(u8Y=99Z(u0wJD^eJS+^&3tV>`dO_t<4jNuS0}Ki1&lOSk z@=)ms0o00B3)SrW9g@UMtwE7tb1*9)i3EoLAD?9s+&D1yhI`d;TV1MNkSbk#^*4#* z9o2bOS(&Z4_{S+GlSV8P_0tRK&Yt7RF~6eN+0tK{c_`w&v?@b?U+OR(TnCVyU$@7m;%0+no3P5CaV@Tbs=NW|YEdg$|W?2y}{kCg|~kK`S;f zB0`}o@y3~i!HQW|2ph`lqQB%nV+SguvGa3(cC_=0m7j~)|HLHF0bsY}w?Qddy78`% zT~75!9=E>~;d4Vj*U7QMi{*~%)AQt>sn5>_;qMbcZ~aBV>oHPwaMH3})o(iPfOC^~ z98TMl#BozURK?Bxymqo*DzdS;Sw(ys@%ApvU@5AHFIejwFc`St`we8YfB~c%t2TtJ z>*XfWvtiBPsewe}sDBMC4`$78JZLbqv;L2x6;S&+br)^!j zZm074s_`7I8<K;b0ZcB0WG zfY#>+;fB#d^~boln-dkesKZ6(d*C+#UDA1F;N|4|x0+x>p@y87469I9`9cajB7(-q z$VdY#D<)ps;@b}&KICy&+1Z$2b7OierU)X~v8xK$vI1-brYj9$_TXWVJU^c;xDk?4AUS1w=_V!kNd<(Jg_aiMF8k0(BI0K`s ztgI`$)m^ivnH@7XKU=%Ck*i%{+JRyyypYKb#Et9BX`}Bo zeNB3kWD{5nK%N$yoQc1LdpZM|8VewmmSGC)!3Qk2`R5W8Yzf;HR10=AN|lu*RbQnh zCH;Y_gh@SB>BE+Rfr07il*~L>cwA;fqAmCZEL8=!W`VB&&+6;zn;7+fL@#Yr|7)Ov z%wHoPtyT^L);4FevI!lGigap~oOT;POiaJYBXPzAZ&6Wd^(8KENO<@xc=wmNdAEtN zgkjD#7}Q`DB$yh<=-ov`Kr*u+o@(|R045;_^)*m+SwTW09N-^csPS41k4=gis*_C~ ziNR6gH^}|+rP6Geijq<+_?H5-(wd&0UL8F&SC-2er2CQdwYnMtq}{gQ#~jxGHa9O} zI3{bX1N{9#l8RBYTvSzdVieQ3JF!7Fb~smF-H!7^QF?khtYhc(vCvGK`}YG-557}; zkU1+k6+`6p+qB>m#1E+lcMXn@=XOjM=o}svjerYR_$5tUbtIfl71z`lpOEkmoI(>5 zX1oy?1xNB%J&9}pH~fA9U1Iuoc5H+tL`D5J%KbyYzu!N-m}?1u1K=FOXLGG8i;2Cl zVr?8T2t{Q^2a5rxNWw>xJP84?H)LZB%U*DCanaC7$S&Nh3D9I3f!m-l2NbakS$GFM z4b8w8k?j9Dfa-_6GPm=zRL`lK;lxehbvE((%w{SF!ita+>rLhYQbd-&Ef;aLi z1Qx!5Yx8mq2~md#|8_^nGhmhpAr$b`@s7`#m1n1|WdQi&v|p^?`UsvoH!eBCrw`n7 z_&1#m^3(9|%ESMkzwKMN3D^@2Qac?OQUqe;-i>n!BmAec^X=hnQp@|$=w0D4gs*G< zyPN|kC!4ns4)-9IaXL}*)9MqX(YX1wx2WN?Fs)p(=kNaKm@CZARSO_iF#mge?WYll zE~jPV=K<79HZLsjlH1yu8xiRo0P;X|4OD~z7+(PDMDP}8a_9&eqVoPV;{Q5B-=W*4 z59^dP?c6>sUk-#Z{neItYO!vvV@&jC`er@W>hR^H+dq5uY-)PCiuj(87$Bjv$}nLC z_=p4GcE>M2tgEkY%n2ExiuC_>l|NP> z?AxgDokI8cU3UJ$(T#zM3T?d#nzE)KxP?v%2i1*@{Ed&m463{xFPVlT1r804jfgG3 z?nSS?U({e!swyjshFoGvBT+P*>ZRPMNcGX$uzdailT-hNv(ze6fB|{;D6Un)ZG_4I zoRjqqt9D?l%=x?E6y$YW6SeMWZ>OW8g!6;R`e9jF8RU=Ow1-RpwtZAoGgU<0)mlbe z+8TP#)uUqEQkaCTfyhG+6tx3t8+;wKmkDe$tYhKl_k?DK;0Kid5(Vbtr$^*Xki!DI zKngA8X0AY5as{=vwiX1;Z0ziy8=#boj0)cX_}gv=D;ZQ#kPw0nYx=k^?!l1^9dfAp z;J6Ma;_4e5j1+hTX?tpFYB-1v4jgj&fNl3VB?T1jp_^0=mDj|~inM{Xb-zWPnakyw zBQtY|$f@jqV*!{U2@7&Gse;~m(G{=Dxu~pLl%6U)fCdZ8c@gpP`Xwbdn^>h43}QW~ zWtp3sTUvT9cH!39KqDD7H8t>qqKULYKIUTf`l<$G(eZ-Cthaxc`OZ}RSGE(f@E+jY z!l#w_6iiI0;tQ?p9;Eg6cXsA|r`7pp^P{IHC29^LOX$TsffSI76q%?DRCezde4=i@ViKp zt<8=@GGs`1jg65broB|sItUG_KC{pW5dsVlPar2JmreZ~+X(rxn@nJpvYeGwB4yHd zG-bHv%_xFA;+t6z0Li=`aYG9r;R6NQCEfqi;FbwPgGI;~j~qmRJ}4x$_fkK&LDc*Z z1B2QR)^lfHW!3}8Ja_S_Xq#O4VUmP{2ijfMof-yLaE>3zD@?PkV0sE_)U!ag;=3`PRf_X?;C8HrCkOybhcJ*7+~I zF)6cnL`3h5jQj|!6&~~^FY8lYeg&L53R%oS3q$S6g5fLrVY*e~H4H#8#Y#b^V?qK> z6IAOfZ5s|w^TXSV*GcN!wJ73_+O_SDLj2e1e@7p907%qwT<<%McuFAq|Bge2VEzwa zTtS&`Wa0VRRbEd|4HEa^5mpaupiP%-V^?Gwws&u zYZ94^kEGND{Vu}%<`hDNioSaFj@J*B$9e1XO{j!C&!kU-_YB_hse^=;K>^9ty}U)Z znPQDkU=9##FZikujwpEdZw-9#@j$QBf9B_LJnal@ zYBkl~p`obmMW~*AMH24n>48)591`iH_iG_F15@7*QPRM`SES6_hhP+#^qXV^^@2J? z5$Pt7>Kow&XByJ^=5}_hWUalumawZplU2FM3DN`tY4XUBsz+~hQ>%g|KKna6koKxQ zdkRDekSBo^zy+?^8ahf>Oj_D(t~{wFBl>N`p>k%q_T|Za*SZz~ZB(X(ATA;c&Ov!+^4uo+~)~BMPQdCp~lv8#M zurWF-fx^TEjUC6wk9T$ybY7-n+Cc$mLN#aP?&`*d2Y8J^X@X$bxxuhA!7(`ojSyEh zZ{0zHd3#kBra}kxp1gpY*ZoFh0o^9UwjkiV@wp!{!1ko1^v_R&dTV@q4O|dXQmm6V zLE$md-Mf5n_Cq=C-aR1mAW6-BcR5^Bov7L#z03V}ARSDH`Wj?Ig{D7@pskfdLVIZj zFS!@^6hV&9$jPA=rB!WVW@rd0=~Crfxxqy;GBSwY%>Pu&kh%QC_;I$J)E@P5CFVm(24^3z(INkg$znR2U&L(hu{>3!yy)0 z3&g}I;AA^|wuY8L(C{;v>LoN+grlyx;ki)pA#|Pn@`b|M+FDkxNN+KqWRCo{5C?d^ zj)|~bz#_nRLa6|HdEA$mm#;+y5qALrftV0jUI4kl6owESR%BvoDiaPWhxe;5a8AL} z1Bf6B8FZI|8Bi^n|NHmpxiA^WOMuG#tD2~qs%j1SKHz;v`*O&V($e;hj#Xd3+BiDON=v_4yL;~*1iZg~{Fno~>0qFycJtIwlmJ4} z&ycxaN8%V2gCd@yCL^+DE#Ls55wL!N>qF%Rx>mVRmZBh3=pj^yhst8H_w^MleQW5k z0>>?`IucOAjJM*3`1tu(QOqIMhRFTq9UyeWM#`Fd^q5g=1nOx(__D9nfndzk)U=I9 zIY%b&)2G)>399!_VP}H8vV0D$eYx!oMp;%$iV*TO&|f8i$IZ{4+jdoLFSoPvCI zM#kd*#-gde$x>t9$xp-wKAD&zhkcP0>cfXJGBRM+)Rj%44g4L*gRRL%(1NF55M{vaC9z&XfP3xo@R+QJCyoOme>VGAiS~ zE4_F7t^bgpOxV;jdxiTnDKn-LxHpbubNE zzD@YH51|hcxcj*5oz;)XT6tN*NbIt_i$dT~A6gfLE>{&`TV=R)zro79`9Dfwso7Kh!S2Mv%gU-mCu8U%K=)u4q4Iark#mEM9ii}m?8Y%pzZKV;;9wbHnFT-4 zP0T@CE{D@SW5}HW2NecEUdGkdw#?sOOhpBUZQK%^G$_=hc^)MO2X{jOJobnaNO2(0 z(arcEvoZr5W>6FN_iuv&@#^^MZK2{DJOBsFcrn#z8e1^|ZtA*tmT5zN;-Z>lx$+o9 zVTwHCXju@(R3&ZD4ImIVk&!va8MiV5L72+o&*R@_Xa>b#JRI*G6Lg(#2MSCPwy@aO7D{>#5|zlJ-qGbY8<-?VXg(SO?|nw^mC zAC8~SDXz})Wlcy-+;!n&iZ49zqxB(#0*vM4Kjs$=^Q1UMuEuF)VN%_Zb+<-nsNQ3z z3NY|^OW|Jzvz$G)>KA|;wRJzDt>=fpo3(j0x~oLAvzp(az-i9;6@u}v!$swf3)Rpt zy)pZ=KJx(ukL`Y8!Q;&9-tUYm;1$S*cML9M<=lrzPpgI1szlHoCHi|{i7pL1$AwqW z%Ab3d0Z5CW!kPu;KOwH2+6`7gHyy`(A_$+}5~O@zt(9jE{Y}p2sNu06bh3myC0tx( zVQC3f?&3`vA@zIro0HcT=9NWC6W{+4_teTk34y0TwIjDVU>p$^bl$yVgi<992^#QL zh?nN)eZLf^zA0lr%X>e9mX3Ey^Bx;NX!}z0#rpb({1&nJH=jg{HdlNwh2avbq0oB^ z^2c!gID_^0gNA}|kjR%P`1MK_17_Jg#TjLadHkTE=?%(5N_o=9jLE8hH|d&oldDC1 zF`(Zf)EgmDmI29FV2BBs#spC$d5j$Rz;)vcQ63&PE_G7b_&>odq4+%cZua+zTnOT)T(_xZC7I6VCU0A-_q-p@geAT`Srd+FxAXU2!e|`M}B26GzD?|BrQ$xt~?*042>%J^R z3bQH#jxE!82x1Xxn=@=vROVD#v}q=|l3sJ=K_^LLDX@jw9|`32s56_`MnLL~l-tg4 z=4k)_DeTMRseapTcO$7#ndjYPj--?fk*P9g&RAq9L&nUKp_HM?9GR8OvqD9wkeLud z*v5(-p>)>n_nh}UpL3q`zW(SFHv8-Qz3=P3uC>;6t%VL#g-2%!#Heo!W?6N7nps-g znd5p=8c&O~zlLk5H&^TD>&<5ZKic=5K703;MDAN|*=+JhncYHN?oL2{I1c5NO~3Wj zWv{TH(e$`dp}M|dvh~!;P{Q5ZYs;fl50qw7Xy!{eIS7QeyUwKl>qehgx>Dg|Eg?tp zyl$27(Y`t3Pc?t<_X}$kav7oS#A^s1x}rVX2{NCLMSM9H!C(@>zonN)%}=k$+r+k> zYmG&t`VN=S=y+cQEq-ke*-ZR=KBGE=k(X7q|PKJ54v+%|4wW z;V!VVg2wyD6k|*y=Nn&14U;S2rp#S#!pX>f}{!+Q@TSC1SyZ+llA5PBE7|O^Q6IN8y#kJst+cqd`19D!ldAMo&GXj)vg1-=BPE7uGp_^@mtfS z-}!FV;Upt-G}ou3i6f72{Q`m}{VD4Rog!XR5&O9hOC!zC7C9O{z&0m!fcoZ_K3}<@ z3|w6Gna|F}rJV5}iVF+d0Kq(B|7q+V$1R;#)wvy9`4n@6Re1BL8;&bHbeA6GOWFn<2Ow(eT0^5>1Rf8Ng?4^pXrb#@qQfH~hp$aBS@ zKc&kQ$}h4KoCiN*DD>hiT*(epS>mLJ7v3XE`d#5LAUPXEO(-?^Kihfg_y=og_PqXn zk_yMy#vKZ~nUanfiMm~FD>7{VV%HNH+cCp^3Y;E|-4BGAQy3_k2+N>C0?bnlnz9Gqt;uQU&)_ zT)|8L24xe5rG0^2o9JCtE!{t+HEMoOZY> zf=mMdf)t%5xG!Q8!7fd^uxiI)HPAGK6N==y0dFK)K~aEw+V{z)IK8{O4~Hm>FymzX zYgG2O+XWCb*>%@`EKtrYf%7i|H=%l&0W4r)K{TNZ4?^(k>+7Rs5qhqWu|Zp+bsvMG?fg?KBoNZSv(a*k_}s^tO<+I594b?KY!kViEm z=o5%?)XZ;rUqUmAP}s}hK_RP-aqYd4~#B|ZtmX(v!1jr6OH8l}s!_vY6DwFVAw?-G1QDblDuxp6m1kOgG zZ&XyUpBwP3fq+3{N)o>WQU&nlx5Y(@|L^s{Kuk4g8U=Tp!0SOmHM>i`=GXk0A3NF^ zN$($cps2V_$ zwA$CTL`E}^GPE2Jb7^If7+S)Mc z6z)kt>s4_txlCp6h;yY7)2+JU*0W=M<&J+)p@K(Dh>gW~Q?AG!47#&!`7*TY^~3^P z4UElCr&$#mNO5Vncj;wV8;$3|<{5s_sKtwruRA}(ZDHKGQ=X0kY*tf4gEj6*5aq}p zs<@JdGTk|pPgr{p{)-#zau{# zj?)wTpU|woUDdnYt~;MnSC3wu-uxrZU-Ot=(0A79*x2S`a9I2ns8`Y)G?zGlg42uH zw(pg{5gOWnE;BF-Xn@3R38L?P9!^}fY8tCq^(NkVzWQJs$xaac*$%qecD~<-jm*r< z_U_ygEHC+0D_GqS=wYLa(P`sn?#&$?3&0)2JlTm~;HW@W6G<;eiwUxsBL%6o)RY)L@fcW&+&+lZ88jSe>cGuBzH9kIG@AD!m zGEnDa%`7f@|M=* zm>y^yu!BD&%Q{+FWt7NYy;>0&Nw<&Uu1bFWeQ`1An9D^kuPU&yzi580tf(g(3I7lP zL#9mihffR|UcQuD?A`ho^Uxb7mo|2H>-7`qAzYx)Sd8@(MUB5mSCBiI5T7ZaSfmygefF{hUdG9jC9VhMWj=^Xc+o6-|%*@92{*!Qqrwn?d z9A)RN@5khr3&>nte3na=*RFJWtiRh(!>w!IZW}##ZfviN4+~?-4hoDG2~MRB6nT4& z`cPBTv07R8r2Q-*teaz5fBG@yxWW0-dy)Rq__U1mS58L8AFz+7oB?+AHqBu8DHn@$ ztKCgEt?RaXi_I{YL|7i6FoZ5~Zm{}NNsD>J5%!j>j0}@1*U1hI*?}70vFcozf<3i@Hs2C05#hgmijI~Q6g=IoR2-kpb6bTU-YurfOb+%`7~HZVJOz`Xd8 zCSSTWYi?X(q7+;ukl(6|K2^$LolR1reh zV&IuvZu#zRMABV&+e7t;4Hg;_f>pk#E}NokTjJ&7r&G&4`GLhDZ4LNLWZE zFbc!Zw{DdAGEV$TgPy7&_%+D82)$IS=bqg)69oruxH{Z5p%fUWtQaqG& z38wqMAxCK6c{g*(VY9M@*FC}0do$<44!Ey^($OR#w(zH!SV+I_JUwAD*0*%fj1s! ziIKgDi3uE5gyC6z9^>7K^-ju0X@@s#*npC7Kbjez{CzSYR#w`y5V1&9oc_T3W=g5p3CXc_`8mf42rc(#yH>vc&Pl;sSwC784a!Kw&crMC2lSA1|LPtY9fh@nfq?-KOeZH%VdGLeU3GPJP0dyyvT8lX)n0Dc+?twE;%YB# z4yO$qGQxw1O-|-!U~rx4e2i1eV~lxYq$-BzVtpU3QlLN^0Ag{mwg!;;Kkb+V*bcwF zY519t^(0cL47mpiLit@IlltpQj(bh8sLojQHA)u}XDn!Zw=IfpI(d`dR!gK>A(FZI zVabbh+?`Sh$Wr&fY^+TUfNoM|A^v zYpp4xw3~w2;vr#+0qLJn)P^b>@eAKFW+v0Z*g&eqg(K9K#{4l_rp|e+n^83G+&0aQ zMN4WV$ApzuhL&VSsOT%k$>sTI#-9JM{&u0EB_x5d7BnUSteNI~2VcLf66Kuan;5>% z{)lep`}|8TbfsKZ1xU-7|5#b~KQfsdrx4b%T@6 z>@N1{xtFqs!jl;U5*R}2yHA2`6;{(?+9Z-*Qlr+lUhqKPbbnL8$+O=e%O_b&p1y1l znd^AMVY=Vu{GqB)e)6tvNBbKZo`i;g`iqXJ?0%0YI6_tMA^FdHVVPPrE^PrhLN146X8rJ}N?JX50BwrU5eBZ#Fw*;P_Hd5W?)VJ$vd35;y1L zpsPogYh{IpQChjVFEcP>c0<6)4x1l(WdSX|gw?d3$pcKbkyg_mk`&GtOJJvvDGCz} zHMM;~DDB`S8q734F_D#-si&iZLGG^0+(!txakF zT*=DpMB&4&0nMaFKRw66))MnKA|k;yVltRmcI^jQ>z94pLfIeAim&(QYyZ4lr6QL1 zvfE}~?s}q&$?I}*a!`%l=TcKqF{pC2MY{`#%k)>U8%6@&9!H}j*c#d=+9z|g3a0du z)jE;^Yni&du?rVcH~9z>$VYy6ahGgrC{bSf-G4wgNv3FGEq^A?_EYSYvRQW5Cj|vj zTu{}29e5VQm2XfO$OUOMz&;RG@d-cbh5+mL%7=YOQ`I6(q9TaKA26bjkYq z(Z7@QcZXMxv^8y*y?OooNlxJl(!Q3&9Ywt&E_;%T?7sqK^ibJjcW#L*vtgD_^&~dd zsdnSGp%$qqwbiYEnsh!jY$mi0k;n!;+g}`*6k#mqTAA{=+1#OgWLBi;+eF134T~57 zLcrR)6O|O7qhIDe1huc+d;iG#zxKV1P)QqnySQkR zf<%Bi=?u&&Bg!xOTU6|l(pw@Bj(fm?O7_ok3#YUf>KE{rKQ8(Q++BD$bmb-eoBFh$ z-Rnfv{T~~gb23iwCY7&#^x+!i4StuLaH~t2@Tl=gP7c(ZqTY*_MMUzGydQ*x(dg^z zdkM`}SXIQ-MG;S88_&+p4!HOd(io(G9#uW~XrMuD`+I7a+MCdlAl5Ge$%mOQTqd*% zpFVR2YE@KS-r`jhJ`kg>(;LJKOMAuKpiY1>#SMq*$b*L;U1aj1yJm9X=^ys5sR^~m zWeh%T72yv%FY=tmfTG&O=yT|bA)5+|j0CK*3qM}(L!c5QiuxB zcmn*x5sm{5yZ!6)1$7%6K4N-ao&!az^%~uZgP>}mTPhu6zivvThMi9@QYd%_a-(pz z7L+)A7^j92?I4Lyvl=Fq&cRCM@83h+ zbK^*5Gp@#h1q6J!(jHFS>o#Q!^+!)=e);eSw_n6-O3*!T*R;NEJK=?xcGJ!vGuc3t znZRd$|Ngy64to8|W*wwrY9g3a4w!96F0ss9H_FY+&aQBqW|FpI$;`-j4JysXs_w*o zPIlAa8yE-HIL28WU0m#v;IC4W(lhgJ zH$2!?8ZIs_jAY+$l>Ow%{I_of-#AZ5*mP|f{}pSqu5J4xZ%Ek`m6a3n4ml1=`J7xj zg@+ANbofCx4pF-X73Sfhfhm#9=6l5wdE)7#mLa2A`Gdn zZr-c8EAaYtH%Sx7nR?9prO(&jSY%API&w2Up3N&UI$G&mG%Ym|GbY;^bVzF5$`YA* zHP`5O)6{MsVvH_tifz4qC^|)B=SDgDAowI)J8(VyRk1|;e8k0kGe+x?o0$iZZ#%#a`Iv18h7n^`DgrYex^&WNs z1VY&CRkv=PNnq-$Jxe-_EOA=@HRplH@#KnPomgLFp;F^96mBBVe_c@exN;wauz4Z>n6bpI7N!pau1+fz9T3d%hZKpIeTA(dN?|><3 zCxPs;-JYFOJtrAfM|Vnhtp__k!5=cz$kfZh3immb>}=dkoub=U)3p zCI1#R0s!+r-aw0jqR3=2s#qYBJLQi!eB$c@sJN%G2sc`!=I03F8L6RwLx=t*%;af zG(TSazBs@s9pryJ@H$jFvGDAu69xW{)2SEO7T(`y{-$f`eIu~QyvaWD)Tf}t!6^qt z@*MDh(a{Uzf^QlcsHmtQ>It4$hSugo{NWtmhe#8FsOaeHC(L~TS^_A00pkMB3?3v9 z$ec8+)*m8oGSLsbc&VVI6n+2xjZ}kk=bqtgUF75ah%ZRw6%@RYGYq>;@ zmY|?Uf%Ry*Y3NmoM6X4DRn8gbq*=1^%Y-n83UiuPX^P>u`gDn*C5YiBG{G@3ThA>@ zArN690s2Ye3j;#Pj*0d8`HmWkAB;0E#2upt$wqjwE;hLzMlccFvRq6|N?*f0p+p8! zGP}3x;O)|Xz@5(s9H5dA?$UUu+k>y30#(*ngg}Kax+*VYh2RjYtg7mRz7Y(K$n$UV z%E7JNkbR=TACiTN%1Y_gnPS9+peRhwz=&tqfurXO6a|n70Erc#a9diwBGZ^%2y3IU zXG__)tu9;VNmLZQ?JcjgV58i{g%;Ul;6*0GD!F<{x0mt}aRAl;dBDH}k$dklRpYdQ zk{cTf-FpW;U%y7etaCovzA$Ef0lssV5p#PuWP>^S4WwD#=4hpIKuexi8Jd*u; zckUdt>&ylZ0cpBXvkj3<1<@1QDitg(zX0N-gdJNAgk_wtX^l#L(3NV)ET`Aj`a=fP zrX=~q{3ErexRUPKvo2(k7Jmk-@cnm|8dBfuaiCxt#tG_5K&7xk;TP7a1jmbq1)hR3 z#U?Q(rbnjAutP9n6g|W`#>U2=t(sq28=@I}5wsz$!tUNZ$(6&Wf$j;8l`VRuo*74R zXyTo^k&C!|Kn@$p8$W~c*oh=Aw;r-70Jk~xmPS5*die@$IoMVA-A zeYeH1#)EDlOJI*YoJjJHpdXo-SVALPxnp`7O-MK;4ukc9TX-s;Smc2=R^jwF5uD!G zxVU8bxA>bfM88Z-G>G0VFE3Y0yAu)aF!pK8bcZ~i)Z{MgSboL`c8LfJ7H;n3MUkGi z{5C#YuWRCdy}L}rk|)*j`0m`;&!q@pjKqaR7Do&YyFeaHxov+15goQFpjR&tq3Bs- zKXRFrO}|p?U^vlRa0lu+^<FerxdVAkfzS->Lm#tQ{w~h3a)2(iV$NX#T+RN_Ri=sU+yQRQi z)Jkgu07@U6&fuX2wi!)>Y>|=XHL`0s2P_YG48`#Ut-WY$yk)8(#LxeoOcpn4sH;m$ zPYdbht->RpAM8T!={^zI7j>L`uwmu*vw2g+wt~T+U zy!@;7j5BqwUTxmS!ovuEI5-sX@bU2xi53X8`)N{SskdyYM7N@s*9u^FY^Ct-I5DgD zAY(B+=P+BixQyjd8_N>;RV7-Eqfv2j{P*6eGM`qu&C=G{N&Yr`a=~M6DjJW|Rni2% z0ABU-Bbogf>P;&FfdWr(G{q5!hVxhUf}( zjjfKjtgHE0$58bmURo8chuBA5nAUSZzhKK5Z>u&t`|7lbs{65bnSR)qEdA#cYkaTt zLyKd9-jrB5(~PU8CPD!W@#I&dO$j8U{Peh_#Wdw-d&9Hs2fSms%4J2jV`~v!)W9|S z?R%OdZKbBY!`_o^2PP?1Iwj!B#U9GL$cDHAU9-qB=WpzS=T15~wAYTsT|aS#mTKZfit|9xzLR-CZ5}u4)KD6ka{!bV_$s z9(Y&4hpwKUMp8CA-rysHOaR`6z_I!7y9Lxy-iJVNVgVL_584fsDb+o!(415=sKDaE$WbYPxC4`|Rk3u<`gcIv&c;g;L~PDAn@>1qG)EH!DYQ*p6u7 zt?`i)OM8xKQk;4uke~93|L+^dLnuf%LH7iWO!@hL#$wDAwBQy7$ug87Txe|QF#_0u zmymWeH#Z{Gtw#ZYQwk9kb?_PH@JF{d+%O4N)+H&I^ehu2Ct4#Hnq z;6a!LORio&5~D6V){raC&h67b@GaDyWhI7`OZAQc`~=UY4mac^WvJH%C+P^W6s^U! zx&k4=uvG2G96xWI&e*AV$I39o0`8#|_d`1$!c zR{w}jNH~|EFT}^k6ZCU=Ic|<6*9o@+2c5_C@y!8wHg<9)t7&1v8gxX3Y;fPvR2dAuGVzgs#(LOAyj2KfpS%Jm5 zXJ3&iDyOyGfA9c}yq1_0g|RajiFA-?2{kts-ty<`bO=jGwlH+3e~r;`ph}S%HoTd~ zG1ol(J;B_Cl3|}%h|?LFnNg*FoS)xgBd#%(SDrnk;MPtJ%y%5c2|-} zzmt=5U=1h=5v|grl<^t;(IAsDtgAr^klT{|RvzmQZX|r%+*lMuk1F1Gzw0_ml2C~{ z@_^}c73w$;Qo=Z><#Fd$a}q`vk!UBTgt1zi)b>_590mDVuX*MaMw4=nfp0no`I`{=}jJL@|?s15qCcJ zmg!Ks=Pmkq=&Mf$=T|QpAy0wYywzf!kw7QK`QYS^T-D)W zQ&W9U>`UE52v2o~G-Xk#M~oUoTX7Y-Wf2I}FE^bfDQ&BJYgv?;xnGL`PVziqWpg_M z#uUDvUVa!d#f~5$7I%B}Xv8Rs(xbNR>u3p=Y1_6}QPqhev)gtEB^zp|_wQ-KmCzBE z+GZoprgrI4j%j}RpZw|(_R_SFcNrb+tGOXrts0-&js{1t5BBv5#57uou{IV7HM9;( z1d=xs%0ft+2SO}ZqOCT`Y4R}uKEj!x?a*OY>Qz11%{gPNSp2H-B=KQ)AzR&95-;1= zNgV;k#v<{m?@}#AUMmyfpZ+6LgAXQzteQO{g zpjyuLIv3Hmi*@tS)dsPru4$&LxVxk=t(DqAtu@zmkN)7daD8#w!?bPn+0*WE+_&+z z4DE9s8woF{eS|5?`slWQ&iv48e>c@<(ZBWN|C?+WaBO^W|J6Jb3RnW+v&5UU{^7e% zOy{TqIB9&E-l0V8t|mMSpr<3a2R_bmIGV via a socket or a file. This package allows one to easily encode + * any graph-like data structure and send it for visualization to + * OracleLab's Ideal Graph Visualizer tool. Assuming you already have your own data + * structure that contains nodes and edges among them, creating a + * {@link org.graalvm.graphio.GraphOutput} specialized for your data is a matter of implementing a + * single interface: + * + * {@link org.graalvm.graphio.GraphSnippets#acmeGraphStructure} + * + * The {@link org.graalvm.graphio.GraphStructure} interface defines the set of operations that are + * needed by the graph protocol to encode a graph into the IGV expected format. The + * graph structure is implemented as a so called + * singletonizer API pattern: there is no + * need to change your data structures or implement some special interfaces - everything needed is + * provided by implementing the {@link org.graalvm.graphio.GraphStructure} operations. + *

+ * The next step is to turn this graph structure into an instance of + * {@link org.graalvm.graphio.GraphOutput}. To do so use the associated + * {@link org.graalvm.graphio.GraphOutput.Builder builder} just like shown in the following method: + * + * {@link org.graalvm.graphio.GraphSnippets#buildOutput} + * + * Now you are ready to dump your graph into IGV. Where to obtain the right channel? One + * option is to create a {@link java.nio.channels.FileChannel} and dump the data into a file + * (preferrably with .bgv extension). The other is to open a socket to port + * 4445 (the default port IGV listens to) and dump the data there. Here is an + * example: + * + * {@link org.graalvm.graphio.GraphSnippets#dump} + * + * Call the {@code dump} method with pointer to file {@code diamond.bgv} and then you can open the + * file in IGV. The result will look like this: + *

+ * + *

+ * You can verify the behavior directly in the IGV by downloading + * diamond.bgv file generated from the above diamond structure + * graph. + *

+ * The primary IGV focus is on graphs used by Graal compiler. As such they aren't plain + * graphs, but contain various compiler oriented attributes: + *

    + *
  • {@linkplain org.graalvm.graphio.GraphBlocks code blocks} information
  • + *
  • {@linkplain org.graalvm.graphio.GraphElements method and fields} information
  • + *
  • Advanced support for {@linkplain org.graalvm.graphio.GraphTypes recognizing types}
  • + *
+ * all these additional interfaces ({@link org.graalvm.graphio.GraphBlocks}, + * {@link org.graalvm.graphio.GraphElements} and {@link org.graalvm.graphio.GraphTypes}) are + * optional - they don't have to be provided. As such they can be specified via + * {@link org.graalvm.graphio.GraphOutput.Builder} instance methods, which may, but need not be + * called at all. Here is an example: + * + * {@link org.graalvm.graphio.GraphSnippets#buildAll} + * + * All these interfaces follow the + * singletonizer API pattern again - e.g. + * no need to change your existing data structures, just implement the operations provided by the + * interfaces you pass into the builder. By combining these interfaces together you can get as rich, + * colorful, source linked graphs as Graal compiler produces to describe its optimizations. + */ +package org.graalvm.graphio; diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.word/.checkstyle_checks.xml b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.word/.checkstyle_checks.xml new file mode 100644 index 00000000000..44c996f7eea --- /dev/null +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.word/.checkstyle_checks.xml @@ -0,0 +1,240 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + From 10a1edcc220ac4926b9fe61ff4ac05cbb5704338 Mon Sep 17 00:00:00 2001 From: Jamsheed Mohammed C M Date: Mon, 6 Nov 2017 21:28:03 -0800 Subject: [PATCH 19/61] 8152470: Add COMPILER2_OR_JVMCI definition Defined(COMPILER2) || INCLUDE_JVMCI changed replaced with COMPILER2_OR_JVMCI Reviewed-by: kvn --- src/hotspot/cpu/aarch64/frame_aarch64.cpp | 4 ++-- .../cpu/aarch64/sharedRuntime_aarch64.cpp | 13 ++++++------- src/hotspot/cpu/arm/compiledIC_arm.cpp | 4 ++-- src/hotspot/cpu/x86/frame_x86.cpp | 4 ++-- src/hotspot/cpu/x86/globals_x86.hpp | 4 ++-- src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp | 8 ++++---- src/hotspot/cpu/x86/vm_version_x86.cpp | 4 ++-- src/hotspot/os_cpu/bsd_x86/thread_bsd_x86.cpp | 4 ++-- .../os_cpu/linux_x86/thread_linux_x86.cpp | 4 ++-- .../os_cpu/windows_x86/thread_windows_x86.cpp | 4 ++-- src/hotspot/share/code/scopeDesc.cpp | 4 ++-- src/hotspot/share/compiler/oopMap.cpp | 12 ++++++------ src/hotspot/share/compiler/oopMap.hpp | 4 ++-- .../gc/cms/concurrentMarkSweepGeneration.cpp | 6 +++--- src/hotspot/share/gc/g1/g1CollectedHeap.cpp | 6 +++--- src/hotspot/share/gc/g1/g1GCPhaseTimes.cpp | 2 +- src/hotspot/share/gc/g1/g1MarkSweep.cpp | 8 ++++---- src/hotspot/share/gc/parallel/psMarkSweep.cpp | 6 +++--- .../share/gc/parallel/psParallelCompact.cpp | 6 +++--- src/hotspot/share/gc/parallel/psScavenge.cpp | 4 ++-- src/hotspot/share/gc/serial/genMarkSweep.cpp | 2 +- src/hotspot/share/gc/shared/collectedHeap.cpp | 4 ++-- .../share/gc/shared/genCollectedHeap.cpp | 4 ++-- src/hotspot/share/oops/method.hpp | 4 ++-- src/hotspot/share/oops/methodCounters.hpp | 14 +++++++------- src/hotspot/share/runtime/arguments.cpp | 2 +- src/hotspot/share/runtime/deoptimization.cpp | 18 +++++++++--------- src/hotspot/share/runtime/deoptimization.hpp | 4 ++-- src/hotspot/share/runtime/frame.cpp | 2 +- src/hotspot/share/runtime/globals.hpp | 4 ++-- src/hotspot/share/runtime/rframe.cpp | 2 +- src/hotspot/share/runtime/sharedRuntime.cpp | 4 ++-- src/hotspot/share/runtime/thread.cpp | 2 +- 33 files changed, 88 insertions(+), 89 deletions(-) diff --git a/src/hotspot/cpu/aarch64/frame_aarch64.cpp b/src/hotspot/cpu/aarch64/frame_aarch64.cpp index ab53d646d80..6df308b3eb5 100644 --- a/src/hotspot/cpu/aarch64/frame_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/frame_aarch64.cpp @@ -431,11 +431,11 @@ frame frame::sender_for_interpreter_frame(RegisterMap* map) const { // This is the sp before any possible extension (adapter/locals). intptr_t* unextended_sp = interpreter_frame_sender_sp(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (map->update_map()) { update_map_with_saved_link(map, (intptr_t**) addr_at(link_offset)); } -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI return frame(sender_sp, unextended_sp, link(), sender_pc()); } diff --git a/src/hotspot/cpu/aarch64/sharedRuntime_aarch64.cpp b/src/hotspot/cpu/aarch64/sharedRuntime_aarch64.cpp index ac5eb020361..9b67fe7a5f2 100644 --- a/src/hotspot/cpu/aarch64/sharedRuntime_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/sharedRuntime_aarch64.cpp @@ -41,7 +41,7 @@ #ifdef COMPILER1 #include "c1/c1_Runtime1.hpp" #endif -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI #include "adfiles/ad_aarch64.hpp" #include "opto/runtime.hpp" #endif @@ -114,7 +114,7 @@ class RegisterSaver { }; OopMap* RegisterSaver::save_live_registers(MacroAssembler* masm, int additional_frame_words, int* total_frame_words, bool save_vectors) { -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (save_vectors) { // Save upper half of vector registers int vect_words = 32 * 8 / wordSize; @@ -2688,7 +2688,7 @@ uint SharedRuntime::out_preserve_stack_slots() { return 0; } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI //------------------------------generate_uncommon_trap_blob-------------------- void SharedRuntime::generate_uncommon_trap_blob() { // Allocate space for the code @@ -2894,7 +2894,7 @@ void SharedRuntime::generate_uncommon_trap_blob() { } #endif } -#endif // COMPILER2 +#endif // COMPILER2_OR_JVMCI //------------------------------generate_handler_blob------ @@ -3070,8 +3070,7 @@ RuntimeStub* SharedRuntime::generate_resolve_blob(address destination, const cha return RuntimeStub::new_runtime_stub(name, &buffer, frame_complete, frame_size_in_words, oop_maps, true); } - -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // This is here instead of runtime_x86_64.cpp because it uses SimpleRuntimeFrame // //------------------------------generate_exception_blob--------------------------- @@ -3200,4 +3199,4 @@ void OptoRuntime::generate_exception_blob() { // Set exception blob _exception_blob = ExceptionBlob::create(&buffer, oop_maps, SimpleRuntimeFrame::framesize >> 1); } -#endif // COMPILER2 +#endif // COMPILER2_OR_JVMCI diff --git a/src/hotspot/cpu/arm/compiledIC_arm.cpp b/src/hotspot/cpu/arm/compiledIC_arm.cpp index abbd673b6ba..211f5d06963 100644 --- a/src/hotspot/cpu/arm/compiledIC_arm.cpp +++ b/src/hotspot/cpu/arm/compiledIC_arm.cpp @@ -33,7 +33,7 @@ #include "runtime/safepoint.hpp" // ---------------------------------------------------------------------------- -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI #define __ _masm. // emit call stub, compiled java to interpreter address CompiledStaticCall::emit_to_interp_stub(CodeBuffer &cbuf, address mark) { @@ -89,7 +89,7 @@ address CompiledStaticCall::emit_to_interp_stub(CodeBuffer &cbuf, address mark) int CompiledStaticCall::reloc_to_interp_stub() { return 10; // 4 in emit_to_interp_stub + 1 in Java_Static_Call } -#endif // COMPILER2 || JVMCI +#endif // COMPILER2_OR_JVMCI // size of C2 call stub, compiled java to interpretor int CompiledStaticCall::to_interp_stub_size() { diff --git a/src/hotspot/cpu/x86/frame_x86.cpp b/src/hotspot/cpu/x86/frame_x86.cpp index 918a413adee..26e24371adc 100644 --- a/src/hotspot/cpu/x86/frame_x86.cpp +++ b/src/hotspot/cpu/x86/frame_x86.cpp @@ -436,11 +436,11 @@ frame frame::sender_for_interpreter_frame(RegisterMap* map) const { // This is the sp before any possible extension (adapter/locals). intptr_t* unextended_sp = interpreter_frame_sender_sp(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (map->update_map()) { update_map_with_saved_link(map, (intptr_t**) addr_at(link_offset)); } -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI return frame(sender_sp, unextended_sp, link(), sender_pc()); } diff --git a/src/hotspot/cpu/x86/globals_x86.hpp b/src/hotspot/cpu/x86/globals_x86.hpp index 80344a3c8cf..b383af47006 100644 --- a/src/hotspot/cpu/x86/globals_x86.hpp +++ b/src/hotspot/cpu/x86/globals_x86.hpp @@ -46,11 +46,11 @@ define_pd_global(uintx, CodeCacheSegmentSize, 64 TIERED_ONLY(+64)); // Tiered // the the vep is aligned at CodeEntryAlignment whereas c2 only aligns // the uep and the vep doesn't get real alignment but just slops on by // only assured that the entry instruction meets the 5 byte size requirement. -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI define_pd_global(intx, CodeEntryAlignment, 32); #else define_pd_global(intx, CodeEntryAlignment, 16); -#endif // COMPILER2 +#endif // COMPILER2_OR_JVMCI define_pd_global(intx, OptoLoopAlignment, 16); define_pd_global(intx, InlineFrequencyCount, 100); define_pd_global(intx, InlineSmallCode, 1000); diff --git a/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp b/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp index 996ae27cbc9..b7cf3ad898b 100644 --- a/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp +++ b/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp @@ -151,7 +151,7 @@ OopMap* RegisterSaver::save_live_registers(MacroAssembler* masm, int additional_ if (UseAVX < 3) { num_xmm_regs = num_xmm_regs/2; } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (save_vectors) { assert(UseAVX > 0, "Vectors larger than 16 byte long are supported only with AVX"); assert(MaxVectorSize <= 64, "Only up to 64 byte long vectors are supported"); @@ -260,7 +260,7 @@ OopMap* RegisterSaver::save_live_registers(MacroAssembler* masm, int additional_ } } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (save_vectors) { off = ymm0_off; int delta = ymm1_off - off; @@ -270,7 +270,7 @@ OopMap* RegisterSaver::save_live_registers(MacroAssembler* masm, int additional_ off += delta; } } -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI // %%% These should all be a waste but we'll keep things as they were for now if (true) { @@ -323,7 +323,7 @@ void RegisterSaver::restore_live_registers(MacroAssembler* masm, bool restore_ve __ addptr(rsp, frame::arg_reg_save_area_bytes); } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (restore_vectors) { assert(UseAVX > 0, "Vectors larger than 16 byte long are supported only with AVX"); assert(MaxVectorSize <= 64, "Only up to 64 byte long vectors are supported"); diff --git a/src/hotspot/cpu/x86/vm_version_x86.cpp b/src/hotspot/cpu/x86/vm_version_x86.cpp index 22e7b653f22..f322d60a6ed 100644 --- a/src/hotspot/cpu/x86/vm_version_x86.cpp +++ b/src/hotspot/cpu/x86/vm_version_x86.cpp @@ -944,7 +944,7 @@ void VM_Version::get_processor_features() { } } #endif -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (MaxVectorSize > 0) { if (!is_power_of_2(MaxVectorSize)) { warning("MaxVectorSize must be a power of 2"); @@ -996,7 +996,7 @@ void VM_Version::get_processor_features() { } #endif // COMPILER2 && ASSERT } -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI #ifdef COMPILER2 #ifdef _LP64 diff --git a/src/hotspot/os_cpu/bsd_x86/thread_bsd_x86.cpp b/src/hotspot/os_cpu/bsd_x86/thread_bsd_x86.cpp index 7f5838f806f..ea06c7d03bc 100644 --- a/src/hotspot/os_cpu/bsd_x86/thread_bsd_x86.cpp +++ b/src/hotspot/os_cpu/bsd_x86/thread_bsd_x86.cpp @@ -73,7 +73,7 @@ bool JavaThread::pd_get_top_frame(frame* fr_addr, void* ucontext, bool isInJava) frame ret_frame(ret_sp, ret_fp, addr.pc()); if (!ret_frame.safe_for_sender(jt)) { -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // C2 and JVMCI use ebp as a general register see if NULL fp helps frame ret_frame2(ret_sp, NULL, addr.pc()); if (!ret_frame2.safe_for_sender(jt)) { @@ -84,7 +84,7 @@ bool JavaThread::pd_get_top_frame(frame* fr_addr, void* ucontext, bool isInJava) #else // nothing else to try if the frame isn't good return false; -#endif /* COMPILER2 || INCLUDE_JVMCI */ +#endif // COMPILER2_OR_JVMCI } *fr_addr = ret_frame; return true; diff --git a/src/hotspot/os_cpu/linux_x86/thread_linux_x86.cpp b/src/hotspot/os_cpu/linux_x86/thread_linux_x86.cpp index b98a851908f..fbdcef60406 100644 --- a/src/hotspot/os_cpu/linux_x86/thread_linux_x86.cpp +++ b/src/hotspot/os_cpu/linux_x86/thread_linux_x86.cpp @@ -74,7 +74,7 @@ bool JavaThread::pd_get_top_frame(frame* fr_addr, void* ucontext, bool isInJava) frame ret_frame(ret_sp, ret_fp, addr.pc()); if (!ret_frame.safe_for_sender(jt)) { -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // C2 and JVMCI use ebp as a general register see if NULL fp helps frame ret_frame2(ret_sp, NULL, addr.pc()); if (!ret_frame2.safe_for_sender(jt)) { @@ -85,7 +85,7 @@ bool JavaThread::pd_get_top_frame(frame* fr_addr, void* ucontext, bool isInJava) #else // nothing else to try if the frame isn't good return false; -#endif /* COMPILER2 || INCLUDE_JVMCI */ +#endif // COMPILER2_OR_JVMCI } *fr_addr = ret_frame; return true; diff --git a/src/hotspot/os_cpu/windows_x86/thread_windows_x86.cpp b/src/hotspot/os_cpu/windows_x86/thread_windows_x86.cpp index 9926a89666f..1e457ce4bf9 100644 --- a/src/hotspot/os_cpu/windows_x86/thread_windows_x86.cpp +++ b/src/hotspot/os_cpu/windows_x86/thread_windows_x86.cpp @@ -81,7 +81,7 @@ bool JavaThread::pd_get_top_frame(frame* fr_addr, void* ucontext, bool isInJava) frame ret_frame(ret_sp, ret_fp, addr.pc()); if (!ret_frame.safe_for_sender(jt)) { -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // C2 and JVMCI use ebp as a general register see if NULL fp helps frame ret_frame2(ret_sp, NULL, addr.pc()); if (!ret_frame2.safe_for_sender(jt)) { @@ -92,7 +92,7 @@ bool JavaThread::pd_get_top_frame(frame* fr_addr, void* ucontext, bool isInJava) #else // nothing else to try if the frame isn't good return false; -#endif /* COMPILER2 || INCLUDE_JVMCI */ +#endif // COMPILER2_OR_JVMCI } *fr_addr = ret_frame; return true; diff --git a/src/hotspot/share/code/scopeDesc.cpp b/src/hotspot/share/code/scopeDesc.cpp index 27eb37ad550..4807f61d74c 100644 --- a/src/hotspot/share/code/scopeDesc.cpp +++ b/src/hotspot/share/code/scopeDesc.cpp @@ -228,7 +228,7 @@ void ScopeDesc::print_on(outputStream* st, PcDesc* pd) const { } } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (NOT_JVMCI(DoEscapeAnalysis &&) is_top() && _objects != NULL) { st->print_cr(" Objects"); for (int i = 0; i < _objects->length(); i++) { @@ -239,7 +239,7 @@ void ScopeDesc::print_on(outputStream* st, PcDesc* pd) const { st->cr(); } } -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI } #endif diff --git a/src/hotspot/share/compiler/oopMap.cpp b/src/hotspot/share/compiler/oopMap.cpp index e412e4bf2d0..c36c4ff3ff0 100644 --- a/src/hotspot/share/compiler/oopMap.cpp +++ b/src/hotspot/share/compiler/oopMap.cpp @@ -268,9 +268,9 @@ static void add_derived_oop(oop* base, oop* derived) { #if !defined(TIERED) && !defined(INCLUDE_JVMCI) COMPILER1_PRESENT(ShouldNotReachHere();) #endif // !defined(TIERED) && !defined(INCLUDE_JVMCI) -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::add(derived, base); -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI } @@ -461,12 +461,12 @@ bool ImmutableOopMap::has_derived_pointer() const { #if !defined(TIERED) && !defined(INCLUDE_JVMCI) COMPILER1_PRESENT(return false); #endif // !TIERED -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI OopMapStream oms(this,OopMapValue::derived_oop_value); return oms.is_done(); #else return false; -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI } #endif //PRODUCT @@ -726,7 +726,7 @@ ImmutableOopMapSet* ImmutableOopMapSet::build_from(const OopMapSet* oopmap_set) //------------------------------DerivedPointerTable--------------------------- -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI class DerivedPointerEntry : public CHeapObj { private: @@ -819,4 +819,4 @@ void DerivedPointerTable::update_pointers() { _active = false; } -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI diff --git a/src/hotspot/share/compiler/oopMap.hpp b/src/hotspot/share/compiler/oopMap.hpp index 5428f3bdee2..6c9fe4ee299 100644 --- a/src/hotspot/share/compiler/oopMap.hpp +++ b/src/hotspot/share/compiler/oopMap.hpp @@ -427,7 +427,7 @@ private: // oops, it is filled in with references to all locations that contains a // derived oop (assumed to be very few). When the GC is complete, the derived // pointers are updated based on their base pointers new value and an offset. -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI class DerivedPointerTable : public AllStatic { friend class VMStructs; private: @@ -463,6 +463,6 @@ class DerivedPointerTableDeactivate: public StackObj { } } }; -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI #endif // SHARE_VM_COMPILER_OOPMAP_HPP diff --git a/src/hotspot/share/gc/cms/concurrentMarkSweepGeneration.cpp b/src/hotspot/share/gc/cms/concurrentMarkSweepGeneration.cpp index 0901117f646..c49112957b0 100644 --- a/src/hotspot/share/gc/cms/concurrentMarkSweepGeneration.cpp +++ b/src/hotspot/share/gc/cms/concurrentMarkSweepGeneration.cpp @@ -2296,7 +2296,7 @@ bool CMSCollector::verify_after_remark() { // way with the marking information used by GC. NoRefDiscovery no_discovery(ref_processor()); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTableDeactivate dpt_deact; #endif @@ -2869,7 +2869,7 @@ void CMSCollector::checkpointRootsInitialWork() { print_eden_and_survivor_chunk_arrays(); { -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTableDeactivate dpt_deact; #endif if (CMSParallelInitialMarkEnabled) { @@ -4171,7 +4171,7 @@ void CMSCollector::checkpointRootsFinalWork() { print_eden_and_survivor_chunk_arrays(); { -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTableDeactivate dpt_deact; #endif diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp index 8c2ea13bf4a..79bb1c91b1c 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp @@ -2597,7 +2597,7 @@ void G1CollectedHeap::gc_epilogue(bool full) { // FIXME: what is this about? // I'm ignoring the "fill_newgen()" call if "alloc_event_enabled" // is set. -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI assert(DerivedPointerTable::is_empty(), "derived pointer present"); #endif // always_do_update_barrier = true; @@ -3010,7 +3010,7 @@ G1CollectedHeap::do_collection_pause_at_safepoint(double target_pause_time_ms) { _verifier->check_bitmaps("GC Start"); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::clear(); #endif @@ -4439,7 +4439,7 @@ void G1CollectedHeap::post_evacuate_collection_set(EvacuationInfo& evacuation_in purge_code_root_memory(); redirty_logged_cards(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI double start = os::elapsedTime(); DerivedPointerTable::update_pointers(); g1_policy()->phase_times()->record_derived_pointer_table_update_time((os::elapsedTime() - start) * 1000.0); diff --git a/src/hotspot/share/gc/g1/g1GCPhaseTimes.cpp b/src/hotspot/share/gc/g1/g1GCPhaseTimes.cpp index c939a8459ab..979b4836d8c 100644 --- a/src/hotspot/share/gc/g1/g1GCPhaseTimes.cpp +++ b/src/hotspot/share/gc/g1/g1GCPhaseTimes.cpp @@ -414,7 +414,7 @@ double G1GCPhaseTimes::print_post_evacuate_collection_set() const { debug_time("Redirty Cards", _recorded_redirty_logged_cards_time_ms); trace_phase(_gc_par_phases[RedirtyCards]); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI debug_time("DerivedPointerTable Update", _cur_derived_pointer_table_update_time_ms); #endif diff --git a/src/hotspot/share/gc/g1/g1MarkSweep.cpp b/src/hotspot/share/gc/g1/g1MarkSweep.cpp index 9fd8fa62790..9b2b18274bb 100644 --- a/src/hotspot/share/gc/g1/g1MarkSweep.cpp +++ b/src/hotspot/share/gc/g1/g1MarkSweep.cpp @@ -62,7 +62,7 @@ void G1MarkSweep::invoke_at_safepoint(ReferenceProcessor* rp, assert(SafepointSynchronize::is_at_safepoint(), "must be at a safepoint"); HandleMark hm; // Discard invalid handles created during gc -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::clear(); #endif #ifdef ASSERT @@ -96,7 +96,7 @@ void G1MarkSweep::invoke_at_safepoint(ReferenceProcessor* rp, // Prepare compaction. mark_sweep_phase2(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // Don't add any more derived pointers during phase3 DerivedPointerTable::set_active(false); #endif @@ -111,7 +111,7 @@ void G1MarkSweep::invoke_at_safepoint(ReferenceProcessor* rp, BiasedLocking::restore_marks(); GenMarkSweep::deallocate_stacks(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // Now update the derived pointers. DerivedPointerTable::update_pointers(); #endif @@ -204,7 +204,7 @@ void G1MarkSweep::mark_sweep_phase1(bool& marked_for_unloading, if (VerifyDuringGC) { HandleMark hm; // handle scope -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTableDeactivate dpt_deact; #endif g1h->prepare_for_verify(); diff --git a/src/hotspot/share/gc/parallel/psMarkSweep.cpp b/src/hotspot/share/gc/parallel/psMarkSweep.cpp index db566bf0ad0..185cbb2f62d 100644 --- a/src/hotspot/share/gc/parallel/psMarkSweep.cpp +++ b/src/hotspot/share/gc/parallel/psMarkSweep.cpp @@ -192,7 +192,7 @@ bool PSMarkSweep::invoke_no_policy(bool clear_all_softrefs) { allocate_stacks(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::clear(); #endif @@ -203,7 +203,7 @@ bool PSMarkSweep::invoke_no_policy(bool clear_all_softrefs) { mark_sweep_phase2(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // Don't add any more derived pointers during phase3 assert(DerivedPointerTable::is_active(), "Sanity"); DerivedPointerTable::set_active(false); @@ -252,7 +252,7 @@ bool PSMarkSweep::invoke_no_policy(bool clear_all_softrefs) { CodeCache::gc_epilogue(); JvmtiExport::gc_epilogue(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::update_pointers(); #endif diff --git a/src/hotspot/share/gc/parallel/psParallelCompact.cpp b/src/hotspot/share/gc/parallel/psParallelCompact.cpp index 9fea2221bfc..c46f0d41aa0 100644 --- a/src/hotspot/share/gc/parallel/psParallelCompact.cpp +++ b/src/hotspot/share/gc/parallel/psParallelCompact.cpp @@ -1032,7 +1032,7 @@ void PSParallelCompact::post_compact() CodeCache::gc_epilogue(); JvmtiExport::gc_epilogue(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::update_pointers(); #endif @@ -1783,7 +1783,7 @@ bool PSParallelCompact::invoke_no_policy(bool maximum_heap_compaction) { CodeCache::gc_prologue(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::clear(); #endif @@ -1799,7 +1799,7 @@ bool PSParallelCompact::invoke_no_policy(bool maximum_heap_compaction) { && GCCause::is_user_requested_gc(gc_cause); summary_phase(vmthread_cm, maximum_heap_compaction || max_on_system_gc); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI assert(DerivedPointerTable::is_active(), "Sanity"); DerivedPointerTable::set_active(false); #endif diff --git a/src/hotspot/share/gc/parallel/psScavenge.cpp b/src/hotspot/share/gc/parallel/psScavenge.cpp index 3cf3e2302a0..64c0b7486ab 100644 --- a/src/hotspot/share/gc/parallel/psScavenge.cpp +++ b/src/hotspot/share/gc/parallel/psScavenge.cpp @@ -331,7 +331,7 @@ bool PSScavenge::invoke_no_policy() { save_to_space_top_before_gc(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::clear(); #endif @@ -601,7 +601,7 @@ bool PSScavenge::invoke_no_policy() { assert(young_gen->to_space()->is_empty(), "to space should be empty now"); } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI DerivedPointerTable::update_pointers(); #endif diff --git a/src/hotspot/share/gc/serial/genMarkSweep.cpp b/src/hotspot/share/gc/serial/genMarkSweep.cpp index 8f22aea7a19..70156a6b70d 100644 --- a/src/hotspot/share/gc/serial/genMarkSweep.cpp +++ b/src/hotspot/share/gc/serial/genMarkSweep.cpp @@ -92,7 +92,7 @@ void GenMarkSweep::invoke_at_safepoint(ReferenceProcessor* rp, bool clear_all_so mark_sweep_phase2(); // Don't add any more derived pointers during phase3 -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI assert(DerivedPointerTable::is_active(), "Sanity"); DerivedPointerTable::set_active(false); #endif diff --git a/src/hotspot/share/gc/shared/collectedHeap.cpp b/src/hotspot/share/gc/shared/collectedHeap.cpp index f5d93a400e6..5a0f729f80e 100644 --- a/src/hotspot/share/gc/shared/collectedHeap.cpp +++ b/src/hotspot/share/gc/shared/collectedHeap.cpp @@ -241,7 +241,7 @@ void CollectedHeap::set_barrier_set(BarrierSet* barrier_set) { void CollectedHeap::pre_initialize() { // Used for ReduceInitialCardMarks (when COMPILER2 is used); // otherwise remains unused. -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI _defer_initial_card_mark = is_server_compilation_mode_vm() && ReduceInitialCardMarks && can_elide_tlab_store_barriers() && (DeferInitialCardMark || card_mark_must_follow_store()); #else @@ -545,7 +545,7 @@ void CollectedHeap::ensure_parsability(bool retire_tlabs) { " to threads list is doomed to failure!"); for (JavaThread *thread = Threads::first(); thread; thread = thread->next()) { if (use_tlab) thread->tlab().make_parsable(retire_tlabs); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // The deferred store barriers must all have been flushed to the // card-table (or other remembered set structure) before GC starts // processing the card-table (or other remembered set). diff --git a/src/hotspot/share/gc/shared/genCollectedHeap.cpp b/src/hotspot/share/gc/shared/genCollectedHeap.cpp index d6624936a4a..6c9f07b9744 100644 --- a/src/hotspot/share/gc/shared/genCollectedHeap.cpp +++ b/src/hotspot/share/gc/shared/genCollectedHeap.cpp @@ -1067,11 +1067,11 @@ class GenGCEpilogueClosure: public GenCollectedHeap::GenClosure { }; void GenCollectedHeap::gc_epilogue(bool full) { -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI assert(DerivedPointerTable::is_empty(), "derived pointer present"); size_t actual_gap = pointer_delta((HeapWord*) (max_uintx-3), *(end_addr())); guarantee(is_client_compilation_mode_vm() || actual_gap > (size_t)FastAllocateSizeLimit, "inline allocation wraps"); -#endif /* COMPILER2 || INCLUDE_JVMCI */ +#endif // COMPILER2_OR_JVMCI resize_all_tlabs(); diff --git a/src/hotspot/share/oops/method.hpp b/src/hotspot/share/oops/method.hpp index 0a7ee37f100..6866b833fdd 100644 --- a/src/hotspot/share/oops/method.hpp +++ b/src/hotspot/share/oops/method.hpp @@ -271,7 +271,7 @@ class Method : public Metadata { int highest_osr_comp_level() const; void set_highest_osr_comp_level(int level); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // Count of times method was exited via exception while interpreting void interpreter_throwout_increment(TRAPS) { MethodCounters* mcs = get_method_counters(CHECK); @@ -426,7 +426,7 @@ class Method : public Metadata { return (mcs == NULL) ? 0 : mcs->interpreter_invocation_count(); } } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI int increment_interpreter_invocation_count(TRAPS) { if (TieredCompilation) ShouldNotReachHere(); MethodCounters* mcs = get_method_counters(CHECK_0); diff --git a/src/hotspot/share/oops/methodCounters.hpp b/src/hotspot/share/oops/methodCounters.hpp index 5b97cba61a8..1d526ca46d0 100644 --- a/src/hotspot/share/oops/methodCounters.hpp +++ b/src/hotspot/share/oops/methodCounters.hpp @@ -40,7 +40,7 @@ class MethodCounters : public Metadata { #if INCLUDE_AOT Method* _method; // Back link to Method #endif -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI int _interpreter_invocation_count; // Count of times invoked (reused as prev_event_count in tiered) u2 _interpreter_throwout_count; // Count of times method was exited via exception while interpreting #endif @@ -130,7 +130,7 @@ class MethodCounters : public Metadata { MetaspaceObj::Type type() const { return MethodCountersType; } void clear_counters(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI int interpreter_invocation_count() { return _interpreter_invocation_count; @@ -154,7 +154,7 @@ class MethodCounters : public Metadata { _interpreter_throwout_count = count; } -#else // defined(COMPILER2) || INCLUDE_JVMCI +#else // COMPILER2_OR_JVMCI int interpreter_invocation_count() { return 0; @@ -170,7 +170,7 @@ class MethodCounters : public Metadata { assert(count == 0, "count must be 0"); } -#endif // defined(COMPILER2) || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI #if INCLUDE_JVMTI u2 number_of_breakpoints() const { return _number_of_breakpoints; } @@ -213,7 +213,7 @@ class MethodCounters : public Metadata { return byte_offset_of(MethodCounters, _nmethod_age); } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI static ByteSize interpreter_invocation_counter_offset() { return byte_offset_of(MethodCounters, _interpreter_invocation_count); @@ -223,14 +223,14 @@ class MethodCounters : public Metadata { return offset_of(MethodCounters, _interpreter_invocation_count); } -#else // defined(COMPILER2) || INCLUDE_JVMCI +#else // COMPILER2_OR_JVMCI static ByteSize interpreter_invocation_counter_offset() { ShouldNotReachHere(); return in_ByteSize(0); } -#endif // defined(COMPILER2) || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI static ByteSize invocation_counter_offset() { return byte_offset_of(MethodCounters, _invocation_counter); diff --git a/src/hotspot/share/runtime/arguments.cpp b/src/hotspot/share/runtime/arguments.cpp index 263091cf981..c1590ea2ac7 100644 --- a/src/hotspot/share/runtime/arguments.cpp +++ b/src/hotspot/share/runtime/arguments.cpp @@ -1860,7 +1860,7 @@ void Arguments::set_ergonomics_flags() { #endif select_gc(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // Shared spaces work fine with other GCs but causes bytecode rewriting // to be disabled, which hurts interpreter performance and decreases // server performance. When -server is specified, keep the default off diff --git a/src/hotspot/share/runtime/deoptimization.cpp b/src/hotspot/share/runtime/deoptimization.cpp index 087e26f5019..fb93fbdd712 100644 --- a/src/hotspot/share/runtime/deoptimization.cpp +++ b/src/hotspot/share/runtime/deoptimization.cpp @@ -192,7 +192,7 @@ Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread bool realloc_failures = false; -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // Reallocate the non-escaping objects and restore their fields. Then // relock objects if synchronization on them was eliminated. #ifndef INCLUDE_JVMCI @@ -282,7 +282,7 @@ Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread } } #endif // INCLUDE_JVMCI -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI ScopeDesc* trap_scope = chunk->at(0)->scope(); Handle exceptionObject; @@ -303,7 +303,7 @@ Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread NoSafepointVerifier no_safepoint; vframeArray* array = create_vframeArray(thread, deoptee, &map, chunk, realloc_failures); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI if (realloc_failures) { pop_frames_failed_reallocs(thread, array); } @@ -792,7 +792,7 @@ int Deoptimization::deoptimize_dependents() { Deoptimization::DeoptAction Deoptimization::_unloaded_action = Deoptimization::Action_reinterpret; -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI bool Deoptimization::realloc_objects(JavaThread* thread, frame* fr, GrowableArray* objects, TRAPS) { Handle pending_exception(THREAD, thread->pending_exception()); const char* exception_file = thread->exception_file(); @@ -1151,7 +1151,7 @@ void Deoptimization::print_objects(GrowableArray* objects, bool rea } } #endif -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI vframeArray* Deoptimization::create_vframeArray(JavaThread* thread, frame fr, RegisterMap *reg_map, GrowableArray* chunk, bool realloc_failures) { Events::log(thread, "DEOPT PACKING pc=" INTPTR_FORMAT " sp=" INTPTR_FORMAT, p2i(fr.pc()), p2i(fr.sp())); @@ -1211,7 +1211,7 @@ vframeArray* Deoptimization::create_vframeArray(JavaThread* thread, frame fr, Re return array; } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI void Deoptimization::pop_frames_failed_reallocs(JavaThread* thread, vframeArray* array) { // Reallocation of some scalar replaced objects failed. Record // that we need to pop all the interpreter frames for the @@ -1443,7 +1443,7 @@ Deoptimization::get_method_data(JavaThread* thread, const methodHandle& m, return mdo; } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI void Deoptimization::load_class_by_index(const constantPoolHandle& constant_pool, int index, TRAPS) { // in case of an unresolved klass entry, load the class. if (constant_pool->tag_at(index).is_unresolved_klass()) { @@ -2350,7 +2350,7 @@ void Deoptimization::print_statistics() { if (xtty != NULL) xtty->tail("statistics"); } } -#else // COMPILER2 || INCLUDE_JVMCI +#else // COMPILER2_OR_JVMCI // Stubs for C1 only system. @@ -2386,4 +2386,4 @@ const char* Deoptimization::format_trap_state(char* buf, size_t buflen, return buf; } -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI diff --git a/src/hotspot/share/runtime/deoptimization.hpp b/src/hotspot/share/runtime/deoptimization.hpp index 0f151301e5c..2d4e9ee6f6e 100644 --- a/src/hotspot/share/runtime/deoptimization.hpp +++ b/src/hotspot/share/runtime/deoptimization.hpp @@ -151,7 +151,7 @@ class Deoptimization : AllStatic { // executing in a particular CodeBlob if UseBiasedLocking is enabled static void revoke_biases_of_monitors(CodeBlob* cb); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI JVMCI_ONLY(public:) // Support for restoring non-escaping objects @@ -162,7 +162,7 @@ JVMCI_ONLY(public:) static void relock_objects(GrowableArray* monitors, JavaThread* thread, bool realloc_failures); static void pop_frames_failed_reallocs(JavaThread* thread, vframeArray* array); NOT_PRODUCT(static void print_objects(GrowableArray* objects, bool realloc_failures);) -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI public: static vframeArray* create_vframeArray(JavaThread* thread, frame fr, RegisterMap *reg_map, GrowableArray* chunk, bool realloc_failures); diff --git a/src/hotspot/share/runtime/frame.cpp b/src/hotspot/share/runtime/frame.cpp index 1ae48bdedf9..49516ea8d40 100644 --- a/src/hotspot/share/runtime/frame.cpp +++ b/src/hotspot/share/runtime/frame.cpp @@ -1148,7 +1148,7 @@ void frame::verify(const RegisterMap* map) { // make sure we have the right receiver type } } -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI assert(DerivedPointerTable::is_empty(), "must be empty before verify"); #endif oops_do_internal(&VerifyOopClosure::verify_oop, NULL, (RegisterMap*)map, false); diff --git a/src/hotspot/share/runtime/globals.hpp b/src/hotspot/share/runtime/globals.hpp index 557605d2108..e2c33aec46f 100644 --- a/src/hotspot/share/runtime/globals.hpp +++ b/src/hotspot/share/runtime/globals.hpp @@ -99,11 +99,11 @@ define_pd_global(uint64_t,MaxRAM, 1ULL*G); #define CI_COMPILER_COUNT 0 #else -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI #define CI_COMPILER_COUNT 2 #else #define CI_COMPILER_COUNT 1 -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI #endif // no compilers diff --git a/src/hotspot/share/runtime/rframe.cpp b/src/hotspot/share/runtime/rframe.cpp index 40283896d85..5b948ac48cb 100644 --- a/src/hotspot/share/runtime/rframe.cpp +++ b/src/hotspot/share/runtime/rframe.cpp @@ -155,7 +155,7 @@ void InterpretedRFrame::init() { void RFrame::print(const char* kind) { #ifndef PRODUCT -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI int cnt = top_method()->interpreter_invocation_count(); #else int cnt = top_method()->invocation_count(); diff --git a/src/hotspot/share/runtime/sharedRuntime.cpp b/src/hotspot/share/runtime/sharedRuntime.cpp index 7cb09c201e4..8e4c90f9199 100644 --- a/src/hotspot/share/runtime/sharedRuntime.cpp +++ b/src/hotspot/share/runtime/sharedRuntime.cpp @@ -102,13 +102,13 @@ void SharedRuntime::generate_stubs() { _resolve_static_call_blob = generate_resolve_blob(CAST_FROM_FN_PTR(address, SharedRuntime::resolve_static_call_C), "resolve_static_call"); _resolve_static_call_entry = _resolve_static_call_blob->entry_point(); -#if defined(COMPILER2) || INCLUDE_JVMCI +#if COMPILER2_OR_JVMCI // Vectors are generated only by C2 and JVMCI. bool support_wide = is_wide_vector(MaxVectorSize); if (support_wide) { _polling_page_vectors_safepoint_handler_blob = generate_handler_blob(CAST_FROM_FN_PTR(address, SafepointSynchronize::handle_polling_page_exception), POLL_AT_VECTOR_LOOP); } -#endif // COMPILER2 || INCLUDE_JVMCI +#endif // COMPILER2_OR_JVMCI _polling_page_safepoint_handler_blob = generate_handler_blob(CAST_FROM_FN_PTR(address, SafepointSynchronize::handle_polling_page_exception), POLL_AT_LOOP); _polling_page_return_handler_blob = generate_handler_blob(CAST_FROM_FN_PTR(address, SafepointSynchronize::handle_polling_page_exception), POLL_AT_RETURN); diff --git a/src/hotspot/share/runtime/thread.cpp b/src/hotspot/share/runtime/thread.cpp index 9ffff8a0405..635625b1655 100644 --- a/src/hotspot/share/runtime/thread.cpp +++ b/src/hotspot/share/runtime/thread.cpp @@ -3724,7 +3724,7 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) { } // initialize compiler(s) -#if defined(COMPILER1) || defined(COMPILER2) || INCLUDE_JVMCI +#if defined(COMPILER1) || COMPILER2_OR_JVMCI CompileBroker::compilation_init(CHECK_JNI_ERR); #endif From d7b2fafd3dc47c8c4c5a545176fe30ebfd24bc06 Mon Sep 17 00:00:00 2001 From: Thomas Schatzl Date: Tue, 7 Nov 2017 13:51:18 +0100 Subject: [PATCH 20/61] 8190703: TestSystemGCWith* infrequently times out on SPARC Give the test a timeout after which it ends on its own. Reviewed-by: shade, sjohanss --- .../gc/stress/systemgc/TestSystemGC.java | 22 +++++++++++++------ .../stress/systemgc/TestSystemGCWithCMS.java | 5 +++-- .../stress/systemgc/TestSystemGCWithG1.java | 5 +++-- .../systemgc/TestSystemGCWithParallel.java | 5 +++-- .../systemgc/TestSystemGCWithSerial.java | 5 +++-- 5 files changed, 27 insertions(+), 15 deletions(-) diff --git a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGC.java b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGC.java index 818ddd16342..cf30bd72424 100644 --- a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGC.java +++ b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGC.java @@ -103,6 +103,8 @@ class SystemGCTask extends Exitable implements Runnable { } public class TestSystemGC { + private static long endTime; + private static final int numGroups = 7; private static final int numGCsPerGroup = 4; @@ -133,8 +135,11 @@ public class TestSystemGC { for (int i = 0; i < numGroups; i++) { for (int j = 0; j < numGCsPerGroup; j++) { - System.gc(); - ThreadUtils.sleep(getDelayMS(i)); + System.gc(); + if (System.currentTimeMillis() >= endTime) { + return; + } + ThreadUtils.sleep(getDelayMS(i)); } } } @@ -159,7 +164,7 @@ public class TestSystemGC { } private static void runAllPhases() { - for (int i = 0; i < 4; i++) { + for (int i = 0; i < 4 && System.currentTimeMillis() < endTime; i++) { SystemGCTask gcTask = (i % 2 == 1) ? createSystemGCTask(numGroups / 3) : null; ShortLivedAllocationTask shortTask = @@ -181,12 +186,15 @@ public class TestSystemGC { } } - public static void main(String[] args) { + public static void main(String[] args) throws Exception { + if (args.length == 0) { + throw new IllegalArgumentException("Must specify timeout in seconds as first argument"); + } + int timeout = Integer.parseInt(args[0]) * 1000; + System.out.println("Running with timeout of " + timeout + "ms"); + endTime = System.currentTimeMillis() + timeout; // First allocate the long lived objects and then run all phases. populateLongLived(); runAllPhases(); - if (args.length > 0 && args[0].equals("long")) { - runAllPhases(); - } } } diff --git a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithCMS.java b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithCMS.java index 51506aed4f4..a722f3ad632 100644 --- a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithCMS.java +++ b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithCMS.java @@ -24,14 +24,15 @@ /* * @test TestSystemGCWithCMS + * @bug 8190703 * @key gc * @key stress * @requires vm.gc.ConcMarkSweep * @summary Stress the CMS GC full GC by allocating objects of different lifetimes concurrently with System.gc(). - * @run main/othervm/timeout=300 -Xlog:gc*=info -Xmx512m -XX:+UseConcMarkSweepGC TestSystemGCWithCMS + * @run main/othervm/timeout=300 -Xlog:gc*=info -Xmx512m -XX:+UseConcMarkSweepGC TestSystemGCWithCMS 270 */ public class TestSystemGCWithCMS { - public static void main(String[] args) { + public static void main(String[] args) throws Exception { TestSystemGC.main(args); } } diff --git a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithG1.java b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithG1.java index 63f89d1cf1e..1dec06daf7d 100644 --- a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithG1.java +++ b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithG1.java @@ -24,14 +24,15 @@ /* * @test TestSystemGCWithG1 + * @bug 8190703 * @key gc * @key stress * @requires vm.gc.G1 * @summary Stress the G1 GC full GC by allocating objects of different lifetimes concurrently with System.gc(). - * @run main/othervm/timeout=300 -Xlog:gc*=info -Xmx512m -XX:+UseG1GC TestSystemGCWithG1 + * @run main/othervm/timeout=300 -Xlog:gc*=info -Xmx512m -XX:+UseG1GC TestSystemGCWithG1 270 */ public class TestSystemGCWithG1 { - public static void main(String[] args) { + public static void main(String[] args) throws Exception { TestSystemGC.main(args); } } diff --git a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithParallel.java b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithParallel.java index 73596095c44..66db1b2c07b 100644 --- a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithParallel.java +++ b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithParallel.java @@ -24,14 +24,15 @@ /* * @test TestSystemGCWithParallel + * @bug 8190703 * @key gc * @key stress * @requires vm.gc.Parallel * @summary Stress the Parallel GC full GC by allocating objects of different lifetimes concurrently with System.gc(). - * @run main/othervm/timeout=300 -Xlog:gc*=info -Xmx512m -XX:+UseParallelGC TestSystemGCWithParallel + * @run main/othervm/timeout=300 -Xlog:gc=info -Xmx512m -XX:+UseParallelGC TestSystemGCWithParallel 270 */ public class TestSystemGCWithParallel { - public static void main(String[] args) { + public static void main(String[] args) throws Exception { TestSystemGC.main(args); } } diff --git a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithSerial.java b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithSerial.java index 8d7f8304230..fd80607adee 100644 --- a/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithSerial.java +++ b/test/hotspot/jtreg/gc/stress/systemgc/TestSystemGCWithSerial.java @@ -24,14 +24,15 @@ /* * @test TestSystemGCWithSerial + * @bug 8190703 * @key gc * @key stress * @requires vm.gc.Serial * @summary Stress the Serial GC full GC by allocating objects of different lifetimes concurrently with System.gc(). - * @run main/othervm/timeout=300 -Xlog:gc*=info -Xmx512m -XX:+UseSerialGC TestSystemGCWithSerial + * @run main/othervm/timeout=300 -Xlog:gc*=info -Xmx512m -XX:+UseSerialGC TestSystemGCWithSerial 270 */ public class TestSystemGCWithSerial { - public static void main(String[] args) { + public static void main(String[] args) throws Exception { TestSystemGC.main(args); } } From 75d5090c81986910bbcfb25cb8f5f2721909ad43 Mon Sep 17 00:00:00 2001 From: Bob Vandette Date: Tue, 7 Nov 2017 10:30:53 -0500 Subject: [PATCH 21/61] 8190283: Default heap sizing options select a MaxHeapSize larger than available physical memory in some cases Reviewed-by: tschatzl, sjohanss --- src/hotspot/share/runtime/arguments.cpp | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/hotspot/share/runtime/arguments.cpp b/src/hotspot/share/runtime/arguments.cpp index c1590ea2ac7..0e630e2f823 100644 --- a/src/hotspot/share/runtime/arguments.cpp +++ b/src/hotspot/share/runtime/arguments.cpp @@ -2089,9 +2089,10 @@ void Arguments::set_heap_size() { // respecting the maximum and minimum sizes of the heap. if (FLAG_IS_DEFAULT(MaxHeapSize)) { julong reasonable_max = (julong)((phys_mem * MaxRAMPercentage) / 100); - if (phys_mem <= (julong)((MaxHeapSize * MinRAMPercentage) / 100)) { + const julong reasonable_min = (julong)((phys_mem * MinRAMPercentage) / 100); + if (reasonable_min < MaxHeapSize) { // Small physical memory, so use a minimum fraction of it for the heap - reasonable_max = (julong)((phys_mem * MinRAMPercentage) / 100); + reasonable_max = reasonable_min; } else { // Not-small physical memory, so require a heap at least // as large as MaxHeapSize From 92ae7c8cfde1e4902ab88b289d1c00eafc93533b Mon Sep 17 00:00:00 2001 From: Zhengyu Gu Date: Tue, 7 Nov 2017 09:37:45 -0500 Subject: [PATCH 22/61] 8189688: NMT: Report per-class load metadata information Report per-class loader metadata info via NMT jcmd metadata sub-command Reviewed-by: stuefe, coleenp --- src/hotspot/share/memory/metaspace.cpp | 250 ++++++++++++++++++-- src/hotspot/share/memory/metaspace.hpp | 4 + src/hotspot/share/runtime/vm_operations.cpp | 6 +- src/hotspot/share/runtime/vm_operations.hpp | 12 + src/hotspot/share/services/nmtDCmd.cpp | 18 +- src/hotspot/share/services/nmtDCmd.hpp | 3 +- 6 files changed, 273 insertions(+), 20 deletions(-) diff --git a/src/hotspot/share/memory/metaspace.cpp b/src/hotspot/share/memory/metaspace.cpp index 9cbe3c07279..76d2baa738f 100644 --- a/src/hotspot/share/memory/metaspace.cpp +++ b/src/hotspot/share/memory/metaspace.cpp @@ -108,6 +108,18 @@ static ChunkIndex next_chunk_index(ChunkIndex i) { return (ChunkIndex) (i+1); } +static const char* scale_unit(size_t scale) { + switch(scale) { + case 1: return "BYTES"; + case K: return "KB"; + case M: return "MB"; + case G: return "GB"; + default: + ShouldNotReachHere(); + return NULL; + } +} + volatile intptr_t MetaspaceGC::_capacity_until_GC = 0; uint MetaspaceGC::_shrink_factor = 0; bool MetaspaceGC::_should_concurrent_collect = false; @@ -176,7 +188,7 @@ class ChunkManager : public CHeapObj { void locked_get_statistics(ChunkManagerStatistics* stat) const; void get_statistics(ChunkManagerStatistics* stat) const; - static void print_statistics(const ChunkManagerStatistics* stat, outputStream* out); + static void print_statistics(const ChunkManagerStatistics* stat, outputStream* out, size_t scale); public: @@ -283,7 +295,7 @@ class ChunkManager : public CHeapObj { // Prints composition for both non-class and (if available) // class chunk manager. - static void print_all_chunkmanagers(outputStream* out); + static void print_all_chunkmanagers(outputStream* out, size_t scale = 1); }; class SmallBlocks : public CHeapObj { @@ -1724,7 +1736,6 @@ bool Metadebug::test_metadata_failure() { #endif // ChunkManager methods - size_t ChunkManager::free_chunks_total_words() { return _free_chunks_total; } @@ -2057,22 +2068,45 @@ void ChunkManager::get_statistics(ChunkManagerStatistics* stat) const { locked_get_statistics(stat); } -void ChunkManager::print_statistics(const ChunkManagerStatistics* stat, outputStream* out) { +void ChunkManager::print_statistics(const ChunkManagerStatistics* stat, outputStream* out, size_t scale) { size_t total = 0; + assert(scale == 1 || scale == K || scale == M || scale == G, "Invalid scale"); + + const char* unit = scale_unit(scale); for (ChunkIndex i = ZeroIndex; i < NumberOfFreeLists; i = next_chunk_index(i)) { - out->print_cr(" " SIZE_FORMAT " %s (" SIZE_FORMAT " bytes) chunks, total " SIZE_FORMAT " bytes", - stat->num_by_type[i], chunk_size_name(i), - stat->single_size_by_type[i], - stat->total_size_by_type[i]); + out->print(" " SIZE_FORMAT " %s (" SIZE_FORMAT " bytes) chunks, total ", + stat->num_by_type[i], chunk_size_name(i), + stat->single_size_by_type[i]); + if (scale == 1) { + out->print_cr(SIZE_FORMAT " bytes", stat->total_size_by_type[i]); + } else { + out->print_cr("%.2f%s", (float)stat->total_size_by_type[i] / scale, unit); + } + total += stat->total_size_by_type[i]; } - out->print_cr(" " SIZE_FORMAT " humongous chunks, total " SIZE_FORMAT " bytes", - stat->num_humongous_chunks, stat->total_size_humongous_chunks); + + total += stat->total_size_humongous_chunks; - out->print_cr(" total size: " SIZE_FORMAT ".", total); + + if (scale == 1) { + out->print_cr(" " SIZE_FORMAT " humongous chunks, total " SIZE_FORMAT " bytes", + stat->num_humongous_chunks, stat->total_size_humongous_chunks); + + out->print_cr(" total size: " SIZE_FORMAT " bytes.", total); + } else { + out->print_cr(" " SIZE_FORMAT " humongous chunks, total %.2f%s", + stat->num_humongous_chunks, + (float)stat->total_size_humongous_chunks / scale, unit); + + out->print_cr(" total size: %.2f%s.", (float)total / scale, unit); + } + } -void ChunkManager::print_all_chunkmanagers(outputStream* out) { +void ChunkManager::print_all_chunkmanagers(outputStream* out, size_t scale) { + assert(scale == 1 || scale == K || scale == M || scale == G, "Invalid scale"); + // Note: keep lock protection only to retrieving statistics; keep printing // out of lock protection ChunkManagerStatistics stat; @@ -2080,7 +2114,7 @@ void ChunkManager::print_all_chunkmanagers(outputStream* out) { const ChunkManager* const non_class_cm = Metaspace::chunk_manager_metadata(); if (non_class_cm != NULL) { non_class_cm->get_statistics(&stat); - ChunkManager::print_statistics(&stat, out); + ChunkManager::print_statistics(&stat, out, scale); } else { out->print_cr("unavailable."); } @@ -2088,7 +2122,7 @@ void ChunkManager::print_all_chunkmanagers(outputStream* out) { const ChunkManager* const class_cm = Metaspace::chunk_manager_class(); if (class_cm != NULL) { class_cm->get_statistics(&stat); - ChunkManager::print_statistics(&stat, out); + ChunkManager::print_statistics(&stat, out, scale); } else { out->print_cr("unavailable."); } @@ -2981,6 +3015,194 @@ void MetaspaceAux::print_waste(outputStream* out) { } } +class MetadataStats VALUE_OBJ_CLASS_SPEC { +private: + size_t _capacity; + size_t _used; + size_t _free; + size_t _waste; + +public: + MetadataStats() : _capacity(0), _used(0), _free(0), _waste(0) { } + MetadataStats(size_t capacity, size_t used, size_t free, size_t waste) + : _capacity(capacity), _used(used), _free(free), _waste(waste) { } + + void add(const MetadataStats& stats) { + _capacity += stats.capacity(); + _used += stats.used(); + _free += stats.free(); + _waste += stats.waste(); + } + + size_t capacity() const { return _capacity; } + size_t used() const { return _used; } + size_t free() const { return _free; } + size_t waste() const { return _waste; } + + void print_on(outputStream* out, size_t scale) const; +}; + + +void MetadataStats::print_on(outputStream* out, size_t scale) const { + const char* unit = scale_unit(scale); + out->print_cr("capacity=%10.2f%s used=%10.2f%s free=%10.2f%s waste=%10.2f%s", + (float)capacity() / scale, unit, + (float)used() / scale, unit, + (float)free() / scale, unit, + (float)waste() / scale, unit); +} + +class PrintCLDMetaspaceInfoClosure : public CLDClosure { +private: + outputStream* _out; + size_t _scale; + + size_t _total_count; + MetadataStats _total_metadata; + MetadataStats _total_class; + + size_t _total_anon_count; + MetadataStats _total_anon_metadata; + MetadataStats _total_anon_class; + +public: + PrintCLDMetaspaceInfoClosure(outputStream* out, size_t scale = K) + : _out(out), _scale(scale), _total_count(0), _total_anon_count(0) { } + + ~PrintCLDMetaspaceInfoClosure() { + print_summary(); + } + + void do_cld(ClassLoaderData* cld) { + assert(SafepointSynchronize::is_at_safepoint(), "Must be at a safepoint"); + + if (cld->is_unloading()) return; + Metaspace* msp = cld->metaspace_or_null(); + if (msp == NULL) { + return; + } + + bool anonymous = false; + if (cld->is_anonymous()) { + _out->print_cr("ClassLoader: for anonymous class"); + anonymous = true; + } else { + ResourceMark rm; + _out->print_cr("ClassLoader: %s", cld->loader_name()); + } + + print_metaspace(msp, anonymous); + _out->cr(); + } + +private: + void print_metaspace(Metaspace* msp, bool anonymous); + void print_summary() const; +}; + +void PrintCLDMetaspaceInfoClosure::print_metaspace(Metaspace* msp, bool anonymous){ + assert(msp != NULL, "Sanity"); + SpaceManager* vsm = msp->vsm(); + const char* unit = scale_unit(_scale); + + size_t capacity = vsm->sum_capacity_in_chunks_in_use() * BytesPerWord; + size_t used = vsm->sum_used_in_chunks_in_use() * BytesPerWord; + size_t free = vsm->sum_free_in_chunks_in_use() * BytesPerWord; + size_t waste = vsm->sum_waste_in_chunks_in_use() * BytesPerWord; + + _total_count ++; + MetadataStats metadata_stats(capacity, used, free, waste); + _total_metadata.add(metadata_stats); + + if (anonymous) { + _total_anon_count ++; + _total_anon_metadata.add(metadata_stats); + } + + _out->print(" Metadata "); + metadata_stats.print_on(_out, _scale); + + if (Metaspace::using_class_space()) { + vsm = msp->class_vsm(); + + capacity = vsm->sum_capacity_in_chunks_in_use() * BytesPerWord; + used = vsm->sum_used_in_chunks_in_use() * BytesPerWord; + free = vsm->sum_free_in_chunks_in_use() * BytesPerWord; + waste = vsm->sum_waste_in_chunks_in_use() * BytesPerWord; + + MetadataStats class_stats(capacity, used, free, waste); + _total_class.add(class_stats); + + if (anonymous) { + _total_anon_class.add(class_stats); + } + + _out->print(" Class data "); + class_stats.print_on(_out, _scale); + } +} + +void PrintCLDMetaspaceInfoClosure::print_summary() const { + const char* unit = scale_unit(_scale); + _out->cr(); + _out->print_cr("Summary:"); + + MetadataStats total; + total.add(_total_metadata); + total.add(_total_class); + + _out->print(" Total class loaders=" SIZE_FORMAT_W(6) " ", _total_count); + total.print_on(_out, _scale); + + _out->print(" Metadata "); + _total_metadata.print_on(_out, _scale); + + if (Metaspace::using_class_space()) { + _out->print(" Class data "); + _total_class.print_on(_out, _scale); + } + _out->cr(); + + MetadataStats total_anon; + total_anon.add(_total_anon_metadata); + total_anon.add(_total_anon_class); + + _out->print("For anonymous classes=" SIZE_FORMAT_W(6) " ", _total_anon_count); + total_anon.print_on(_out, _scale); + + _out->print(" Metadata "); + _total_anon_metadata.print_on(_out, _scale); + + if (Metaspace::using_class_space()) { + _out->print(" Class data "); + _total_anon_class.print_on(_out, _scale); + } +} + +void MetaspaceAux::print_metadata_for_nmt(outputStream* out, size_t scale) { + const char* unit = scale_unit(scale); + out->print_cr("Metaspaces:"); + out->print_cr(" Metadata space: reserved=" SIZE_FORMAT_W(10) "%s committed=" SIZE_FORMAT_W(10) "%s", + reserved_bytes(Metaspace::NonClassType) / scale, unit, + committed_bytes(Metaspace::NonClassType) / scale, unit); + if (Metaspace::using_class_space()) { + out->print_cr(" Class space: reserved=" SIZE_FORMAT_W(10) "%s committed=" SIZE_FORMAT_W(10) "%s", + reserved_bytes(Metaspace::ClassType) / scale, unit, + committed_bytes(Metaspace::ClassType) / scale, unit); + } + + out->cr(); + ChunkManager::print_all_chunkmanagers(out, scale); + + out->cr(); + out->print_cr("Per-classloader metadata:"); + out->cr(); + + PrintCLDMetaspaceInfoClosure cl(out, scale); + ClassLoaderDataGraph::cld_do(&cl); +} + + // Dump global metaspace things from the end of ClassLoaderDataGraph void MetaspaceAux::dump(outputStream* out) { out->print_cr("All Metaspace:"); diff --git a/src/hotspot/share/memory/metaspace.hpp b/src/hotspot/share/memory/metaspace.hpp index bda5190c314..7f18c616801 100644 --- a/src/hotspot/share/memory/metaspace.hpp +++ b/src/hotspot/share/memory/metaspace.hpp @@ -63,6 +63,7 @@ class MetaspaceTracer; class MetaWord; class Mutex; class outputStream; +class PrintCLDMetaspaceInfoClosure; class SpaceManager; class VirtualSpaceList; @@ -87,6 +88,7 @@ class Metaspace : public CHeapObj { friend class MetaspaceAux; friend class MetaspaceShared; friend class CollectorPolicy; + friend class PrintCLDMetaspaceInfoClosure; public: enum MetadataType { @@ -347,6 +349,8 @@ class MetaspaceAux : AllStatic { return min_chunk_size_words() * BytesPerWord; } + static void print_metadata_for_nmt(outputStream* out, size_t scale = K); + static bool has_chunk_free_list(Metaspace::MetadataType mdtype); static MetaspaceChunkFreeListSummary chunk_free_list_summary(Metaspace::MetadataType mdtype); diff --git a/src/hotspot/share/runtime/vm_operations.cpp b/src/hotspot/share/runtime/vm_operations.cpp index 95c9943aefb..e9ad8af4314 100644 --- a/src/hotspot/share/runtime/vm_operations.cpp +++ b/src/hotspot/share/runtime/vm_operations.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2017, 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 @@ -230,6 +230,10 @@ void VM_PrintJNI::doit() { JNIHandles::print_on(_out); } +void VM_PrintMetadata::doit() { + MetaspaceAux::print_metadata_for_nmt(_out, _scale); +} + VM_FindDeadlocks::~VM_FindDeadlocks() { if (_deadlocks != NULL) { DeadlockCycle* cycle = _deadlocks; diff --git a/src/hotspot/share/runtime/vm_operations.hpp b/src/hotspot/share/runtime/vm_operations.hpp index 52e4606cf35..b325d484a02 100644 --- a/src/hotspot/share/runtime/vm_operations.hpp +++ b/src/hotspot/share/runtime/vm_operations.hpp @@ -111,6 +111,7 @@ template(ThreadsSuspendJVMTI) \ template(ICBufferFull) \ template(ScavengeMonitors) \ + template(PrintMetadata) \ class VM_Operation: public CHeapObj { public: @@ -374,6 +375,17 @@ class VM_PrintJNI: public VM_Operation { void doit(); }; +class VM_PrintMetadata : public VM_Operation { + private: + outputStream* _out; + size_t _scale; + public: + VM_PrintMetadata(outputStream* out, size_t scale) : _out(out), _scale(scale) {}; + + VMOp_Type type() const { return VMOp_PrintMetadata; } + void doit(); +}; + class DeadlockCycle; class VM_FindDeadlocks: public VM_Operation { private: diff --git a/src/hotspot/share/services/nmtDCmd.cpp b/src/hotspot/share/services/nmtDCmd.cpp index 6620bde60b0..fe329b78f01 100644 --- a/src/hotspot/share/services/nmtDCmd.cpp +++ b/src/hotspot/share/services/nmtDCmd.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2017, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -24,6 +24,8 @@ #include "precompiled.hpp" #include "memory/resourceArea.hpp" #include "runtime/mutexLocker.hpp" +#include "runtime/vmThread.hpp" +#include "runtime/vm_operations.hpp" #include "services/nmtDCmd.hpp" #include "services/memReporter.hpp" #include "services/memTracker.hpp" @@ -38,6 +40,8 @@ NMTDCmd::NMTDCmd(outputStream* output, _detail("detail", "request runtime to report memory allocation >= " "1K by each callsite.", "BOOLEAN", false, "false"), + _metadata("metadata", "request runtime to report metadata information", + "BOOLEAN", false, "false"), _baseline("baseline", "request runtime to baseline current memory usage, " \ "so it can be compared against in later time.", "BOOLEAN", false, "false"), @@ -57,6 +61,7 @@ NMTDCmd::NMTDCmd(outputStream* output, "STRING", false, "KB") { _dcmdparser.add_dcmd_option(&_summary); _dcmdparser.add_dcmd_option(&_detail); + _dcmdparser.add_dcmd_option(&_metadata); _dcmdparser.add_dcmd_option(&_baseline); _dcmdparser.add_dcmd_option(&_summary_diff); _dcmdparser.add_dcmd_option(&_detail_diff); @@ -92,6 +97,7 @@ void NMTDCmd::execute(DCmdSource source, TRAPS) { int nopt = 0; if (_summary.is_set() && _summary.value()) { ++nopt; } if (_detail.is_set() && _detail.value()) { ++nopt; } + if (_metadata.is_set() && _metadata.value()) { ++nopt; } if (_baseline.is_set() && _baseline.value()) { ++nopt; } if (_summary_diff.is_set() && _summary_diff.value()) { ++nopt; } if (_detail_diff.is_set() && _detail_diff.value()) { ++nopt; } @@ -100,7 +106,7 @@ void NMTDCmd::execute(DCmdSource source, TRAPS) { if (nopt > 1) { output()->print_cr("At most one of the following option can be specified: " \ - "summary, detail, baseline, summary.diff, detail.diff, shutdown"); + "summary, detail, metadata, baseline, summary.diff, detail.diff, shutdown"); return; } else if (nopt == 0) { if (_summary.is_set()) { @@ -118,9 +124,13 @@ void NMTDCmd::execute(DCmdSource source, TRAPS) { report(true, scale_unit); } else if (_detail.value()) { if (!check_detail_tracking_level(output())) { - return; - } + return; + } report(false, scale_unit); + } else if (_metadata.value()) { + size_t scale = get_scale(_scale.value()); + VM_PrintMetadata op(output(), scale); + VMThread::execute(&op); } else if (_baseline.value()) { MemBaseline& baseline = MemTracker::get_baseline(); if (!baseline.baseline(MemTracker::tracking_level() != NMT_detail)) { diff --git a/src/hotspot/share/services/nmtDCmd.hpp b/src/hotspot/share/services/nmtDCmd.hpp index df1ab367fd1..c1ede016b8a 100644 --- a/src/hotspot/share/services/nmtDCmd.hpp +++ b/src/hotspot/share/services/nmtDCmd.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2017, 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 @@ -39,6 +39,7 @@ class NMTDCmd: public DCmdWithParser { protected: DCmdArgument _summary; DCmdArgument _detail; + DCmdArgument _metadata; DCmdArgument _baseline; DCmdArgument _summary_diff; DCmdArgument _detail_diff; From 5865d3062feae76bbf61bc6e42044cd675367e2d Mon Sep 17 00:00:00 2001 From: Chris Plummer Date: Tue, 7 Nov 2017 14:13:18 -0800 Subject: [PATCH 23/61] 8059334: nsk/jvmti/scenarios/hotswap/HS201/hs201t001 fails with exit code 0 after timeout Check if we are in "interp only" mode before returning an nm for an OSR request Reviewed-by: sspitsyn, dcubed, kvn --- src/hotspot/share/interpreter/interpreterRuntime.cpp | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/hotspot/share/interpreter/interpreterRuntime.cpp b/src/hotspot/share/interpreter/interpreterRuntime.cpp index 4288922afa3..d322cfd33af 100644 --- a/src/hotspot/share/interpreter/interpreterRuntime.cpp +++ b/src/hotspot/share/interpreter/interpreterRuntime.cpp @@ -921,6 +921,14 @@ nmethod* InterpreterRuntime::frequency_counter_overflow(JavaThread* thread, addr int bci = method->bci_from(last_frame.bcp()); nm = method->lookup_osr_nmethod_for(bci, CompLevel_none, false); } + if (nm != NULL && thread->is_interp_only_mode()) { + // Normally we never get an nm if is_interp_only_mode() is true, because + // policy()->event has a check for this and won't compile the method when + // true. However, it's possible for is_interp_only_mode() to become true + // during the compilation. We don't want to return the nm in that case + // because we want to continue to execute interpreted. + nm = NULL; + } #ifndef PRODUCT if (TraceOnStackReplacement) { if (nm != NULL) { From c2d94a0073ab26e31f109450a5a295ca73884d67 Mon Sep 17 00:00:00 2001 From: Jiangli Zhou Date: Tue, 7 Nov 2017 18:48:17 -0500 Subject: [PATCH 24/61] 8189840: CheckCachedResolvedReferencesApp has no cached resolved references Add WhiteBox.areOpenArchiveHeapObjectsMapped() for CheckCachedResolvedReferencesApp to check and skip test when open archive heap objects are not mapped. Reviewed-by: iklam, ccheung --- src/hotspot/share/prims/whitebox.cpp | 5 +++++ test/lib/sun/hotspot/WhiteBox.java | 1 + 2 files changed, 6 insertions(+) diff --git a/src/hotspot/share/prims/whitebox.cpp b/src/hotspot/share/prims/whitebox.cpp index f7cb20b9ab8..a6b04e79b37 100644 --- a/src/hotspot/share/prims/whitebox.cpp +++ b/src/hotspot/share/prims/whitebox.cpp @@ -1715,6 +1715,10 @@ WB_ENTRY(jobject, WB_GetResolvedReferences(JNIEnv* env, jobject wb, jclass clazz } WB_END +WB_ENTRY(jboolean, WB_AreOpenArchiveHeapObjectsMapped(JNIEnv* env)) + return MetaspaceShared::open_archive_heap_region_mapped(); +WB_END + WB_ENTRY(jboolean, WB_IsCDSIncludedInVmBuild(JNIEnv* env)) #if INCLUDE_CDS return true; @@ -2031,6 +2035,7 @@ static JNINativeMethod methods[] = { {CC"isSharedClass", CC"(Ljava/lang/Class;)Z", (void*)&WB_IsSharedClass }, {CC"areSharedStringsIgnored", CC"()Z", (void*)&WB_AreSharedStringsIgnored }, {CC"getResolvedReferences", CC"(Ljava/lang/Class;)Ljava/lang/Object;", (void*)&WB_GetResolvedReferences}, + {CC"areOpenArchiveHeapObjectsMapped", CC"()Z", (void*)&WB_AreOpenArchiveHeapObjectsMapped}, {CC"isCDSIncludedInVmBuild", CC"()Z", (void*)&WB_IsCDSIncludedInVmBuild }, {CC"clearInlineCaches0", CC"(Z)V", (void*)&WB_ClearInlineCaches }, {CC"addCompilerDirective", CC"(Ljava/lang/String;)I", diff --git a/test/lib/sun/hotspot/WhiteBox.java b/test/lib/sun/hotspot/WhiteBox.java index 49d04f5e68e..64efdea559f 100644 --- a/test/lib/sun/hotspot/WhiteBox.java +++ b/test/lib/sun/hotspot/WhiteBox.java @@ -525,6 +525,7 @@ public class WhiteBox { public native boolean areSharedStringsIgnored(); public native boolean isCDSIncludedInVmBuild(); public native Object getResolvedReferences(Class c); + public native boolean areOpenArchiveHeapObjectsMapped(); // Compiler Directive public native int addCompilerDirective(String compDirect); From 72a1ec38f3f9b4a46daad91c59944e8375fcca3e Mon Sep 17 00:00:00 2001 From: Jesper Wilhelmsson Date: Tue, 7 Nov 2017 23:18:16 +0100 Subject: [PATCH 25/61] 8190865: Require jtreg 4.2 b10 Reviewed-by: dcubed, erikj --- make/conf/jib-profiles.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/make/conf/jib-profiles.js b/make/conf/jib-profiles.js index e5dd458f806..cdba1241893 100644 --- a/make/conf/jib-profiles.js +++ b/make/conf/jib-profiles.js @@ -1060,7 +1060,7 @@ var getJibProfilesDependencies = function (input, common) { jtreg: { server: "javare", revision: "4.2", - build_number: "b09", + build_number: "b10", checksum_file: "MD5_VALUES", file: "jtreg_bin-4.2.zip", environment_name: "JT_HOME", From 6eeb777697dc524f38047c2af8c53cb4a2cfec80 Mon Sep 17 00:00:00 2001 From: Ujwal Vangapally Date: Tue, 7 Nov 2017 22:05:44 -0800 Subject: [PATCH 26/61] 8044122: MBean access to the PID Added new API getPid for getting process ID Reviewed-by: mchung, rriggs --- .../java/lang/management/RuntimeMXBean.java | 19 +++++++ .../RuntimeMXBean/ProcessIdTest.java | 49 +++++++++++++++++++ 2 files changed, 68 insertions(+) create mode 100644 test/jdk/java/lang/management/RuntimeMXBean/ProcessIdTest.java diff --git a/src/java.management/share/classes/java/lang/management/RuntimeMXBean.java b/src/java.management/share/classes/java/lang/management/RuntimeMXBean.java index 495b41f43db..c59a31457fb 100644 --- a/src/java.management/share/classes/java/lang/management/RuntimeMXBean.java +++ b/src/java.management/share/classes/java/lang/management/RuntimeMXBean.java @@ -25,6 +25,9 @@ package java.lang.management; +import java.security.AccessController; +import java.security.PrivilegedAction; + /** * The management interface for the runtime system of * the Java virtual machine. @@ -60,6 +63,22 @@ package java.lang.management; * @since 1.5 */ public interface RuntimeMXBean extends PlatformManagedObject { + /** + * Returns the {@linkplain ProcessHandle#pid process ID} representing + * the running Java virtual machine. + * + * @implSpec The default implementation returns {@link ProcessHandle#pid process ID} + * of the {@linkplain ProcessHandle#current current process}. + * + * @return the process ID representing the running Java virtual machine. + * + * @since 10 + */ + public default long getPid() { + return AccessController.doPrivileged((PrivilegedAction) + () -> ProcessHandle.current().pid()); + } + /** * Returns the name representing the running Java virtual machine. * The returned name string can be any arbitrary string and diff --git a/test/jdk/java/lang/management/RuntimeMXBean/ProcessIdTest.java b/test/jdk/java/lang/management/RuntimeMXBean/ProcessIdTest.java new file mode 100644 index 00000000000..4f2f586862e --- /dev/null +++ b/test/jdk/java/lang/management/RuntimeMXBean/ProcessIdTest.java @@ -0,0 +1,49 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 8044122 + * @summary check the correctness of process ID returned by RuntimeMXBean.getPid() + * @run main ProcessIdTest + */ + +import java.lang.management.ManagementFactory; +import java.lang.management.RuntimeMXBean; +import java.lang.ProcessHandle; + +public class ProcessIdTest { + public static void main(String args[]) { + RuntimeMXBean mbean = ManagementFactory.getRuntimeMXBean(); + long mbeanPid = mbean.getPid(); + long pid = ProcessHandle.current().pid(); + long pid1 = Long.parseLong(mbean.getName().split("@")[0]); + if(mbeanPid != pid || mbeanPid != pid1) { + throw new RuntimeException("Incorrect process ID returned"); + } + + System.out.println("Test Passed"); + } + +} + From 0f101e2c7fe75876a106a7dd0011bebe1d9c3f32 Mon Sep 17 00:00:00 2001 From: Thomas Stuefe Date: Tue, 24 Oct 2017 14:34:14 +0200 Subject: [PATCH 27/61] 8189864: Provide an ascii map to visualize metaspace fragmentation Reviewed-by: goetz, coleenp --- src/hotspot/share/memory/metachunk.cpp | 2 +- src/hotspot/share/memory/metachunk.hpp | 4 +- src/hotspot/share/memory/metaspace.cpp | 137 ++++++++++++++++++++++++- src/hotspot/share/memory/metaspace.hpp | 4 + 4 files changed, 138 insertions(+), 9 deletions(-) diff --git a/src/hotspot/share/memory/metachunk.cpp b/src/hotspot/share/memory/metachunk.cpp index 45528faa180..c9740e94537 100644 --- a/src/hotspot/share/memory/metachunk.cpp +++ b/src/hotspot/share/memory/metachunk.cpp @@ -55,8 +55,8 @@ Metachunk::Metachunk(size_t word_size, _container(container) { _top = initial_top(); -#ifdef ASSERT set_is_tagged_free(false); +#ifdef ASSERT mangle(uninitMetaWordVal); #endif } diff --git a/src/hotspot/share/memory/metachunk.hpp b/src/hotspot/share/memory/metachunk.hpp index 5b92827586f..d8b3d11900b 100644 --- a/src/hotspot/share/memory/metachunk.hpp +++ b/src/hotspot/share/memory/metachunk.hpp @@ -102,7 +102,7 @@ class Metachunk : public Metabase { // Current allocation top. MetaWord* _top; - DEBUG_ONLY(bool _is_tagged_free;) + bool _is_tagged_free; MetaWord* initial_top() const { return (MetaWord*)this + overhead(); } MetaWord* top() const { return _top; } @@ -138,10 +138,8 @@ class Metachunk : public Metabase { size_t used_word_size() const; size_t free_word_size() const; -#ifdef ASSERT bool is_tagged_free() { return _is_tagged_free; } void set_is_tagged_free(bool v) { _is_tagged_free = v; } -#endif bool contains(const void* ptr) { return bottom() <= ptr && ptr < _top; } diff --git a/src/hotspot/share/memory/metaspace.cpp b/src/hotspot/share/memory/metaspace.cpp index 76d2baa738f..5671e12c2eb 100644 --- a/src/hotspot/share/memory/metaspace.cpp +++ b/src/hotspot/share/memory/metaspace.cpp @@ -492,6 +492,7 @@ class VirtualSpaceNode : public CHeapObj { #endif void print_on(outputStream* st) const; + void print_map(outputStream* st, bool is_class) const; }; #define assert_is_aligned(value, alignment) \ @@ -543,6 +544,94 @@ void VirtualSpaceNode::purge(ChunkManager* chunk_manager) { } } +void VirtualSpaceNode::print_map(outputStream* st, bool is_class) const { + + // Format: + // + // . .. . . .. + // SSxSSMMMMMMMMMMMMMMMMsssXX + // 112114444444444444444 + // . .. . . .. + // SSxSSMMMMMMMMMMMMMMMMsssXX + // 112114444444444444444 + + if (bottom() == top()) { + return; + } + + // First line: dividers for every med-chunk-sized interval + // Second line: a dot for the start of a chunk + // Third line: a letter per chunk type (x,s,m,h), uppercase if in use. + + const size_t spec_chunk_size = is_class ? ClassSpecializedChunk : SpecializedChunk; + const size_t small_chunk_size = is_class ? ClassSmallChunk : SmallChunk; + const size_t med_chunk_size = is_class ? ClassMediumChunk : MediumChunk; + + int line_len = 100; + const size_t section_len = align_up(spec_chunk_size * line_len, med_chunk_size); + line_len = (int)(section_len / spec_chunk_size); + + char* line1 = (char*)os::malloc(line_len, mtInternal); + char* line2 = (char*)os::malloc(line_len, mtInternal); + char* line3 = (char*)os::malloc(line_len, mtInternal); + int pos = 0; + const MetaWord* p = bottom(); + const Metachunk* chunk = (const Metachunk*)p; + const MetaWord* chunk_end = p + chunk->word_size(); + while (p < top()) { + if (pos == line_len) { + pos = 0; + st->fill_to(22); + st->print_raw(line1, line_len); + st->cr(); + st->fill_to(22); + st->print_raw(line2, line_len); + st->cr(); + } + if (pos == 0) { + st->print(PTR_FORMAT ":", p2i(p)); + } + if (p == chunk_end) { + chunk = (Metachunk*)p; + chunk_end = p + chunk->word_size(); + } + if (p == (const MetaWord*)chunk) { + // chunk starts. + line1[pos] = '.'; + } else { + line1[pos] = ' '; + } + // Line 2: chunk type (x=spec, s=small, m=medium, h=humongous), uppercase if + // chunk is in use. + const bool chunk_is_free = ((Metachunk*)chunk)->is_tagged_free(); + if (chunk->word_size() == spec_chunk_size) { + line2[pos] = chunk_is_free ? 'x' : 'X'; + } else if (chunk->word_size() == small_chunk_size) { + line2[pos] = chunk_is_free ? 's' : 'S'; + } else if (chunk->word_size() == med_chunk_size) { + line2[pos] = chunk_is_free ? 'm' : 'M'; + } else if (chunk->word_size() > med_chunk_size) { + line2[pos] = chunk_is_free ? 'h' : 'H'; + } else { + ShouldNotReachHere(); + } + p += spec_chunk_size; + pos ++; + } + if (pos > 0) { + st->fill_to(22); + st->print_raw(line1, pos); + st->cr(); + st->fill_to(22); + st->print_raw(line2, pos); + st->cr(); + } + os::free(line1); + os::free(line2); + os::free(line3); +} + + #ifdef ASSERT uintx VirtualSpaceNode::container_count_slow() { uintx count = 0; @@ -649,6 +738,7 @@ class VirtualSpaceList : public CHeapObj { void purge(ChunkManager* chunk_manager); void print_on(outputStream* st) const; + void print_map(outputStream* st) const; class VirtualSpaceListIterator : public StackObj { VirtualSpaceNode* _virtual_spaces; @@ -1461,6 +1551,18 @@ void VirtualSpaceList::print_on(outputStream* st) const { } } +void VirtualSpaceList::print_map(outputStream* st) const { + VirtualSpaceNode* list = virtual_space_list(); + VirtualSpaceListIterator iter(list); + unsigned i = 0; + while (iter.repeat()) { + st->print_cr("Node %u:", i); + VirtualSpaceNode* node = iter.get_next(); + node->print_map(st, this->is_class()); + i ++; + } +} + // MetaspaceGC methods // VM_CollectForMetadataAllocation is the vm operation used to GC. @@ -1934,11 +2036,10 @@ Metachunk* ChunkManager::free_chunks_get(size_t word_size) { // Remove it from the links to this freelist chunk->set_next(NULL); chunk->set_prev(NULL); -#ifdef ASSERT + // Chunk is no longer on any freelist. Setting to false make container_count_slow() // work. chunk->set_is_tagged_free(false); -#endif chunk->container()->inc_container_count(); slow_locked_verify(); @@ -2006,7 +2107,7 @@ void ChunkManager::return_single_chunk(ChunkIndex index, Metachunk* chunk) { chunk_size_name(index), p2i(chunk), chunk->word_size()); } chunk->container()->dec_container_count(); - DEBUG_ONLY(chunk->set_is_tagged_free(true);) + chunk->set_is_tagged_free(true); // Chunk has been added; update counters. account_for_added_chunk(chunk); @@ -2945,9 +3046,9 @@ void MetaspaceAux::print_on(outputStream* out, Metaspace::MetadataType mdtype) { size_t free_bytes = free_bytes_slow(mdtype); size_t used_and_free = used_bytes + free_bytes + free_chunks_capacity_bytes; - out->print_cr(" Chunk accounting: used in chunks " SIZE_FORMAT + out->print_cr(" Chunk accounting: (used in chunks " SIZE_FORMAT "K + unused in chunks " SIZE_FORMAT "K + " - " capacity in free chunks " SIZE_FORMAT "K = " SIZE_FORMAT + " capacity in free chunks " SIZE_FORMAT "K) = " SIZE_FORMAT "K capacity in allocated chunks " SIZE_FORMAT "K", used_bytes / K, free_bytes / K, @@ -3211,6 +3312,31 @@ void MetaspaceAux::dump(outputStream* out) { print_waste(out); } +// Prints an ASCII representation of the given space. +void MetaspaceAux::print_metaspace_map(outputStream* out, Metaspace::MetadataType mdtype) { + MutexLockerEx cl(SpaceManager::expand_lock(), Mutex::_no_safepoint_check_flag); + const bool for_class = mdtype == Metaspace::ClassType ? true : false; + VirtualSpaceList* const vsl = for_class ? Metaspace::class_space_list() : Metaspace::space_list(); + if (vsl != NULL) { + if (for_class) { + if (!Metaspace::using_class_space()) { + out->print_cr("No Class Space."); + return; + } + out->print_raw("---- Metaspace Map (Class Space) ----"); + } else { + out->print_raw("---- Metaspace Map (Non-Class Space) ----"); + } + // Print legend: + out->cr(); + out->print_cr("Chunk Types (uppercase chunks are in use): x-specialized, s-small, m-medium, h-humongous."); + out->cr(); + VirtualSpaceList* const vsl = for_class ? Metaspace::class_space_list() : Metaspace::space_list(); + vsl->print_map(out); + out->cr(); + } +} + void MetaspaceAux::verify_free_chunks() { Metaspace::chunk_manager_metadata()->verify(); if (Metaspace::using_class_space()) { @@ -3849,6 +3975,7 @@ void Metaspace::report_metadata_oome(ClassLoaderData* loader_data, size_t word_s } LogStream ls(log.info()); MetaspaceAux::dump(&ls); + MetaspaceAux::print_metaspace_map(&ls, mdtype); ChunkManager::print_all_chunkmanagers(&ls); } diff --git a/src/hotspot/share/memory/metaspace.hpp b/src/hotspot/share/memory/metaspace.hpp index 7f18c616801..b9e86bd15b0 100644 --- a/src/hotspot/share/memory/metaspace.hpp +++ b/src/hotspot/share/memory/metaspace.hpp @@ -361,6 +361,10 @@ class MetaspaceAux : AllStatic { static void print_class_waste(outputStream* out); static void print_waste(outputStream* out); + + // Prints an ASCII representation of the given space. + static void print_metaspace_map(outputStream* out, Metaspace::MetadataType mdtype); + static void dump(outputStream* out); static void verify_free_chunks(); // Checks that the values returned by allocated_capacity_bytes() and From e878b3272bbd61f9f0a260d255a1c174af5d845b Mon Sep 17 00:00:00 2001 From: Dmitry Chuyko Date: Thu, 2 Nov 2017 18:44:44 +0300 Subject: [PATCH 28/61] 8189176: AARCH64: Improve _updateBytesCRC32 intrinsic Reviewed-by: aph --- src/hotspot/cpu/aarch64/assembler_aarch64.hpp | 11 ++ .../cpu/aarch64/c1_LIRAssembler_aarch64.cpp | 4 +- .../cpu/aarch64/macroAssembler_aarch64.cpp | 152 ++++++++++++------ .../cpu/aarch64/macroAssembler_aarch64.hpp | 3 + .../templateInterpreterGenerator_aarch64.cpp | 4 +- 5 files changed, 121 insertions(+), 53 deletions(-) diff --git a/src/hotspot/cpu/aarch64/assembler_aarch64.hpp b/src/hotspot/cpu/aarch64/assembler_aarch64.hpp index 6dc260f5a54..ab1bcd55817 100644 --- a/src/hotspot/cpu/aarch64/assembler_aarch64.hpp +++ b/src/hotspot/cpu/aarch64/assembler_aarch64.hpp @@ -1489,6 +1489,17 @@ public: #undef INSN + // Aliases for short forms of orn +void mvn(Register Rd, Register Rm, + enum shift_kind kind = LSL, unsigned shift = 0) { + orn(Rd, zr, Rm, kind, shift); +} + +void mvnw(Register Rd, Register Rm, + enum shift_kind kind = LSL, unsigned shift = 0) { + ornw(Rd, zr, Rm, kind, shift); +} + // Add/subtract (shifted register) #define INSN(NAME, size, op) \ void NAME(Register Rd, Register Rn, Register Rm, \ diff --git a/src/hotspot/cpu/aarch64/c1_LIRAssembler_aarch64.cpp b/src/hotspot/cpu/aarch64/c1_LIRAssembler_aarch64.cpp index b7ff274e212..36fefc866aa 100644 --- a/src/hotspot/cpu/aarch64/c1_LIRAssembler_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/c1_LIRAssembler_aarch64.cpp @@ -2657,9 +2657,9 @@ void LIR_Assembler::emit_updatecrc32(LIR_OpUpdateCRC32* op) { __ adrp(res, ExternalAddress(StubRoutines::crc_table_addr()), offset); if (offset) __ add(res, res, offset); - __ ornw(crc, zr, crc); // ~crc + __ mvnw(crc, crc); // ~crc __ update_byte_crc32(crc, val, res); - __ ornw(res, zr, crc); // ~crc + __ mvnw(res, crc); // ~crc } void LIR_Assembler::emit_profile_type(LIR_OpProfileType* op) { diff --git a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp index b4dfe38ed93..646aa3e7ebd 100644 --- a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp @@ -2929,6 +2929,105 @@ void MacroAssembler::update_word_crc32(Register crc, Register v, Register tmp, eor(crc, crc, tmp); } +void MacroAssembler::kernel_crc32_using_crc32(Register crc, Register buf, + Register len, Register tmp0, Register tmp1, Register tmp2, + Register tmp3) { + Label CRC_by64_loop, CRC_by4_loop, CRC_by1_loop, CRC_less64, CRC_by64_pre, CRC_by32_loop, CRC_less32, L_exit; + assert_different_registers(crc, buf, len, tmp0, tmp1, tmp2, tmp3); + + mvnw(crc, crc); + + subs(len, len, 128); + br(Assembler::GE, CRC_by64_pre); + BIND(CRC_less64); + adds(len, len, 128-32); + br(Assembler::GE, CRC_by32_loop); + BIND(CRC_less32) + adds(len, len, 32-4); + br(Assembler::GE, CRC_by4_loop); + adds(len, len, 4); + br(Assembler::GT, CRC_by1_loop); + b(L_exit); + + BIND(CRC_by32_loop); + ldp(tmp0, tmp1, Address(post(buf, 16))); + subs(len, len, 32); + crc32x(crc, crc, tmp0); + ldr(tmp2, Address(post(buf, 8))); + crc32x(crc, crc, tmp1); + ldr(tmp3, Address(post(buf, 8))); + crc32x(crc, crc, tmp2); + crc32x(crc, crc, tmp3); + br(Assembler::GE, CRC_by32_loop); + cmn(len, 32); + br(Assembler::NE, CRC_less32); + b(L_exit); + + BIND(CRC_by4_loop); + ldrw(tmp0, Address(post(buf, 4))); + subs(len, len, 4); + crc32w(crc, crc, tmp0); + br(Assembler::GE, CRC_by4_loop); + adds(len, len, 4); + br(Assembler::LE, L_exit); + BIND(CRC_by1_loop); + ldrb(tmp0, Address(post(buf, 1))); + subs(len, len, 1); + crc32b(crc, crc, tmp0); + br(Assembler::GT, CRC_by1_loop); + b(L_exit); + + BIND(CRC_by64_pre); + sub(buf, buf, 8); + ldp(tmp0, tmp1, Address(buf, 8)); + crc32x(crc, crc, tmp0); + ldr(tmp2, Address(buf, 24)); + crc32x(crc, crc, tmp1); + ldr(tmp3, Address(buf, 32)); + crc32x(crc, crc, tmp2); + ldr(tmp0, Address(buf, 40)); + crc32x(crc, crc, tmp3); + ldr(tmp1, Address(buf, 48)); + crc32x(crc, crc, tmp0); + ldr(tmp2, Address(buf, 56)); + crc32x(crc, crc, tmp1); + ldr(tmp3, Address(pre(buf, 64))); + + b(CRC_by64_loop); + + align(CodeEntryAlignment); + BIND(CRC_by64_loop); + subs(len, len, 64); + crc32x(crc, crc, tmp2); + ldr(tmp0, Address(buf, 8)); + crc32x(crc, crc, tmp3); + ldr(tmp1, Address(buf, 16)); + crc32x(crc, crc, tmp0); + ldr(tmp2, Address(buf, 24)); + crc32x(crc, crc, tmp1); + ldr(tmp3, Address(buf, 32)); + crc32x(crc, crc, tmp2); + ldr(tmp0, Address(buf, 40)); + crc32x(crc, crc, tmp3); + ldr(tmp1, Address(buf, 48)); + crc32x(crc, crc, tmp0); + ldr(tmp2, Address(buf, 56)); + crc32x(crc, crc, tmp1); + ldr(tmp3, Address(pre(buf, 64))); + br(Assembler::GE, CRC_by64_loop); + + // post-loop + crc32x(crc, crc, tmp2); + crc32x(crc, crc, tmp3); + + sub(len, len, 64); + add(buf, buf, 8); + cmn(len, 128); + br(Assembler::NE, CRC_less64); + BIND(L_exit); + mvnw(crc, crc); +} + /** * @param crc register containing existing CRC (32-bit) * @param buf register pointing to input byte buffer (byte*) @@ -2942,58 +3041,13 @@ void MacroAssembler::kernel_crc32(Register crc, Register buf, Register len, Label L_by16, L_by16_loop, L_by4, L_by4_loop, L_by1, L_by1_loop, L_exit; unsigned long offset; - ornw(crc, zr, crc); - if (UseCRC32) { - Label CRC_by64_loop, CRC_by4_loop, CRC_by1_loop; - - subs(len, len, 64); - br(Assembler::GE, CRC_by64_loop); - adds(len, len, 64-4); - br(Assembler::GE, CRC_by4_loop); - adds(len, len, 4); - br(Assembler::GT, CRC_by1_loop); - b(L_exit); - - BIND(CRC_by4_loop); - ldrw(tmp, Address(post(buf, 4))); - subs(len, len, 4); - crc32w(crc, crc, tmp); - br(Assembler::GE, CRC_by4_loop); - adds(len, len, 4); - br(Assembler::LE, L_exit); - BIND(CRC_by1_loop); - ldrb(tmp, Address(post(buf, 1))); - subs(len, len, 1); - crc32b(crc, crc, tmp); - br(Assembler::GT, CRC_by1_loop); - b(L_exit); - - align(CodeEntryAlignment); - BIND(CRC_by64_loop); - subs(len, len, 64); - ldp(tmp, tmp3, Address(post(buf, 16))); - crc32x(crc, crc, tmp); - crc32x(crc, crc, tmp3); - ldp(tmp, tmp3, Address(post(buf, 16))); - crc32x(crc, crc, tmp); - crc32x(crc, crc, tmp3); - ldp(tmp, tmp3, Address(post(buf, 16))); - crc32x(crc, crc, tmp); - crc32x(crc, crc, tmp3); - ldp(tmp, tmp3, Address(post(buf, 16))); - crc32x(crc, crc, tmp); - crc32x(crc, crc, tmp3); - br(Assembler::GE, CRC_by64_loop); - adds(len, len, 64-4); - br(Assembler::GE, CRC_by4_loop); - adds(len, len, 4); - br(Assembler::GT, CRC_by1_loop); - BIND(L_exit); - ornw(crc, zr, crc); + kernel_crc32_using_crc32(crc, buf, len, table0, table1, table2, table3); return; } + mvnw(crc, crc); + adrp(table0, ExternalAddress(StubRoutines::crc_table_addr()), offset); if (offset) add(table0, table0, offset); add(table1, table0, 1*256*sizeof(juint)); @@ -3171,7 +3225,7 @@ void MacroAssembler::kernel_crc32(Register crc, Register buf, Register len, adds(len, len, 4); br(Assembler::GT, L_by1_loop); BIND(L_exit); - ornw(crc, zr, crc); + mvnw(crc, crc); } /** diff --git a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.hpp b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.hpp index d0ca968bd44..640f88525d9 100644 --- a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.hpp +++ b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.hpp @@ -1261,6 +1261,9 @@ private: Register yz_idx1, Register yz_idx2, Register tmp, Register tmp3, Register tmp4, Register tmp7, Register product_hi); + void kernel_crc32_using_crc32(Register crc, Register buf, + Register len, Register tmp0, Register tmp1, Register tmp2, + Register tmp3); public: void multiply_to_len(Register x, Register xlen, Register y, Register ylen, Register z, Register zlen, Register tmp1, Register tmp2, Register tmp3, diff --git a/src/hotspot/cpu/aarch64/templateInterpreterGenerator_aarch64.cpp b/src/hotspot/cpu/aarch64/templateInterpreterGenerator_aarch64.cpp index 242a0715e8b..b601af31a35 100644 --- a/src/hotspot/cpu/aarch64/templateInterpreterGenerator_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/templateInterpreterGenerator_aarch64.cpp @@ -984,9 +984,9 @@ address TemplateInterpreterGenerator::generate_CRC32_update_entry() { __ adrp(tbl, ExternalAddress(StubRoutines::crc_table_addr()), offset); __ add(tbl, tbl, offset); - __ ornw(crc, zr, crc); // ~crc + __ mvnw(crc, crc); // ~crc __ update_byte_crc32(crc, val, tbl); - __ ornw(crc, zr, crc); // ~crc + __ mvnw(crc, crc); // ~crc // result in c_rarg0 From a043febf6fac76028dfe85735619530376ca15ba Mon Sep 17 00:00:00 2001 From: Gerard Ziemski Date: Thu, 2 Nov 2017 11:00:34 -0500 Subject: [PATCH 29/61] 8184765: Dynamically resize SystemDictionary Implemented dynamic resizing, which triggers when load factor is too high Reviewed-by: coleenp, rehn --- .../share/classfile/classLoaderData.cpp | 44 ++++---- .../share/classfile/classLoaderData.hpp | 2 + src/hotspot/share/classfile/dictionary.cpp | 82 ++++++++++++-- src/hotspot/share/classfile/dictionary.hpp | 18 ++- .../share/classfile/systemDictionary.cpp | 71 ++++++------ .../share/classfile/systemDictionary.hpp | 9 +- src/hotspot/share/runtime/globals.hpp | 4 +- src/hotspot/share/runtime/safepoint.cpp | 9 ++ src/hotspot/share/runtime/safepoint.hpp | 1 + src/hotspot/share/utilities/hashtable.cpp | 43 +++++++ src/hotspot/share/utilities/hashtable.hpp | 3 +- .../runtime/CommandLine/VMOptionWarning.java | 6 +- .../jtreg/runtime/LoadClass/TestResize.java | 103 +++++++++++++++++ .../runtime/LoadClass/TriggerResize.java | 105 ++++++++++++++++++ 14 files changed, 415 insertions(+), 85 deletions(-) create mode 100644 test/hotspot/jtreg/runtime/LoadClass/TestResize.java create mode 100644 test/hotspot/jtreg/runtime/LoadClass/TriggerResize.java diff --git a/src/hotspot/share/classfile/classLoaderData.cpp b/src/hotspot/share/classfile/classLoaderData.cpp index 761782bb35b..06c5d947cd0 100644 --- a/src/hotspot/share/classfile/classLoaderData.cpp +++ b/src/hotspot/share/classfile/classLoaderData.cpp @@ -604,40 +604,27 @@ ModuleEntryTable* ClassLoaderData::modules() { const int _boot_loader_dictionary_size = 1009; const int _default_loader_dictionary_size = 107; -const int _prime_array_size = 8; // array of primes for system dictionary size -const int _average_depth_goal = 3; // goal for lookup length -const int _primelist[_prime_array_size] = {107, 1009, 2017, 4049, 5051, 10103, 20201, 40423}; - -// Calculate a "good" dictionary size based -// on predicted or current loaded classes count. -static int calculate_dictionary_size(int classcount) { - int newsize = _primelist[0]; - if (classcount > 0 && !DumpSharedSpaces) { - int index = 0; - int desiredsize = classcount/_average_depth_goal; - for (newsize = _primelist[index]; index < _prime_array_size -1; - newsize = _primelist[++index]) { - if (desiredsize <= newsize) { - break; - } - } - } - return newsize; -} Dictionary* ClassLoaderData::create_dictionary() { assert(!is_anonymous(), "anonymous class loader data do not have a dictionary"); int size; + bool resizable = false; if (_the_null_class_loader_data == NULL) { size = _boot_loader_dictionary_size; + resizable = true; } else if (class_loader()->is_a(SystemDictionary::reflect_DelegatingClassLoader_klass())) { size = 1; // there's only one class in relection class loader and no initiated classes } else if (is_system_class_loader_data()) { - size = calculate_dictionary_size(PredictedLoadedClassCount); + size = _boot_loader_dictionary_size; + resizable = true; } else { size = _default_loader_dictionary_size; + resizable = true; } - return new Dictionary(this, size); + if (!DynamicallyResizeSystemDictionaries || DumpSharedSpaces || UseSharedSpaces) { + resizable = false; + } + return new Dictionary(this, size, resizable); } // Unloading support @@ -1325,6 +1312,19 @@ void ClassLoaderDataGraph::purge() { } } +int ClassLoaderDataGraph::resize_if_needed() { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint!"); + int resized = 0; + if (Dictionary::does_any_dictionary_needs_resizing()) { + FOR_ALL_DICTIONARY(cld) { + if (cld->dictionary()->resize_if_needed()) { + resized++; + } + } + } + return resized; +} + void ClassLoaderDataGraph::post_class_unload_events() { #if INCLUDE_TRACE assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint!"); diff --git a/src/hotspot/share/classfile/classLoaderData.hpp b/src/hotspot/share/classfile/classLoaderData.hpp index 7c912ba57bb..fc142452165 100644 --- a/src/hotspot/share/classfile/classLoaderData.hpp +++ b/src/hotspot/share/classfile/classLoaderData.hpp @@ -143,6 +143,8 @@ class ClassLoaderDataGraph : public AllStatic { } } + static int resize_if_needed(); + static bool has_metaspace_oom() { return _metaspace_oom; } static void set_metaspace_oom(bool value) { _metaspace_oom = value; } diff --git a/src/hotspot/share/classfile/dictionary.cpp b/src/hotspot/share/classfile/dictionary.cpp index c500c47734e..e0c23db63bc 100644 --- a/src/hotspot/share/classfile/dictionary.cpp +++ b/src/hotspot/share/classfile/dictionary.cpp @@ -29,6 +29,7 @@ #include "classfile/protectionDomainCache.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/systemDictionaryShared.hpp" +#include "gc/shared/gcLocker.hpp" #include "logging/log.hpp" #include "logging/logStream.hpp" #include "memory/iterator.hpp" @@ -39,6 +40,11 @@ #include "runtime/orderAccess.inline.hpp" #include "utilities/hashtable.inline.hpp" +// Optimization: if any dictionary needs resizing, we set this flag, +// so that we dont't have to walk all dictionaries to check if any actually +// needs resizing, which is costly to do at Safepoint. +bool Dictionary::_some_dictionary_needs_resizing = false; + size_t Dictionary::entry_size() { if (DumpSharedSpaces) { return SystemDictionaryShared::dictionary_entry_size(); @@ -47,15 +53,17 @@ size_t Dictionary::entry_size() { } } -Dictionary::Dictionary(ClassLoaderData* loader_data, int table_size) - : _loader_data(loader_data), Hashtable(table_size, (int)entry_size()) { +Dictionary::Dictionary(ClassLoaderData* loader_data, int table_size, bool resizable) + : _loader_data(loader_data), _resizable(resizable), _needs_resizing(false), + Hashtable(table_size, (int)entry_size()) { }; Dictionary::Dictionary(ClassLoaderData* loader_data, int table_size, HashtableBucket* t, - int number_of_entries) - : _loader_data(loader_data), Hashtable(table_size, (int)entry_size(), t, number_of_entries) { + int number_of_entries, bool resizable) + : _loader_data(loader_data), _resizable(resizable), _needs_resizing(false), + Hashtable(table_size, (int)entry_size(), t, number_of_entries) { }; Dictionary::~Dictionary() { @@ -96,6 +104,60 @@ void Dictionary::free_entry(DictionaryEntry* entry) { FREE_C_HEAP_ARRAY(char, entry); } +const int _resize_load_trigger = 5; // load factor that will trigger the resize +const double _resize_factor = 2.0; // by how much we will resize using current number of entries +const int _resize_max_size = 40423; // the max dictionary size allowed +const int _primelist[] = {107, 1009, 2017, 4049, 5051, 10103, 20201, _resize_max_size}; +const int _prime_array_size = sizeof(_primelist)/sizeof(int); + +// Calculate next "good" dictionary size based on requested count +static int calculate_dictionary_size(int requested) { + int newsize = _primelist[0]; + int index = 0; + for (newsize = _primelist[index]; index < (_prime_array_size - 1); + newsize = _primelist[++index]) { + if (requested <= newsize) { + break; + } + } + return newsize; +} + +bool Dictionary::does_any_dictionary_needs_resizing() { + return Dictionary::_some_dictionary_needs_resizing; +} + +void Dictionary::check_if_needs_resize() { + if (_resizable == true) { + if (number_of_entries() > (_resize_load_trigger*table_size())) { + _needs_resizing = true; + Dictionary::_some_dictionary_needs_resizing = true; + } + } +} + +bool Dictionary::resize_if_needed() { + int desired_size = 0; + if (_needs_resizing == true) { + desired_size = calculate_dictionary_size((int)(_resize_factor*number_of_entries())); + if (desired_size >= _resize_max_size) { + desired_size = _resize_max_size; + // We have reached the limit, turn resizing off + _resizable = false; + } + if ((desired_size != 0) && (desired_size != table_size())) { + if (!resize(desired_size)) { + // Something went wrong, turn resizing off + _resizable = false; + } + } + } + + _needs_resizing = false; + Dictionary::_some_dictionary_needs_resizing = false; + + return (desired_size != 0); +} bool DictionaryEntry::contains_protection_domain(oop protection_domain) const { #ifdef ASSERT @@ -264,14 +326,16 @@ void Dictionary::classes_do(MetaspaceClosure* it) { // also cast to volatile; we do this to ensure store order is maintained // by the compilers. -void Dictionary::add_klass(int index, unsigned int hash, Symbol* class_name, +void Dictionary::add_klass(unsigned int hash, Symbol* class_name, InstanceKlass* obj) { assert_locked_or_safepoint(SystemDictionary_lock); assert(obj != NULL, "adding NULL obj"); assert(obj->name() == class_name, "sanity check on name"); DictionaryEntry* entry = new_entry(hash, obj); + int index = hash_to_index(hash); add_entry(index, entry); + check_if_needs_resize(); } @@ -299,8 +363,11 @@ DictionaryEntry* Dictionary::get_entry(int index, unsigned int hash, } -InstanceKlass* Dictionary::find(int index, unsigned int hash, Symbol* name, +InstanceKlass* Dictionary::find(unsigned int hash, Symbol* name, Handle protection_domain) { + NoSafepointVerifier nsv; + + int index = hash_to_index(hash); DictionaryEntry* entry = get_entry(index, hash, name); if (entry != NULL && entry->is_valid_protection_domain(protection_domain)) { return entry->instance_klass(); @@ -350,9 +417,10 @@ void Dictionary::add_protection_domain(int index, unsigned int hash, } -bool Dictionary::is_valid_protection_domain(int index, unsigned int hash, +bool Dictionary::is_valid_protection_domain(unsigned int hash, Symbol* name, Handle protection_domain) { + int index = hash_to_index(hash); DictionaryEntry* entry = get_entry(index, hash, name); return entry->is_valid_protection_domain(protection_domain); } diff --git a/src/hotspot/share/classfile/dictionary.hpp b/src/hotspot/share/classfile/dictionary.hpp index d4f56df6787..be11f90005b 100644 --- a/src/hotspot/share/classfile/dictionary.hpp +++ b/src/hotspot/share/classfile/dictionary.hpp @@ -43,6 +43,11 @@ class BoolObjectClosure; class Dictionary : public Hashtable { friend class VMStructs; + static bool _some_dictionary_needs_resizing; + bool _resizable; + bool _needs_resizing; + void check_if_needs_resize(); + ClassLoaderData* _loader_data; // backpointer to owning loader ClassLoaderData* loader_data() const { return _loader_data; } @@ -51,13 +56,16 @@ class Dictionary : public Hashtable { protected: static size_t entry_size(); public: - Dictionary(ClassLoaderData* loader_data, int table_size); - Dictionary(ClassLoaderData* loader_data, int table_size, HashtableBucket* t, int number_of_entries); + Dictionary(ClassLoaderData* loader_data, int table_size, bool resizable = false); + Dictionary(ClassLoaderData* loader_data, int table_size, HashtableBucket* t, int number_of_entries, bool resizable = false); ~Dictionary(); + static bool does_any_dictionary_needs_resizing(); + bool resize_if_needed(); + DictionaryEntry* new_entry(unsigned int hash, InstanceKlass* klass); - void add_klass(int index, unsigned int hash, Symbol* class_name, InstanceKlass* obj); + void add_klass(unsigned int hash, Symbol* class_name, InstanceKlass* obj); InstanceKlass* find_class(int index, unsigned int hash, Symbol* name); @@ -79,8 +87,8 @@ public: void do_unloading(); // Protection domains - InstanceKlass* find(int index, unsigned int hash, Symbol* name, Handle protection_domain); - bool is_valid_protection_domain(int index, unsigned int hash, + InstanceKlass* find(unsigned int hash, Symbol* name, Handle protection_domain); + bool is_valid_protection_domain(unsigned int hash, Symbol* name, Handle protection_domain); void add_protection_domain(int index, unsigned int hash, diff --git a/src/hotspot/share/classfile/systemDictionary.cpp b/src/hotspot/share/classfile/systemDictionary.cpp index 9eabbf518d1..4bc3914eed5 100644 --- a/src/hotspot/share/classfile/systemDictionary.cpp +++ b/src/hotspot/share/classfile/systemDictionary.cpp @@ -371,7 +371,6 @@ Klass* SystemDictionary::resolve_super_or_fail(Symbol* child_name, ClassLoaderData* loader_data = class_loader_data(class_loader); Dictionary* dictionary = loader_data->dictionary(); unsigned int d_hash = dictionary->compute_hash(child_name); - int d_index = dictionary->hash_to_index(d_hash); unsigned int p_hash = placeholders()->compute_hash(child_name); int p_index = placeholders()->hash_to_index(p_hash); // can't throw error holding a lock @@ -379,7 +378,7 @@ Klass* SystemDictionary::resolve_super_or_fail(Symbol* child_name, bool throw_circularity_error = false; { MutexLocker mu(SystemDictionary_lock, THREAD); - Klass* childk = find_class(d_index, d_hash, child_name, dictionary); + Klass* childk = find_class(d_hash, child_name, dictionary); Klass* quicksuperk; // to support // loading: if child done loading, just return superclass // if class_name, & class_loader don't match: @@ -487,9 +486,9 @@ void SystemDictionary::validate_protection_domain(InstanceKlass* klass, Symbol* kn = klass->name(); unsigned int d_hash = dictionary->compute_hash(kn); - int d_index = dictionary->hash_to_index(d_hash); MutexLocker mu(SystemDictionary_lock, THREAD); + int d_index = dictionary->hash_to_index(d_hash); dictionary->add_protection_domain(d_index, d_hash, klass, protection_domain, THREAD); } @@ -555,7 +554,6 @@ InstanceKlass* SystemDictionary::handle_parallel_super_load( ClassLoaderData* loader_data = class_loader_data(class_loader); Dictionary* dictionary = loader_data->dictionary(); unsigned int d_hash = dictionary->compute_hash(name); - int d_index = dictionary->hash_to_index(d_hash); unsigned int p_hash = placeholders()->compute_hash(name); int p_index = placeholders()->hash_to_index(p_hash); @@ -579,7 +577,7 @@ InstanceKlass* SystemDictionary::handle_parallel_super_load( if (!class_loader.is_null() && is_parallelCapable(class_loader)) { MutexLocker mu(SystemDictionary_lock, THREAD); // Check if classloading completed while we were loading superclass or waiting - return find_class(d_index, d_hash, name, dictionary); + return find_class(d_hash, name, dictionary); } // must loop to both handle other placeholder updates @@ -589,7 +587,7 @@ InstanceKlass* SystemDictionary::handle_parallel_super_load( while (super_load_in_progress) { MutexLocker mu(SystemDictionary_lock, THREAD); // Check if classloading completed while we were loading superclass or waiting - InstanceKlass* check = find_class(d_index, d_hash, name, dictionary); + InstanceKlass* check = find_class(d_hash, name, dictionary); if (check != NULL) { // Klass is already loaded, so just return it return check; @@ -670,6 +668,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, class_loader = Handle(THREAD, java_lang_ClassLoader::non_reflection_class_loader(class_loader())); ClassLoaderData *loader_data = register_loader(class_loader, CHECK_NULL); Dictionary* dictionary = loader_data->dictionary(); + unsigned int d_hash = dictionary->compute_hash(name); // Do lookup to see if class already exist and the protection domain // has the right access @@ -677,11 +676,10 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, // All subsequent calls use find_class, and set has_loaded_class so that // before we return a result we call out to java to check for valid protection domain // to allow returning the Klass* and add it to the pd_set if it is valid - unsigned int d_hash = dictionary->compute_hash(name); - int d_index = dictionary->hash_to_index(d_hash); - Klass* probe = dictionary->find(d_index, d_hash, name, protection_domain); - if (probe != NULL) return probe; - + { + Klass* probe = dictionary->find(d_hash, name, protection_domain); + if (probe != NULL) return probe; + } // Non-bootstrap class loaders will call out to class loader and // define via jvm/jni_DefineClass which will acquire the @@ -716,7 +714,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, { MutexLocker mu(SystemDictionary_lock, THREAD); - InstanceKlass* check = find_class(d_index, d_hash, name, dictionary); + InstanceKlass* check = find_class(d_hash, name, dictionary); if (check != NULL) { // Klass is already loaded, so just return it class_has_been_loaded = true; @@ -800,7 +798,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, double_lock_wait(lockObject, THREAD); } // Check if classloading completed while we were waiting - InstanceKlass* check = find_class(d_index, d_hash, name, dictionary); + InstanceKlass* check = find_class(d_hash, name, dictionary); if (check != NULL) { // Klass is already loaded, so just return it k = check; @@ -825,7 +823,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, // i.e. now that we hold the LOAD_INSTANCE token on loading this class/CL // one final check if the load has already completed // class loaders holding the ObjectLock shouldn't find the class here - InstanceKlass* check = find_class(d_index, d_hash, name, dictionary); + InstanceKlass* check = find_class(d_hash, name, dictionary); if (check != NULL) { // Klass is already loaded, so return it after checking/adding protection domain k = check; @@ -858,7 +856,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, if (k == NULL && HAS_PENDING_EXCEPTION && PENDING_EXCEPTION->is_a(SystemDictionary::LinkageError_klass())) { MutexLocker mu(SystemDictionary_lock, THREAD); - InstanceKlass* check = find_class(d_index, d_hash, name, dictionary); + InstanceKlass* check = find_class(d_hash, name, dictionary); if (check != NULL) { // Klass is already loaded, so just use it k = check; @@ -873,7 +871,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, if (!HAS_PENDING_EXCEPTION && k != NULL && k->class_loader() != class_loader()) { - check_constraints(d_index, d_hash, k, class_loader, false, THREAD); + check_constraints(d_hash, k, class_loader, false, THREAD); // Need to check for a PENDING_EXCEPTION again; check_constraints // can throw and doesn't use the CHECK macro. @@ -881,7 +879,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, { // Grabbing the Compile_lock prevents systemDictionary updates // during compilations. MutexLocker mu(Compile_lock, THREAD); - update_dictionary(d_index, d_hash, p_index, p_hash, + update_dictionary(d_hash, p_index, p_hash, k, class_loader, THREAD); } @@ -923,7 +921,7 @@ Klass* SystemDictionary::resolve_instance_class_or_null(Symbol* name, if (protection_domain() == NULL) return k; // Check the protection domain has the right access - if (dictionary->is_valid_protection_domain(d_index, d_hash, name, + if (dictionary->is_valid_protection_domain(d_hash, name, protection_domain)) { return k; } @@ -965,8 +963,7 @@ Klass* SystemDictionary::find(Symbol* class_name, Dictionary* dictionary = loader_data->dictionary(); unsigned int d_hash = dictionary->compute_hash(class_name); - int d_index = dictionary->hash_to_index(d_hash); - return dictionary->find(d_index, d_hash, class_name, + return dictionary->find(d_hash, class_name, protection_domain); } @@ -1644,8 +1641,7 @@ void SystemDictionary::define_instance_class(InstanceKlass* k, TRAPS) { Symbol* name_h = k->name(); Dictionary* dictionary = loader_data->dictionary(); unsigned int d_hash = dictionary->compute_hash(name_h); - int d_index = dictionary->hash_to_index(d_hash); - check_constraints(d_index, d_hash, k, class_loader_h, true, CHECK); + check_constraints(d_hash, k, class_loader_h, true, CHECK); // Register class just loaded with class loader (placed in Vector) // Note we do this before updating the dictionary, as this can @@ -1673,7 +1669,7 @@ void SystemDictionary::define_instance_class(InstanceKlass* k, TRAPS) { // Add to systemDictionary - so other classes can see it. // Grabs and releases SystemDictionary_lock - update_dictionary(d_index, d_hash, p_index, p_hash, + update_dictionary(d_hash, p_index, p_hash, k, class_loader_h, THREAD); } k->eager_initialize(THREAD); @@ -1715,7 +1711,6 @@ InstanceKlass* SystemDictionary::find_or_define_instance_class(Symbol* class_nam Dictionary* dictionary = loader_data->dictionary(); unsigned int d_hash = dictionary->compute_hash(name_h); - int d_index = dictionary->hash_to_index(d_hash); // Hold SD lock around find_class and placeholder creation for DEFINE_CLASS unsigned int p_hash = placeholders()->compute_hash(name_h); @@ -1726,7 +1721,7 @@ InstanceKlass* SystemDictionary::find_or_define_instance_class(Symbol* class_nam MutexLocker mu(SystemDictionary_lock, THREAD); // First check if class already defined if (UnsyncloadClass || (is_parallelDefine(class_loader))) { - InstanceKlass* check = find_class(d_index, d_hash, name_h, dictionary); + InstanceKlass* check = find_class(d_hash, name_h, dictionary); if (check != NULL) { return check; } @@ -1748,7 +1743,7 @@ InstanceKlass* SystemDictionary::find_or_define_instance_class(Symbol* class_nam placeholders()->find_and_remove(p_index, p_hash, name_h, loader_data, PlaceholderTable::DEFINE_CLASS, THREAD); SystemDictionary_lock->notify_all(); #ifdef ASSERT - InstanceKlass* check = find_class(d_index, d_hash, name_h, dictionary); + InstanceKlass* check = find_class(d_hash, name_h, dictionary); assert(check != NULL, "definer missed recording success"); #endif return probe->instance_klass(); @@ -1823,10 +1818,11 @@ void SystemDictionary::check_loader_lock_contention(Handle loader_lock, TRAPS) { // ---------------------------------------------------------------------------- // Lookup -InstanceKlass* SystemDictionary::find_class(int index, unsigned int hash, +InstanceKlass* SystemDictionary::find_class(unsigned int hash, Symbol* class_name, Dictionary* dictionary) { assert_locked_or_safepoint(SystemDictionary_lock); + int index = dictionary->hash_to_index(hash); return dictionary->find_class(index, hash, class_name); } @@ -1856,8 +1852,7 @@ InstanceKlass* SystemDictionary::find_class(Symbol* class_name, ClassLoaderData* Dictionary* dictionary = loader_data->dictionary(); unsigned int d_hash = dictionary->compute_hash(class_name); - int d_index = dictionary->hash_to_index(d_hash); - return find_class(d_index, d_hash, class_name, dictionary); + return find_class(d_hash, class_name, dictionary); } @@ -2210,7 +2205,7 @@ BasicType SystemDictionary::box_klass_type(Klass* k) { // if defining is true, then LinkageError if already in dictionary // if initiating loader, then ok if InstanceKlass matches existing entry -void SystemDictionary::check_constraints(int d_index, unsigned int d_hash, +void SystemDictionary::check_constraints(unsigned int d_hash, InstanceKlass* k, Handle class_loader, bool defining, TRAPS) { @@ -2222,7 +2217,7 @@ void SystemDictionary::check_constraints(int d_index, unsigned int d_hash, MutexLocker mu(SystemDictionary_lock, THREAD); - InstanceKlass* check = find_class(d_index, d_hash, name, loader_data->dictionary()); + InstanceKlass* check = find_class(d_hash, name, loader_data->dictionary()); if (check != NULL) { // if different InstanceKlass - duplicate class definition, // else - ok, class loaded by a different thread in parallel, @@ -2270,7 +2265,7 @@ void SystemDictionary::check_constraints(int d_index, unsigned int d_hash, // Update class loader data dictionary - done after check_constraint and add_to_hierachy // have been called. -void SystemDictionary::update_dictionary(int d_index, unsigned int d_hash, +void SystemDictionary::update_dictionary(unsigned int d_hash, int p_index, unsigned int p_hash, InstanceKlass* k, Handle class_loader, @@ -2305,13 +2300,13 @@ void SystemDictionary::update_dictionary(int d_index, unsigned int d_hash, // Make a new dictionary entry. Dictionary* dictionary = loader_data->dictionary(); - InstanceKlass* sd_check = find_class(d_index, d_hash, name, dictionary); + InstanceKlass* sd_check = find_class(d_hash, name, dictionary); if (sd_check == NULL) { - dictionary->add_klass(d_index, d_hash, name, k); + dictionary->add_klass(d_hash, name, k); notice_modification(); } #ifdef ASSERT - sd_check = find_class(d_index, d_hash, name, dictionary); + sd_check = find_class(d_hash, name, dictionary); assert (sd_check != NULL, "should have entry in dictionary"); // Note: there may be a placeholder entry: for circularity testing // or for parallel defines @@ -2388,16 +2383,14 @@ bool SystemDictionary::add_loader_constraint(Symbol* class_name, Dictionary* dictionary1 = loader_data1->dictionary(); unsigned int d_hash1 = dictionary1->compute_hash(constraint_name); - int d_index1 = dictionary1->hash_to_index(d_hash1); Dictionary* dictionary2 = loader_data2->dictionary(); unsigned int d_hash2 = dictionary2->compute_hash(constraint_name); - int d_index2 = dictionary2->hash_to_index(d_hash2); { MutexLocker mu_s(SystemDictionary_lock, THREAD); - InstanceKlass* klass1 = find_class(d_index1, d_hash1, constraint_name, dictionary1); - InstanceKlass* klass2 = find_class(d_index2, d_hash2, constraint_name, dictionary2); + InstanceKlass* klass1 = find_class(d_hash1, constraint_name, dictionary1); + InstanceKlass* klass2 = find_class(d_hash2, constraint_name, dictionary2); return constraints()->add_entry(constraint_name, klass1, class_loader1, klass2, class_loader2); } diff --git a/src/hotspot/share/classfile/systemDictionary.hpp b/src/hotspot/share/classfile/systemDictionary.hpp index c890afaf321..9047b2a0d39 100644 --- a/src/hotspot/share/classfile/systemDictionary.hpp +++ b/src/hotspot/share/classfile/systemDictionary.hpp @@ -655,11 +655,8 @@ protected: // Setup link to hierarchy static void add_to_hierarchy(InstanceKlass* k, TRAPS); - // We pass in the hashtable index so we can calculate it outside of - // the SystemDictionary_lock. - // Basic find on loaded classes - static InstanceKlass* find_class(int index, unsigned int hash, + static InstanceKlass* find_class(unsigned int hash, Symbol* name, Dictionary* dictionary); static InstanceKlass* find_class(Symbol* class_name, ClassLoaderData* loader_data); @@ -685,10 +682,10 @@ protected: static void initialize_preloaded_classes(TRAPS); // Class loader constraints - static void check_constraints(int index, unsigned int hash, + static void check_constraints(unsigned int hash, InstanceKlass* k, Handle loader, bool defining, TRAPS); - static void update_dictionary(int d_index, unsigned int d_hash, + static void update_dictionary(unsigned int d_hash, int p_index, unsigned int p_hash, InstanceKlass* k, Handle loader, TRAPS); diff --git a/src/hotspot/share/runtime/globals.hpp b/src/hotspot/share/runtime/globals.hpp index ddef624a3f7..a8f00ba895d 100644 --- a/src/hotspot/share/runtime/globals.hpp +++ b/src/hotspot/share/runtime/globals.hpp @@ -1144,8 +1144,8 @@ public: notproduct(bool, PrintSystemDictionaryAtExit, false, \ "Print the system dictionary at exit") \ \ - experimental(intx, PredictedLoadedClassCount, 0, \ - "Experimental: Tune loaded class cache starting size") \ + diagnostic(bool, DynamicallyResizeSystemDictionaries, true, \ + "Dynamically resize system dictionaries as needed") \ \ diagnostic(bool, UnsyncloadClass, false, \ "Unstable: VM calls loadClass unsynchronized. Custom " \ diff --git a/src/hotspot/share/runtime/safepoint.cpp b/src/hotspot/share/runtime/safepoint.cpp index 505803b2c6e..5acd062d463 100644 --- a/src/hotspot/share/runtime/safepoint.cpp +++ b/src/hotspot/share/runtime/safepoint.cpp @@ -23,6 +23,7 @@ */ #include "precompiled.hpp" +#include "classfile/classLoaderData.hpp" #include "classfile/stringTable.hpp" #include "classfile/symbolTable.hpp" #include "classfile/systemDictionary.hpp" @@ -618,6 +619,14 @@ public: ClassLoaderDataGraph::purge_if_needed(); event_safepoint_cleanup_task_commit(event, name); } + + if (!_subtasks.is_task_claimed(SafepointSynchronize::SAFEPOINT_CLEANUP_SYSTEM_DICTIONARY_RESIZE)) { + const char* name = "resizing system dictionaries"; + EventSafepointCleanupTask event; + TraceTime timer(name, TRACETIME_LOG(Info, safepoint, cleanup)); + ClassLoaderDataGraph::resize_if_needed(); + event_safepoint_cleanup_task_commit(event, name); + } _subtasks.all_tasks_completed(_num_workers); } }; diff --git a/src/hotspot/share/runtime/safepoint.hpp b/src/hotspot/share/runtime/safepoint.hpp index 9a0e3c886b0..561431bb637 100644 --- a/src/hotspot/share/runtime/safepoint.hpp +++ b/src/hotspot/share/runtime/safepoint.hpp @@ -83,6 +83,7 @@ class SafepointSynchronize : AllStatic { SAFEPOINT_CLEANUP_SYMBOL_TABLE_REHASH, SAFEPOINT_CLEANUP_STRING_TABLE_REHASH, SAFEPOINT_CLEANUP_CLD_PURGE, + SAFEPOINT_CLEANUP_SYSTEM_DICTIONARY_RESIZE, // Leave this one last. SAFEPOINT_CLEANUP_NUM_TASKS }; diff --git a/src/hotspot/share/utilities/hashtable.cpp b/src/hotspot/share/utilities/hashtable.cpp index 625616887b3..2195920c1a2 100644 --- a/src/hotspot/share/utilities/hashtable.cpp +++ b/src/hotspot/share/utilities/hashtable.cpp @@ -264,6 +264,49 @@ static int literal_size(oop obj) { } } +template bool BasicHashtable::resize(int new_size) { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint"); + + // Allocate new buckets + HashtableBucket* buckets_new = NEW_C_HEAP_ARRAY2_RETURN_NULL(HashtableBucket, new_size, F, CURRENT_PC); + if (buckets_new == NULL) { + return false; + } + + // Clear the new buckets + for (int i = 0; i < new_size; i++) { + buckets_new[i].clear(); + } + + int table_size_old = _table_size; + // hash_to_index() uses _table_size, so switch the sizes now + _table_size = new_size; + + // Move entries from the old table to a new table + for (int index_old = 0; index_old < table_size_old; index_old++) { + for (BasicHashtableEntry* p = _buckets[index_old].get_entry(); p != NULL; ) { + BasicHashtableEntry* next = p->next(); + bool keep_shared = p->is_shared(); + int index_new = hash_to_index(p->hash()); + + p->set_next(buckets_new[index_new].get_entry()); + buckets_new[index_new].set_entry(p); + + if (keep_shared) { + p->set_shared(); + } + p = next; + } + } + + // The old backets now can be released + BasicHashtable::free_buckets(); + + // Switch to the new storage + _buckets = buckets_new; + + return true; +} // Dump footprint and bucket length statistics // diff --git a/src/hotspot/share/utilities/hashtable.hpp b/src/hotspot/share/utilities/hashtable.hpp index fab276f27a0..20b8a8ff30e 100644 --- a/src/hotspot/share/utilities/hashtable.hpp +++ b/src/hotspot/share/utilities/hashtable.hpp @@ -237,6 +237,8 @@ public: int number_of_entries() const { return _number_of_entries; } + bool resize(int new_size); + template void verify_table(const char* table_name) PRODUCT_RETURN; }; @@ -281,7 +283,6 @@ public: HashtableEntry** bucket_addr(int i) { return (HashtableEntry**)BasicHashtable::bucket_addr(i); } - }; template class RehashableHashtable : public Hashtable { diff --git a/test/hotspot/jtreg/runtime/CommandLine/VMOptionWarning.java b/test/hotspot/jtreg/runtime/CommandLine/VMOptionWarning.java index c9c6f2187bc..8cb2a422598 100644 --- a/test/hotspot/jtreg/runtime/CommandLine/VMOptionWarning.java +++ b/test/hotspot/jtreg/runtime/CommandLine/VMOptionWarning.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2014, 2017, 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 @@ -36,9 +36,9 @@ import jdk.test.lib.Platform; public class VMOptionWarning { public static void main(String[] args) throws Exception { - ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-XX:+PredictedLoadedClassCount", "-version"); + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-XX:+AlwaysSafeConstructors", "-version"); OutputAnalyzer output = new OutputAnalyzer(pb.start()); - output.shouldContain("Error: VM option 'PredictedLoadedClassCount' is experimental and must be enabled via -XX:+UnlockExperimentalVMOptions."); + output.shouldContain("Error: VM option 'AlwaysSafeConstructors' is experimental and must be enabled via -XX:+UnlockExperimentalVMOptions."); if (Platform.isDebugBuild()) { System.out.println("Skip the rest of the tests on debug builds since diagnostic, develop, and notproduct options are available on debug builds."); diff --git a/test/hotspot/jtreg/runtime/LoadClass/TestResize.java b/test/hotspot/jtreg/runtime/LoadClass/TestResize.java new file mode 100644 index 00000000000..31bd0f33e10 --- /dev/null +++ b/test/hotspot/jtreg/runtime/LoadClass/TestResize.java @@ -0,0 +1,103 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 8184765 + * @summary make sure the SystemDictionary gets resized when load factor is too high + * @library /test/lib + * @modules java.base/jdk.internal.misc + * java.management + * @compile TriggerResize.java + * @run driver TestResize + */ + +import java.lang.ProcessBuilder; +import java.lang.Process; +import jdk.test.lib.process.ProcessTools; +import java.io.BufferedReader; +import java.io.InputStreamReader; +import java.util.Scanner; + +public class TestResize { + + static double MAX_LOAD_FACTOR = 5.0; // see _resize_load_trigger in dictionary.cpp + + static int getInt(String string) { + int start = 0; + for (int i = 0; i < string.length(); i++) { + if (!Character.isDigit(string.charAt(i))) { + start++; + } else { + break; + } + } + int end = start; + for (int i = end; i < string.length(); i++) { + if (Character.isDigit(string.charAt(i))) { + end++; + } else { + break; + } + } + return Integer.parseInt(string.substring(start, end)); + } + + static void analyzeOutputOn(ProcessBuilder pb) throws Exception { + pb.redirectErrorStream(true); + Process process = pb.start(); + BufferedReader rd = new BufferedReader(new InputStreamReader(process.getInputStream())); + String line = rd.readLine(); + while (line != null) { + if (line.startsWith("Java dictionary (")) { + // ex. "Java dictionary (table_size=107, classes=6)" + // ex. "Java dictionary (table_size=20201, classes=50002)" + Scanner scanner = new Scanner(line); + scanner.next(); + scanner.next(); + int table_size = getInt(scanner.next()); + int classes = getInt(scanner.next()); + scanner.close(); + + double loadFactor = (double)classes / (double)table_size; + if (loadFactor > MAX_LOAD_FACTOR) { + throw new RuntimeException("Load factor too high, expected MAX "+MAX_LOAD_FACTOR+", got "+loadFactor); + } else { + System.out.println("PASS table_size:"+table_size+", classes:"+classes+" OK"); + } + } + line = rd.readLine(); + } + int retval = process.waitFor(); + if (retval != 0) { + throw new RuntimeException("Error: test returned non-zero value"); + } + } + + public static void main(String[] args) throws Exception { + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-XX:+PrintSystemDictionaryAtExit", + "TriggerResize", + "50000"); + analyzeOutputOn(pb); + } +} diff --git a/test/hotspot/jtreg/runtime/LoadClass/TriggerResize.java b/test/hotspot/jtreg/runtime/LoadClass/TriggerResize.java new file mode 100644 index 00000000000..8ea5f66fd71 --- /dev/null +++ b/test/hotspot/jtreg/runtime/LoadClass/TriggerResize.java @@ -0,0 +1,105 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +import java.lang.ClassLoader; + +public class TriggerResize extends ClassLoader +{ + static private int[] DATA = // bytes for "class TestCase00000 {}" + { + -54, -2, -70, -66, 0, 0, 0, 52, 0, 13, // 0 + 10, 0, 3, 0, 10, 7, 0, 11, 7, 0, // 10 + 12, 1, 0, 6, 60, 105, 110, 105, 116, 62, // 20 + 1, 0, 3, 40, 41, 86, 1, 0, 4, 67, // 30 + 111, 100, 101, 1, 0, 15, 76, 105, 110, 101, // 40 + 78, 117, 109, 98, 101, 114, 84, 97, 98, 108, // 50 + 101, 1, 0, 10, 83, 111, 117, 114, 99, 101, // 60 + 70, 105, 108, 101, 1, 0, 18, 84, 101, 115, // 70 + 116, 67, 97, 115, 101, 48, 48, 48, 48, 48, // 80 + 46, 106, 97, 118, 97, 12, 0, 4, 0, 5, // 90 + 1, 0, 13, 84, 101, 115, 116, 67, 97, 115, // 100 + 101, 48, 48, 48, 48, 48, 1, 0, 16, 106, // 110 + 97, 118, 97, 47, 108, 97, 110, 103, 47, 79, // 120 + 98, 106, 101, 99, 116, 0, 32, 0, 2, 0, // 130 + 3, 0, 0, 0, 0, 0, 1, 0, 0, 0, // 140 + 4, 0, 5, 0, 1, 0, 6, 0, 0, 0, // 150 + 29, 0, 1, 0, 1, 0, 0, 0, 5, 42, // 160 + -73, 0, 1, -79, 0, 0, 0, 1, 0, 7, // 170 + 0, 0, 0, 6, 0, 1, 0, 0, 0, 1, // 180 + 0, 1, 0, 8, 0, 0, 0, 2, 0, 9 // 190 + }; + + static private int INDEX1 = 85; + static private int INDEX2 = 111; + static private int BASE = 48; + + public TriggerResize() + { + super(); + } + + public void load(int index) + { + byte[] bytes = new byte[TriggerResize.DATA.length]; + for (int i=0; i= 1) { + Integer i = new Integer(args[0]); + count = i.intValue(); + } + + TriggerResize test = new TriggerResize(); + for (int i = 0; i <= count; i++) + { + test.load(i); + } + + // trigger safepoint to resize the SystemDictionary if needed + System.gc(); + } +} From df4f3bc82e4bd094ad9ce710aa0041cb4c391530 Mon Sep 17 00:00:00 2001 From: Coleen Phillimore Date: Fri, 3 Nov 2017 07:44:13 -0400 Subject: [PATCH 30/61] 8190617: test/jdk/sun/tools/jhsdb/BasicLauncherTest.java fails Change type expected by SA for PerfMemory::_initialize to int. Reviewed-by: dholmes, sspitsyn --- .../share/classes/sun/jvm/hotspot/runtime/PerfMemory.java | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/PerfMemory.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/PerfMemory.java index aa9a78f59d3..1b4bd2a7e5c 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/PerfMemory.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/PerfMemory.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2004, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2004, 2017, 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 @@ -35,7 +35,7 @@ public class PerfMemory { private static AddressField topField; private static CIntegerField capacityField; private static AddressField prologueField; - private static JIntField initializedField; + private static CIntegerField initializedField; static { VM.registerVMInitializedObserver(new Observer() { @@ -52,7 +52,7 @@ public class PerfMemory { topField = type.getAddressField("_top"); capacityField = type.getCIntegerField("_capacity"); prologueField = type.getAddressField("_prologue"); - initializedField = type.getJIntField("_initialized"); + initializedField = type.getCIntegerField("_initialized"); } // Accessors @@ -73,7 +73,7 @@ public class PerfMemory { } public static boolean initialized() { - return ((int) initializedField.getValue()) != 0; + return (initializedField.getValue()) != 0; } public static PerfDataPrologue prologue() { From c39d2071f017954f6cc8958df55d99b1657597f2 Mon Sep 17 00:00:00 2001 From: Calvin Cheung Date: Fri, 3 Nov 2017 10:48:26 -0700 Subject: [PATCH 31/61] 8187347: Do not abort CDS archive creation when some classes are unverifiable Deprecating the IgnoreUnverifiableClassesDuringDump vm option in JDK10 and set its default value to true Reviewed-by: dholmes, jiangli --- src/hotspot/share/runtime/arguments.cpp | 1 + src/hotspot/share/runtime/globals.hpp | 2 +- test/hotspot/jtreg/runtime/CommandLine/VMDeprecatedOptions.java | 1 + 3 files changed, 3 insertions(+), 1 deletion(-) diff --git a/src/hotspot/share/runtime/arguments.cpp b/src/hotspot/share/runtime/arguments.cpp index e3fe0d287d7..263091cf981 100644 --- a/src/hotspot/share/runtime/arguments.cpp +++ b/src/hotspot/share/runtime/arguments.cpp @@ -382,6 +382,7 @@ static SpecialFlag const special_jvm_flags[] = { { "MaxRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() }, { "MinRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() }, { "InitialRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() }, + { "IgnoreUnverifiableClassesDuringDump", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() }, // --- Deprecated alias flags (see also aliased_jvm_flags) - sorted by obsolete_in then expired_in: { "DefaultMaxRAMFraction", JDK_Version::jdk(8), JDK_Version::undefined(), JDK_Version::undefined() }, diff --git a/src/hotspot/share/runtime/globals.hpp b/src/hotspot/share/runtime/globals.hpp index a8f00ba895d..557605d2108 100644 --- a/src/hotspot/share/runtime/globals.hpp +++ b/src/hotspot/share/runtime/globals.hpp @@ -3922,7 +3922,7 @@ public: "Average number of symbols per bucket in shared table") \ range(2, 246) \ \ - diagnostic(bool, IgnoreUnverifiableClassesDuringDump, false, \ + diagnostic(bool, IgnoreUnverifiableClassesDuringDump, true, \ "Do not quit -Xshare:dump even if we encounter unverifiable " \ "classes. Just exclude them from the shared dictionary.") \ \ diff --git a/test/hotspot/jtreg/runtime/CommandLine/VMDeprecatedOptions.java b/test/hotspot/jtreg/runtime/CommandLine/VMDeprecatedOptions.java index ab0b29d95ed..f7f1c685ed5 100644 --- a/test/hotspot/jtreg/runtime/CommandLine/VMDeprecatedOptions.java +++ b/test/hotspot/jtreg/runtime/CommandLine/VMDeprecatedOptions.java @@ -91,5 +91,6 @@ public class VMDeprecatedOptions { public static void main(String[] args) throws Throwable { testDeprecated(DEPRECATED_OPTIONS); // Make sure that each deprecated option is mentioned in the output. testDeprecatedDiagnostic("UnsyncloadClass", "false"); + testDeprecatedDiagnostic("IgnoreUnverifiableClassesDuringDump", "false"); } } From c07d8e198102dc6a263cadddeb40d2f60c4a6b78 Mon Sep 17 00:00:00 2001 From: Ben Walsh Date: Fri, 3 Nov 2017 17:09:25 -0700 Subject: [PATCH 32/61] 8189731: Disable CFLH when there are no transformers Enable CFLH only when there are transformers Reviewed-by: sspitsyn, alanb --- make/mapfiles/libinstrument/mapfile-vers | 1 + .../sun/instrument/InstrumentationImpl.java | 14 ++++++-- .../InstrumentationImplNativeMethods.c | 13 ++++++- .../native/libinstrument/InvocationAdapter.c | 2 +- .../share/native/libinstrument/JPLISAgent.c | 35 ++++++++++++------- .../share/native/libinstrument/JPLISAgent.h | 11 ++++-- 6 files changed, 57 insertions(+), 19 deletions(-) diff --git a/make/mapfiles/libinstrument/mapfile-vers b/make/mapfiles/libinstrument/mapfile-vers index c51a8af2dc7..f89d009670a 100644 --- a/make/mapfiles/libinstrument/mapfile-vers +++ b/make/mapfiles/libinstrument/mapfile-vers @@ -31,6 +31,7 @@ SUNWprivate_1.1 { Agent_OnAttach; Java_sun_instrument_InstrumentationImpl_isModifiableClass0; Java_sun_instrument_InstrumentationImpl_isRetransformClassesSupported0; + Java_sun_instrument_InstrumentationImpl_setHasTransformers; Java_sun_instrument_InstrumentationImpl_setHasRetransformableTransformers; Java_sun_instrument_InstrumentationImpl_retransformClasses0; Java_sun_instrument_InstrumentationImpl_getAllLoadedClasses0; diff --git a/src/java.instrument/share/classes/sun/instrument/InstrumentationImpl.java b/src/java.instrument/share/classes/sun/instrument/InstrumentationImpl.java index 4669dcac8a0..5cf0c21361f 100644 --- a/src/java.instrument/share/classes/sun/instrument/InstrumentationImpl.java +++ b/src/java.instrument/share/classes/sun/instrument/InstrumentationImpl.java @@ -103,6 +103,9 @@ public class InstrumentationImpl implements Instrumentation { } } else { mTransformerManager.addTransformer(transformer); + if (mTransformerManager.getTransformerCount() == 1) { + setHasTransformers(mNativeAgent, true); + } } } @@ -114,8 +117,12 @@ public class InstrumentationImpl implements Instrumentation { TransformerManager mgr = findTransformerManager(transformer); if (mgr != null) { mgr.removeTransformer(transformer); - if (mgr.isRetransformable() && mgr.getTransformerCount() == 0) { - setHasRetransformableTransformers(mNativeAgent, false); + if (mgr.getTransformerCount() == 0) { + if (mgr.isRetransformable()) { + setHasRetransformableTransformers(mNativeAgent, false); + } else { + setHasTransformers(mNativeAgent, false); + } } return true; } @@ -361,6 +368,9 @@ public class InstrumentationImpl implements Instrumentation { private native boolean isRetransformClassesSupported0(long nativeAgent); + private native void + setHasTransformers(long nativeAgent, boolean has); + private native void setHasRetransformableTransformers(long nativeAgent, boolean has); diff --git a/src/java.instrument/share/native/libinstrument/InstrumentationImplNativeMethods.c b/src/java.instrument/share/native/libinstrument/InstrumentationImplNativeMethods.c index 7b2ca5de9e2..7d8937dc508 100644 --- a/src/java.instrument/share/native/libinstrument/InstrumentationImplNativeMethods.c +++ b/src/java.instrument/share/native/libinstrument/InstrumentationImplNativeMethods.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2017, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -76,6 +76,17 @@ Java_sun_instrument_InstrumentationImpl_isRetransformClassesSupported0 return isRetransformClassesSupported(jnienv, (JPLISAgent*)(intptr_t)agent); } +/* + * Class: sun_instrument_InstrumentationImpl + * Method: setHasTransformers + * Signature: (Z)V + */ +JNIEXPORT void JNICALL +Java_sun_instrument_InstrumentationImpl_setHasTransformers + (JNIEnv * jnienv, jobject implThis, jlong agent, jboolean has) { + setHasTransformers(jnienv, (JPLISAgent*)(intptr_t)agent, has); +} + /* * Class: sun_instrument_InstrumentationImpl * Method: setHasRetransformableTransformers diff --git a/src/java.instrument/share/native/libinstrument/InvocationAdapter.c b/src/java.instrument/share/native/libinstrument/InvocationAdapter.c index dad02634dc8..2845bb5edb3 100644 --- a/src/java.instrument/share/native/libinstrument/InvocationAdapter.c +++ b/src/java.instrument/share/native/libinstrument/InvocationAdapter.c @@ -395,7 +395,7 @@ DEF_Agent_OnAttach(JavaVM* vm, char *args, void * reserved) { jplis_assert(success); /* - * Turn on the ClassFileLoadHook. + * Setup ClassFileLoadHook handler. */ if (success) { success = setLivePhaseEventHandlers(agent); diff --git a/src/java.instrument/share/native/libinstrument/JPLISAgent.c b/src/java.instrument/share/native/libinstrument/JPLISAgent.c index 01edc8d451d..b6485d23f53 100644 --- a/src/java.instrument/share/native/libinstrument/JPLISAgent.c +++ b/src/java.instrument/share/native/libinstrument/JPLISAgent.c @@ -404,8 +404,8 @@ processJavaStart( JPLISAgent * agent, /* - * Then turn off the VMInit handler and turn on the ClassFileLoadHook. - * This way it is on before anyone registers a transformer. + * Register a handler for ClassFileLoadHook (without enabling this event). + * Turn off the VMInit handler. */ if ( result ) { result = setLivePhaseEventHandlers(agent); @@ -649,17 +649,6 @@ setLivePhaseEventHandlers( JPLISAgent * agent) { jplis_assert(jvmtierror == JVMTI_ERROR_NONE); } - if ( jvmtierror == JVMTI_ERROR_NONE ) { - /* turn on ClassFileLoadHook */ - jvmtierror = (*jvmtienv)->SetEventNotificationMode( - jvmtienv, - JVMTI_ENABLE, - JVMTI_EVENT_CLASS_FILE_LOAD_HOOK, - NULL /* all threads */); - check_phase_ret_false(jvmtierror); - jplis_assert(jvmtierror == JVMTI_ERROR_NONE); - } - return (jvmtierror == JVMTI_ERROR_NONE); } @@ -1096,6 +1085,21 @@ isRetransformClassesSupported(JNIEnv * jnienv, JPLISAgent * agent) { return agent->mRetransformEnvironment.mIsRetransformer; } +void +setHasTransformers(JNIEnv * jnienv, JPLISAgent * agent, jboolean has) { + jvmtiEnv * jvmtienv = jvmti(agent); + jvmtiError jvmtierror; + + jplis_assert(jvmtienv != NULL); + jvmtierror = (*jvmtienv)->SetEventNotificationMode( + jvmtienv, + has? JVMTI_ENABLE : JVMTI_DISABLE, + JVMTI_EVENT_CLASS_FILE_LOAD_HOOK, + NULL /* all threads */); + check_phase_ret(jvmtierror); + jplis_assert(jvmtierror == JVMTI_ERROR_NONE); +} + void setHasRetransformableTransformers(JNIEnv * jnienv, JPLISAgent * agent, jboolean has) { jvmtiEnv * retransformerEnv = retransformableEnvironment(agent); @@ -1107,6 +1111,7 @@ setHasRetransformableTransformers(JNIEnv * jnienv, JPLISAgent * agent, jboolean has? JVMTI_ENABLE : JVMTI_DISABLE, JVMTI_EVENT_CLASS_FILE_LOAD_HOOK, NULL /* all threads */); + check_phase_ret(jvmtierror); jplis_assert(jvmtierror == JVMTI_ERROR_NONE); } @@ -1185,6 +1190,10 @@ retransformClasses(JNIEnv * jnienv, JPLISAgent * agent, jobjectArray classes) { deallocate(retransformerEnv, (void*)classArray); } + /* Return back if we executed the JVMTI API in a wrong phase + */ + check_phase_ret(errorCode); + if (errorCode != JVMTI_ERROR_NONE) { createAndThrowThrowableFromJVMTIErrorCode(jnienv, errorCode); } diff --git a/src/java.instrument/share/native/libinstrument/JPLISAgent.h b/src/java.instrument/share/native/libinstrument/JPLISAgent.h index 08d61636839..1ee046d69a0 100644 --- a/src/java.instrument/share/native/libinstrument/JPLISAgent.h +++ b/src/java.instrument/share/native/libinstrument/JPLISAgent.h @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2017, 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 @@ -120,7 +120,11 @@ eventHandlerVMInit( jvmtiEnv * jvmtienv, JNIEnv * jnienv, jthread thread); -/* ClassFileLoadHook event handler. Installed during VMInit, then left in place forever. */ +/* + * ClassFileLoadHook event handler. + * Enabled when the first transformer is added; + * Disabled when the last transformer is removed. + */ extern void JNICALL eventHandlerClassFileLoadHook( jvmtiEnv * jvmtienv, JNIEnv * jnienv, @@ -240,6 +244,9 @@ isModifiableClass(JNIEnv * jnienv, JPLISAgent * agent, jclass clazz); extern jboolean isRetransformClassesSupported(JNIEnv * jnienv, JPLISAgent * agent); +extern void +setHasTransformers(JNIEnv * jnienv, JPLISAgent * agent, jboolean has); + extern void setHasRetransformableTransformers(JNIEnv * jnienv, JPLISAgent * agent, jboolean has); From 249ea8369e037656768740bb6b42fbd81216e0a1 Mon Sep 17 00:00:00 2001 From: Erik Gahlin Date: Sun, 5 Nov 2017 03:43:15 +0100 Subject: [PATCH 33/61] 8189425: Minor updates in support of closed changes Reviewed-by: mgronlun, dholmes --- src/hotspot/share/runtime/handles.cpp | 2 +- src/hotspot/share/runtime/mutexLocker.cpp | 2 -- src/hotspot/share/runtime/mutexLocker.hpp | 1 - src/hotspot/share/utilities/growableArray.hpp | 2 +- 4 files changed, 2 insertions(+), 5 deletions(-) diff --git a/src/hotspot/share/runtime/handles.cpp b/src/hotspot/share/runtime/handles.cpp index 6edf4d02f46..9c684065fb9 100644 --- a/src/hotspot/share/runtime/handles.cpp +++ b/src/hotspot/share/runtime/handles.cpp @@ -99,7 +99,7 @@ static uintx chunk_oops_do(OopClosure* f, Chunk* chunk, char* chunk_top) { while (bottom < top) { // This test can be moved up but for now check every oop. - assert(oopDesc::is_oop(*bottom), "handle should point to oop"); + assert(oopDesc::is_oop(*bottom, true), "handle should point to oop"); f->do_oop(bottom++); } diff --git a/src/hotspot/share/runtime/mutexLocker.cpp b/src/hotspot/share/runtime/mutexLocker.cpp index 5927b527f73..711578c254a 100644 --- a/src/hotspot/share/runtime/mutexLocker.cpp +++ b/src/hotspot/share/runtime/mutexLocker.cpp @@ -129,7 +129,6 @@ Mutex* JfrStacktrace_lock = NULL; Monitor* JfrMsg_lock = NULL; Mutex* JfrBuffer_lock = NULL; Mutex* JfrStream_lock = NULL; -Mutex* JfrThreadGroups_lock = NULL; #endif #ifndef SUPPORTS_NATIVE_CX8 @@ -280,7 +279,6 @@ void mutex_init() { #if INCLUDE_TRACE def(JfrMsg_lock , PaddedMonitor, leaf, true, Monitor::_safepoint_check_always); def(JfrBuffer_lock , PaddedMutex , leaf, true, Monitor::_safepoint_check_never); - def(JfrThreadGroups_lock , PaddedMutex , leaf, true, Monitor::_safepoint_check_always); def(JfrStream_lock , PaddedMutex , leaf+1, true, Monitor::_safepoint_check_never); // ensure to rank lower than 'safepoint' def(JfrStacktrace_lock , PaddedMutex , special, true, Monitor::_safepoint_check_sometimes); #endif diff --git a/src/hotspot/share/runtime/mutexLocker.hpp b/src/hotspot/share/runtime/mutexLocker.hpp index 7612059eff5..65e54459552 100644 --- a/src/hotspot/share/runtime/mutexLocker.hpp +++ b/src/hotspot/share/runtime/mutexLocker.hpp @@ -128,7 +128,6 @@ extern Mutex* JfrStacktrace_lock; // used to guard access to the extern Monitor* JfrMsg_lock; // protects JFR messaging extern Mutex* JfrBuffer_lock; // protects JFR buffer operations extern Mutex* JfrStream_lock; // protects JFR stream access -extern Mutex* JfrThreadGroups_lock; // protects JFR access to Thread Groups #endif #ifndef SUPPORTS_NATIVE_CX8 diff --git a/src/hotspot/share/utilities/growableArray.hpp b/src/hotspot/share/utilities/growableArray.hpp index beec6427ba5..f65f45cede7 100644 --- a/src/hotspot/share/utilities/growableArray.hpp +++ b/src/hotspot/share/utilities/growableArray.hpp @@ -402,7 +402,7 @@ template class GrowableArray : public GenericGrowableArray { // matching key according to the static compare function. Insert // that element is not already in the list. Assumes the list is // already sorted according to compare function. - template E insert_sorted(E& key) { + template E insert_sorted(const E& key) { bool found; int location = find_sorted(key, found); if (!found) { From 5a9432a16b4cb1755f17fe0fabcc5db6a57b18b1 Mon Sep 17 00:00:00 2001 From: Erik Gahlin Date: Sun, 5 Nov 2017 06:11:10 +0100 Subject: [PATCH 34/61] 8189440: Event tracing macros for allocation and weak oops processing Reviewed-by: dholmes, mgronlun --- src/hotspot/share/gc/shared/allocTracer.cpp | 7 +++++-- src/hotspot/share/gc/shared/allocTracer.hpp | 4 ++-- src/hotspot/share/gc/shared/collectedHeap.cpp | 2 +- src/hotspot/share/gc/shared/collectedHeap.inline.hpp | 2 +- src/hotspot/share/gc/shared/weakProcessor.cpp | 3 +++ src/hotspot/share/runtime/jniHandles.cpp | 1 + src/hotspot/share/trace/traceMacros.hpp | 2 ++ 7 files changed, 15 insertions(+), 6 deletions(-) diff --git a/src/hotspot/share/gc/shared/allocTracer.cpp b/src/hotspot/share/gc/shared/allocTracer.cpp index b7050110278..a1efeed104e 100644 --- a/src/hotspot/share/gc/shared/allocTracer.cpp +++ b/src/hotspot/share/gc/shared/allocTracer.cpp @@ -26,9 +26,11 @@ #include "gc/shared/allocTracer.hpp" #include "runtime/handles.hpp" #include "trace/tracing.hpp" +#include "trace/traceMacros.hpp" #include "utilities/globalDefinitions.hpp" -void AllocTracer::send_allocation_outside_tlab_event(Klass* klass, size_t alloc_size) { +void AllocTracer::send_allocation_outside_tlab(Klass* klass, HeapWord* obj, size_t alloc_size, Thread* thread) { + TRACE_ALLOCATION(obj, alloc_size, thread); EventObjectAllocationOutsideTLAB event; if (event.should_commit()) { event.set_objectClass(klass); @@ -37,7 +39,8 @@ void AllocTracer::send_allocation_outside_tlab_event(Klass* klass, size_t alloc_ } } -void AllocTracer::send_allocation_in_new_tlab_event(Klass* klass, size_t tlab_size, size_t alloc_size) { +void AllocTracer::send_allocation_in_new_tlab(Klass* klass, HeapWord* obj, size_t tlab_size, size_t alloc_size, Thread* thread) { + TRACE_ALLOCATION(obj, tlab_size, thread); EventObjectAllocationInNewTLAB event; if (event.should_commit()) { event.set_objectClass(klass); diff --git a/src/hotspot/share/gc/shared/allocTracer.hpp b/src/hotspot/share/gc/shared/allocTracer.hpp index 82a42325c29..593d4538f86 100644 --- a/src/hotspot/share/gc/shared/allocTracer.hpp +++ b/src/hotspot/share/gc/shared/allocTracer.hpp @@ -30,8 +30,8 @@ class AllocTracer : AllStatic { public: - static void send_allocation_outside_tlab_event(Klass* klass, size_t alloc_size); - static void send_allocation_in_new_tlab_event(Klass* klass, size_t tlab_size, size_t alloc_size); + static void send_allocation_outside_tlab(Klass* klass, HeapWord* obj, size_t alloc_size, Thread* thread); + static void send_allocation_in_new_tlab(Klass* klass, HeapWord* obj, size_t tlab_size, size_t alloc_size, Thread* thread); static void send_allocation_requiring_gc_event(size_t size, uint gcId); }; diff --git a/src/hotspot/share/gc/shared/collectedHeap.cpp b/src/hotspot/share/gc/shared/collectedHeap.cpp index b97ae63ca19..f5d93a400e6 100644 --- a/src/hotspot/share/gc/shared/collectedHeap.cpp +++ b/src/hotspot/share/gc/shared/collectedHeap.cpp @@ -313,7 +313,7 @@ HeapWord* CollectedHeap::allocate_from_tlab_slow(Klass* klass, Thread* thread, s return NULL; } - AllocTracer::send_allocation_in_new_tlab_event(klass, new_tlab_size * HeapWordSize, size * HeapWordSize); + AllocTracer::send_allocation_in_new_tlab(klass, obj, new_tlab_size * HeapWordSize, size * HeapWordSize, thread); if (ZeroTLAB) { // ..and clear it. diff --git a/src/hotspot/share/gc/shared/collectedHeap.inline.hpp b/src/hotspot/share/gc/shared/collectedHeap.inline.hpp index d776ad984a6..cb6cb96a819 100644 --- a/src/hotspot/share/gc/shared/collectedHeap.inline.hpp +++ b/src/hotspot/share/gc/shared/collectedHeap.inline.hpp @@ -156,7 +156,7 @@ HeapWord* CollectedHeap::common_mem_allocate_noinit(Klass* klass, size_t size, T "Unexpected exception, will result in uninitialized storage"); THREAD->incr_allocated_bytes(size * HeapWordSize); - AllocTracer::send_allocation_outside_tlab_event(klass, size * HeapWordSize); + AllocTracer::send_allocation_outside_tlab(klass, result, size * HeapWordSize, THREAD); return result; } diff --git a/src/hotspot/share/gc/shared/weakProcessor.cpp b/src/hotspot/share/gc/shared/weakProcessor.cpp index a04aada9e13..c230b14d549 100644 --- a/src/hotspot/share/gc/shared/weakProcessor.cpp +++ b/src/hotspot/share/gc/shared/weakProcessor.cpp @@ -26,10 +26,13 @@ #include "gc/shared/weakProcessor.hpp" #include "prims/jvmtiExport.hpp" #include "runtime/jniHandles.hpp" +#include "trace/tracing.hpp" +#include "trace/traceMacros.hpp" void WeakProcessor::weak_oops_do(BoolObjectClosure* is_alive, OopClosure* keep_alive) { JNIHandles::weak_oops_do(is_alive, keep_alive); JvmtiExport::weak_oops_do(is_alive, keep_alive); + TRACE_WEAK_OOPS_DO(is_alive, keep_alive); } void WeakProcessor::oops_do(OopClosure* closure) { diff --git a/src/hotspot/share/runtime/jniHandles.cpp b/src/hotspot/share/runtime/jniHandles.cpp index 2f5049fcd27..76f13e620da 100644 --- a/src/hotspot/share/runtime/jniHandles.cpp +++ b/src/hotspot/share/runtime/jniHandles.cpp @@ -30,6 +30,7 @@ #include "runtime/jniHandles.hpp" #include "runtime/mutexLocker.hpp" #include "runtime/thread.inline.hpp" +#include "trace/traceMacros.hpp" #include "utilities/align.hpp" #if INCLUDE_ALL_GCS #include "gc/g1/g1SATBCardTableModRefBS.hpp" diff --git a/src/hotspot/share/trace/traceMacros.hpp b/src/hotspot/share/trace/traceMacros.hpp index 33f38320e81..8a734a9964c 100644 --- a/src/hotspot/share/trace/traceMacros.hpp +++ b/src/hotspot/share/trace/traceMacros.hpp @@ -41,6 +41,8 @@ extern "C" void JNICALL trace_register_natives(JNIEnv*, jclass); #define TRACE_REGISTER_NATIVES ((void*)((address_word)(&trace_register_natives))) #define TRACE_START() JNI_OK #define TRACE_INITIALIZE() JNI_OK +#define TRACE_ALLOCATION(obj, size, thread) +#define TRACE_WEAK_OOPS_DO(is_alive, f) #define TRACE_VM_EXIT() #define TRACE_VM_ERROR() #define TRACE_SUSPEND_THREAD(t) From 8ce6d0239ab248c782ae623612667c75392d5c26 Mon Sep 17 00:00:00 2001 From: Doug Simon Date: Sun, 5 Nov 2017 22:27:57 +0100 Subject: [PATCH 35/61] 8177397: [JVMCI] remove unnecessary synchronization Reviewed-by: thartmann, kvn --- .../src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java index abb3e43998d..7447cd2850e 100644 --- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java +++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedObjectTypeImpl.java @@ -538,7 +538,7 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem return result; } - synchronized HotSpotResolvedJavaField createField(JavaType type, long offset, int rawFlags, int index) { + HotSpotResolvedJavaField createField(JavaType type, long offset, int rawFlags, int index) { return new HotSpotResolvedJavaFieldImpl(this, type, offset, rawFlags, index); } From 168565de206c78271b56d905f379c4251d07e65a Mon Sep 17 00:00:00 2001 From: Dmitry Chuyko Date: Mon, 6 Nov 2017 00:29:08 +0300 Subject: [PATCH 36/61] 8190745: AARCH64: fix for JDK-8189176 may break a build Add missed semicolon Reviewed-by: dsamersoff --- src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp index 646aa3e7ebd..792cc2a5176 100644 --- a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp @@ -2942,7 +2942,7 @@ void MacroAssembler::kernel_crc32_using_crc32(Register crc, Register buf, BIND(CRC_less64); adds(len, len, 128-32); br(Assembler::GE, CRC_by32_loop); - BIND(CRC_less32) + BIND(CRC_less32); adds(len, len, 32-4); br(Assembler::GE, CRC_by4_loop); adds(len, len, 4); From ef98509665e602b0b81886e1e3a8011cb57ef3f2 Mon Sep 17 00:00:00 2001 From: Jamsheed Mohammed C M Date: Mon, 6 Nov 2017 00:30:36 -0800 Subject: [PATCH 37/61] 8167408: Invalid critical JNI function lookup Made correction to arg_size calculation in NativeLookup::lookup_critical_entry Reviewed-by: dholmes, dlong, mdoerr, vlivanov --- make/test/JtregNativeHotspot.gmk | 1 + src/hotspot/share/prims/nativeLookup.cpp | 10 ++-- .../criticalnatives/lookup/LookUp.java | 59 +++++++++++++++++++ .../criticalnatives/lookup/libCNLookUp.c | 35 +++++++++++ 4 files changed, 101 insertions(+), 4 deletions(-) create mode 100644 test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup/LookUp.java create mode 100644 test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup/libCNLookUp.c diff --git a/make/test/JtregNativeHotspot.gmk b/make/test/JtregNativeHotspot.gmk index 504c3b2d9b1..a9effa53d03 100644 --- a/make/test/JtregNativeHotspot.gmk +++ b/make/test/JtregNativeHotspot.gmk @@ -63,6 +63,7 @@ BUILD_HOTSPOT_JTREG_NATIVE_SRC += \ $(TOPDIR)/test/hotspot/jtreg/runtime/RedefineTests \ $(TOPDIR)/test/hotspot/jtreg/compiler/floatingpoint/ \ $(TOPDIR)/test/hotspot/jtreg/compiler/calls \ + $(TOPDIR)/test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup \ $(TOPDIR)/test/hotspot/jtreg/serviceability/jvmti/GetOwnedMonitorInfo \ $(TOPDIR)/test/hotspot/jtreg/serviceability/jvmti/GetNamedModule \ $(TOPDIR)/test/hotspot/jtreg/serviceability/jvmti/IsModifiableModule \ diff --git a/src/hotspot/share/prims/nativeLookup.cpp b/src/hotspot/share/prims/nativeLookup.cpp index dfe88062887..c840b388ef0 100644 --- a/src/hotspot/share/prims/nativeLookup.cpp +++ b/src/hotspot/share/prims/nativeLookup.cpp @@ -293,10 +293,12 @@ address NativeLookup::lookup_critical_entry(const methodHandle& method) { char* critical_name = critical_jni_name(method); // Compute argument size - int args_size = 1 // JNIEnv - + (method->is_static() ? 1 : 0) // class for static methods - + method->size_of_parameters(); // actual parameters - + int args_size = method->size_of_parameters(); + for (SignatureStream ss(signature); !ss.at_return_type(); ss.next()) { + if (ss.is_array()) { + args_size += T_INT_size; // array length parameter + } + } // 1) Try JNI short style entry = lookup_critical_style(method, critical_name, "", args_size, true); diff --git a/test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup/LookUp.java b/test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup/LookUp.java new file mode 100644 index 00000000000..17c9a0f9b90 --- /dev/null +++ b/test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup/LookUp.java @@ -0,0 +1,59 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + + +/* @test + * @bug 8167408 + * @run main/othervm/native -Xcomp compiler.runtime.criticalnatives.lookup.LookUp + */ +package compiler.runtime.criticalnatives.lookup; +public class LookUp { + static { + System.loadLibrary("CNLookUp"); + } + static native void m1(byte a1, long a2, char a3, int a4, float a5, double a6, byte[] result); + static native void m2(int a1, int[] a2, long a3, long[] a4, float a5,float[] a6, double a7, double[] a8, byte result[]); + public static void main(String args[]) throws Exception { + test(); + } + private static void test() throws Exception { + int[] l1 = { 1111, 2222, 3333 }; + long[] l2 = { 4444L, 5555L, 6666L }; + float[] l3 = { 7777.0F, 8888.0F, 9999.0F }; + double[] l4 = { 4545.0D, 5656.0D, 6767.0D }; + byte[] result = { -1 }; + m1((byte)0xA, 4444444455555555L, 'A', 12345678, 343434.0F, 6666666677777777.0D, result); + check(result[0]); + result[0] = -1; + m2(12345678, l1, 4444444455555555L, l2, 343434.0F, l3, 6666666677777777.0D, l4, result); + check(result[0]); + } + private static void check(byte result) throws Exception { + if (result != 2) { + if (result == 1) { + throw new Exception("critical native arguments mismatch"); + } + throw new Exception("critical native lookup failed"); + } + } +} diff --git a/test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup/libCNLookUp.c b/test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup/libCNLookUp.c new file mode 100644 index 00000000000..6111f14c5aa --- /dev/null +++ b/test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup/libCNLookUp.c @@ -0,0 +1,35 @@ +#include "jni.h" +JNIEXPORT void JNICALL JavaCritical_compiler_runtime_criticalnatives_lookup_LookUp_m1 + (jbyte a1, jlong a2, jchar a3, jint a4, jfloat a5, jdouble a6, jint result_length, jbyte* result) { + jint l1 = (jint) a5; + jlong l2 = (jlong) a6; + + if (a1 != 0xA || a2 != 4444444455555555LL || a3 != 0x41 || a4 != 12345678 || l1 != 343434 || l2 != 6666666677777777LL || + result_length != 1 || result[0] != -1) { + result[0] = 1; + } else { + result[0] = 2; + } +} + +JNIEXPORT void JNICALL JavaCritical_compiler_runtime_criticalnatives_lookup_LookUp_m2 + (jint a1, jint a2_length, jint* a2, jlong a3, jint a4_length, jlong* a4, jfloat a5, jint a6_length, jfloat* a6, jdouble a7, + jint a8_length, jdouble* a8, jint result_length, jbyte* result) { + jint l1 = (jint) a5; + jlong l2 = (jlong) a7; + + if (a1 != 12345678 || a2_length != 3 || a2[0] != 1111 || a3 != 4444444455555555LL || a4_length != 3 || a4[0] != 4444 || + l1 != 343434 || a6_length != 3 || 7777 != (jint)a6[0] || l2 != 6666666677777777LL || a8_length != 3 || 4545 != (jlong)a8[0] || + result_length != 1 || result[0] != -1) { + result[0] = 1; + } else { + result[0] = 2; + } +} + +JNIEXPORT void JNICALL Java_compiler_runtime_criticalnatives_lookup_LookUp_m1 + (JNIEnv * env, jclass jclazz, jbyte a3, jlong a4, jchar a5, jint a6, jfloat a7, jdouble a8, jbyteArray result) {} + +JNIEXPORT void JNICALL Java_compiler_runtime_criticalnatives_lookup_LookUp_m2 + (JNIEnv * env, jclass jclazz, jint a3, jintArray a4, jlong a5, jlongArray a6, jfloat a7, jfloatArray a8, jdouble a9, jdoubleArray a10, jbyteArray result) {} + From 16963a0d5cd51a9b883d863104ecea80d24faccf Mon Sep 17 00:00:00 2001 From: Stefan Johansson Date: Mon, 6 Nov 2017 11:11:44 +0100 Subject: [PATCH 38/61] 8190353: [Testbug] test/hotspot/jtreg/gc/logging/TestPrintReferences.java can still fail Reviewed-by: sangheki, eosterlund --- .../jtreg/gc/logging/TestPrintReferences.java | 24 ++++++++++++------- 1 file changed, 15 insertions(+), 9 deletions(-) diff --git a/test/hotspot/jtreg/gc/logging/TestPrintReferences.java b/test/hotspot/jtreg/gc/logging/TestPrintReferences.java index 864163354a7..b5437cadc3d 100644 --- a/test/hotspot/jtreg/gc/logging/TestPrintReferences.java +++ b/test/hotspot/jtreg/gc/logging/TestPrintReferences.java @@ -137,15 +137,21 @@ public class TestPrintReferences { // Actual value: SoftReference(5.55) = phase1(1.85) + phase2(1.85) + phase3(1.85) // Log value: SoftReference(5.6) = phase1(1.9) + phase2(1.9) + phase3(1.9) // When checked: 5.6 < 5.7 (sum of phase1~3) - public static boolean approximatelyEqual(BigDecimal phaseTime, BigDecimal sumOfSubPhasesTime, BigDecimal tolerance) { - BigDecimal abs = phaseTime.subtract(sumOfSubPhasesTime).abs(); + // Because of this we need method to verify that our measurements and calculations are valid. + public static boolean greaterThanOrApproximatelyEqual(BigDecimal phaseTime, BigDecimal sumOfSubPhasesTime, BigDecimal tolerance) { + if (phaseTime.compareTo(sumOfSubPhasesTime) >= 0) { + // phaseTime is greater than or equal. + return true; + } - int result = abs.compareTo(tolerance); + BigDecimal diff = sumOfSubPhasesTime.subtract(phaseTime); + if (diff.compareTo(tolerance) <= 0) { + // Difference is within tolerance, so approximately equal. + return true; + } - // result == -1, abs is less than tolerance. - // result == 0, abs is equal to tolerance. - // result == 1, abs is greater than tolerance. - return (result != 1); + // sumOfSubPhasesTime is greater than phaseTime and not within tolerance. + return false; } public static BigDecimal checkPhaseTime(String refType) { @@ -160,7 +166,7 @@ public class TestPrintReferences { // If there are 3 sub-phases, we should allow 0.1 tolerance. final BigDecimal toleranceFor3SubPhases = BigDecimal.valueOf(0.1); - if (!approximatelyEqual(phaseTime, sumOfSubPhasesTime, toleranceFor3SubPhases)) { + if (!greaterThanOrApproximatelyEqual(phaseTime, sumOfSubPhasesTime, toleranceFor3SubPhases)) { throw new RuntimeException(refType +" time(" + phaseTime + "ms) is less than the sum(" + sumOfSubPhasesTime + "ms) of each phases"); } @@ -181,7 +187,7 @@ public class TestPrintReferences { // If there are 4 sub-phases, we should allow 0.2 tolerance. final BigDecimal toleranceFor4SubPhases = BigDecimal.valueOf(0.2); - if (!approximatelyEqual(refProcTime, sumOfSubPhasesTime, toleranceFor4SubPhases)) { + if (!greaterThanOrApproximatelyEqual(refProcTime, sumOfSubPhasesTime, toleranceFor4SubPhases)) { throw new RuntimeException("Reference Processing time(" + refProcTime + "ms) is less than the sum(" + sumOfSubPhasesTime + "ms) of each phases"); } From 18e9e806885109c785ea0b82c0f78ce0767ecc6c Mon Sep 17 00:00:00 2001 From: Doug Simon Date: Mon, 6 Nov 2017 09:44:42 +0100 Subject: [PATCH 39/61] 8190415: [JVMCI] JVMCIRuntime::adjust_comp_level must not swallow ThreadDeath Reviewed-by: never, thartmann --- src/hotspot/.mx.jvmci/mx_jvmci.py | 21 +++++++----------- src/hotspot/share/jvmci/jvmciRuntime.cpp | 27 ++++++++++++++++-------- 2 files changed, 26 insertions(+), 22 deletions(-) diff --git a/src/hotspot/.mx.jvmci/mx_jvmci.py b/src/hotspot/.mx.jvmci/mx_jvmci.py index 5b582b3d45d..98f4f6e3485 100644 --- a/src/hotspot/.mx.jvmci/mx_jvmci.py +++ b/src/hotspot/.mx.jvmci/mx_jvmci.py @@ -42,11 +42,6 @@ _suite = mx.suite('jvmci') JVMCI_VERSION = 9 -""" -Top level directory of the JDK source workspace. -""" -_jdkSourceRoot = dirname(_suite.dir) - _JVMCI_JDK_TAG = 'jvmci' _minVersion = mx.VersionSpec('1.9') @@ -145,7 +140,7 @@ def isJVMCIEnabled(vm): return True def _makehelp(): - return subprocess.check_output([mx.gmake_cmd(), 'help'], cwd=_jdkSourceRoot) + return subprocess.check_output([mx.gmake_cmd(), 'help'], cwd=_get_jdk_dir()) def _runmake(args): """run the JDK make process @@ -155,12 +150,12 @@ To build hotspot and import it into the JDK: "mx make hotspot import-hotspot" jdkBuildDir = _get_jdk_build_dir() if not exists(jdkBuildDir): - # JDK9 must be bootstrapped with a JDK8 - compliance = mx.JavaCompliance('8') - jdk8 = mx.get_jdk(compliance.exactMatch, versionDescription=compliance.value) + # JDK10 must be bootstrapped with a JDK9 + compliance = mx.JavaCompliance('9') + jdk9 = mx.get_jdk(compliance.exactMatch, versionDescription=compliance.value) cmd = ['sh', 'configure', '--with-debug-level=' + _vm.debugLevel, '--with-native-debug-symbols=external', '--disable-precompiled-headers', '--with-jvm-features=graal', - '--with-jvm-variants=' + _vm.jvmVariant, '--disable-warnings-as-errors', '--with-boot-jdk=' + jdk8.home, '--with-jvm-features=graal'] - mx.run(cmd, cwd=_jdkSourceRoot) + '--with-jvm-variants=' + _vm.jvmVariant, '--disable-warnings-as-errors', '--with-boot-jdk=' + jdk9.home, '--with-jvm-features=graal'] + mx.run(cmd, cwd=_get_jdk_dir()) cmd = [mx.gmake_cmd(), 'CONF=' + _vm.debugLevel] if mx.get_opts().verbose: cmd.append('LOG=debug') @@ -170,11 +165,11 @@ To build hotspot and import it into the JDK: "mx make hotspot import-hotspot" if not mx.get_opts().verbose: mx.log('--------------- make execution ----------------------') - mx.log('Working directory: ' + _jdkSourceRoot) + mx.log('Working directory: ' + _get_jdk_dir()) mx.log('Command line: ' + ' '.join(cmd)) mx.log('-----------------------------------------------------') - mx.run(cmd, cwd=_jdkSourceRoot) + mx.run(cmd, cwd=_get_jdk_dir()) def _runmultimake(args): """run the JDK make process for one or more configurations""" diff --git a/src/hotspot/share/jvmci/jvmciRuntime.cpp b/src/hotspot/share/jvmci/jvmciRuntime.cpp index e64551e08a7..2f9678d1119 100644 --- a/src/hotspot/share/jvmci/jvmciRuntime.cpp +++ b/src/hotspot/share/jvmci/jvmciRuntime.cpp @@ -825,24 +825,33 @@ CompLevel JVMCIRuntime::adjust_comp_level_inner(const methodHandle& method, bool if (compiler != NULL && compiler->is_bootstrapping()) { return level; } - if (!is_HotSpotJVMCIRuntime_initialized() || !_comp_level_adjustment) { + if (!is_HotSpotJVMCIRuntime_initialized() || _comp_level_adjustment == JVMCIRuntime::none) { // JVMCI cannot participate in compilation scheduling until // JVMCI is initialized and indicates it wants to participate. return level; } #define CHECK_RETURN THREAD); \ -if (HAS_PENDING_EXCEPTION) { \ - Handle exception(THREAD, PENDING_EXCEPTION); \ - CLEAR_PENDING_EXCEPTION; \ -\ - java_lang_Throwable::java_printStackTrace(exception, THREAD); \ if (HAS_PENDING_EXCEPTION) { \ + Handle exception(THREAD, PENDING_EXCEPTION); \ CLEAR_PENDING_EXCEPTION; \ + \ + if (exception->is_a(SystemDictionary::ThreadDeath_klass())) { \ + /* In the special case of ThreadDeath, we need to reset the */ \ + /* pending async exception so that it is propagated. */ \ + thread->set_pending_async_exception(exception()); \ + return level; \ + } \ + tty->print("Uncaught exception while adjusting compilation level: "); \ + java_lang_Throwable::print(exception(), tty); \ + tty->cr(); \ + java_lang_Throwable::print_stack_trace(exception, tty); \ + if (HAS_PENDING_EXCEPTION) { \ + CLEAR_PENDING_EXCEPTION; \ + } \ + return level; \ } \ - return level; \ -} \ -(void)(0 + (void)(0 Thread* THREAD = thread; From c2935fa4a4a59898ff652844b9ee4b7729b9809a Mon Sep 17 00:00:00 2001 From: Leo Korinth Date: Mon, 6 Nov 2017 09:41:54 +0100 Subject: [PATCH 40/61] 8186502: Assert when range testing G1RefProcDrainInterval on 64-bit systems Reviewed-by: sjohanss, sangheki --- src/hotspot/share/gc/g1/g1_globals.hpp | 6 +++--- .../common/optionsvalidation/JVMOptionsUtils.java | 3 +++ 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/src/hotspot/share/gc/g1/g1_globals.hpp b/src/hotspot/share/gc/g1/g1_globals.hpp index d69c1712433..f8b2a8e2fe0 100644 --- a/src/hotspot/share/gc/g1/g1_globals.hpp +++ b/src/hotspot/share/gc/g1/g1_globals.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2017, 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 @@ -74,10 +74,10 @@ "in milliseconds.") \ range(1.0, DBL_MAX) \ \ - product(intx, G1RefProcDrainInterval, 10, \ + product(int, G1RefProcDrainInterval, 10, \ "The number of discovered reference objects to process before " \ "draining concurrent marking work queues.") \ - range(1, max_intx) \ + range(1, INT_MAX) \ \ experimental(double, G1LastPLABAverageOccupancy, 50.0, \ "The expected average occupancy of the last PLAB in " \ diff --git a/test/hotspot/jtreg/runtime/CommandLine/OptionsValidation/common/optionsvalidation/JVMOptionsUtils.java b/test/hotspot/jtreg/runtime/CommandLine/OptionsValidation/common/optionsvalidation/JVMOptionsUtils.java index 336e115b4b9..63531bbf4e3 100644 --- a/test/hotspot/jtreg/runtime/CommandLine/OptionsValidation/common/optionsvalidation/JVMOptionsUtils.java +++ b/test/hotspot/jtreg/runtime/CommandLine/OptionsValidation/common/optionsvalidation/JVMOptionsUtils.java @@ -216,6 +216,9 @@ public class JVMOptionsUtils { case "CMSPrecleanDenominator": option.addPrepend("-XX:CMSPrecleanNumerator=" + ((new Integer(option.getMin())) - 1)); break; + case "G1RefProcDrainInterval": + option.addPrepend("-XX:+ExplicitGCInvokesConcurrent"); + break; case "InitialTenuringThreshold": option.addPrepend("-XX:MaxTenuringThreshold=" + option.getMax()); break; From 34deb9cc2afbcdbed7aa2275afb441434b0d8497 Mon Sep 17 00:00:00 2001 From: Jamsheed Mohammed C M Date: Mon, 6 Nov 2017 04:14:22 -0800 Subject: [PATCH 41/61] 8167409: Invalid value passed to critical JNI function The tmp reg used to break the shuffling cycle (handled in ComputeMoveOrder) is set to 64 bit. Reviewed-by: dlong --- make/test/JtregNativeHotspot.gmk | 1 + src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp | 2 +- .../argumentcorruption/CheckLongArgs.java | 60 +++++++++++++++++++ .../argumentcorruption/libCNCheckLongArgs.c | 30 ++++++++++ 4 files changed, 92 insertions(+), 1 deletion(-) create mode 100644 test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption/CheckLongArgs.java create mode 100644 test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption/libCNCheckLongArgs.c diff --git a/make/test/JtregNativeHotspot.gmk b/make/test/JtregNativeHotspot.gmk index a9effa53d03..9f5c979c6df 100644 --- a/make/test/JtregNativeHotspot.gmk +++ b/make/test/JtregNativeHotspot.gmk @@ -64,6 +64,7 @@ BUILD_HOTSPOT_JTREG_NATIVE_SRC += \ $(TOPDIR)/test/hotspot/jtreg/compiler/floatingpoint/ \ $(TOPDIR)/test/hotspot/jtreg/compiler/calls \ $(TOPDIR)/test/hotspot/jtreg/compiler/runtime/criticalnatives/lookup \ + $(TOPDIR)/test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption \ $(TOPDIR)/test/hotspot/jtreg/serviceability/jvmti/GetOwnedMonitorInfo \ $(TOPDIR)/test/hotspot/jtreg/serviceability/jvmti/GetNamedModule \ $(TOPDIR)/test/hotspot/jtreg/serviceability/jvmti/IsModifiableModule \ diff --git a/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp b/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp index 12f5f76a451..996ae27cbc9 100644 --- a/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp +++ b/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp @@ -2183,7 +2183,7 @@ nmethod* SharedRuntime::generate_native_wrapper(MacroAssembler* masm, // critical natives they are offset down. GrowableArray arg_order(2 * total_in_args); VMRegPair tmp_vmreg; - tmp_vmreg.set1(rbx->as_VMReg()); + tmp_vmreg.set2(rbx->as_VMReg()); if (!is_critical_native) { for (int i = total_in_args - 1, c_arg = total_c_args - 1; i >= 0; i--, c_arg--) { diff --git a/test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption/CheckLongArgs.java b/test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption/CheckLongArgs.java new file mode 100644 index 00000000000..cbe0235c4e0 --- /dev/null +++ b/test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption/CheckLongArgs.java @@ -0,0 +1,60 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + + +/* @test + * @bug 8167409 + * @run main/othervm/native -Xcomp compiler.runtime.criticalnatives.argumentcorruption.CheckLongArgs + */ +package compiler.runtime.criticalnatives.argumentcorruption; +public class CheckLongArgs { + static { + System.loadLibrary("CNCheckLongArgs"); + } + static native void m1(long a1, long a2, long a3, long a4, long a5, long a6, long a7, long a8, byte[] result); + static native void m2(long a1, int[] a2, long a3, int[] a4, long a5, int[] a6, long a7, int[] a8, long a9, byte[] result); + public static void main(String args[]) throws Exception { + test(); + } + private static void test() throws Exception { + int[] l1 = { 1111, 2222, 3333 }; + int[] l2 = { 4444, 5555, 6666 }; + int[] l3 = { 7777, 8888, 9999 }; + int[] l4 = { 1010, 2020, 3030 }; + byte[] result = { -1 }; + m1(1111111122222222L, 3333333344444444L, 5555555566666666L, 7777777788888888L, 9999999900000000L, 1212121234343434L, + 5656565678787878L, 9090909012121212L, result); + check(result[0]); + result[0] = -1; + m2(1111111122222222L, l1, 3333333344444444L, l2, 5555555566666666L, l3, 7777777788888888L, l4, 9999999900000000L, result); + check(result[0]); + } + private static void check(byte result) throws Exception { + if (result != 2) { + if (result == 1) { + throw new Exception("critical native arguments mismatch"); + } + throw new Exception("critical native lookup failed"); + } + } +} diff --git a/test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption/libCNCheckLongArgs.c b/test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption/libCNCheckLongArgs.c new file mode 100644 index 00000000000..c805d75af38 --- /dev/null +++ b/test/hotspot/jtreg/compiler/runtime/criticalnatives/argumentcorruption/libCNCheckLongArgs.c @@ -0,0 +1,30 @@ +#include "jni.h" +JNIEXPORT void JNICALL JavaCritical_compiler_runtime_criticalnatives_argumentcorruption_CheckLongArgs_m1 + (jlong a1, jlong a2, jlong a3, jlong a4, jlong a5, jlong a6, jlong a7, jlong a8,jint result_length,jbyte* result) { + + if (a1 != 1111111122222222LL || a2 != 3333333344444444LL || a3 != 5555555566666666LL || a4 != 7777777788888888LL || + a5 != 9999999900000000LL || a6 != 1212121234343434LL || a7 != 5656565678787878LL || a8 != 9090909012121212LL || + result_length != 1 || result[0] != -1) { + result[0] = 1; + } else { + result[0] = 2; + } +} + +JNIEXPORT void JNICALL JavaCritical_compiler_runtime_criticalnatives_argumentcorruption_CheckLongArgs_m2 + (jlong a1, jint a2_length, jint* a2, jlong a3, jint a4_length, jint* a4, jlong a5, jint a6_length, jint* a6, jlong a7, + jint a8_length, jint* a8, jlong a9, jint result_length, jbyte* result) { + if (a1 != 1111111122222222LL || a2_length != 3 || a2[0] != 1111 || a3 != 3333333344444444LL || a4_length != 3 || a4[0] != 4444 || + a5 != 5555555566666666LL || a6_length != 3 || a6[0] != 7777 || a7 != 7777777788888888LL || a8_length != 3 || a8[0] != 1010 || a9 != 9999999900000000LL || + result_length != 1 || result[0] != -1) { + result[0] = 1; + } else { + result[0] = 2; + } +} + +JNIEXPORT void JNICALL Java_compiler_runtime_criticalnatives_argumentcorruption_CheckLongArgs_m1 + (JNIEnv * env, jclass jclazz, jlong a3, jlong a4, jlong a5, jlong a6, jlong a7, jlong a8, jlong a9, jlong a10, jbyteArray result) {} + +JNIEXPORT void JNICALL Java_compiler_runtime_criticalnatives_argumentcorruption_CheckLongArgs_m2 + (JNIEnv * env, jclass jclazz, jlong a3, jintArray a4, jlong a5, jintArray a6, jlong a7, jintArray a8, jlong a9, jintArray a10, jlong a11, jbyteArray result) {} From b59c920e12599eba2ec1bd9143e1feeaba1c90cb Mon Sep 17 00:00:00 2001 From: Thomas Schatzl Date: Mon, 6 Nov 2017 14:24:31 +0100 Subject: [PATCH 42/61] 8149127: Rename g1/concurrentMarkThread.* to g1/g1ConcurrentMarkThread.* Reviewed-by: sjohanss, sangheki --- src/hotspot/share/gc/g1/g1CollectedHeap.cpp | 18 ++--- src/hotspot/share/gc/g1/g1CollectedHeap.hpp | 6 +- ...entG1Refine.cpp => g1ConcurrentRefine.cpp} | 72 +++++++++---------- ...entG1Refine.hpp => g1ConcurrentRefine.hpp} | 22 +++--- ...hread.cpp => g1ConcurrentRefineThread.cpp} | 32 +++++---- ...hread.hpp => g1ConcurrentRefineThread.hpp} | 20 +++--- src/hotspot/share/gc/g1/g1DefaultPolicy.cpp | 4 +- src/hotspot/share/gc/g1/g1RemSet.cpp | 4 +- src/hotspot/share/gc/g1/g1RemSetSummary.cpp | 16 ++--- src/hotspot/share/gc/g1/heapRegionManager.cpp | 2 +- src/hotspot/share/gc/g1/heapRegionRemSet.cpp | 2 +- 11 files changed, 100 insertions(+), 98 deletions(-) rename src/hotspot/share/gc/g1/{concurrentG1Refine.cpp => g1ConcurrentRefine.cpp} (86%) rename src/hotspot/share/gc/g1/{concurrentG1Refine.hpp => g1ConcurrentRefine.hpp} (87%) rename src/hotspot/share/gc/g1/{concurrentG1RefineThread.cpp => g1ConcurrentRefineThread.cpp} (85%) rename src/hotspot/share/gc/g1/{concurrentG1RefineThread.hpp => g1ConcurrentRefineThread.hpp} (82%) diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp index e6b8e91190e..150c3a655ac 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp @@ -29,14 +29,14 @@ #include "code/codeCache.hpp" #include "code/icBuffer.hpp" #include "gc/g1/bufferingOopClosure.hpp" -#include "gc/g1/concurrentG1Refine.hpp" -#include "gc/g1/concurrentG1RefineThread.hpp" #include "gc/g1/concurrentMarkThread.inline.hpp" #include "gc/g1/g1Allocator.inline.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp" #include "gc/g1/g1CollectionSet.hpp" #include "gc/g1/g1CollectorPolicy.hpp" #include "gc/g1/g1CollectorState.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" +#include "gc/g1/g1ConcurrentRefineThread.hpp" #include "gc/g1/g1EvacStats.inline.hpp" #include "gc/g1/g1FullGCScope.hpp" #include "gc/g1/g1GCPhaseTimes.hpp" @@ -1554,7 +1554,7 @@ G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* collector_policy) : _bot(NULL), _hot_card_cache(NULL), _g1_rem_set(NULL), - _cg1r(NULL), + _cr(NULL), _g1mm(NULL), _preserved_marks_set(true /* in_c_heap */), _secondary_free_list("Secondary Free List", new SecondaryFreeRegionListMtSafeChecker()), @@ -1633,7 +1633,7 @@ G1RegionToSpaceMapper* G1CollectedHeap::create_aux_memory_mapper(const char* des jint G1CollectedHeap::initialize_concurrent_refinement() { jint ecode = JNI_OK; - _cg1r = ConcurrentG1Refine::create(&ecode); + _cr = G1ConcurrentRefine::create(&ecode); return ecode; } @@ -1791,8 +1791,8 @@ jint G1CollectedHeap::initialize() { JavaThread::dirty_card_queue_set().initialize(DirtyCardQ_CBL_mon, DirtyCardQ_FL_lock, - (int)concurrent_g1_refine()->yellow_zone(), - (int)concurrent_g1_refine()->red_zone(), + (int)concurrent_refine()->yellow_zone(), + (int)concurrent_refine()->red_zone(), Shared_DirtyCardQ_lock, NULL, // fl_owner true); // init_free_ids @@ -1836,7 +1836,7 @@ void G1CollectedHeap::stop() { // Stop all concurrent threads. We do this to make sure these threads // do not continue to execute and access resources (e.g. logging) // that are destroyed during shutdown. - _cg1r->stop(); + _cr->stop(); _cmThread->stop(); if (G1StringDedup::is_enabled()) { G1StringDedup::stop(); @@ -2436,7 +2436,7 @@ void G1CollectedHeap::print_gc_threads_on(outputStream* st) const { _cmThread->print_on(st); st->cr(); _cm->print_worker_threads_on(st); - _cg1r->print_worker_threads_on(st); // also prints the sample thread + _cr->print_worker_threads_on(st); // also prints the sample thread if (G1StringDedup::is_enabled()) { G1StringDedup::print_worker_threads_on(st); } @@ -2446,7 +2446,7 @@ void G1CollectedHeap::gc_threads_do(ThreadClosure* tc) const { workers()->threads_do(tc); tc->do_thread(_cmThread); _cm->threads_do(tc); - _cg1r->threads_do(tc); // also iterates over the sample thread + _cr->threads_do(tc); // also iterates over the sample thread if (G1StringDedup::is_enabled()) { G1StringDedup::threads_do(tc); } diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.hpp b/src/hotspot/share/gc/g1/g1CollectedHeap.hpp index fd014af03e8..fc7ea3e9183 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.hpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.hpp @@ -76,7 +76,7 @@ class G1RemSet; class HeapRegionRemSetIterator; class G1ConcurrentMark; class ConcurrentMarkThread; -class ConcurrentG1Refine; +class G1ConcurrentRefine; class GenerationCounters; class STWGCTimer; class G1NewTracer; @@ -806,7 +806,7 @@ protected: ConcurrentMarkThread* _cmThread; // The concurrent refiner. - ConcurrentG1Refine* _cg1r; + G1ConcurrentRefine* _cr; // The parallel task queues RefToScanQueueSet *_task_queues; @@ -1389,7 +1389,7 @@ public: // Refinement - ConcurrentG1Refine* concurrent_g1_refine() const { return _cg1r; } + G1ConcurrentRefine* concurrent_refine() const { return _cr; } // Optimized nmethod scanning support routines diff --git a/src/hotspot/share/gc/g1/concurrentG1Refine.cpp b/src/hotspot/share/gc/g1/g1ConcurrentRefine.cpp similarity index 86% rename from src/hotspot/share/gc/g1/concurrentG1Refine.cpp rename to src/hotspot/share/gc/g1/g1ConcurrentRefine.cpp index 056c0b1a424..ac1271067c8 100644 --- a/src/hotspot/share/gc/g1/concurrentG1Refine.cpp +++ b/src/hotspot/share/gc/g1/g1ConcurrentRefine.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2017, 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 @@ -23,8 +23,8 @@ */ #include "precompiled.hpp" -#include "gc/g1/concurrentG1Refine.hpp" -#include "gc/g1/concurrentG1RefineThread.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" +#include "gc/g1/g1ConcurrentRefineThread.hpp" #include "gc/g1/g1YoungRemSetSamplingThread.hpp" #include "logging/log.hpp" #include "runtime/java.hpp" @@ -97,7 +97,7 @@ static Thresholds calc_thresholds(size_t green_zone, size_t yellow_zone, uint worker_i) { double yellow_size = yellow_zone - green_zone; - double step = yellow_size / ConcurrentG1Refine::thread_num(); + double step = yellow_size / G1ConcurrentRefine::thread_num(); if (worker_i == 0) { // Potentially activate worker 0 more aggressively, to keep // available buffers near green_zone value. When yellow_size is @@ -112,7 +112,7 @@ static Thresholds calc_thresholds(size_t green_zone, green_zone + deactivate_offset); } -ConcurrentG1Refine::ConcurrentG1Refine(size_t green_zone, +G1ConcurrentRefine::G1ConcurrentRefine(size_t green_zone, size_t yellow_zone, size_t red_zone, size_t min_yellow_zone_size) : @@ -129,7 +129,7 @@ ConcurrentG1Refine::ConcurrentG1Refine(size_t green_zone, static size_t calc_min_yellow_zone_size() { size_t step = G1ConcRefinementThresholdStep; - uint n_workers = ConcurrentG1Refine::thread_num(); + uint n_workers = G1ConcurrentRefine::thread_num(); if ((max_yellow_zone / step) < n_workers) { return max_yellow_zone; } else { @@ -169,7 +169,7 @@ static size_t calc_init_red_zone(size_t green, size_t yellow) { return MIN2(yellow + size, max_red_zone); } -ConcurrentG1Refine* ConcurrentG1Refine::create(jint* ecode) { +G1ConcurrentRefine* G1ConcurrentRefine::create(jint* ecode) { size_t min_yellow_zone_size = calc_min_yellow_zone_size(); size_t green_zone = calc_init_green_zone(); size_t yellow_zone = calc_init_yellow_zone(green_zone, min_yellow_zone_size); @@ -182,31 +182,31 @@ ConcurrentG1Refine* ConcurrentG1Refine::create(jint* ecode) { "min yellow size: " SIZE_FORMAT, green_zone, yellow_zone, red_zone, min_yellow_zone_size); - ConcurrentG1Refine* cg1r = new ConcurrentG1Refine(green_zone, - yellow_zone, - red_zone, - min_yellow_zone_size); + G1ConcurrentRefine* cr = new G1ConcurrentRefine(green_zone, + yellow_zone, + red_zone, + min_yellow_zone_size); - if (cg1r == NULL) { + if (cr == NULL) { *ecode = JNI_ENOMEM; - vm_shutdown_during_initialization("Could not create ConcurrentG1Refine"); + vm_shutdown_during_initialization("Could not create G1ConcurrentRefine"); return NULL; } - cg1r->_threads = NEW_C_HEAP_ARRAY_RETURN_NULL(ConcurrentG1RefineThread*, cg1r->_n_worker_threads, mtGC); - if (cg1r->_threads == NULL) { + cr->_threads = NEW_C_HEAP_ARRAY_RETURN_NULL(G1ConcurrentRefineThread*, cr->_n_worker_threads, mtGC); + if (cr->_threads == NULL) { *ecode = JNI_ENOMEM; - vm_shutdown_during_initialization("Could not allocate an array for ConcurrentG1RefineThread"); + vm_shutdown_during_initialization("Could not allocate an array for G1ConcurrentRefineThread"); return NULL; } uint worker_id_offset = DirtyCardQueueSet::num_par_ids(); - ConcurrentG1RefineThread *next = NULL; - for (uint i = cg1r->_n_worker_threads - 1; i != UINT_MAX; i--) { + G1ConcurrentRefineThread *next = NULL; + for (uint i = cr->_n_worker_threads - 1; i != UINT_MAX; i--) { Thresholds thresholds = calc_thresholds(green_zone, yellow_zone, i); - ConcurrentG1RefineThread* t = - new ConcurrentG1RefineThread(cg1r, + G1ConcurrentRefineThread* t = + new G1ConcurrentRefineThread(cr, next, worker_id_offset, i, @@ -215,34 +215,34 @@ ConcurrentG1Refine* ConcurrentG1Refine::create(jint* ecode) { assert(t != NULL, "Conc refine should have been created"); if (t->osthread() == NULL) { *ecode = JNI_ENOMEM; - vm_shutdown_during_initialization("Could not create ConcurrentG1RefineThread"); + vm_shutdown_during_initialization("Could not create G1ConcurrentRefineThread"); return NULL; } - assert(t->cg1r() == cg1r, "Conc refine thread should refer to this"); - cg1r->_threads[i] = t; + assert(t->cr() == cr, "Conc refine thread should refer to this"); + cr->_threads[i] = t; next = t; } - cg1r->_sample_thread = new G1YoungRemSetSamplingThread(); - if (cg1r->_sample_thread->osthread() == NULL) { + cr->_sample_thread = new G1YoungRemSetSamplingThread(); + if (cr->_sample_thread->osthread() == NULL) { *ecode = JNI_ENOMEM; vm_shutdown_during_initialization("Could not create G1YoungRemSetSamplingThread"); return NULL; } *ecode = JNI_OK; - return cg1r; + return cr; } -void ConcurrentG1Refine::stop() { +void G1ConcurrentRefine::stop() { for (uint i = 0; i < _n_worker_threads; i++) { _threads[i]->stop(); } _sample_thread->stop(); } -void ConcurrentG1Refine::update_thread_thresholds() { +void G1ConcurrentRefine::update_thread_thresholds() { for (uint i = 0; i < _n_worker_threads; i++) { Thresholds thresholds = calc_thresholds(_green_zone, _yellow_zone, i); _threads[i]->update_thresholds(activation_level(thresholds), @@ -250,31 +250,31 @@ void ConcurrentG1Refine::update_thread_thresholds() { } } -ConcurrentG1Refine::~ConcurrentG1Refine() { +G1ConcurrentRefine::~G1ConcurrentRefine() { for (uint i = 0; i < _n_worker_threads; i++) { delete _threads[i]; } - FREE_C_HEAP_ARRAY(ConcurrentG1RefineThread*, _threads); + FREE_C_HEAP_ARRAY(G1ConcurrentRefineThread*, _threads); delete _sample_thread; } -void ConcurrentG1Refine::threads_do(ThreadClosure *tc) { +void G1ConcurrentRefine::threads_do(ThreadClosure *tc) { worker_threads_do(tc); tc->do_thread(_sample_thread); } -void ConcurrentG1Refine::worker_threads_do(ThreadClosure * tc) { +void G1ConcurrentRefine::worker_threads_do(ThreadClosure * tc) { for (uint i = 0; i < _n_worker_threads; i++) { tc->do_thread(_threads[i]); } } -uint ConcurrentG1Refine::thread_num() { +uint G1ConcurrentRefine::thread_num() { return G1ConcRefinementThreads; } -void ConcurrentG1Refine::print_worker_threads_on(outputStream* st) const { +void G1ConcurrentRefine::print_worker_threads_on(outputStream* st) const { for (uint i = 0; i < _n_worker_threads; ++i) { _threads[i]->print_on(st); st->cr(); @@ -312,7 +312,7 @@ static size_t calc_new_red_zone(size_t green, size_t yellow) { return MIN2(yellow + (yellow - green), max_red_zone); } -void ConcurrentG1Refine::update_zones(double update_rs_time, +void G1ConcurrentRefine::update_zones(double update_rs_time, size_t update_rs_processed_buffers, double goal_ms) { log_trace( CTRL_TAGS )("Updating Refinement Zones: " @@ -338,7 +338,7 @@ void ConcurrentG1Refine::update_zones(double update_rs_time, _green_zone, _yellow_zone, _red_zone); } -void ConcurrentG1Refine::adjust(double update_rs_time, +void G1ConcurrentRefine::adjust(double update_rs_time, size_t update_rs_processed_buffers, double goal_ms) { DirtyCardQueueSet& dcqs = JavaThread::dirty_card_queue_set(); diff --git a/src/hotspot/share/gc/g1/concurrentG1Refine.hpp b/src/hotspot/share/gc/g1/g1ConcurrentRefine.hpp similarity index 87% rename from src/hotspot/share/gc/g1/concurrentG1Refine.hpp rename to src/hotspot/share/gc/g1/g1ConcurrentRefine.hpp index 346599cbd89..e352fca236a 100644 --- a/src/hotspot/share/gc/g1/concurrentG1Refine.hpp +++ b/src/hotspot/share/gc/g1/g1ConcurrentRefine.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2017, 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 @@ -22,23 +22,23 @@ * */ -#ifndef SHARE_VM_GC_G1_CONCURRENTG1REFINE_HPP -#define SHARE_VM_GC_G1_CONCURRENTG1REFINE_HPP +#ifndef SHARE_VM_GC_G1_G1CONCURRENTREFINE_HPP +#define SHARE_VM_GC_G1_G1CONCURRENTREFINE_HPP #include "memory/allocation.hpp" #include "utilities/globalDefinitions.hpp" // Forward decl class CardTableEntryClosure; -class ConcurrentG1RefineThread; +class G1ConcurrentRefineThread; class G1YoungRemSetSamplingThread; class outputStream; class ThreadClosure; -class ConcurrentG1Refine: public CHeapObj { +class G1ConcurrentRefine : public CHeapObj { G1YoungRemSetSamplingThread* _sample_thread; - ConcurrentG1RefineThread** _threads; + G1ConcurrentRefineThread** _threads; uint _n_worker_threads; /* * The value of the update buffer queue length falls into one of 3 zones: @@ -62,7 +62,7 @@ class ConcurrentG1Refine: public CHeapObj { size_t _red_zone; size_t _min_yellow_zone_size; - ConcurrentG1Refine(size_t green_zone, + G1ConcurrentRefine(size_t green_zone, size_t yellow_zone, size_t red_zone, size_t min_yellow_zone_size); @@ -76,11 +76,11 @@ class ConcurrentG1Refine: public CHeapObj { void update_thread_thresholds(); public: - ~ConcurrentG1Refine(); + ~G1ConcurrentRefine(); - // Returns ConcurrentG1Refine instance if succeeded to create/initialize ConcurrentG1Refine and ConcurrentG1RefineThread. + // Returns a G1ConcurrentRefine instance if succeeded to create/initialize G1ConcurrentRefine and G1ConcurrentRefineThreads. // Otherwise, returns NULL with error code. - static ConcurrentG1Refine* create(jint* ecode); + static G1ConcurrentRefine* create(jint* ecode); void stop(); @@ -104,4 +104,4 @@ class ConcurrentG1Refine: public CHeapObj { size_t red_zone() const { return _red_zone; } }; -#endif // SHARE_VM_GC_G1_CONCURRENTG1REFINE_HPP +#endif // SHARE_VM_GC_G1_G1CONCURRENTREFINE_HPP diff --git a/src/hotspot/share/gc/g1/concurrentG1RefineThread.cpp b/src/hotspot/share/gc/g1/g1ConcurrentRefineThread.cpp similarity index 85% rename from src/hotspot/share/gc/g1/concurrentG1RefineThread.cpp rename to src/hotspot/share/gc/g1/g1ConcurrentRefineThread.cpp index 8af13dc078d..f62e0c369e8 100644 --- a/src/hotspot/share/gc/g1/concurrentG1RefineThread.cpp +++ b/src/hotspot/share/gc/g1/g1ConcurrentRefineThread.cpp @@ -23,8 +23,8 @@ */ #include "precompiled.hpp" -#include "gc/g1/concurrentG1Refine.hpp" -#include "gc/g1/concurrentG1RefineThread.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" +#include "gc/g1/g1ConcurrentRefineThread.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp" #include "gc/g1/g1RemSet.hpp" #include "gc/shared/suspendibleThreadSet.hpp" @@ -33,17 +33,19 @@ #include "runtime/handles.inline.hpp" #include "runtime/mutexLocker.hpp" -ConcurrentG1RefineThread:: -ConcurrentG1RefineThread(ConcurrentG1Refine* cg1r, ConcurrentG1RefineThread *next, - uint worker_id_offset, uint worker_id, - size_t activate, size_t deactivate) : +G1ConcurrentRefineThread::G1ConcurrentRefineThread(G1ConcurrentRefine* cr, + G1ConcurrentRefineThread *next, + uint worker_id_offset, + uint worker_id, + size_t activate, + size_t deactivate) : ConcurrentGCThread(), _worker_id_offset(worker_id_offset), _worker_id(worker_id), _active(false), _next(next), _monitor(NULL), - _cg1r(cg1r), + _cr(cr), _vtime_accum(0.0), _activation_threshold(activate), _deactivation_threshold(deactivate) @@ -65,26 +67,26 @@ ConcurrentG1RefineThread(ConcurrentG1Refine* cg1r, ConcurrentG1RefineThread *nex create_and_start(); } -void ConcurrentG1RefineThread::update_thresholds(size_t activate, +void G1ConcurrentRefineThread::update_thresholds(size_t activate, size_t deactivate) { assert(deactivate < activate, "precondition"); _activation_threshold = activate; _deactivation_threshold = deactivate; } -void ConcurrentG1RefineThread::wait_for_completed_buffers() { +void G1ConcurrentRefineThread::wait_for_completed_buffers() { MutexLockerEx x(_monitor, Mutex::_no_safepoint_check_flag); while (!should_terminate() && !is_active()) { _monitor->wait(Mutex::_no_safepoint_check_flag); } } -bool ConcurrentG1RefineThread::is_active() { +bool G1ConcurrentRefineThread::is_active() { DirtyCardQueueSet& dcqs = JavaThread::dirty_card_queue_set(); return is_primary() ? dcqs.process_completed_buffers() : _active; } -void ConcurrentG1RefineThread::activate() { +void G1ConcurrentRefineThread::activate() { MutexLockerEx x(_monitor, Mutex::_no_safepoint_check_flag); if (!is_primary()) { set_active(true); @@ -95,7 +97,7 @@ void ConcurrentG1RefineThread::activate() { _monitor->notify(); } -void ConcurrentG1RefineThread::deactivate() { +void G1ConcurrentRefineThread::deactivate() { MutexLockerEx x(_monitor, Mutex::_no_safepoint_check_flag); if (!is_primary()) { set_active(false); @@ -105,7 +107,7 @@ void ConcurrentG1RefineThread::deactivate() { } } -void ConcurrentG1RefineThread::run_service() { +void G1ConcurrentRefineThread::run_service() { _vtime_start = os::elapsedVTime(); while (!should_terminate()) { @@ -132,7 +134,7 @@ void ConcurrentG1RefineThread::run_service() { size_t curr_buffer_num = dcqs.completed_buffers_num(); // If the number of the buffers falls down into the yellow zone, // that means that the transition period after the evacuation pause has ended. - if (dcqs.completed_queue_padding() > 0 && curr_buffer_num <= cg1r()->yellow_zone()) { + if (dcqs.completed_queue_padding() > 0 && curr_buffer_num <= cr()->yellow_zone()) { dcqs.set_completed_queue_padding(0); } @@ -168,7 +170,7 @@ void ConcurrentG1RefineThread::run_service() { log_debug(gc, refine)("Stopping %d", _worker_id); } -void ConcurrentG1RefineThread::stop_service() { +void G1ConcurrentRefineThread::stop_service() { MutexLockerEx x(_monitor, Mutex::_no_safepoint_check_flag); _monitor->notify(); } diff --git a/src/hotspot/share/gc/g1/concurrentG1RefineThread.hpp b/src/hotspot/share/gc/g1/g1ConcurrentRefineThread.hpp similarity index 82% rename from src/hotspot/share/gc/g1/concurrentG1RefineThread.hpp rename to src/hotspot/share/gc/g1/g1ConcurrentRefineThread.hpp index db835f96c76..fbc10fcfb4c 100644 --- a/src/hotspot/share/gc/g1/concurrentG1RefineThread.hpp +++ b/src/hotspot/share/gc/g1/g1ConcurrentRefineThread.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2017, 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 @@ -22,19 +22,19 @@ * */ -#ifndef SHARE_VM_GC_G1_CONCURRENTG1REFINETHREAD_HPP -#define SHARE_VM_GC_G1_CONCURRENTG1REFINETHREAD_HPP +#ifndef SHARE_VM_GC_G1_G1CONCURRENTREFINETHREAD_HPP +#define SHARE_VM_GC_G1_G1CONCURRENTREFINETHREAD_HPP #include "gc/g1/dirtyCardQueue.hpp" #include "gc/shared/concurrentGCThread.hpp" // Forward Decl. class CardTableEntryClosure; -class ConcurrentG1Refine; +class G1ConcurrentRefine; // One or more G1 Concurrent Refinement Threads may be active if concurrent // refinement is in progress. -class ConcurrentG1RefineThread: public ConcurrentGCThread { +class G1ConcurrentRefineThread: public ConcurrentGCThread { friend class VMStructs; friend class G1CollectedHeap; @@ -47,9 +47,9 @@ class ConcurrentG1RefineThread: public ConcurrentGCThread { // when the number of the rset update buffer crosses a certain threshold. A successor // would self-deactivate when the number of the buffers falls below the threshold. bool _active; - ConcurrentG1RefineThread* _next; + G1ConcurrentRefineThread* _next; Monitor* _monitor; - ConcurrentG1Refine* _cg1r; + G1ConcurrentRefine* _cr; // This thread's activation/deactivation thresholds size_t _activation_threshold; @@ -69,7 +69,7 @@ class ConcurrentG1RefineThread: public ConcurrentGCThread { public: // Constructor - ConcurrentG1RefineThread(ConcurrentG1Refine* cg1r, ConcurrentG1RefineThread* next, + G1ConcurrentRefineThread(G1ConcurrentRefine* cr, G1ConcurrentRefineThread* next, uint worker_id_offset, uint worker_id, size_t activate, size_t deactivate); @@ -79,7 +79,7 @@ public: // Total virtual time so far. double vtime_accum() { return _vtime_accum; } - ConcurrentG1Refine* cg1r() { return _cg1r; } + G1ConcurrentRefine* cr() { return _cr; } }; -#endif // SHARE_VM_GC_G1_CONCURRENTG1REFINETHREAD_HPP +#endif // SHARE_VM_GC_G1_G1CONCURRENTREFINETHREAD_HPP diff --git a/src/hotspot/share/gc/g1/g1DefaultPolicy.cpp b/src/hotspot/share/gc/g1/g1DefaultPolicy.cpp index 945db6e0491..409c8ba99aa 100644 --- a/src/hotspot/share/gc/g1/g1DefaultPolicy.cpp +++ b/src/hotspot/share/gc/g1/g1DefaultPolicy.cpp @@ -23,12 +23,12 @@ */ #include "precompiled.hpp" -#include "gc/g1/concurrentG1Refine.hpp" #include "gc/g1/concurrentMarkThread.inline.hpp" #include "gc/g1/g1Analytics.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp" #include "gc/g1/g1CollectionSet.hpp" #include "gc/g1/g1ConcurrentMark.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" #include "gc/g1/g1DefaultPolicy.hpp" #include "gc/g1/g1HotCardCache.hpp" #include "gc/g1/g1IHOPControl.hpp" @@ -745,7 +745,7 @@ void G1DefaultPolicy::record_collection_pause_end(double pause_time_ms, size_t c } else { update_rs_time_goal_ms -= scan_hcc_time_ms; } - _g1->concurrent_g1_refine()->adjust(average_time_ms(G1GCPhaseTimes::UpdateRS) - scan_hcc_time_ms, + _g1->concurrent_refine()->adjust(average_time_ms(G1GCPhaseTimes::UpdateRS) - scan_hcc_time_ms, phase_times()->sum_thread_work_items(G1GCPhaseTimes::UpdateRS), update_rs_time_goal_ms); diff --git a/src/hotspot/share/gc/g1/g1RemSet.cpp b/src/hotspot/share/gc/g1/g1RemSet.cpp index 76f54d85004..435958bef69 100644 --- a/src/hotspot/share/gc/g1/g1RemSet.cpp +++ b/src/hotspot/share/gc/g1/g1RemSet.cpp @@ -23,10 +23,10 @@ */ #include "precompiled.hpp" -#include "gc/g1/concurrentG1Refine.hpp" #include "gc/g1/dirtyCardQueue.hpp" #include "gc/g1/g1BlockOffsetTable.inline.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" #include "gc/g1/g1FromCardCache.hpp" #include "gc/g1/g1GCPhaseTimes.hpp" #include "gc/g1/g1HotCardCache.hpp" @@ -298,7 +298,7 @@ G1RemSet::~G1RemSet() { } uint G1RemSet::num_par_rem_sets() { - return MAX2(DirtyCardQueueSet::num_par_ids() + ConcurrentG1Refine::thread_num(), ParallelGCThreads); + return MAX2(DirtyCardQueueSet::num_par_ids() + G1ConcurrentRefine::thread_num(), ParallelGCThreads); } void G1RemSet::initialize(size_t capacity, uint max_regions) { diff --git a/src/hotspot/share/gc/g1/g1RemSetSummary.cpp b/src/hotspot/share/gc/g1/g1RemSetSummary.cpp index 2123fc0a2a5..542946a4a7b 100644 --- a/src/hotspot/share/gc/g1/g1RemSetSummary.cpp +++ b/src/hotspot/share/gc/g1/g1RemSetSummary.cpp @@ -23,9 +23,9 @@ */ #include "precompiled.hpp" -#include "gc/g1/concurrentG1Refine.hpp" -#include "gc/g1/concurrentG1RefineThread.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" +#include "gc/g1/g1ConcurrentRefineThread.hpp" #include "gc/g1/g1RemSet.inline.hpp" #include "gc/g1/g1RemSetSummary.hpp" #include "gc/g1/g1YoungRemSetSamplingThread.hpp" @@ -45,7 +45,7 @@ public: } virtual void do_thread(Thread* t) { - ConcurrentG1RefineThread* crt = (ConcurrentG1RefineThread*) t; + G1ConcurrentRefineThread* crt = (G1ConcurrentRefineThread*) t; _summary->set_rs_thread_vtime(_counter, crt->vtime_accum()); _counter++; } @@ -59,12 +59,12 @@ void G1RemSetSummary::update() { _num_coarsenings = HeapRegionRemSet::n_coarsenings(); - ConcurrentG1Refine * cg1r = G1CollectedHeap::heap()->concurrent_g1_refine(); + G1ConcurrentRefine * cr = G1CollectedHeap::heap()->concurrent_refine(); if (_rs_threads_vtimes != NULL) { GetRSThreadVTimeClosure p(this); - cg1r->worker_threads_do(&p); + cr->worker_threads_do(&p); } - set_sampling_thread_vtime(cg1r->sampling_thread()->vtime_accum()); + set_sampling_thread_vtime(cr->sampling_thread()->vtime_accum()); } void G1RemSetSummary::set_rs_thread_vtime(uint thread, double value) { @@ -85,7 +85,7 @@ G1RemSetSummary::G1RemSetSummary() : _num_processed_buf_mutator(0), _num_processed_buf_rs_threads(0), _num_coarsenings(0), - _num_vtimes(ConcurrentG1Refine::thread_num()), + _num_vtimes(G1ConcurrentRefine::thread_num()), _rs_threads_vtimes(NEW_C_HEAP_ARRAY(double, _num_vtimes, mtGC)), _sampling_thread_vtime(0.0f) { @@ -98,7 +98,7 @@ G1RemSetSummary::G1RemSetSummary(G1RemSet* rem_set) : _num_processed_buf_mutator(0), _num_processed_buf_rs_threads(0), _num_coarsenings(0), - _num_vtimes(ConcurrentG1Refine::thread_num()), + _num_vtimes(G1ConcurrentRefine::thread_num()), _rs_threads_vtimes(NEW_C_HEAP_ARRAY(double, _num_vtimes, mtGC)), _sampling_thread_vtime(0.0f) { update(); diff --git a/src/hotspot/share/gc/g1/heapRegionManager.cpp b/src/hotspot/share/gc/g1/heapRegionManager.cpp index 9260b9178df..c1b8c2b404e 100644 --- a/src/hotspot/share/gc/g1/heapRegionManager.cpp +++ b/src/hotspot/share/gc/g1/heapRegionManager.cpp @@ -23,8 +23,8 @@ */ #include "precompiled.hpp" -#include "gc/g1/concurrentG1Refine.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" #include "gc/g1/heapRegion.hpp" #include "gc/g1/heapRegionManager.inline.hpp" #include "gc/g1/heapRegionSet.inline.hpp" diff --git a/src/hotspot/share/gc/g1/heapRegionRemSet.cpp b/src/hotspot/share/gc/g1/heapRegionRemSet.cpp index 6ea136ad40f..603be129590 100644 --- a/src/hotspot/share/gc/g1/heapRegionRemSet.cpp +++ b/src/hotspot/share/gc/g1/heapRegionRemSet.cpp @@ -23,9 +23,9 @@ */ #include "precompiled.hpp" -#include "gc/g1/concurrentG1Refine.hpp" #include "gc/g1/g1BlockOffsetTable.inline.hpp" #include "gc/g1/g1CollectedHeap.inline.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" #include "gc/g1/g1CardLiveData.inline.hpp" #include "gc/g1/heapRegionManager.inline.hpp" #include "gc/g1/heapRegionRemSet.hpp" From 5815fc9d54f677fdaa9818b97b5540e7e0c8ab09 Mon Sep 17 00:00:00 2001 From: Thomas Schatzl Date: Mon, 6 Nov 2017 14:25:18 +0100 Subject: [PATCH 43/61] 8140255: Move the management of G1YoungRemSetSamplingThread from G1ConcurrentRefine Reviewed-by: sjohanss, sangheki --- src/hotspot/share/gc/g1/g1CollectedHeap.cpp | 23 +++++++++++++++++-- src/hotspot/share/gc/g1/g1CollectedHeap.hpp | 6 +++++ .../share/gc/g1/g1ConcurrentRefine.cpp | 21 +---------------- .../share/gc/g1/g1ConcurrentRefine.hpp | 13 ++--------- src/hotspot/share/gc/g1/g1RemSetSummary.cpp | 7 +++--- 5 files changed, 34 insertions(+), 36 deletions(-) diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp index 150c3a655ac..8c2ea13bf4a 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp @@ -54,6 +54,7 @@ #include "gc/g1/g1SerialFullCollector.hpp" #include "gc/g1/g1StringDedup.hpp" #include "gc/g1/g1YCTypes.hpp" +#include "gc/g1/g1YoungRemSetSamplingThread.hpp" #include "gc/g1/heapRegion.inline.hpp" #include "gc/g1/heapRegionRemSet.hpp" #include "gc/g1/heapRegionSet.inline.hpp" @@ -1541,6 +1542,7 @@ void G1CollectedHeap::shrink(size_t shrink_bytes) { G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* collector_policy) : CollectedHeap(), + _young_gen_sampling_thread(NULL), _collector_policy(collector_policy), _gc_timer_stw(new (ResourceObj::C_HEAP, mtGC) STWGCTimer()), _gc_tracer_stw(new (ResourceObj::C_HEAP, mtGC) G1NewTracer()), @@ -1637,6 +1639,15 @@ jint G1CollectedHeap::initialize_concurrent_refinement() { return ecode; } +jint G1CollectedHeap::initialize_young_gen_sampling_thread() { + _young_gen_sampling_thread = new G1YoungRemSetSamplingThread(); + if (_young_gen_sampling_thread->osthread() == NULL) { + vm_shutdown_during_initialization("Could not create G1YoungRemSetSamplingThread"); + return JNI_ENOMEM; + } + return JNI_OK; +} + jint G1CollectedHeap::initialize() { CollectedHeap::pre_initialize(); os::enable_vtime(); @@ -1789,6 +1800,11 @@ jint G1CollectedHeap::initialize() { return ecode; } + ecode = initialize_young_gen_sampling_thread(); + if (ecode != JNI_OK) { + return ecode; + } + JavaThread::dirty_card_queue_set().initialize(DirtyCardQ_CBL_mon, DirtyCardQ_FL_lock, (int)concurrent_refine()->yellow_zone(), @@ -1837,6 +1853,7 @@ void G1CollectedHeap::stop() { // do not continue to execute and access resources (e.g. logging) // that are destroyed during shutdown. _cr->stop(); + _young_gen_sampling_thread->stop(); _cmThread->stop(); if (G1StringDedup::is_enabled()) { G1StringDedup::stop(); @@ -2436,7 +2453,8 @@ void G1CollectedHeap::print_gc_threads_on(outputStream* st) const { _cmThread->print_on(st); st->cr(); _cm->print_worker_threads_on(st); - _cr->print_worker_threads_on(st); // also prints the sample thread + _cr->print_threads_on(st); + _young_gen_sampling_thread->print_on(st); if (G1StringDedup::is_enabled()) { G1StringDedup::print_worker_threads_on(st); } @@ -2446,7 +2464,8 @@ void G1CollectedHeap::gc_threads_do(ThreadClosure* tc) const { workers()->threads_do(tc); tc->do_thread(_cmThread); _cm->threads_do(tc); - _cr->threads_do(tc); // also iterates over the sample thread + _cr->threads_do(tc); + tc->do_thread(_young_gen_sampling_thread); if (G1StringDedup::is_enabled()) { G1StringDedup::threads_do(tc); } diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.hpp b/src/hotspot/share/gc/g1/g1CollectedHeap.hpp index fc7ea3e9183..92c6b3c86de 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.hpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.hpp @@ -73,6 +73,7 @@ class G1CollectorPolicy; class G1Policy; class G1HotCardCache; class G1RemSet; +class G1YoungRemSetSamplingThread; class HeapRegionRemSetIterator; class G1ConcurrentMark; class ConcurrentMarkThread; @@ -142,6 +143,8 @@ class G1CollectedHeap : public CollectedHeap { friend class G1CheckCSetFastTableClosure; private: + G1YoungRemSetSamplingThread* _young_gen_sampling_thread; + WorkGang* _workers; G1CollectorPolicy* _collector_policy; @@ -553,6 +556,8 @@ protected: // during GC into global variables. void merge_per_thread_state_info(G1ParScanThreadStateSet* per_thread_states); public: + G1YoungRemSetSamplingThread* sampling_thread() const { return _young_gen_sampling_thread; } + WorkGang* workers() const { return _workers; } G1Allocator* allocator() { @@ -959,6 +964,7 @@ public: private: jint initialize_concurrent_refinement(); + jint initialize_young_gen_sampling_thread(); public: // Initialize the G1CollectedHeap to have the initial and // maximum sizes and remembered and barrier sets diff --git a/src/hotspot/share/gc/g1/g1ConcurrentRefine.cpp b/src/hotspot/share/gc/g1/g1ConcurrentRefine.cpp index ac1271067c8..8e39ff06698 100644 --- a/src/hotspot/share/gc/g1/g1ConcurrentRefine.cpp +++ b/src/hotspot/share/gc/g1/g1ConcurrentRefine.cpp @@ -25,7 +25,6 @@ #include "precompiled.hpp" #include "gc/g1/g1ConcurrentRefine.hpp" #include "gc/g1/g1ConcurrentRefineThread.hpp" -#include "gc/g1/g1YoungRemSetSamplingThread.hpp" #include "logging/log.hpp" #include "runtime/java.hpp" #include "runtime/thread.hpp" @@ -117,7 +116,6 @@ G1ConcurrentRefine::G1ConcurrentRefine(size_t green_zone, size_t red_zone, size_t min_yellow_zone_size) : _threads(NULL), - _sample_thread(NULL), _n_worker_threads(thread_num()), _green_zone(green_zone), _yellow_zone(yellow_zone), @@ -224,13 +222,6 @@ G1ConcurrentRefine* G1ConcurrentRefine::create(jint* ecode) { next = t; } - cr->_sample_thread = new G1YoungRemSetSamplingThread(); - if (cr->_sample_thread->osthread() == NULL) { - *ecode = JNI_ENOMEM; - vm_shutdown_during_initialization("Could not create G1YoungRemSetSamplingThread"); - return NULL; - } - *ecode = JNI_OK; return cr; } @@ -239,7 +230,6 @@ void G1ConcurrentRefine::stop() { for (uint i = 0; i < _n_worker_threads; i++) { _threads[i]->stop(); } - _sample_thread->stop(); } void G1ConcurrentRefine::update_thread_thresholds() { @@ -255,16 +245,9 @@ G1ConcurrentRefine::~G1ConcurrentRefine() { delete _threads[i]; } FREE_C_HEAP_ARRAY(G1ConcurrentRefineThread*, _threads); - - delete _sample_thread; } void G1ConcurrentRefine::threads_do(ThreadClosure *tc) { - worker_threads_do(tc); - tc->do_thread(_sample_thread); -} - -void G1ConcurrentRefine::worker_threads_do(ThreadClosure * tc) { for (uint i = 0; i < _n_worker_threads; i++) { tc->do_thread(_threads[i]); } @@ -274,13 +257,11 @@ uint G1ConcurrentRefine::thread_num() { return G1ConcRefinementThreads; } -void G1ConcurrentRefine::print_worker_threads_on(outputStream* st) const { +void G1ConcurrentRefine::print_threads_on(outputStream* st) const { for (uint i = 0; i < _n_worker_threads; ++i) { _threads[i]->print_on(st); st->cr(); } - _sample_thread->print_on(st); - st->cr(); } static size_t calc_new_green_zone(size_t green, diff --git a/src/hotspot/share/gc/g1/g1ConcurrentRefine.hpp b/src/hotspot/share/gc/g1/g1ConcurrentRefine.hpp index e352fca236a..b64d4e3ee80 100644 --- a/src/hotspot/share/gc/g1/g1ConcurrentRefine.hpp +++ b/src/hotspot/share/gc/g1/g1ConcurrentRefine.hpp @@ -31,13 +31,10 @@ // Forward decl class CardTableEntryClosure; class G1ConcurrentRefineThread; -class G1YoungRemSetSamplingThread; class outputStream; class ThreadClosure; class G1ConcurrentRefine : public CHeapObj { - G1YoungRemSetSamplingThread* _sample_thread; - G1ConcurrentRefineThread** _threads; uint _n_worker_threads; /* @@ -86,18 +83,12 @@ class G1ConcurrentRefine : public CHeapObj { void adjust(double update_rs_time, size_t update_rs_processed_buffers, double goal_ms); - // Iterate over all concurrent refinement threads + // Iterate over all concurrent refinement threads applying the given closure. void threads_do(ThreadClosure *tc); - // Iterate over all worker refinement threads - void worker_threads_do(ThreadClosure * tc); - - // The RS sampling thread has nothing to do with refinement, but is here for now. - G1YoungRemSetSamplingThread * sampling_thread() const { return _sample_thread; } - static uint thread_num(); - void print_worker_threads_on(outputStream* st) const; + void print_threads_on(outputStream* st) const; size_t green_zone() const { return _green_zone; } size_t yellow_zone() const { return _yellow_zone; } diff --git a/src/hotspot/share/gc/g1/g1RemSetSummary.cpp b/src/hotspot/share/gc/g1/g1RemSetSummary.cpp index 542946a4a7b..25825114ace 100644 --- a/src/hotspot/share/gc/g1/g1RemSetSummary.cpp +++ b/src/hotspot/share/gc/g1/g1RemSetSummary.cpp @@ -59,12 +59,13 @@ void G1RemSetSummary::update() { _num_coarsenings = HeapRegionRemSet::n_coarsenings(); - G1ConcurrentRefine * cr = G1CollectedHeap::heap()->concurrent_refine(); + G1CollectedHeap* g1h = G1CollectedHeap::heap(); + G1ConcurrentRefine* cg1r = g1h->concurrent_refine(); if (_rs_threads_vtimes != NULL) { GetRSThreadVTimeClosure p(this); - cr->worker_threads_do(&p); + cg1r->threads_do(&p); } - set_sampling_thread_vtime(cr->sampling_thread()->vtime_accum()); + set_sampling_thread_vtime(g1h->sampling_thread()->vtime_accum()); } void G1RemSetSummary::set_rs_thread_vtime(uint thread, double value) { From 16fb073895f666c152233c0199c8b831d2e3a3b3 Mon Sep 17 00:00:00 2001 From: Dmitry Samersoff Date: Mon, 6 Nov 2017 14:31:22 +0000 Subject: [PATCH 44/61] 8163011: AArch64: NMT detail stack trace cleanup Makes aarch64 behavior similar to x86 one Reviewed-by: aph, drwhite --- .../os_cpu/linux_aarch64/os_linux_aarch64.cpp | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp b/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp index 939d782f6ee..0c28eb492ac 100644 --- a/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp +++ b/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp @@ -233,8 +233,17 @@ frame os::get_sender_for_C_frame(frame* fr) { } intptr_t* _get_previous_fp() { - register intptr_t **ebp __asm__ (SPELL_REG_FP); - return (intptr_t*) *ebp; // we want what it points to. + register intptr_t **fp __asm__ (SPELL_REG_FP); + + // fp is for this frame (_get_previous_fp). We want the fp for the + // caller of os::current_frame*(), so go up two frames. However, for + // optimized builds, _get_previous_fp() will be inlined, so only go + // up 1 frame in that case. + #ifdef _NMT_NOINLINE_ + return **(intptr_t***)fp; + #else + return *fp; + #endif } From 67b21ffa1dc66007f4f28141b788211bfae297a6 Mon Sep 17 00:00:00 2001 From: Doug Simon Date: Mon, 6 Nov 2017 12:17:59 +0100 Subject: [PATCH 45/61] 8186478: [JVMCI] rename HotSpotResolvedJavaMethod#setNotInlineableOrCompileable Reviewed-by: kvn, thartmann --- src/hotspot/share/jvmci/jvmciCompilerToVM.cpp | 4 ++-- .../jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java | 2 +- .../src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethod.java | 2 +- .../src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethodImpl.java | 4 ++-- .../jdk/vm/ci/hotspot/CompilerToVMHelper.java | 4 ++-- .../compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java | 4 ++-- 6 files changed, 10 insertions(+), 10 deletions(-) diff --git a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp index bccb7fdcff8..46fb9a650a3 100644 --- a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp +++ b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp @@ -1000,7 +1000,7 @@ C2V_VMENTRY(jlong, getMaxCallTargetOffset, (JNIEnv*, jobject, jlong addr)) return -1; C2V_END -C2V_VMENTRY(void, setNotInlineableOrCompileable,(JNIEnv *, jobject, jobject jvmci_method)) +C2V_VMENTRY(void, setNotInlinableOrCompilable,(JNIEnv *, jobject, jobject jvmci_method)) methodHandle method = CompilerToVM::asMethod(jvmci_method); method->set_not_c1_compilable(); method->set_not_c2_compilable(); @@ -1820,7 +1820,7 @@ JNINativeMethod CompilerToVM::methods[] = { {CC "getImplementor", CC "(" HS_RESOLVED_KLASS ")" HS_RESOLVED_KLASS, FN_PTR(getImplementor)}, {CC "getStackTraceElement", CC "(" HS_RESOLVED_METHOD "I)" STACK_TRACE_ELEMENT, FN_PTR(getStackTraceElement)}, {CC "methodIsIgnoredBySecurityStackWalk", CC "(" HS_RESOLVED_METHOD ")Z", FN_PTR(methodIsIgnoredBySecurityStackWalk)}, - {CC "setNotInlineableOrCompileable", CC "(" HS_RESOLVED_METHOD ")V", FN_PTR(setNotInlineableOrCompileable)}, + {CC "setNotInlinableOrCompilable", CC "(" HS_RESOLVED_METHOD ")V", FN_PTR(setNotInlinableOrCompilable)}, {CC "isCompilable", CC "(" HS_RESOLVED_METHOD ")Z", FN_PTR(isCompilable)}, {CC "hasNeverInlineDirective", CC "(" HS_RESOLVED_METHOD ")Z", FN_PTR(hasNeverInlineDirective)}, {CC "shouldInlineMethod", CC "(" HS_RESOLVED_METHOD ")Z", FN_PTR(shouldInlineMethod)}, diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java index 979ee934df5..a2626583bef 100644 --- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java +++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/CompilerToVM.java @@ -470,7 +470,7 @@ final class CompilerToVM { /** * Sets flags on {@code method} indicating that it should never be inlined or compiled by the VM. */ - native void setNotInlineableOrCompileable(HotSpotResolvedJavaMethodImpl method); + native void setNotInlinableOrCompilable(HotSpotResolvedJavaMethodImpl method); /** * Invalidates the profiling information for {@code method} and (re)initializes it such that diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethod.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethod.java index 398f6241717..60ccabfba5f 100644 --- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethod.java +++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethod.java @@ -59,7 +59,7 @@ public interface HotSpotResolvedJavaMethod extends ResolvedJavaMethod { /** * Sets flags on {@code method} indicating that it should never be inlined or compiled by the VM. */ - void setNotInlineableOrCompileable(); + void setNotInlinableOrCompilable(); /** * Returns true if this method is one of the special methods that is ignored by security stack diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethodImpl.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethodImpl.java index 6fb8ce8a6fa..80d0b266e95 100644 --- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethodImpl.java +++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotResolvedJavaMethodImpl.java @@ -320,8 +320,8 @@ final class HotSpotResolvedJavaMethodImpl extends HotSpotMethod implements HotSp /** * Sets flags on {@code method} indicating that it should never be inlined or compiled by the VM. */ - public void setNotInlineableOrCompileable() { - compilerToVM().setNotInlineableOrCompileable(this); + public void setNotInlinableOrCompilable() { + compilerToVM().setNotInlinableOrCompilable(this); } /** diff --git a/test/hotspot/jtreg/compiler/jvmci/common/patches/jdk.internal.vm.ci/jdk/vm/ci/hotspot/CompilerToVMHelper.java b/test/hotspot/jtreg/compiler/jvmci/common/patches/jdk.internal.vm.ci/jdk/vm/ci/hotspot/CompilerToVMHelper.java index 79c9cd7891f..a05497db0fd 100644 --- a/test/hotspot/jtreg/compiler/jvmci/common/patches/jdk.internal.vm.ci/jdk/vm/ci/hotspot/CompilerToVMHelper.java +++ b/test/hotspot/jtreg/compiler/jvmci/common/patches/jdk.internal.vm.ci/jdk/vm/ci/hotspot/CompilerToVMHelper.java @@ -224,8 +224,8 @@ public class CompilerToVMHelper { return CTVM.getLocalVariableTableStart((HotSpotResolvedJavaMethodImpl)method); } - public static void setNotInlineableOrCompileable(HotSpotResolvedJavaMethod method) { - CTVM.setNotInlineableOrCompileable((HotSpotResolvedJavaMethodImpl)method); + public static void setNotInlinableOrCompilable(HotSpotResolvedJavaMethod method) { + CTVM.setNotInlinableOrCompilable((HotSpotResolvedJavaMethodImpl)method); } public static void reprofile(HotSpotResolvedJavaMethod method) { diff --git a/test/hotspot/jtreg/compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java b/test/hotspot/jtreg/compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java index 20211643dde..9ef807b1a69 100644 --- a/test/hotspot/jtreg/compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java +++ b/test/hotspot/jtreg/compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java @@ -71,10 +71,10 @@ public class DoNotInlineOrCompileTest { boolean hasNeverInlineDirective = CompilerToVMHelper.hasNeverInlineDirective(method); Asserts.assertFalse(hasNeverInlineDirective, "Unexpected initial " + "value of property 'hasNeverInlineDirective'"); - CompilerToVMHelper.setNotInlineableOrCompileable(method); + CompilerToVMHelper.setNotInlinableOrCompilable(method); hasNeverInlineDirective = CompilerToVMHelper.hasNeverInlineDirective(method); Asserts.assertTrue(hasNeverInlineDirective, aMethod - + " : hasNeverInlineDirective is false even after setNotInlineableOrCompileable'"); + + " : hasNeverInlineDirective is false even after setNotInlinableOrCompilable'"); } private static List createTestCases() { From f37bbe430d08c27237b978bf50202ea20a1a4c41 Mon Sep 17 00:00:00 2001 From: Doug Simon Date: Mon, 6 Nov 2017 12:53:55 +0100 Subject: [PATCH 46/61] 8188102: [JVMCI] Convert special JVMCI oops in nmethod to jweak values Reviewed-by: never, kvn, kbarrett --- src/hotspot/.mx.jvmci/.project | 18 -- src/hotspot/.mx.jvmci/.pydevproject | 12 -- src/hotspot/share/code/nmethod.cpp | 172 +++++++++--------- src/hotspot/share/code/nmethod.hpp | 54 ++++-- .../share/gc/g1/g1SATBCardTableModRefBS.cpp | 49 ----- .../share/gc/g1/g1SATBCardTableModRefBS.hpp | 3 - src/hotspot/share/gc/shared/barrierSet.hpp | 3 - src/hotspot/share/jvmci/jvmciCompilerToVM.cpp | 9 +- src/hotspot/share/jvmci/jvmciEnv.cpp | 4 +- src/hotspot/share/runtime/deoptimization.cpp | 16 +- src/hotspot/share/runtime/jniHandles.cpp | 5 + src/hotspot/share/runtime/jniHandles.hpp | 1 + .../src/jdk/vm/ci/hotspot/HotSpotNmethod.java | 16 +- 13 files changed, 146 insertions(+), 216 deletions(-) delete mode 100644 src/hotspot/.mx.jvmci/.project delete mode 100644 src/hotspot/.mx.jvmci/.pydevproject diff --git a/src/hotspot/.mx.jvmci/.project b/src/hotspot/.mx.jvmci/.project deleted file mode 100644 index 5d4c97b68bc..00000000000 --- a/src/hotspot/.mx.jvmci/.project +++ /dev/null @@ -1,18 +0,0 @@ - - - mx.jvmci - - - mx - - - - org.python.pydev.PyDevBuilder - - - - - - org.python.pydev.pythonNature - - diff --git a/src/hotspot/.mx.jvmci/.pydevproject b/src/hotspot/.mx.jvmci/.pydevproject deleted file mode 100644 index 3f852ee834a..00000000000 --- a/src/hotspot/.mx.jvmci/.pydevproject +++ /dev/null @@ -1,12 +0,0 @@ - - -Default -python 2.7 - -/mx.jvmci - - -/mx - - - diff --git a/src/hotspot/share/code/nmethod.cpp b/src/hotspot/share/code/nmethod.cpp index 9a1559f53df..412331bdad0 100644 --- a/src/hotspot/share/code/nmethod.cpp +++ b/src/hotspot/share/code/nmethod.cpp @@ -409,6 +409,7 @@ void nmethod::init_defaults() { #if INCLUDE_JVMCI _jvmci_installed_code = NULL; _speculation_log = NULL; + _jvmci_installed_code_triggers_unloading = false; #endif } @@ -461,8 +462,8 @@ nmethod* nmethod::new_nmethod(const methodHandle& method, AbstractCompiler* compiler, int comp_level #if INCLUDE_JVMCI - , Handle installed_code, - Handle speculationLog + , jweak installed_code, + jweak speculationLog #endif ) { @@ -642,8 +643,8 @@ nmethod::nmethod( AbstractCompiler* compiler, int comp_level #if INCLUDE_JVMCI - , Handle installed_code, - Handle speculation_log + , jweak installed_code, + jweak speculation_log #endif ) : CompiledMethod(method, "nmethod", type, nmethod_size, sizeof(nmethod), code_buffer, offsets->value(CodeOffsets::Frame_Complete), frame_size, oop_maps, false), @@ -671,8 +672,14 @@ nmethod::nmethod( set_ctable_begin(header_begin() + _consts_offset); #if INCLUDE_JVMCI - _jvmci_installed_code = installed_code(); - _speculation_log = (instanceOop)speculation_log(); + _jvmci_installed_code = installed_code; + _speculation_log = speculation_log; + oop obj = JNIHandles::resolve(installed_code); + if (obj == NULL || (obj->is_a(HotSpotNmethod::klass()) && HotSpotNmethod::isDefault(obj))) { + _jvmci_installed_code_triggers_unloading = false; + } else { + _jvmci_installed_code_triggers_unloading = true; + } if (compiler->is_jvmci()) { // JVMCI might not produce any stub sections @@ -1026,8 +1033,6 @@ void nmethod::make_unloaded(BoolObjectClosure* is_alive, oop cause) { " unloadable, Method*(" INTPTR_FORMAT "), cause(" INTPTR_FORMAT ")", p2i(this), p2i(_method), p2i(cause)); - if (!Universe::heap()->is_gc_active()) - cause->klass()->print_on(&ls); } // Unlink the osr method, so we do not look this up again if (is_osr_method()) { @@ -1077,14 +1082,8 @@ void nmethod::make_unloaded(BoolObjectClosure* is_alive, oop cause) { #if INCLUDE_JVMCI // The method can only be unloaded after the pointer to the installed code // Java wrapper is no longer alive. Here we need to clear out this weak - // reference to the dead object. Nulling out the reference has to happen - // after the method is unregistered since the original value may be still - // tracked by the rset. + // reference to the dead object. maybe_invalidate_installed_code(); - // Clear these out after the nmethod has been unregistered and any - // updates to the InstalledCode instance have been performed. - _jvmci_installed_code = NULL; - _speculation_log = NULL; #endif // The Method* is gone at this point @@ -1246,10 +1245,6 @@ bool nmethod::make_not_entrant_or_zombie(unsigned int state) { MutexLockerEx mu(CodeCache_lock, Mutex::_no_safepoint_check_flag); if (nmethod_needs_unregister) { Universe::heap()->unregister_nmethod(this); -#ifdef JVMCI - _jvmci_installed_code = NULL; - _speculation_log = NULL; -#endif } flush_dependencies(NULL); } @@ -1314,6 +1309,11 @@ void nmethod::flush() { CodeCache::drop_scavenge_root_nmethod(this); } +#if INCLUDE_JVMCI + assert(_jvmci_installed_code == NULL, "should have been nulled out when transitioned to zombie"); + assert(_speculation_log == NULL, "should have been nulled out when transitioned to zombie"); +#endif + CodeBlob::flush(); CodeCache::free(this); } @@ -1500,29 +1500,18 @@ bool nmethod::do_unloading_oops(address low_boundary, BoolObjectClosure* is_aliv #if INCLUDE_JVMCI bool nmethod::do_unloading_jvmci(BoolObjectClosure* is_alive, bool unloading_occurred) { - bool is_unloaded = false; - // Follow JVMCI method - BarrierSet* bs = Universe::heap()->barrier_set(); if (_jvmci_installed_code != NULL) { - if (_jvmci_installed_code->is_a(HotSpotNmethod::klass()) && HotSpotNmethod::isDefault(_jvmci_installed_code)) { - if (!is_alive->do_object_b(_jvmci_installed_code)) { + if (JNIHandles::is_global_weak_cleared(_jvmci_installed_code)) { + if (_jvmci_installed_code_triggers_unloading) { + // jweak reference processing has already cleared the referent + make_unloaded(is_alive, NULL); + return true; + } else { clear_jvmci_installed_code(); } - } else { - if (can_unload(is_alive, (oop*)&_jvmci_installed_code, unloading_occurred)) { - return true; - } } } - - if (_speculation_log != NULL) { - if (!is_alive->do_object_b(_speculation_log)) { - bs->write_ref_nmethod_pre(&_speculation_log, this); - _speculation_log = NULL; - bs->write_ref_nmethod_post(&_speculation_log, this); - } - } - return is_unloaded; + return false; } #endif @@ -1594,15 +1583,6 @@ void nmethod::oops_do(OopClosure* f, bool allow_zombie) { // (See comment above.) } -#if INCLUDE_JVMCI - if (_jvmci_installed_code != NULL) { - f->do_oop((oop*) &_jvmci_installed_code); - } - if (_speculation_log != NULL) { - f->do_oop((oop*) &_speculation_log); - } -#endif - RelocIterator iter(this, low_boundary); while (iter.next()) { @@ -2860,44 +2840,50 @@ void nmethod::print_statistics() { #if INCLUDE_JVMCI void nmethod::clear_jvmci_installed_code() { - // write_ref_method_pre/post can only be safely called at a - // safepoint or while holding the CodeCache_lock - assert(CodeCache_lock->is_locked() || - SafepointSynchronize::is_at_safepoint(), "should be performed under a lock for consistency"); + assert_locked_or_safepoint(Patching_lock); if (_jvmci_installed_code != NULL) { - // This must be done carefully to maintain nmethod remembered sets properly - BarrierSet* bs = Universe::heap()->barrier_set(); - bs->write_ref_nmethod_pre(&_jvmci_installed_code, this); + JNIHandles::destroy_weak_global(_jvmci_installed_code); _jvmci_installed_code = NULL; - bs->write_ref_nmethod_post(&_jvmci_installed_code, this); + } +} + +void nmethod::clear_speculation_log() { + assert_locked_or_safepoint(Patching_lock); + if (_speculation_log != NULL) { + JNIHandles::destroy_weak_global(_speculation_log); + _speculation_log = NULL; } } void nmethod::maybe_invalidate_installed_code() { assert(Patching_lock->is_locked() || SafepointSynchronize::is_at_safepoint(), "should be performed under a lock for consistency"); - oop installed_code = jvmci_installed_code(); + oop installed_code = JNIHandles::resolve(_jvmci_installed_code); if (installed_code != NULL) { + // Update the values in the InstalledCode instance if it still refers to this nmethod nmethod* nm = (nmethod*)InstalledCode::address(installed_code); - if (nm == NULL || nm != this) { - // The link has been broken or the InstalledCode instance is - // associated with another nmethod so do nothing. - return; - } - if (!is_alive()) { - // Break the link between nmethod and InstalledCode such that the nmethod - // can subsequently be flushed safely. The link must be maintained while - // the method could have live activations since invalidateInstalledCode - // might want to invalidate all existing activations. - InstalledCode::set_address(installed_code, 0); - InstalledCode::set_entryPoint(installed_code, 0); - } else if (is_not_entrant()) { - // Remove the entry point so any invocation will fail but keep - // the address link around that so that existing activations can - // be invalidated. - InstalledCode::set_entryPoint(installed_code, 0); + if (nm == this) { + if (!is_alive()) { + // Break the link between nmethod and InstalledCode such that the nmethod + // can subsequently be flushed safely. The link must be maintained while + // the method could have live activations since invalidateInstalledCode + // might want to invalidate all existing activations. + InstalledCode::set_address(installed_code, 0); + InstalledCode::set_entryPoint(installed_code, 0); + } else if (is_not_entrant()) { + // Remove the entry point so any invocation will fail but keep + // the address link around that so that existing activations can + // be invalidated. + InstalledCode::set_entryPoint(installed_code, 0); + } } } + if (!is_alive()) { + // Clear these out after the nmethod has been unregistered and any + // updates to the InstalledCode instance have been performed. + clear_jvmci_installed_code(); + clear_speculation_log(); + } } void nmethod::invalidate_installed_code(Handle installedCode, TRAPS) { @@ -2916,45 +2902,49 @@ void nmethod::invalidate_installed_code(Handle installedCode, TRAPS) { { MutexLockerEx pl(Patching_lock, Mutex::_no_safepoint_check_flag); // This relationship can only be checked safely under a lock - assert(nm == NULL || !nm->is_alive() || nm->jvmci_installed_code() == installedCode(), "sanity check"); + assert(!nm->is_alive() || nm->jvmci_installed_code() == installedCode(), "sanity check"); } #endif if (nm->is_alive()) { - // The nmethod state machinery maintains the link between the - // HotSpotInstalledCode and nmethod* so as long as the nmethod appears to be - // alive assume there is work to do and deoptimize the nmethod. + // Invalidating the InstalledCode means we want the nmethod + // to be deoptimized. nm->mark_for_deoptimization(); VM_Deoptimize op; VMThread::execute(&op); } + // Multiple threads could reach this point so we now need to + // lock and re-check the link to the nmethod so that only one + // thread clears it. MutexLockerEx pl(Patching_lock, Mutex::_no_safepoint_check_flag); - // Check that it's still associated with the same nmethod and break - // the link if it is. if (InstalledCode::address(installedCode) == nativeMethod) { - InstalledCode::set_address(installedCode, 0); + InstalledCode::set_address(installedCode, 0); } } +oop nmethod::jvmci_installed_code() { + return JNIHandles::resolve(_jvmci_installed_code); +} + +oop nmethod::speculation_log() { + return JNIHandles::resolve(_speculation_log); +} + char* nmethod::jvmci_installed_code_name(char* buf, size_t buflen) { if (!this->is_compiled_by_jvmci()) { return NULL; } - oop installedCode = this->jvmci_installed_code(); - if (installedCode != NULL) { - oop installedCodeName = NULL; - if (installedCode->is_a(InstalledCode::klass())) { - installedCodeName = InstalledCode::name(installedCode); + oop installed_code = JNIHandles::resolve(_jvmci_installed_code); + if (installed_code != NULL) { + oop installed_code_name = NULL; + if (installed_code->is_a(InstalledCode::klass())) { + installed_code_name = InstalledCode::name(installed_code); } - if (installedCodeName != NULL) { - return java_lang_String::as_utf8_string(installedCodeName, buf, (int)buflen); - } else { - jio_snprintf(buf, buflen, "null"); - return buf; + if (installed_code_name != NULL) { + return java_lang_String::as_utf8_string(installed_code_name, buf, (int)buflen); } } - jio_snprintf(buf, buflen, "noInstalledCode"); - return buf; + return NULL; } #endif diff --git a/src/hotspot/share/code/nmethod.hpp b/src/hotspot/share/code/nmethod.hpp index 600c34fcb23..73fe7e236ec 100644 --- a/src/hotspot/share/code/nmethod.hpp +++ b/src/hotspot/share/code/nmethod.hpp @@ -63,9 +63,22 @@ class nmethod : public CompiledMethod { jmethodID _jmethod_id; // Cache of method()->jmethod_id() #if INCLUDE_JVMCI - // Needed to keep nmethods alive that are not the default nmethod for the associated Method. - oop _jvmci_installed_code; - oop _speculation_log; + // A weak reference to an InstalledCode object associated with + // this nmethod. + jweak _jvmci_installed_code; + + // A weak reference to a SpeculationLog object associated with + // this nmethod. + jweak _speculation_log; + + // Determines whether this nmethod is unloaded when the + // referent in _jvmci_installed_code is cleared. This + // will be false if the referent is initialized to a + // HotSpotNMethod object whose isDefault field is true. + // That is, installed code other than a "default" + // HotSpotNMethod causes nmethod unloading. + // This field is ignored once _jvmci_installed_code is NULL. + bool _jvmci_installed_code_triggers_unloading; #endif // To support simple linked-list chaining of nmethods: @@ -192,8 +205,8 @@ class nmethod : public CompiledMethod { AbstractCompiler* compiler, int comp_level #if INCLUDE_JVMCI - , Handle installed_code, - Handle speculation_log + , jweak installed_code, + jweak speculation_log #endif ); @@ -236,8 +249,8 @@ class nmethod : public CompiledMethod { AbstractCompiler* compiler, int comp_level #if INCLUDE_JVMCI - , Handle installed_code = Handle(), - Handle speculation_log = Handle() + , jweak installed_code = NULL, + jweak speculation_log = NULL #endif ); @@ -433,27 +446,46 @@ public: void set_method(Method* method) { _method = method; } #if INCLUDE_JVMCI - oop jvmci_installed_code() { return _jvmci_installed_code ; } + // Gets the InstalledCode object associated with this nmethod + // which may be NULL if this nmethod was not compiled by JVMCI + // or the weak reference has been cleared. + oop jvmci_installed_code(); + + // Copies the value of the name field in the InstalledCode + // object (if any) associated with this nmethod into buf. + // Returns the value of buf if it was updated otherwise NULL. char* jvmci_installed_code_name(char* buf, size_t buflen); - // Update the state of any InstalledCode instance associated with + // Updates the state of the InstalledCode (if any) associated with // this nmethod based on the current value of _state. void maybe_invalidate_installed_code(); - // Helper function to invalidate InstalledCode instances + // Deoptimizes the nmethod (if any) in the address field of a given + // InstalledCode object. The address field is zeroed upon return. static void invalidate_installed_code(Handle installed_code, TRAPS); - oop speculation_log() { return _speculation_log ; } + // Gets the SpeculationLog object associated with this nmethod + // which may be NULL if this nmethod was not compiled by JVMCI + // or the weak reference has been cleared. + oop speculation_log(); private: + // Deletes the weak reference (if any) to the InstalledCode object + // associated with this nmethod. void clear_jvmci_installed_code(); + // Deletes the weak reference (if any) to the SpeculationLog object + // associated with this nmethod. + void clear_speculation_log(); + public: #endif protected: virtual bool do_unloading_oops(address low_boundary, BoolObjectClosure* is_alive, bool unloading_occurred); #if INCLUDE_JVMCI + // See comment for _jvmci_installed_code_triggers_unloading field. + // Returns whether this nmethod was unloaded. virtual bool do_unloading_jvmci(BoolObjectClosure* is_alive, bool unloading_occurred); #endif diff --git a/src/hotspot/share/gc/g1/g1SATBCardTableModRefBS.cpp b/src/hotspot/share/gc/g1/g1SATBCardTableModRefBS.cpp index c8ff35ddf63..4ed16f6e7a4 100644 --- a/src/hotspot/share/gc/g1/g1SATBCardTableModRefBS.cpp +++ b/src/hotspot/share/gc/g1/g1SATBCardTableModRefBS.cpp @@ -214,52 +214,3 @@ G1SATBCardTableLoggingModRefBS::invalidate(MemRegion mr) { } } } - -void G1SATBCardTableModRefBS::write_ref_nmethod_post(oop* dst, nmethod* nm) { - oop obj = oopDesc::load_heap_oop(dst); - if (obj != NULL) { - G1CollectedHeap* g1h = G1CollectedHeap::heap(); - HeapRegion* hr = g1h->heap_region_containing(obj); - hr->add_strong_code_root(nm); - } -} - -class G1EnsureLastRefToRegion : public OopClosure { - G1CollectedHeap* _g1h; - HeapRegion* _hr; - oop* _dst; - - bool _value; -public: - G1EnsureLastRefToRegion(G1CollectedHeap* g1h, HeapRegion* hr, oop* dst) : - _g1h(g1h), _hr(hr), _dst(dst), _value(true) {} - - void do_oop(oop* p) { - if (_value && p != _dst) { - oop obj = oopDesc::load_heap_oop(p); - if (obj != NULL) { - HeapRegion* hr = _g1h->heap_region_containing(obj); - if (hr == _hr) { - // Another reference to the same region. - _value = false; - } - } - } - } - void do_oop(narrowOop* p) { ShouldNotReachHere(); } - bool value() const { return _value; } -}; - -void G1SATBCardTableModRefBS::write_ref_nmethod_pre(oop* dst, nmethod* nm) { - oop obj = oopDesc::load_heap_oop(dst); - if (obj != NULL) { - G1CollectedHeap* g1h = G1CollectedHeap::heap(); - HeapRegion* hr = g1h->heap_region_containing(obj); - G1EnsureLastRefToRegion ensure_last_ref(g1h, hr, dst); - nm->oops_do(&ensure_last_ref); - if (ensure_last_ref.value()) { - // Last reference to this region, remove the nmethod from the rset. - hr->remove_strong_code_root(nm); - } - } -} diff --git a/src/hotspot/share/gc/g1/g1SATBCardTableModRefBS.hpp b/src/hotspot/share/gc/g1/g1SATBCardTableModRefBS.hpp index 65315484aeb..b1703f86880 100644 --- a/src/hotspot/share/gc/g1/g1SATBCardTableModRefBS.hpp +++ b/src/hotspot/share/gc/g1/g1SATBCardTableModRefBS.hpp @@ -95,9 +95,6 @@ public: jbyte val = _byte_map[card_index]; return (val & (clean_card_mask_val() | deferred_card_val())) == deferred_card_val(); } - virtual void write_ref_nmethod_pre(oop* dst, nmethod* nm); - virtual void write_ref_nmethod_post(oop* dst, nmethod* nm); - }; template<> diff --git a/src/hotspot/share/gc/shared/barrierSet.hpp b/src/hotspot/share/gc/shared/barrierSet.hpp index eaec4af4fcc..bd8a7e532a5 100644 --- a/src/hotspot/share/gc/shared/barrierSet.hpp +++ b/src/hotspot/share/gc/shared/barrierSet.hpp @@ -120,9 +120,6 @@ public: static void static_write_ref_array_pre(HeapWord* start, size_t count); static void static_write_ref_array_post(HeapWord* start, size_t count); - virtual void write_ref_nmethod_pre(oop* dst, nmethod* nm) {} - virtual void write_ref_nmethod_post(oop* dst, nmethod* nm) {} - protected: virtual void write_ref_array_work(MemRegion mr) = 0; diff --git a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp index 46fb9a650a3..a01da7b52cc 100644 --- a/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp +++ b/src/hotspot/share/jvmci/jvmciCompilerToVM.cpp @@ -1039,7 +1039,7 @@ C2V_VMENTRY(jint, installCode, (JNIEnv *jniEnv, jobject, jobject target, jobject if (result != JVMCIEnv::ok) { assert(cb == NULL, "should be"); } else { - if (!installed_code_handle.is_null()) { + if (installed_code_handle.not_null()) { assert(installed_code_handle->is_a(InstalledCode::klass()), "wrong type"); nmethod::invalidate_installed_code(installed_code_handle, CHECK_0); { @@ -1058,13 +1058,6 @@ C2V_VMENTRY(jint, installCode, (JNIEnv *jniEnv, jobject, jobject target, jobject HotSpotInstalledCode::set_codeSize(installed_code_handle, cb->code_size()); } } - nmethod* nm = cb->as_nmethod_or_null(); - if (nm != NULL && installed_code_handle->is_scavengable()) { - assert(nm->detect_scavenge_root_oops(), "nm should be scavengable if installed_code is scavengable"); - if (!UseG1GC) { - assert(nm->on_scavenge_root_list(), "nm should be on scavengable list"); - } - } } } return result; diff --git a/src/hotspot/share/jvmci/jvmciEnv.cpp b/src/hotspot/share/jvmci/jvmciEnv.cpp index 320ecccd035..6c3b67701ab 100644 --- a/src/hotspot/share/jvmci/jvmciEnv.cpp +++ b/src/hotspot/share/jvmci/jvmciEnv.cpp @@ -521,7 +521,9 @@ JVMCIEnv::CodeInstallResult JVMCIEnv::register_method( debug_info, dependencies, code_buffer, frame_words, oop_map_set, handler_table, &implicit_tbl, - compiler, comp_level, installed_code, speculation_log); + compiler, comp_level, + JNIHandles::make_weak_global(installed_code), + JNIHandles::make_weak_global(speculation_log)); // Free codeBlobs //code_buffer->free_blob(); diff --git a/src/hotspot/share/runtime/deoptimization.cpp b/src/hotspot/share/runtime/deoptimization.cpp index 4197e87e723..087e26f5019 100644 --- a/src/hotspot/share/runtime/deoptimization.cpp +++ b/src/hotspot/share/runtime/deoptimization.cpp @@ -1674,19 +1674,9 @@ JRT_ENTRY(void, Deoptimization::uncommon_trap_inner(JavaThread* thread, jint tra tty->print(" compiler=%s compile_id=%d", nm->compiler_name(), nm->compile_id()); #if INCLUDE_JVMCI if (nm->is_nmethod()) { - oop installedCode = nm->as_nmethod()->jvmci_installed_code(); - if (installedCode != NULL) { - oop installedCodeName = NULL; - if (installedCode->is_a(InstalledCode::klass())) { - installedCodeName = InstalledCode::name(installedCode); - } - if (installedCodeName != NULL) { - tty->print(" (JVMCI: installedCodeName=%s) ", java_lang_String::as_utf8_string(installedCodeName)); - } else { - tty->print(" (JVMCI: installed code has no name) "); - } - } else if (nm->is_compiled_by_jvmci()) { - tty->print(" (JVMCI: no installed code) "); + char* installed_code_name = nm->as_nmethod()->jvmci_installed_code_name(buf, sizeof(buf)); + if (installed_code_name != NULL) { + tty->print(" (JVMCI: installed code name=%s) ", installed_code_name); } } #endif diff --git a/src/hotspot/share/runtime/jniHandles.cpp b/src/hotspot/share/runtime/jniHandles.cpp index 76f13e620da..8819f3c2cfc 100644 --- a/src/hotspot/share/runtime/jniHandles.cpp +++ b/src/hotspot/share/runtime/jniHandles.cpp @@ -127,6 +127,11 @@ oop JNIHandles::resolve_jweak(jweak handle) { template oop JNIHandles::resolve_jweak(jweak); template oop JNIHandles::resolve_jweak(jweak); +bool JNIHandles::is_global_weak_cleared(jweak handle) { + assert(is_jweak(handle), "not a weak handle"); + return guard_value(jweak_ref(handle)) == NULL; +} + void JNIHandles::destroy_global(jobject handle) { if (handle != NULL) { assert(is_global_handle(handle), "Invalid delete of global JNI handle"); diff --git a/src/hotspot/share/runtime/jniHandles.hpp b/src/hotspot/share/runtime/jniHandles.hpp index 71e65dd3491..b7cd7668549 100644 --- a/src/hotspot/share/runtime/jniHandles.hpp +++ b/src/hotspot/share/runtime/jniHandles.hpp @@ -82,6 +82,7 @@ class JNIHandles : AllStatic { // Weak global handles static jobject make_weak_global(Handle obj); static void destroy_weak_global(jobject handle); + static bool is_global_weak_cleared(jweak handle); // Test jweak without resolution // Sentinel marking deleted handles in block. Note that we cannot store NULL as // the sentinel, since clearing weak global JNI refs are done by storing NULL in diff --git a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotNmethod.java b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotNmethod.java index f8aa3cb9310..413ba290741 100644 --- a/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotNmethod.java +++ b/src/jdk.internal.vm.ci/share/classes/jdk.vm.ci.hotspot/src/jdk/vm/ci/hotspot/HotSpotNmethod.java @@ -30,13 +30,10 @@ import jdk.vm.ci.meta.JavaType; import jdk.vm.ci.meta.ResolvedJavaMethod; /** - * Implementation of {@link InstalledCode} for code installed as an nmethod. The nmethod stores a - * weak reference to an instance of this class. This is necessary to keep the nmethod from being - * unloaded while the associated {@link HotSpotNmethod} instance is alive. - *

- * Note that there is no (current) way for the reference from an nmethod to a {@link HotSpotNmethod} - * instance to be anything but weak. This is due to the fact that HotSpot does not treat nmethods as - * strong GC roots. + * Implementation of {@link InstalledCode} for code installed as an nmethod. + * + * When a {@link HotSpotNmethod} dies, it triggers unloading of the nmethod unless + * {@link #isDefault() == true}. */ public class HotSpotNmethod extends HotSpotInstalledCode { @@ -54,6 +51,11 @@ public class HotSpotNmethod extends HotSpotInstalledCode { this.isDefault = isDefault; } + /** + * Determines if the nmethod associated with this object is the compiled entry point for + * {@link #getMethod()}. If {@code false}, then the nmethod is unloaded when the VM determines + * this object has died. + */ public boolean isDefault() { return isDefault; } From 614fca959b312b61c8d7f84184a4b6af5ca846f2 Mon Sep 17 00:00:00 2001 From: Martin Doerr Date: Mon, 6 Nov 2017 17:56:29 +0100 Subject: [PATCH 47/61] 8190781: ppc64 + s390: Fix CriticalJNINatives Reviewed-by: goetz --- src/hotspot/cpu/ppc/assembler_ppc.inline.hpp | 2 +- src/hotspot/cpu/ppc/sharedRuntime_ppc.cpp | 27 +++++++++++++-- src/hotspot/cpu/ppc/vm_version_ppc.cpp | 3 +- src/hotspot/cpu/s390/sharedRuntime_s390.cpp | 35 +++++++++++++++++--- 4 files changed, 59 insertions(+), 8 deletions(-) diff --git a/src/hotspot/cpu/ppc/assembler_ppc.inline.hpp b/src/hotspot/cpu/ppc/assembler_ppc.inline.hpp index 3931b56093e..8188a5a89e1 100644 --- a/src/hotspot/cpu/ppc/assembler_ppc.inline.hpp +++ b/src/hotspot/cpu/ppc/assembler_ppc.inline.hpp @@ -941,7 +941,7 @@ inline void Assembler::vpmsumh( VectorRegister d, VectorRegister a, VectorRegis inline void Assembler::vpmsumw( VectorRegister d, VectorRegister a, VectorRegister b) { emit_int32( VPMSUMW_OPCODE | vrt(d) | vra(a) | vrb(b)); } // Vector Permute and Xor (introduced with Power 8) -inline void Assembler::vpermxor( VectorRegister d, VectorRegister a, VectorRegister b, VectorRegister c) { emit_int32( VPMSUMW_OPCODE | vrt(d) | vra(a) | vrb(b) | vrc(c)); } +inline void Assembler::vpermxor( VectorRegister d, VectorRegister a, VectorRegister b, VectorRegister c) { emit_int32( VPERMXOR_OPCODE | vrt(d) | vra(a) | vrb(b) | vrc(c)); } // Transactional Memory instructions (introduced with Power 8) inline void Assembler::tbegin_() { emit_int32( TBEGIN_OPCODE | rc(1)); } diff --git a/src/hotspot/cpu/ppc/sharedRuntime_ppc.cpp b/src/hotspot/cpu/ppc/sharedRuntime_ppc.cpp index 4b879905afa..6a4da8e8c93 100644 --- a/src/hotspot/cpu/ppc/sharedRuntime_ppc.cpp +++ b/src/hotspot/cpu/ppc/sharedRuntime_ppc.cpp @@ -1469,8 +1469,31 @@ static void save_or_restore_arguments(MacroAssembler* masm, } // Save or restore single word registers. for (int i = 0; i < total_in_args; i++) { - // PPC64: pass ints as longs: must only deal with floats here. - if (in_regs[i].first()->is_FloatRegister()) { + if (in_regs[i].first()->is_Register()) { + int offset = slot * VMRegImpl::stack_slot_size; + // Value lives in an input register. Save it on stack. + switch (in_sig_bt[i]) { + case T_BOOLEAN: + case T_CHAR: + case T_BYTE: + case T_SHORT: + case T_INT: + if (map != NULL) { + __ stw(in_regs[i].first()->as_Register(), offset, R1_SP); + } else { + __ lwa(in_regs[i].first()->as_Register(), offset, R1_SP); + } + slot++; + assert(slot <= stack_slots, "overflow (after INT or smaller stack slot)"); + break; + case T_ARRAY: + case T_LONG: + // handled above + break; + case T_OBJECT: + default: ShouldNotReachHere(); + } + } else if (in_regs[i].first()->is_FloatRegister()) { if (in_sig_bt[i] == T_FLOAT) { int offset = slot * VMRegImpl::stack_slot_size; slot++; diff --git a/src/hotspot/cpu/ppc/vm_version_ppc.cpp b/src/hotspot/cpu/ppc/vm_version_ppc.cpp index a27fb75b395..32aad5bcbc2 100644 --- a/src/hotspot/cpu/ppc/vm_version_ppc.cpp +++ b/src/hotspot/cpu/ppc/vm_version_ppc.cpp @@ -109,7 +109,8 @@ void VM_Version::initialize() { if (PowerArchitecturePPC64 >= 8) { if (FLAG_IS_DEFAULT(SuperwordUseVSX)) { - FLAG_SET_ERGO(bool, SuperwordUseVSX, true); + // TODO: Switch on when it works stable. Currently, MachSpillCopyNode::implementation code is missing. + //FLAG_SET_ERGO(bool, SuperwordUseVSX, true); } } else { if (SuperwordUseVSX) { diff --git a/src/hotspot/cpu/s390/sharedRuntime_s390.cpp b/src/hotspot/cpu/s390/sharedRuntime_s390.cpp index d8c1a63728b..032777b98be 100644 --- a/src/hotspot/cpu/s390/sharedRuntime_s390.cpp +++ b/src/hotspot/cpu/s390/sharedRuntime_s390.cpp @@ -1309,15 +1309,42 @@ static void save_or_restore_arguments(MacroAssembler *masm, } } else { __ z_lg(reg, offset, Z_SP); - slot += VMRegImpl::slots_per_word; - assert(slot <= stack_slots, "overflow (after LONG/ARRAY stack slot)"); } + slot += VMRegImpl::slots_per_word; + assert(slot <= stack_slots, "overflow (after LONG/ARRAY stack slot)"); } } // Save or restore single word registers. for (int i = 0; i < total_in_args; i++) { - if (in_regs[i].first()->is_FloatRegister()) { + if (in_regs[i].first()->is_Register()) { + int offset = slot * VMRegImpl::stack_slot_size; + // Value lives in an input register. Save it on stack. + switch (in_sig_bt[i]) { + case T_BOOLEAN: + case T_CHAR: + case T_BYTE: + case T_SHORT: + case T_INT: { + const Register reg = in_regs[i].first()->as_Register(); + Address stackaddr(Z_SP, offset); + if (map != NULL) { + __ z_st(reg, stackaddr); + } else { + __ z_lgf(reg, stackaddr); + } + slot++; + assert(slot <= stack_slots, "overflow (after INT or smaller stack slot)"); + break; + } + case T_ARRAY: + case T_LONG: + // handled above + break; + case T_OBJECT: + default: ShouldNotReachHere(); + } + } else if (in_regs[i].first()->is_FloatRegister()) { if (in_sig_bt[i] == T_FLOAT) { int offset = slot * VMRegImpl::stack_slot_size; slot++; @@ -1908,7 +1935,7 @@ nmethod *SharedRuntime::generate_native_wrapper(MacroAssembler *masm, case T_ARRAY: if (is_critical_native) { int body_arg = cix; - cix -= 2; // Point to length arg. + cix -= 1; // Point to length arg. unpack_array_argument(masm, in_regs[jix], in_elem_bt[jix], out_regs[body_arg], out_regs[cix], stack_slots); break; } From 9f9e85e9ada89d38e31f607db7b4134aa5cc66e6 Mon Sep 17 00:00:00 2001 From: John Paul Adrian Glaubitz Date: Mon, 6 Nov 2017 19:45:47 +0100 Subject: [PATCH 48/61] 8190570: Zero fails to build on linux-sparc due undefined reference Reviewed-by: coleenp --- src/hotspot/share/runtime/init.cpp | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/hotspot/share/runtime/init.cpp b/src/hotspot/share/runtime/init.cpp index 2386744b15f..ff31ac9da77 100644 --- a/src/hotspot/share/runtime/init.cpp +++ b/src/hotspot/share/runtime/init.cpp @@ -60,7 +60,11 @@ void stubRoutines_init1(); jint universe_init(); // depends on codeCache_init and stubRoutines_init #if INCLUDE_ALL_GCS // depends on universe_init, must be before interpreter_init (currently only on SPARC) +#ifndef ZERO void g1_barrier_stubs_init() NOT_SPARC({}); +#else +void g1_barrier_stubs_init() {}; +#endif #endif void interpreter_init(); // before any methods loaded void invocationCounter_init(); // before any methods loaded From 8c073d124bcdf094b54e096b95c38a51855b2291 Mon Sep 17 00:00:00 2001 From: Igor Ignatyev Date: Wed, 8 Nov 2017 11:44:37 -0800 Subject: [PATCH 49/61] 8190890: remove hotspot_tier1_{compiler,gc}_closed groups Reviewed-by: ehelin, lmesnik --- test/hotspot/jtreg/TEST.groups | 9 --------- 1 file changed, 9 deletions(-) diff --git a/test/hotspot/jtreg/TEST.groups b/test/hotspot/jtreg/TEST.groups index 83d5c498663..45c67bcf60d 100644 --- a/test/hotspot/jtreg/TEST.groups +++ b/test/hotspot/jtreg/TEST.groups @@ -107,15 +107,11 @@ hotspot_tier1_compiler_3 = \ -compiler/loopopts/Test7052494.java \ -compiler/runtime/Test6826736.java -hotspot_tier1_compiler_closed = \ - sanity/ExecuteInternalVMTests.java - hotspot_not_fast_compiler = \ :hotspot_compiler \ -:hotspot_tier1_compiler_1 \ -:hotspot_tier1_compiler_2 \ -:hotspot_tier1_compiler_3 \ - -:hotspot_tier1_compiler_closed hotspot_tier1_gc_1 = \ gc/g1/ @@ -130,9 +126,6 @@ hotspot_tier1_gc_2 = \ -gc/cms/TestMBeanCMS.java \ -gc/metaspace/CompressedClassSpaceSizeInJmapHeap.java -hotspot_tier1_gc_closed = \ - sanity/ExecuteInternalVMTests.java - hotspot_tier1_gc_gcold = \ gc/stress/gcold/TestGCOldWithG1.java gc/stress/gcold/TestGCOldWithCMS.java @@ -204,10 +197,8 @@ hotspot_tier1 = \ :hotspot_tier1_compiler_1 \ :hotspot_tier1_compiler_2 \ :hotspot_tier1_compiler_3 \ - :hotspot_tier1_compiler_closed \ :hotspot_tier1_gc_1 \ :hotspot_tier1_gc_2 \ - :hotspot_tier1_gc_closed \ :hotspot_tier1_gc_gcold \ :hotspot_tier1_runtime \ :hotspot_tier1_serviceability From 483ab7638e2b1c1d4994810a56fc2f9a702d2ee1 Mon Sep 17 00:00:00 2001 From: Paul Sandoz Date: Wed, 8 Nov 2017 12:08:23 -0800 Subject: [PATCH 50/61] 8190887: Stub code marking missing from some x86 generated stub routines Reviewed-by: kvn --- src/hotspot/cpu/x86/stubGenerator_x86_32.cpp | 22 +++++++++++++++++++- src/hotspot/cpu/x86/stubGenerator_x86_64.cpp | 14 +++++++++++++ 2 files changed, 35 insertions(+), 1 deletion(-) diff --git a/src/hotspot/cpu/x86/stubGenerator_x86_32.cpp b/src/hotspot/cpu/x86/stubGenerator_x86_32.cpp index 25eb6384c6b..677b018d838 100644 --- a/src/hotspot/cpu/x86/stubGenerator_x86_32.cpp +++ b/src/hotspot/cpu/x86/stubGenerator_x86_32.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2017, 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 @@ -3433,6 +3433,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmExp() { + StubCodeMark mark(this, "StubRoutines", "libmExp"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -3458,6 +3460,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmLog() { + StubCodeMark mark(this, "StubRoutines", "libmLog"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -3483,6 +3487,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmLog10() { + StubCodeMark mark(this, "StubRoutines", "libmLog10"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -3508,6 +3514,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmPow() { + StubCodeMark mark(this, "StubRoutines", "libmPow"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -3533,6 +3541,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libm_reduce_pi04l() { + StubCodeMark mark(this, "StubRoutines", "libm_reduce_pi04l"); + address start = __ pc(); BLOCK_COMMENT("Entry:"); @@ -3543,6 +3553,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libm_sin_cos_huge() { + StubCodeMark mark(this, "StubRoutines", "libm_sin_cos_huge"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -3556,6 +3568,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmSin() { + StubCodeMark mark(this, "StubRoutines", "libmSin"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -3579,6 +3593,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmCos() { + StubCodeMark mark(this, "StubRoutines", "libmCos"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -3604,6 +3620,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libm_tan_cot_huge() { + StubCodeMark mark(this, "StubRoutines", "libm_tan_cot_huge"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -3617,6 +3635,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmTan() { + StubCodeMark mark(this, "StubRoutines", "libmTan"); + address start = __ pc(); const XMMRegister x0 = xmm0; diff --git a/src/hotspot/cpu/x86/stubGenerator_x86_64.cpp b/src/hotspot/cpu/x86/stubGenerator_x86_64.cpp index fd8ee726abd..7db8ba32981 100644 --- a/src/hotspot/cpu/x86/stubGenerator_x86_64.cpp +++ b/src/hotspot/cpu/x86/stubGenerator_x86_64.cpp @@ -4619,6 +4619,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmExp() { + StubCodeMark mark(this, "StubRoutines", "libmExp"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -4646,6 +4648,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmLog() { + StubCodeMark mark(this, "StubRoutines", "libmLog"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -4674,6 +4678,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmLog10() { + StubCodeMark mark(this, "StubRoutines", "libmLog10"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -4701,6 +4707,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmPow() { + StubCodeMark mark(this, "StubRoutines", "libmPow"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -4731,6 +4739,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmSin() { + StubCodeMark mark(this, "StubRoutines", "libmSin"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -4770,6 +4780,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmCos() { + StubCodeMark mark(this, "StubRoutines", "libmCos"); + address start = __ pc(); const XMMRegister x0 = xmm0; @@ -4809,6 +4821,8 @@ class StubGenerator: public StubCodeGenerator { } address generate_libmTan() { + StubCodeMark mark(this, "StubRoutines", "libmTan"); + address start = __ pc(); const XMMRegister x0 = xmm0; From 3063beded5267ad48e913a8747e6b07e815f2472 Mon Sep 17 00:00:00 2001 From: Ioi Lam Date: Wed, 8 Nov 2017 09:03:24 -0800 Subject: [PATCH 51/61] 8186778: Make obsolete VM options for shared region size control Obsoleted SharedReadOnlySize, SharedMiscCodeSize, SharedMiscDataSize and SharedReadWriteSize Reviewed-by: dholmes --- src/hotspot/share/runtime/arguments.cpp | 4 ++++ src/hotspot/share/runtime/globals.hpp | 12 ------------ 2 files changed, 4 insertions(+), 12 deletions(-) diff --git a/src/hotspot/share/runtime/arguments.cpp b/src/hotspot/share/runtime/arguments.cpp index 0e630e2f823..b95ab1b00cd 100644 --- a/src/hotspot/share/runtime/arguments.cpp +++ b/src/hotspot/share/runtime/arguments.cpp @@ -396,6 +396,10 @@ static SpecialFlag const special_jvm_flags[] = { { "MinSleepInterval", JDK_Version::jdk(9), JDK_Version::jdk(10), JDK_Version::jdk(11) }, { "PermSize", JDK_Version::undefined(), JDK_Version::jdk(8), JDK_Version::undefined() }, { "MaxPermSize", JDK_Version::undefined(), JDK_Version::jdk(8), JDK_Version::undefined() }, + { "SharedReadWriteSize", JDK_Version::undefined(), JDK_Version::jdk(10), JDK_Version::undefined() }, + { "SharedReadOnlySize", JDK_Version::undefined(), JDK_Version::jdk(10), JDK_Version::undefined() }, + { "SharedMiscDataSize", JDK_Version::undefined(), JDK_Version::jdk(10), JDK_Version::undefined() }, + { "SharedMiscCodeSize", JDK_Version::undefined(), JDK_Version::jdk(10), JDK_Version::undefined() }, #ifdef TEST_VERIFY_SPECIAL_JVM_FLAGS { "dep > obs", JDK_Version::jdk(9), JDK_Version::jdk(8), JDK_Version::undefined() }, diff --git a/src/hotspot/share/runtime/globals.hpp b/src/hotspot/share/runtime/globals.hpp index e2c33aec46f..83a9088f5fc 100644 --- a/src/hotspot/share/runtime/globals.hpp +++ b/src/hotspot/share/runtime/globals.hpp @@ -3901,18 +3901,6 @@ public: "If PrintSharedArchiveAndExit is true, also print the shared " \ "dictionary") \ \ - product(size_t, SharedReadWriteSize, 0, \ - "Deprecated") \ - \ - product(size_t, SharedReadOnlySize, 0, \ - "Deprecated") \ - \ - product(size_t, SharedMiscDataSize, 0, \ - "Deprecated") \ - \ - product(size_t, SharedMiscCodeSize, 0, \ - "Deprecated") \ - \ product(size_t, SharedBaseAddress, LP64_ONLY(32*G) \ NOT_LP64(LINUX_ONLY(2*G) NOT_LINUX(0)), \ "Address to allocate shared memory region for class data") \ From 229c048a08eb78aa705470478cdde6ed6a84e493 Mon Sep 17 00:00:00 2001 From: Mikhailo Seledtsov Date: Wed, 8 Nov 2017 20:15:56 -0800 Subject: [PATCH 52/61] 8189213: [TESTBUG] Running jtreg tests on machine without docker shows extra message Removed the message from VMProps.java Reviewed-by: dholmes --- test/jtreg-ext/requires/VMProps.java | 1 - 1 file changed, 1 deletion(-) diff --git a/test/jtreg-ext/requires/VMProps.java b/test/jtreg-ext/requires/VMProps.java index 63c9ae9b049..13632fe7abe 100644 --- a/test/jtreg-ext/requires/VMProps.java +++ b/test/jtreg-ext/requires/VMProps.java @@ -347,7 +347,6 @@ public class VMProps implements Callable> { isSupported = checkDockerSupport(); } catch (Exception e) { isSupported = false; - System.err.println("dockerSupport() threw exception: " + e); } return (isSupported) ? "true" : "false"; From 2aecf1b3210ef21ee8dff7c9c20999da0e25103d Mon Sep 17 00:00:00 2001 From: David Holmes Date: Thu, 9 Nov 2017 00:52:15 -0500 Subject: [PATCH 53/61] 8190881: [TESTBUG] test.runtime.ErrorHandling.TestOnError comment is incomplete Reviewed-by: iignatyev --- test/hotspot/jtreg/runtime/ErrorHandling/TestOnError.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/test/hotspot/jtreg/runtime/ErrorHandling/TestOnError.java b/test/hotspot/jtreg/runtime/ErrorHandling/TestOnError.java index e1dcf38714d..fbb90854c6a 100644 --- a/test/hotspot/jtreg/runtime/ErrorHandling/TestOnError.java +++ b/test/hotspot/jtreg/runtime/ErrorHandling/TestOnError.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -44,7 +44,6 @@ public class TestOnError { String msg = "Test Succeeded"; - // Execute the VM so that a ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( "-XX:-TransmitErrorReport", "-XX:-CreateCoredumpOnCrash", From 02b4fc7985164480441d16fc1616423b033430e8 Mon Sep 17 00:00:00 2001 From: Jini George Date: Thu, 9 Nov 2017 12:12:32 +0530 Subject: [PATCH 54/61] 8189798: SA cleanup - part 1 Avoid varible redefinitions in SA, modify SA varible names to match hotspot ones Reviewed-by: sspitsyn, coleenp, sballal --- .../share/gc/cms/compactibleFreeListSpace.cpp | 5 +- .../share/gc/cms/compactibleFreeListSpace.hpp | 2 + src/hotspot/share/gc/cms/vmStructs_cms.hpp | 4 +- src/hotspot/share/runtime/stackValue.cpp | 6 +- src/hotspot/share/runtime/stackValue.hpp | 42 ++++----- src/hotspot/share/runtime/vmStructs.cpp | 1 + .../hotspot/compiler/ImmutableOopMapPair.java | 14 +-- .../gc/cms/CompactibleFreeListSpace.java | 22 ++--- .../sun/jvm/hotspot/runtime/Arguments.java | 22 ++--- .../jvm/hotspot/runtime/CompilerThread.java | 8 +- .../sun/jvm/hotspot/runtime/JNIid.java | 87 ------------------- .../classes/sun/jvm/hotspot/runtime/VM.java | 8 +- .../Win32AMD64JavaThreadPDAccess.java | 8 +- .../win32_x86/Win32X86JavaThreadPDAccess.java | 8 +- 14 files changed, 78 insertions(+), 159 deletions(-) delete mode 100644 src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/JNIid.java diff --git a/src/hotspot/share/gc/cms/compactibleFreeListSpace.cpp b/src/hotspot/share/gc/cms/compactibleFreeListSpace.cpp index 34116c60433..ca5d056c789 100644 --- a/src/hotspot/share/gc/cms/compactibleFreeListSpace.cpp +++ b/src/hotspot/share/gc/cms/compactibleFreeListSpace.cpp @@ -57,6 +57,7 @@ int CompactibleFreeListSpace::_lockRank = Mutex::leaf + 3; // Defaults are 0 so things will break badly if incorrectly initialized. size_t CompactibleFreeListSpace::IndexSetStart = 0; size_t CompactibleFreeListSpace::IndexSetStride = 0; +size_t CompactibleFreeListSpace::_min_chunk_size_in_bytes = 0; size_t MinChunkSize = 0; @@ -66,8 +67,8 @@ void CompactibleFreeListSpace::set_cms_values() { // MinChunkSize should be a multiple of MinObjAlignment and be large enough // for chunks to contain a FreeChunk. - size_t min_chunk_size_in_bytes = align_up(sizeof(FreeChunk), MinObjAlignmentInBytes); - MinChunkSize = min_chunk_size_in_bytes / BytesPerWord; + _min_chunk_size_in_bytes = align_up(sizeof(FreeChunk), MinObjAlignmentInBytes); + MinChunkSize = _min_chunk_size_in_bytes / BytesPerWord; assert(IndexSetStart == 0 && IndexSetStride == 0, "already set"); IndexSetStart = MinChunkSize; diff --git a/src/hotspot/share/gc/cms/compactibleFreeListSpace.hpp b/src/hotspot/share/gc/cms/compactibleFreeListSpace.hpp index 300afec4fbf..15fa9812dc5 100644 --- a/src/hotspot/share/gc/cms/compactibleFreeListSpace.hpp +++ b/src/hotspot/share/gc/cms/compactibleFreeListSpace.hpp @@ -118,6 +118,7 @@ class CompactibleFreeListSpace: public CompactibleSpace { }; static size_t IndexSetStart; static size_t IndexSetStride; + static size_t _min_chunk_size_in_bytes; private: enum FitStrategyOptions { @@ -134,6 +135,7 @@ class CompactibleFreeListSpace: public CompactibleSpace { // A lock protecting the free lists and free blocks; // mutable because of ubiquity of locking even for otherwise const methods mutable Mutex _freelistLock; + // Locking verifier convenience function void assert_locked() const PRODUCT_RETURN; void assert_locked(const Mutex* lock) const PRODUCT_RETURN; diff --git a/src/hotspot/share/gc/cms/vmStructs_cms.hpp b/src/hotspot/share/gc/cms/vmStructs_cms.hpp index 98197552dc1..307f46d9f48 100644 --- a/src/hotspot/share/gc/cms/vmStructs_cms.hpp +++ b/src/hotspot/share/gc/cms/vmStructs_cms.hpp @@ -30,7 +30,8 @@ static_field) \ nonstatic_field(CompactibleFreeListSpace, _collector, CMSCollector*) \ nonstatic_field(CompactibleFreeListSpace, _bt, BlockOffsetArrayNonContigSpace) \ - \ + static_field(CompactibleFreeListSpace, _min_chunk_size_in_bytes, size_t) \ + nonstatic_field(CMSBitMap, _bmStartWord, HeapWord*) \ nonstatic_field(CMSBitMap, _bmWordSize, size_t) \ nonstatic_field(CMSBitMap, _shifter, const int) \ nonstatic_field(CMSBitMap, _bm, BitMapView) \ @@ -63,6 +64,7 @@ declare_toplevel_type(LinearAllocBlock) #define VM_INT_CONSTANTS_CMS(declare_constant) \ + declare_constant(CompactibleFreeListSpace::IndexSetSize) \ declare_constant(Generation::ConcurrentMarkSweep) \ #endif // SHARE_VM_GC_CMS_VMSTRUCTS_CMS_HPP diff --git a/src/hotspot/share/runtime/stackValue.cpp b/src/hotspot/share/runtime/stackValue.cpp index e4cb824b43a..36ef418719c 100644 --- a/src/hotspot/share/runtime/stackValue.cpp +++ b/src/hotspot/share/runtime/stackValue.cpp @@ -191,12 +191,12 @@ BasicLock* StackValue::resolve_monitor_lock(const frame* fr, Location location) void StackValue::print_on(outputStream* st) const { switch(_type) { case T_INT: - st->print("%d (int) %f (float) %x (hex)", *(int *)&_i, *(float *)&_i, *(int *)&_i); + st->print("%d (int) %f (float) %x (hex)", *(int *)&_integer_value, *(float *)&_integer_value, *(int *)&_integer_value); break; case T_OBJECT: - _o()->print_value_on(st); - st->print(" <" INTPTR_FORMAT ">", p2i((address)_o())); + _handle_value()->print_value_on(st); + st->print(" <" INTPTR_FORMAT ">", p2i((address)_handle_value())); break; case T_CONFLICT: diff --git a/src/hotspot/share/runtime/stackValue.hpp b/src/hotspot/share/runtime/stackValue.hpp index 569ef58f89a..e0efef9a2d5 100644 --- a/src/hotspot/share/runtime/stackValue.hpp +++ b/src/hotspot/share/runtime/stackValue.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2017, 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 @@ -31,63 +31,63 @@ class StackValue : public ResourceObj { private: BasicType _type; - intptr_t _i; // Blank java stack slot value - Handle _o; // Java stack slot value interpreted as a Handle + intptr_t _integer_value; // Blank java stack slot value + Handle _handle_value; // Java stack slot value interpreted as a Handle public: StackValue(intptr_t value) { - _type = T_INT; - _i = value; + _type = T_INT; + _integer_value = value; } StackValue(Handle value, intptr_t scalar_replaced = 0) { - _type = T_OBJECT; - _i = scalar_replaced; - _o = value; - assert(_i == 0 || _o.is_null(), "not null object should not be marked as scalar replaced"); + _type = T_OBJECT; + _integer_value = scalar_replaced; + _handle_value = value; + assert(_integer_value == 0 || _handle_value.is_null(), "not null object should not be marked as scalar replaced"); } StackValue() { - _type = T_CONFLICT; - _i = 0; + _type = T_CONFLICT; + _integer_value = 0; } // Only used during deopt- preserve object type. StackValue(intptr_t o, BasicType t) { assert(t == T_OBJECT, "should not be used"); - _type = t; - _i = o; + _type = t; + _integer_value = o; } Handle get_obj() const { assert(type() == T_OBJECT, "type check"); - return _o; + return _handle_value; } bool obj_is_scalar_replaced() const { assert(type() == T_OBJECT, "type check"); - return _i != 0; + return _integer_value != 0; } void set_obj(Handle value) { assert(type() == T_OBJECT, "type check"); - _o = value; + _handle_value = value; } intptr_t get_int() const { assert(type() == T_INT, "type check"); - return _i; + return _integer_value; } // For special case in deopt. intptr_t get_int(BasicType t) const { assert(t == T_OBJECT && type() == T_OBJECT, "type check"); - return _i; + return _integer_value; } void set_int(intptr_t value) { assert(type() == T_INT, "type check"); - _i = value; + _integer_value = value; } BasicType type() const { return _type; } @@ -95,11 +95,11 @@ class StackValue : public ResourceObj { bool equal(StackValue *value) { if (_type != value->_type) return false; if (_type == T_OBJECT) - return (_o == value->_o); + return (_handle_value == value->_handle_value); else { assert(_type == T_INT, "sanity check"); // [phh] compare only low addressed portions of intptr_t slots - return (*(int *)&_i == *(int *)&value->_i); + return (*(int *)&_integer_value == *(int *)&value->_integer_value); } } diff --git a/src/hotspot/share/runtime/vmStructs.cpp b/src/hotspot/share/runtime/vmStructs.cpp index 8d4c4c235bd..c5ff4cdc584 100644 --- a/src/hotspot/share/runtime/vmStructs.cpp +++ b/src/hotspot/share/runtime/vmStructs.cpp @@ -2175,6 +2175,7 @@ typedef PaddedEnd PaddedObjectMonitor; declare_toplevel_type(vframeArray) \ declare_toplevel_type(vframeArrayElement) \ declare_toplevel_type(Annotations*) \ + declare_type(OopMapValue, StackObj) \ \ /***************/ \ /* Miscellaneous types */ \ diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/compiler/ImmutableOopMapPair.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/compiler/ImmutableOopMapPair.java index fa8b27e61c4..f272a4c13c9 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/compiler/ImmutableOopMapPair.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/compiler/ImmutableOopMapPair.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2017, 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 @@ -34,8 +34,8 @@ import java.util.Observable; import java.util.Observer; public class ImmutableOopMapPair { - private static CIntegerField pcField; - private static CIntegerField offsetField; + private static CIntegerField pcOffsetField; + private static CIntegerField oopmapOffsetField; private static long classSize; static { @@ -57,18 +57,18 @@ public class ImmutableOopMapPair { } public int getPC() { - return (int) pcField.getValue(address); + return (int) pcOffsetField.getValue(address); } public int getOffset() { - return (int) offsetField.getValue(address); + return (int) oopmapOffsetField.getValue(address); } private static void initialize(TypeDataBase db) { Type type = db.lookupType("ImmutableOopMapPair"); - pcField = type.getCIntegerField("_pc_offset"); - offsetField = type.getCIntegerField("_oopmap_offset"); + pcOffsetField = type.getCIntegerField("_pc_offset"); + oopmapOffsetField = type.getCIntegerField("_oopmap_offset"); classSize = type.getSize(); } diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/gc/cms/CompactibleFreeListSpace.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/gc/cms/CompactibleFreeListSpace.java index 86b2d520516..01034aeeb7e 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/gc/cms/CompactibleFreeListSpace.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/gc/cms/CompactibleFreeListSpace.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2017, 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 @@ -41,10 +41,11 @@ public class CompactibleFreeListSpace extends CompactibleSpace { private static AddressField dictionaryField; private static long smallLinearAllocBlockFieldOffset; - private int heapWordSize; // 4 for 32bit, 8 for 64 bits - private int IndexSetStart; // for small indexed list - private int IndexSetSize; - private int IndexSetStride; + private int heapWordSize; // 4 for 32bit, 8 for 64 bits + private int IndexSetStart; // for small indexed list + private int IndexSetSize; + private int IndexSetStride; + private static long MinChunkSizeInBytes; static { VM.registerVMInitializedObserver(new Observer() { @@ -57,8 +58,6 @@ public class CompactibleFreeListSpace extends CompactibleSpace { private static synchronized void initialize(TypeDataBase db) { long sizeofFreeChunk = db.lookupType("FreeChunk").getSize(); VM vm = VM.getVM(); - MinChunkSizeInBytes = numQuanta(sizeofFreeChunk, vm.getMinObjAlignmentInBytes()) * - vm.getMinObjAlignmentInBytes(); Type type = db.lookupType("CompactibleFreeListSpace"); collectorField = type.getAddressField("_collector"); @@ -66,6 +65,7 @@ public class CompactibleFreeListSpace extends CompactibleSpace { dictionaryField = type.getAddressField("_dictionary"); indexedFreeListField = type.getAddressField("_indexedFreeList[0]"); smallLinearAllocBlockFieldOffset = type.getField("_smallLinearAllocBlock").getOffset(); + MinChunkSizeInBytes = (type.getCIntegerField("_min_chunk_size_in_bytes")).getValue(); } public CompactibleFreeListSpace(Address addr) { @@ -74,7 +74,7 @@ public class CompactibleFreeListSpace extends CompactibleSpace { heapWordSize = vm.getHeapWordSize(); IndexSetStart = vm.getMinObjAlignmentInBytes() / heapWordSize; IndexSetStride = IndexSetStart; - IndexSetSize = 257; + IndexSetSize = vm.getIndexSetSize(); } // Accessing block offset table @@ -201,14 +201,8 @@ public class CompactibleFreeListSpace extends CompactibleSpace { // Unlike corresponding VM code, we operate on byte size rather than // HeapWord size for convenience. - private static long numQuanta(long x, long y) { - return ((x+y-1)/y); - } - public static long adjustObjectSizeInBytes(long sizeInBytes) { return Oop.alignObjectSize(Math.max(sizeInBytes, MinChunkSizeInBytes)); } - // FIXME: should I read this directly from VM? - private static long MinChunkSizeInBytes; } diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/Arguments.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/Arguments.java index 8ffbb249e38..858492d4ece 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/Arguments.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/Arguments.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2004, 2006, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2004, 2017, 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 @@ -42,11 +42,11 @@ public class Arguments { } public static String getJVMFlags() { - return buildString(jvmFlagsField, jvmFlagsCount); + return buildString(jvmFlagsArrayField, numJvmFlags); } public static String getJVMArgs() { - return buildString(jvmArgsField, jvmArgsCount); + return buildString(jvmArgsArrayField, numJvmArgs); } public static String getJavaCommand() { @@ -56,20 +56,20 @@ public class Arguments { // Internals only below this point // Fields - private static AddressField jvmFlagsField; - private static AddressField jvmArgsField; + private static AddressField jvmFlagsArrayField; + private static AddressField jvmArgsArrayField; private static AddressField javaCommandField; - private static long jvmFlagsCount; - private static long jvmArgsCount; + private static long numJvmFlags; + private static long numJvmArgs; private static synchronized void initialize(TypeDataBase db) { Type argumentsType = db.lookupType("Arguments"); - jvmFlagsField = argumentsType.getAddressField("_jvm_flags_array"); - jvmArgsField = argumentsType.getAddressField("_jvm_args_array"); + jvmFlagsArrayField = argumentsType.getAddressField("_jvm_flags_array"); + jvmArgsArrayField = argumentsType.getAddressField("_jvm_args_array"); javaCommandField = argumentsType.getAddressField("_java_command"); - jvmArgsCount = argumentsType.getCIntegerField("_num_jvm_args").getValue(); - jvmFlagsCount = argumentsType.getCIntegerField("_num_jvm_flags").getValue(); + numJvmArgs = argumentsType.getCIntegerField("_num_jvm_args").getValue(); + numJvmFlags = argumentsType.getCIntegerField("_num_jvm_flags").getValue(); } private static String buildString(AddressField arrayField, long count) { diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/CompilerThread.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/CompilerThread.java index 7a28c660938..5b1420c9fd4 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/CompilerThread.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/CompilerThread.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2017, 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 @@ -39,19 +39,19 @@ public class CompilerThread extends JavaThread { }); } - private static AddressField _env_field; + private static AddressField envField; private static synchronized void initialize(TypeDataBase db) throws WrongTypeException { Type type = db.lookupType("CompilerThread"); - _env_field = type.getAddressField("_env"); + envField = type.getAddressField("_env"); } private ciEnv _env; public synchronized ciEnv env() { if (_env == null) { - Address v = _env_field.getValue(this.getAddress()); + Address v = envField.getValue(this.getAddress()); if (v != null) { _env = new ciEnv(v); } diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/JNIid.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/JNIid.java deleted file mode 100644 index 7ea55403c6a..00000000000 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/JNIid.java +++ /dev/null @@ -1,87 +0,0 @@ -/* - * Copyright (c) 2002, 2012, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - * - */ - -package sun.jvm.hotspot.runtime; - -import java.util.*; - -import sun.jvm.hotspot.debugger.*; -import sun.jvm.hotspot.oops.*; -import sun.jvm.hotspot.types.*; - -public class JNIid extends VMObject { - private static MetadataField holder; - private static AddressField next; - private static CIntegerField offset; - private static MetadataField resolvedMethod; - private static MetadataField resolvedReceiver; - - private ObjectHeap heap; - - static { - VM.registerVMInitializedObserver(new Observer() { - public void update(Observable o, Object data) { - initialize(VM.getVM().getTypeDataBase()); - } - }); - } - - private static synchronized void initialize(TypeDataBase db) { - - // FIXME: JNIid has been removed as part of redesign of JNI method, - // field ID generation. Please refer to src/share/vm/prims/jniId.hpp/.cpp - // For now, commenting out the below code. - - /*********************************************************** - Type type = db.lookupType("JNIid"); - - holder = type.getOopField("_holder"); - next = type.getAddressField("_next"); - offset = type.getCIntegerField("_offset"); - resolvedMethod = type.getOopField("_resolved_method"); - resolvedReceiver = type.getOopField("_resolved_receiver"); - ***********************************************************/ - } - - public JNIid(Address addr, ObjectHeap heap) { - super(addr); - this.heap = heap; - } - - public JNIid next() { - Address nextAddr = next.getValue(addr); - if (nextAddr == null) { - return null; - } - return new JNIid(nextAddr, heap); - } - - public Klass holder() { return (Klass) holder.getValue(addr); } - public int offset() { return (int) offset.getValue(addr); } - public Method method() { - return ((InstanceKlass) holder()).getMethods().at(offset()); - } - public Method resolvedMethod() { return (Method)resolvedMethod.getValue(addr); } - public Klass resolvedReceiver() { return (Klass) resolvedReceiver.getValue(addr); } -} diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/VM.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/VM.java index bd4934560a6..fe180f189e2 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/VM.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/VM.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2017, 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 @@ -105,6 +105,7 @@ public class VM { private int heapOopSize; private int klassPtrSize; private int oopSize; + private final int IndexSetSize; /** This is only present in a non-core build */ private CodeCache codeCache; /** This is only present in a C1 build */ @@ -376,6 +377,7 @@ public class VM { bytesPerWord = db.lookupIntConstant("BytesPerWord").intValue(); heapWordSize = db.lookupIntConstant("HeapWordSize").intValue(); oopSize = db.lookupIntConstant("oopSize").intValue(); + IndexSetSize = db.lookupIntConstant("CompactibleFreeListSpace::IndexSetSize").intValue(); intType = db.lookupType("int"); uintType = db.lookupType("uint"); @@ -594,6 +596,10 @@ public class VM { return heapOopSize; } + public int getIndexSetSize() { + return IndexSetSize; + } + public int getKlassPtrSize() { return klassPtrSize; } diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/win32_amd64/Win32AMD64JavaThreadPDAccess.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/win32_amd64/Win32AMD64JavaThreadPDAccess.java index 86255a48f6e..ba62cb742ab 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/win32_amd64/Win32AMD64JavaThreadPDAccess.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/win32_amd64/Win32AMD64JavaThreadPDAccess.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2005, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2005, 2017, 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 @@ -43,7 +43,7 @@ public class Win32AMD64JavaThreadPDAccess implements JavaThreadPDAccess { private static AddressField osThreadField; // Field from OSThread - private static Field osThreadThreadIdField; + private static Field threadIdField; // This is currently unneeded but is being kept in case we change // the currentFrameGuess algorithm @@ -64,7 +64,7 @@ public class Win32AMD64JavaThreadPDAccess implements JavaThreadPDAccess { osThreadField = type.getAddressField("_osthread"); type = db.lookupType("OSThread"); - osThreadThreadIdField = type.getField("_thread_id"); + threadIdField = type.getField("_thread_id"); } public Address getLastJavaFP(Address addr) { @@ -130,7 +130,7 @@ public class Win32AMD64JavaThreadPDAccess implements JavaThreadPDAccess { Address osThreadAddr = osThreadField.getValue(addr); // Get the address of the thread_id within the OSThread Address threadIdAddr = - osThreadAddr.addOffsetTo(osThreadThreadIdField.getOffset()); + osThreadAddr.addOffsetTo(threadIdField.getOffset()); JVMDebugger debugger = VM.getVM().getDebugger(); return debugger.getThreadForIdentifierAddress(threadIdAddr); } diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/win32_x86/Win32X86JavaThreadPDAccess.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/win32_x86/Win32X86JavaThreadPDAccess.java index 435e4762335..5b93e88ee9d 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/win32_x86/Win32X86JavaThreadPDAccess.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/win32_x86/Win32X86JavaThreadPDAccess.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2017, 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 @@ -42,7 +42,7 @@ public class Win32X86JavaThreadPDAccess implements JavaThreadPDAccess { private static AddressField osThreadField; // Field from OSThread - private static Field osThreadThreadIdField; + private static Field threadIdField; // This is currently unneeded but is being kept in case we change // the currentFrameGuess algorithm @@ -63,7 +63,7 @@ public class Win32X86JavaThreadPDAccess implements JavaThreadPDAccess { osThreadField = type.getAddressField("_osthread"); type = db.lookupType("OSThread"); - osThreadThreadIdField = type.getField("_thread_id"); + threadIdField = type.getField("_thread_id"); } public Address getLastJavaFP(Address addr) { @@ -129,7 +129,7 @@ public class Win32X86JavaThreadPDAccess implements JavaThreadPDAccess { Address osThreadAddr = osThreadField.getValue(addr); // Get the address of the thread_id within the OSThread Address threadIdAddr = - osThreadAddr.addOffsetTo(osThreadThreadIdField.getOffset()); + osThreadAddr.addOffsetTo(threadIdField.getOffset()); JVMDebugger debugger = VM.getVM().getDebugger(); return debugger.getThreadForIdentifierAddress(threadIdAddr); } From 09c2ca58092d8b7678e1395c4f8861ea622e5d0e Mon Sep 17 00:00:00 2001 From: Stefan Johansson Date: Thu, 9 Nov 2017 10:11:19 +0100 Subject: [PATCH 55/61] 8190711: Assert in G1MMUTracker due to concurrent modification Reviewed-by: tschatzl, rehn, sangheki --- src/hotspot/share/gc/g1/g1MMUTracker.cpp | 11 ----------- src/hotspot/share/gc/g1/g1MMUTracker.hpp | 2 -- 2 files changed, 13 deletions(-) diff --git a/src/hotspot/share/gc/g1/g1MMUTracker.cpp b/src/hotspot/share/gc/g1/g1MMUTracker.cpp index 837eb1576e7..10b30f09366 100644 --- a/src/hotspot/share/gc/g1/g1MMUTracker.cpp +++ b/src/hotspot/share/gc/g1/g1MMUTracker.cpp @@ -112,18 +112,7 @@ void G1MMUTrackerQueue::add_pause(double start, double end) { } } -// basically the _internal call does not remove expired entries -// this is for trying things out in the future and a couple -// of other places (debugging) - double G1MMUTrackerQueue::when_sec(double current_time, double pause_time) { - remove_expired_entries(current_time); - - return when_internal(current_time, pause_time); -} - -double G1MMUTrackerQueue::when_internal(double current_time, - double pause_time) { // if the pause is over the maximum, just assume that it's the maximum double adjusted_pause_time = (pause_time > max_gc_time()) ? max_gc_time() : pause_time; diff --git a/src/hotspot/share/gc/g1/g1MMUTracker.hpp b/src/hotspot/share/gc/g1/g1MMUTracker.hpp index c8a09af2adc..22e2f3a730c 100644 --- a/src/hotspot/share/gc/g1/g1MMUTracker.hpp +++ b/src/hotspot/share/gc/g1/g1MMUTracker.hpp @@ -134,8 +134,6 @@ private: void remove_expired_entries(double current_time); double calculate_gc_time(double current_time); - double when_internal(double current_time, double pause_time); - public: G1MMUTrackerQueue(double time_slice, double max_gc_time); From ac52bdcdd129e35d66a0d2b071755279f4defaf0 Mon Sep 17 00:00:00 2001 From: Brent Christian Date: Thu, 9 Nov 2017 11:13:50 -0800 Subject: [PATCH 56/61] 8185925: StackFrameInfo::walker field can be replaced with bitmap to save footprint 8153682: StackFrameInfo.declaringClass could be removed Reviewed-by: coleenp, mchung --- src/hotspot/share/classfile/javaClasses.cpp | 12 ++------ src/hotspot/share/classfile/javaClasses.hpp | 2 -- src/hotspot/share/classfile/vmSymbols.hpp | 1 - src/hotspot/share/prims/stackwalk.cpp | 3 +- .../classes/java/lang/StackFrameInfo.java | 29 ++++++++++--------- .../share/classes/java/lang/StackWalker.java | 13 +++------ .../java/lang/invoke/MethodHandleImpl.java | 5 ++++ .../internal/misc/JavaLangInvokeAccess.java | 20 ++++++++----- 8 files changed, 42 insertions(+), 43 deletions(-) diff --git a/src/hotspot/share/classfile/javaClasses.cpp b/src/hotspot/share/classfile/javaClasses.cpp index 3499d3e9d01..54367e26335 100644 --- a/src/hotspot/share/classfile/javaClasses.cpp +++ b/src/hotspot/share/classfile/javaClasses.cpp @@ -2258,8 +2258,9 @@ void java_lang_StackFrameInfo::set_method_and_bci(Handle stackFrame, const metho void java_lang_StackFrameInfo::to_stack_trace_element(Handle stackFrame, Handle stack_trace_element, TRAPS) { ResourceMark rm(THREAD); - Handle k (THREAD, stackFrame->obj_field(_declaringClass_offset)); - InstanceKlass* holder = InstanceKlass::cast(java_lang_Class::as_Klass(k())); + Handle mname(THREAD, stackFrame->obj_field(java_lang_StackFrameInfo::_memberName_offset)); + Klass* clazz = java_lang_Class::as_Klass(java_lang_invoke_MemberName::clazz(mname())); + InstanceKlass* holder = InstanceKlass::cast(clazz); Method* method = java_lang_StackFrameInfo::get_method(stackFrame, holder, CHECK); short version = stackFrame->short_field(_version_offset); @@ -2270,7 +2271,6 @@ void java_lang_StackFrameInfo::to_stack_trace_element(Handle stackFrame, Handle void java_lang_StackFrameInfo::compute_offsets() { Klass* k = SystemDictionary::StackFrameInfo_klass(); - compute_offset(_declaringClass_offset, k, vmSymbols::declaringClass_name(), vmSymbols::class_signature()); compute_offset(_memberName_offset, k, vmSymbols::memberName_name(), vmSymbols::object_signature()); compute_offset(_bci_offset, k, vmSymbols::bci_name(), vmSymbols::short_signature()); STACKFRAMEINFO_INJECTED_FIELDS(INJECTED_FIELD_COMPUTE_OFFSET); @@ -3679,7 +3679,6 @@ int java_lang_StackTraceElement::moduleVersion_offset; int java_lang_StackTraceElement::classLoaderName_offset; int java_lang_StackTraceElement::declaringClass_offset; int java_lang_StackTraceElement::declaringClassObject_offset; -int java_lang_StackFrameInfo::_declaringClass_offset; int java_lang_StackFrameInfo::_memberName_offset; int java_lang_StackFrameInfo::_bci_offset; int java_lang_StackFrameInfo::_version_offset; @@ -3732,11 +3731,6 @@ void java_lang_StackTraceElement::set_declaringClassObject(oop element, oop valu element->obj_field_put(declaringClassObject_offset, value); } -// Support for java_lang_StackFrameInfo -void java_lang_StackFrameInfo::set_declaringClass(oop element, oop value) { - element->obj_field_put(_declaringClass_offset, value); -} - void java_lang_StackFrameInfo::set_version(oop element, short value) { element->short_field_put(_version_offset, value); } diff --git a/src/hotspot/share/classfile/javaClasses.hpp b/src/hotspot/share/classfile/javaClasses.hpp index 07736b66685..9c345460612 100644 --- a/src/hotspot/share/classfile/javaClasses.hpp +++ b/src/hotspot/share/classfile/javaClasses.hpp @@ -1370,7 +1370,6 @@ class Backtrace: AllStatic { class java_lang_StackFrameInfo: AllStatic { private: - static int _declaringClass_offset; static int _memberName_offset; static int _bci_offset; static int _version_offset; @@ -1379,7 +1378,6 @@ private: public: // Setters - static void set_declaringClass(oop info, oop value); static void set_method_and_bci(Handle stackFrame, const methodHandle& method, int bci, TRAPS); static void set_bci(oop info, int value); diff --git a/src/hotspot/share/classfile/vmSymbols.hpp b/src/hotspot/share/classfile/vmSymbols.hpp index 88f03e5a205..73fb9296772 100644 --- a/src/hotspot/share/classfile/vmSymbols.hpp +++ b/src/hotspot/share/classfile/vmSymbols.hpp @@ -429,7 +429,6 @@ template(append_name, "append") \ template(klass_name, "klass") \ template(array_klass_name, "array_klass") \ - template(declaringClass_name, "declaringClass") \ template(memberName_name, "memberName") \ template(mid_name, "mid") \ template(cpref_name, "cpref") \ diff --git a/src/hotspot/share/prims/stackwalk.cpp b/src/hotspot/share/prims/stackwalk.cpp index cd638a84dca..3525e8ac277 100644 --- a/src/hotspot/share/prims/stackwalk.cpp +++ b/src/hotspot/share/prims/stackwalk.cpp @@ -272,9 +272,8 @@ objArrayHandle LiveFrameStream::monitors_to_object_array(GrowableArraymethod_holder()->java_mirror()); java_lang_StackFrameInfo::set_method_and_bci(stackFrame, method, bci(), THREAD); } diff --git a/src/java.base/share/classes/java/lang/StackFrameInfo.java b/src/java.base/share/classes/java/lang/StackFrameInfo.java index 3666a834218..10a6989d3c7 100644 --- a/src/java.base/share/classes/java/lang/StackFrameInfo.java +++ b/src/java.base/share/classes/java/lang/StackFrameInfo.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2017, 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 @@ -32,14 +32,12 @@ import java.lang.StackWalker.StackFrame; import java.lang.invoke.MethodType; class StackFrameInfo implements StackFrame { + private final byte RETAIN_CLASS_REF = 0x01; + private final static JavaLangInvokeAccess JLIA = SharedSecrets.getJavaLangInvokeAccess(); - // Footprint improvement: MemberName::clazz can replace - // StackFrameInfo::declaringClass. - - private final StackWalker walker; - private final Class declaringClass; + private final byte flags; private final Object memberName; private final short bci; private volatile StackTraceElement ste; @@ -49,8 +47,7 @@ class StackFrameInfo implements StackFrame { * to use */ StackFrameInfo(StackWalker walker) { - this.walker = walker; - this.declaringClass = null; + this.flags = walker.retainClassRef ? RETAIN_CLASS_REF : 0; this.bci = -1; this.memberName = JLIA.newMemberName(); } @@ -58,20 +55,20 @@ class StackFrameInfo implements StackFrame { // package-private called by StackStreamFactory to skip // the capability check Class declaringClass() { - return declaringClass; + return JLIA.getDeclaringClass(memberName); } // ----- implementation of StackFrame methods @Override public String getClassName() { - return declaringClass.getName(); + return declaringClass().getName(); } @Override public Class getDeclaringClass() { - walker.ensureAccessEnabled(RETAIN_CLASS_REFERENCE); - return declaringClass; + ensureRetainClassRefEnabled(); + return declaringClass(); } @Override @@ -81,7 +78,7 @@ class StackFrameInfo implements StackFrame { @Override public MethodType getMethodType() { - walker.ensureAccessEnabled(RETAIN_CLASS_REFERENCE); + ensureRetainClassRefEnabled(); return JLIA.getMethodType(memberName); } @@ -137,4 +134,10 @@ class StackFrameInfo implements StackFrame { } return s; } + + private void ensureRetainClassRefEnabled() { + if ((flags & RETAIN_CLASS_REF) == 0) { + throw new UnsupportedOperationException("No access to RETAIN_CLASS_REFERENCE"); + } + } } diff --git a/src/java.base/share/classes/java/lang/StackWalker.java b/src/java.base/share/classes/java/lang/StackWalker.java index a079ac53384..af20b79771c 100644 --- a/src/java.base/share/classes/java/lang/StackWalker.java +++ b/src/java.base/share/classes/java/lang/StackWalker.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2017, 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 @@ -296,6 +296,7 @@ public final class StackWalker { private final Set