This commit is contained in:
Alejandro Murillo 2016-07-01 16:50:34 -07:00
commit 2557a71e8c
455 changed files with 7863 additions and 6223 deletions

View File

@ -186,7 +186,8 @@ To build hotspot and import it into the JDK: "mx make hotspot import-hotspot"
# Might be building with JDK8 which has cacerts under jre/ # Might be building with JDK8 which has cacerts under jre/
srcCerts = join(mx.get_jdk(tag='default').home, 'jre', 'lib', 'security', 'cacerts') srcCerts = join(mx.get_jdk(tag='default').home, 'jre', 'lib', 'security', 'cacerts')
dstCerts = join(jdkImageDir, 'lib', 'security', 'cacerts') dstCerts = join(jdkImageDir, 'lib', 'security', 'cacerts')
shutil.copyfile(srcCerts, dstCerts) if srcCerts != dstCerts:
shutil.copyfile(srcCerts, dstCerts)
_create_jdk_bundle(jdkBuildDir, _vm.debugLevel, jdkImageDir) _create_jdk_bundle(jdkBuildDir, _vm.debugLevel, jdkImageDir)

View File

@ -149,7 +149,6 @@ suite = {
"subDir" : "src/jdk.vm.ci/share/classes", "subDir" : "src/jdk.vm.ci/share/classes",
"sourceDirs" : ["src"], "sourceDirs" : ["src"],
"dependencies" : [ "dependencies" : [
"jdk.vm.ci.hotspotvmconfig",
"jdk.vm.ci.common", "jdk.vm.ci.common",
"jdk.vm.ci.runtime", "jdk.vm.ci.runtime",
"jdk.vm.ci.services", "jdk.vm.ci.services",
@ -175,14 +174,6 @@ suite = {
"workingSets" : "API,JVMCI", "workingSets" : "API,JVMCI",
}, },
"jdk.vm.ci.hotspotvmconfig" : {
"subDir" : "src/jdk.vm.ci/share/classes",
"sourceDirs" : ["src"],
"checkstyle" : "jdk.vm.ci.services",
"javaCompliance" : "9",
"workingSets" : "JVMCI,HotSpot",
},
"jdk.vm.ci.hotspot.aarch64" : { "jdk.vm.ci.hotspot.aarch64" : {
"subDir" : "src/jdk.vm.ci/share/classes", "subDir" : "src/jdk.vm.ci/share/classes",
"sourceDirs" : ["src"], "sourceDirs" : ["src"],
@ -248,13 +239,6 @@ suite = {
], ],
}, },
"JVMCI_HOTSPOTVMCONFIG" : {
"subDir" : "src/jdk.vm.ci/share/classes",
"dependencies" : [
"jdk.vm.ci.hotspotvmconfig",
],
},
"JVMCI_HOTSPOT" : { "JVMCI_HOTSPOT" : {
"subDir" : "src/jdk.vm.ci/share/classes", "subDir" : "src/jdk.vm.ci/share/classes",
"dependencies" : [ "dependencies" : [
@ -263,7 +247,6 @@ suite = {
"jdk.vm.ci.hotspot.sparc", "jdk.vm.ci.hotspot.sparc",
], ],
"distDependencies" : [ "distDependencies" : [
"JVMCI_HOTSPOTVMCONFIG",
"JVMCI_SERVICES", "JVMCI_SERVICES",
"JVMCI_API", "JVMCI_API",
], ],

View File

@ -23,7 +23,10 @@
# questions. # questions.
# #
GTEST_TEST_SRC := $(HOTSPOT_TOPDIR)/test/native $(eval $(call IncludeCustomExtension, hotspot, lib/CompileGtest.gmk))
GTEST_TEST_SRC += $(HOTSPOT_TOPDIR)/test/native
GTEST_LAUNCHER_SRC := $(HOTSPOT_TOPDIR)/test/native/gtestLauncher.cpp
GTEST_FRAMEWORK_SRC := $(SRC_ROOT)/test/fmw/gtest GTEST_FRAMEWORK_SRC := $(SRC_ROOT)/test/fmw/gtest
# On Windows, there are no internal debug symbols so must set copying to true # On Windows, there are no internal debug symbols so must set copying to true
@ -65,7 +68,7 @@ $(eval $(call SetupNativeCompilation, BUILD_GTEST_LIBJVM, \
$(BUILD_LIBJVM_ALL_OBJS)), \ $(BUILD_LIBJVM_ALL_OBJS)), \
CFLAGS := $(JVM_CFLAGS) -I$(GTEST_FRAMEWORK_SRC) \ CFLAGS := $(JVM_CFLAGS) -I$(GTEST_FRAMEWORK_SRC) \
-I$(GTEST_FRAMEWORK_SRC)/include \ -I$(GTEST_FRAMEWORK_SRC)/include \
-I$(GTEST_TEST_SRC), \ $(addprefix -I,$(GTEST_TEST_SRC)), \
CFLAGS_windows := /EHsc, \ CFLAGS_windows := /EHsc, \
CFLAGS_solaris := -DGTEST_HAS_EXCEPTIONS=0 -library=stlport4, \ CFLAGS_solaris := -DGTEST_HAS_EXCEPTIONS=0 -library=stlport4, \
CFLAGS_macosx := -DGTEST_OS_MAC=1, \ CFLAGS_macosx := -DGTEST_OS_MAC=1, \
@ -73,7 +76,7 @@ $(eval $(call SetupNativeCompilation, BUILD_GTEST_LIBJVM, \
CXXFLAGS_DEBUG_SYMBOLS := $(JVM_CFLAGS_SYMBOLS), \ CXXFLAGS_DEBUG_SYMBOLS := $(JVM_CFLAGS_SYMBOLS), \
DISABLED_WARNINGS_gcc := undef, \ DISABLED_WARNINGS_gcc := undef, \
DISABLED_WARNINGS_clang := undef switch format-nonliteral \ DISABLED_WARNINGS_clang := undef switch format-nonliteral \
tautological-undefined-compare, \ tautological-undefined-compare $(BUILD_LIBJVM_DISABLED_WARNINGS_clang), \
DISABLED_WARNINGS_solstudio := identexpected, \ DISABLED_WARNINGS_solstudio := identexpected, \
LDFLAGS := $(JVM_LDFLAGS), \ LDFLAGS := $(JVM_LDFLAGS), \
LDFLAGS_solaris := -library=stlport4 $(call SET_SHARED_LIBRARY_ORIGIN), \ LDFLAGS_solaris := -library=stlport4 $(call SET_SHARED_LIBRARY_ORIGIN), \
@ -83,6 +86,8 @@ $(eval $(call SetupNativeCompilation, BUILD_GTEST_LIBJVM, \
USE_MAPFILE_FOR_SYMBOLS := true, \ USE_MAPFILE_FOR_SYMBOLS := true, \
COPY_DEBUG_SYMBOLS := $(GTEST_COPY_DEBUG_SYMBOLS), \ COPY_DEBUG_SYMBOLS := $(GTEST_COPY_DEBUG_SYMBOLS), \
ZIP_EXTERNAL_DEBUG_SYMBOLS := false, \ ZIP_EXTERNAL_DEBUG_SYMBOLS := false, \
PRECOMPILED_HEADER := $(JVM_PRECOMPILED_HEADER), \
PRECOMPILED_HEADER_EXCLUDE := gtest-all.cc gtestMain.cpp, \
)) ))
TARGETS += $(BUILD_GTEST_LIBJVM) TARGETS += $(BUILD_GTEST_LIBJVM)
@ -93,7 +98,7 @@ $(eval $(call SetupNativeCompilation, BUILD_GTEST_LAUNCHER, \
TOOLCHAIN := $(JVM_TOOLCHAIN), \ TOOLCHAIN := $(JVM_TOOLCHAIN), \
PROGRAM := gtestLauncher, \ PROGRAM := gtestLauncher, \
OUTPUT_DIR := $(JVM_OUTPUTDIR)/gtest, \ OUTPUT_DIR := $(JVM_OUTPUTDIR)/gtest, \
EXTRA_FILES := $(GTEST_TEST_SRC)/gtestLauncher.cpp, \ EXTRA_FILES := $(GTEST_LAUNCHER_SRC), \
OBJECT_DIR := $(JVM_OUTPUTDIR)/gtest/launcher-objs, \ OBJECT_DIR := $(JVM_OUTPUTDIR)/gtest/launcher-objs, \
CFLAGS := $(JVM_CFLAGS) -I$(GTEST_FRAMEWORK_SRC) \ CFLAGS := $(JVM_CFLAGS) -I$(GTEST_FRAMEWORK_SRC) \
-I$(GTEST_FRAMEWORK_SRC)/include, \ -I$(GTEST_FRAMEWORK_SRC)/include, \

View File

@ -35,6 +35,7 @@ endif
ifeq ($(OPENJDK_TARGET_OS), linux) ifeq ($(OPENJDK_TARGET_OS), linux)
BUILD_LIBJVM_ostream.cpp_CXXFLAGS := -D_FILE_OFFSET_BITS=64 BUILD_LIBJVM_ostream.cpp_CXXFLAGS := -D_FILE_OFFSET_BITS=64
BUILD_LIBJVM_logFileOutput.cpp_CXXFLAGS := -D_FILE_OFFSET_BITS=64
ifeq ($(OPENJDK_TARGET_CPU_ARCH), x86) ifeq ($(OPENJDK_TARGET_CPU_ARCH), x86)
BUILD_LIBJVM_sharedRuntimeTrig.cpp_CXXFLAGS := -DNO_PCH $(CXX_O_FLAG_NONE) BUILD_LIBJVM_sharedRuntimeTrig.cpp_CXXFLAGS := -DNO_PCH $(CXX_O_FLAG_NONE)

View File

@ -6629,6 +6629,82 @@ operand cmpOpU()
%} %}
%} %}
// used for certain integral comparisons which can be
// converted to cbxx or tbxx instructions
operand cmpOpEqNe()
%{
match(Bool);
match(CmpOp);
op_cost(0);
predicate(n->as_Bool()->_test._test == BoolTest::ne
|| n->as_Bool()->_test._test == BoolTest::eq);
format %{ "" %}
interface(COND_INTER) %{
equal(0x0, "eq");
not_equal(0x1, "ne");
less(0xb, "lt");
greater_equal(0xa, "ge");
less_equal(0xd, "le");
greater(0xc, "gt");
overflow(0x6, "vs");
no_overflow(0x7, "vc");
%}
%}
// used for certain integral comparisons which can be
// converted to cbxx or tbxx instructions
operand cmpOpLtGe()
%{
match(Bool);
match(CmpOp);
op_cost(0);
predicate(n->as_Bool()->_test._test == BoolTest::lt
|| n->as_Bool()->_test._test == BoolTest::ge);
format %{ "" %}
interface(COND_INTER) %{
equal(0x0, "eq");
not_equal(0x1, "ne");
less(0xb, "lt");
greater_equal(0xa, "ge");
less_equal(0xd, "le");
greater(0xc, "gt");
overflow(0x6, "vs");
no_overflow(0x7, "vc");
%}
%}
// used for certain unsigned integral comparisons which can be
// converted to cbxx or tbxx instructions
operand cmpOpUEqNeLtGe()
%{
match(Bool);
match(CmpOp);
op_cost(0);
predicate(n->as_Bool()->_test._test == BoolTest::eq
|| n->as_Bool()->_test._test == BoolTest::ne
|| n->as_Bool()->_test._test == BoolTest::lt
|| n->as_Bool()->_test._test == BoolTest::ge);
format %{ "" %}
interface(COND_INTER) %{
equal(0x0, "eq");
not_equal(0x1, "ne");
less(0xb, "lt");
greater_equal(0xa, "ge");
less_equal(0xd, "le");
greater(0xc, "gt");
overflow(0x6, "vs");
no_overflow(0x7, "vc");
%}
%}
// Special operand allowing long args to int ops to be truncated for free // Special operand allowing long args to int ops to be truncated for free
operand iRegL2I(iRegL reg) %{ operand iRegL2I(iRegL reg) %{
@ -14286,10 +14362,8 @@ instruct branchConU(cmpOpU cmp, rFlagsRegU cr, label lbl)
// shorter than (cmp; branch), have the additional benefit of not // shorter than (cmp; branch), have the additional benefit of not
// killing the flags. // killing the flags.
instruct cmpI_imm0_branch(cmpOp cmp, iRegIorL2I op1, immI0 op2, label labl, rFlagsReg cr) %{ instruct cmpI_imm0_branch(cmpOpEqNe cmp, iRegIorL2I op1, immI0 op2, label labl, rFlagsReg cr) %{
match(If cmp (CmpI op1 op2)); match(If cmp (CmpI op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::ne
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14305,10 +14379,8 @@ instruct cmpI_imm0_branch(cmpOp cmp, iRegIorL2I op1, immI0 op2, label labl, rFla
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct cmpL_imm0_branch(cmpOp cmp, iRegL op1, immL0 op2, label labl, rFlagsReg cr) %{ instruct cmpL_imm0_branch(cmpOpEqNe cmp, iRegL op1, immL0 op2, label labl, rFlagsReg cr) %{
match(If cmp (CmpL op1 op2)); match(If cmp (CmpL op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::ne
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14324,10 +14396,8 @@ instruct cmpL_imm0_branch(cmpOp cmp, iRegL op1, immL0 op2, label labl, rFlagsReg
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct cmpP_imm0_branch(cmpOp cmp, iRegP op1, immP0 op2, label labl, rFlagsReg cr) %{ instruct cmpP_imm0_branch(cmpOpEqNe cmp, iRegP op1, immP0 op2, label labl, rFlagsReg cr) %{
match(If cmp (CmpP op1 op2)); match(If cmp (CmpP op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::ne
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14343,10 +14413,8 @@ instruct cmpP_imm0_branch(cmpOp cmp, iRegP op1, immP0 op2, label labl, rFlagsReg
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct cmpN_imm0_branch(cmpOp cmp, iRegN op1, immN0 op2, label labl, rFlagsReg cr) %{ instruct cmpN_imm0_branch(cmpOpEqNe cmp, iRegN op1, immN0 op2, label labl, rFlagsReg cr) %{
match(If cmp (CmpN op1 op2)); match(If cmp (CmpN op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::ne
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14362,10 +14430,8 @@ instruct cmpN_imm0_branch(cmpOp cmp, iRegN op1, immN0 op2, label labl, rFlagsReg
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct cmpP_narrowOop_imm0_branch(cmpOp cmp, iRegN oop, immP0 zero, label labl, rFlagsReg cr) %{ instruct cmpP_narrowOop_imm0_branch(cmpOpEqNe cmp, iRegN oop, immP0 zero, label labl, rFlagsReg cr) %{
match(If cmp (CmpP (DecodeN oop) zero)); match(If cmp (CmpP (DecodeN oop) zero));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::ne
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14381,12 +14447,8 @@ instruct cmpP_narrowOop_imm0_branch(cmpOp cmp, iRegN oop, immP0 zero, label labl
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct cmpUI_imm0_branch(cmpOpU cmp, iRegIorL2I op1, immI0 op2, label labl, rFlagsRegU cr) %{ instruct cmpUI_imm0_branch(cmpOpUEqNeLtGe cmp, iRegIorL2I op1, immI0 op2, label labl, rFlagsRegU cr) %{
match(If cmp (CmpU op1 op2)); match(If cmp (CmpU op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::ne
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq
|| n->in(1)->as_Bool()->_test._test == BoolTest::gt
|| n->in(1)->as_Bool()->_test._test == BoolTest::le);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14402,12 +14464,8 @@ instruct cmpUI_imm0_branch(cmpOpU cmp, iRegIorL2I op1, immI0 op2, label labl, rF
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct cmpUL_imm0_branch(cmpOpU cmp, iRegL op1, immL0 op2, label labl, rFlagsRegU cr) %{ instruct cmpUL_imm0_branch(cmpOpUEqNeLtGe cmp, iRegL op1, immL0 op2, label labl, rFlagsRegU cr) %{
match(If cmp (CmpU op1 op2)); match(If cmp (CmpU op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::ne
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq
|| n->in(1)->as_Bool()->_test._test == BoolTest::gt
|| n->in(1)->as_Bool()->_test._test == BoolTest::le);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14426,10 +14484,8 @@ instruct cmpUL_imm0_branch(cmpOpU cmp, iRegL op1, immL0 op2, label labl, rFlagsR
// Test bit and Branch // Test bit and Branch
// Patterns for short (< 32KiB) variants // Patterns for short (< 32KiB) variants
instruct cmpL_branch_sign(cmpOp cmp, iRegL op1, immL0 op2, label labl) %{ instruct cmpL_branch_sign(cmpOpLtGe cmp, iRegL op1, immL0 op2, label labl) %{
match(If cmp (CmpL op1 op2)); match(If cmp (CmpL op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::lt
|| n->in(1)->as_Bool()->_test._test == BoolTest::ge);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14444,10 +14500,8 @@ instruct cmpL_branch_sign(cmpOp cmp, iRegL op1, immL0 op2, label labl) %{
ins_short_branch(1); ins_short_branch(1);
%} %}
instruct cmpI_branch_sign(cmpOp cmp, iRegIorL2I op1, immI0 op2, label labl) %{ instruct cmpI_branch_sign(cmpOpLtGe cmp, iRegIorL2I op1, immI0 op2, label labl) %{
match(If cmp (CmpI op1 op2)); match(If cmp (CmpI op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::lt
|| n->in(1)->as_Bool()->_test._test == BoolTest::ge);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14462,11 +14516,9 @@ instruct cmpI_branch_sign(cmpOp cmp, iRegIorL2I op1, immI0 op2, label labl) %{
ins_short_branch(1); ins_short_branch(1);
%} %}
instruct cmpL_branch_bit(cmpOp cmp, iRegL op1, immL op2, immL0 op3, label labl) %{ instruct cmpL_branch_bit(cmpOpEqNe cmp, iRegL op1, immL op2, immL0 op3, label labl) %{
match(If cmp (CmpL (AndL op1 op2) op3)); match(If cmp (CmpL (AndL op1 op2) op3));
predicate((n->in(1)->as_Bool()->_test._test == BoolTest::ne predicate(is_power_of_2(n->in(2)->in(1)->in(2)->get_long()));
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq)
&& is_power_of_2(n->in(2)->in(1)->in(2)->get_long()));
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14481,11 +14533,9 @@ instruct cmpL_branch_bit(cmpOp cmp, iRegL op1, immL op2, immL0 op3, label labl)
ins_short_branch(1); ins_short_branch(1);
%} %}
instruct cmpI_branch_bit(cmpOp cmp, iRegIorL2I op1, immI op2, immI0 op3, label labl) %{ instruct cmpI_branch_bit(cmpOpEqNe cmp, iRegIorL2I op1, immI op2, immI0 op3, label labl) %{
match(If cmp (CmpI (AndI op1 op2) op3)); match(If cmp (CmpI (AndI op1 op2) op3));
predicate((n->in(1)->as_Bool()->_test._test == BoolTest::ne predicate(is_power_of_2(n->in(2)->in(1)->in(2)->get_int()));
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq)
&& is_power_of_2(n->in(2)->in(1)->in(2)->get_int()));
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14501,10 +14551,8 @@ instruct cmpI_branch_bit(cmpOp cmp, iRegIorL2I op1, immI op2, immI0 op3, label l
%} %}
// And far variants // And far variants
instruct far_cmpL_branch_sign(cmpOp cmp, iRegL op1, immL0 op2, label labl) %{ instruct far_cmpL_branch_sign(cmpOpLtGe cmp, iRegL op1, immL0 op2, label labl) %{
match(If cmp (CmpL op1 op2)); match(If cmp (CmpL op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::lt
|| n->in(1)->as_Bool()->_test._test == BoolTest::ge);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14518,10 +14566,8 @@ instruct far_cmpL_branch_sign(cmpOp cmp, iRegL op1, immL0 op2, label labl) %{
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct far_cmpI_branch_sign(cmpOp cmp, iRegIorL2I op1, immI0 op2, label labl) %{ instruct far_cmpI_branch_sign(cmpOpLtGe cmp, iRegIorL2I op1, immI0 op2, label labl) %{
match(If cmp (CmpI op1 op2)); match(If cmp (CmpI op1 op2));
predicate(n->in(1)->as_Bool()->_test._test == BoolTest::lt
|| n->in(1)->as_Bool()->_test._test == BoolTest::ge);
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14535,11 +14581,9 @@ instruct far_cmpI_branch_sign(cmpOp cmp, iRegIorL2I op1, immI0 op2, label labl)
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct far_cmpL_branch_bit(cmpOp cmp, iRegL op1, immL op2, immL0 op3, label labl) %{ instruct far_cmpL_branch_bit(cmpOpEqNe cmp, iRegL op1, immL op2, immL0 op3, label labl) %{
match(If cmp (CmpL (AndL op1 op2) op3)); match(If cmp (CmpL (AndL op1 op2) op3));
predicate((n->in(1)->as_Bool()->_test._test == BoolTest::ne predicate(is_power_of_2(n->in(2)->in(1)->in(2)->get_long()));
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq)
&& is_power_of_2(n->in(2)->in(1)->in(2)->get_long()));
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);
@ -14553,11 +14597,9 @@ instruct far_cmpL_branch_bit(cmpOp cmp, iRegL op1, immL op2, immL0 op3, label la
ins_pipe(pipe_cmp_branch); ins_pipe(pipe_cmp_branch);
%} %}
instruct far_cmpI_branch_bit(cmpOp cmp, iRegIorL2I op1, immI op2, immI0 op3, label labl) %{ instruct far_cmpI_branch_bit(cmpOpEqNe cmp, iRegIorL2I op1, immI op2, immI0 op3, label labl) %{
match(If cmp (CmpI (AndI op1 op2) op3)); match(If cmp (CmpI (AndI op1 op2) op3));
predicate((n->in(1)->as_Bool()->_test._test == BoolTest::ne predicate(is_power_of_2(n->in(2)->in(1)->in(2)->get_int()));
|| n->in(1)->as_Bool()->_test._test == BoolTest::eq)
&& is_power_of_2(n->in(2)->in(1)->in(2)->get_int()));
effect(USE labl); effect(USE labl);
ins_cost(BRANCH_COST); ins_cost(BRANCH_COST);

View File

@ -630,7 +630,11 @@ void TemplateInterpreterGenerator::generate_counter_overflow(Label& do_continue)
__ b(do_continue); __ b(do_continue);
} }
// See if we've got enough room on the stack for locals plus overhead. // See if we've got enough room on the stack for locals plus overhead
// below JavaThread::stack_overflow_limit(). If not, throw a StackOverflowError
// without going through the signal handler, i.e., reserved and yellow zones
// will not be made usable. The shadow zone must suffice to handle the
// overflow.
// The expression stack grows down incrementally, so the normal guard // The expression stack grows down incrementally, so the normal guard
// page mechanism will work for that. // page mechanism will work for that.
// //
@ -674,40 +678,25 @@ void TemplateInterpreterGenerator::generate_stack_overflow_check(void) {
// compute rsp as if this were going to be the last frame on // compute rsp as if this were going to be the last frame on
// the stack before the red zone // the stack before the red zone
const Address stack_base(rthread, Thread::stack_base_offset());
const Address stack_size(rthread, Thread::stack_size_offset());
// locals + overhead, in bytes // locals + overhead, in bytes
__ mov(r0, overhead_size); __ mov(r0, overhead_size);
__ add(r0, r0, r3, Assembler::LSL, Interpreter::logStackElementSize); // 2 slots per parameter. __ add(r0, r0, r3, Assembler::LSL, Interpreter::logStackElementSize); // 2 slots per parameter.
__ ldr(rscratch1, stack_base); const Address stack_limit(rthread, JavaThread::stack_overflow_limit_offset());
__ ldr(rscratch2, stack_size); __ ldr(rscratch1, stack_limit);
#ifdef ASSERT #ifdef ASSERT
Label stack_base_okay, stack_size_okay; Label limit_okay;
// verify that thread stack base is non-zero // Verify that thread stack limit is non-zero.
__ cbnz(rscratch1, stack_base_okay); __ cbnz(rscratch1, limit_okay);
__ stop("stack base is zero"); __ stop("stack overflow limit is zero");
__ bind(stack_base_okay); __ bind(limit_okay);
// verify that thread stack size is non-zero
__ cbnz(rscratch2, stack_size_okay);
__ stop("stack size is zero");
__ bind(stack_size_okay);
#endif #endif
// Add stack base to locals and subtract stack size // Add stack limit to locals.
__ sub(rscratch1, rscratch1, rscratch2); // Stack limit
__ add(r0, r0, rscratch1); __ add(r0, r0, rscratch1);
// Use the bigger size for banging. // Check against the current stack bottom.
const int max_bang_size = MAX2(JavaThread::stack_shadow_zone_size(),
JavaThread::stack_red_zone_size() + JavaThread::stack_yellow_zone_size());
// add in the red and yellow zone sizes
__ add(r0, r0, max_bang_size * 2);
// check against the current stack bottom
__ cmp(sp, r0); __ cmp(sp, r0);
__ br(Assembler::HI, after_frame_check); __ br(Assembler::HI, after_frame_check);
@ -1088,9 +1077,9 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
__ ldr(r2, constMethod); __ ldr(r2, constMethod);
__ load_unsigned_short(r2, size_of_parameters); __ load_unsigned_short(r2, size_of_parameters);
// native calls don't need the stack size check since they have no // Native calls don't need the stack size check since they have no
// expression stack and the arguments are already on the stack and // expression stack and the arguments are already on the stack and
// we only add a handful of words to the stack // we only add a handful of words to the stack.
// rmethod: Method* // rmethod: Method*
// r2: size of parameters // r2: size of parameters

View File

@ -706,9 +706,13 @@ class Assembler : public AbstractAssembler {
TW_OPCODE = (31u << OPCODE_SHIFT | 4u << 1), TW_OPCODE = (31u << OPCODE_SHIFT | 4u << 1),
// Atomics. // Atomics.
LBARX_OPCODE = (31u << OPCODE_SHIFT | 52u << 1),
LHARX_OPCODE = (31u << OPCODE_SHIFT | 116u << 1),
LWARX_OPCODE = (31u << OPCODE_SHIFT | 20u << 1), LWARX_OPCODE = (31u << OPCODE_SHIFT | 20u << 1),
LDARX_OPCODE = (31u << OPCODE_SHIFT | 84u << 1), LDARX_OPCODE = (31u << OPCODE_SHIFT | 84u << 1),
LQARX_OPCODE = (31u << OPCODE_SHIFT | 276u << 1), LQARX_OPCODE = (31u << OPCODE_SHIFT | 276u << 1),
STBCX_OPCODE = (31u << OPCODE_SHIFT | 694u << 1),
STHCX_OPCODE = (31u << OPCODE_SHIFT | 726u << 1),
STWCX_OPCODE = (31u << OPCODE_SHIFT | 150u << 1), STWCX_OPCODE = (31u << OPCODE_SHIFT | 150u << 1),
STDCX_OPCODE = (31u << OPCODE_SHIFT | 214u << 1), STDCX_OPCODE = (31u << OPCODE_SHIFT | 214u << 1),
STQCX_OPCODE = (31u << OPCODE_SHIFT | 182u << 1) STQCX_OPCODE = (31u << OPCODE_SHIFT | 182u << 1)
@ -1796,13 +1800,19 @@ class Assembler : public AbstractAssembler {
inline void waitrsv(); // >=Power7 inline void waitrsv(); // >=Power7
// atomics // atomics
inline void lbarx_unchecked(Register d, Register a, Register b, int eh1 = 0); // >=Power 8
inline void lharx_unchecked(Register d, Register a, Register b, int eh1 = 0); // >=Power 8
inline void lwarx_unchecked(Register d, Register a, Register b, int eh1 = 0); inline void lwarx_unchecked(Register d, Register a, Register b, int eh1 = 0);
inline void ldarx_unchecked(Register d, Register a, Register b, int eh1 = 0); inline void ldarx_unchecked(Register d, Register a, Register b, int eh1 = 0);
inline void lqarx_unchecked(Register d, Register a, Register b, int eh1 = 0); inline void lqarx_unchecked(Register d, Register a, Register b, int eh1 = 0); // >=Power 8
inline bool lxarx_hint_exclusive_access(); inline bool lxarx_hint_exclusive_access();
inline void lbarx( Register d, Register a, Register b, bool hint_exclusive_access = false);
inline void lharx( Register d, Register a, Register b, bool hint_exclusive_access = false);
inline void lwarx( Register d, Register a, Register b, bool hint_exclusive_access = false); inline void lwarx( Register d, Register a, Register b, bool hint_exclusive_access = false);
inline void ldarx( Register d, Register a, Register b, bool hint_exclusive_access = false); inline void ldarx( Register d, Register a, Register b, bool hint_exclusive_access = false);
inline void lqarx( Register d, Register a, Register b, bool hint_exclusive_access = false); inline void lqarx( Register d, Register a, Register b, bool hint_exclusive_access = false);
inline void stbcx_( Register s, Register a, Register b);
inline void sthcx_( Register s, Register a, Register b);
inline void stwcx_( Register s, Register a, Register b); inline void stwcx_( Register s, Register a, Register b);
inline void stdcx_( Register s, Register a, Register b); inline void stdcx_( Register s, Register a, Register b);
inline void stqcx_( Register s, Register a, Register b); inline void stqcx_( Register s, Register a, Register b);
@ -2169,12 +2179,18 @@ class Assembler : public AbstractAssembler {
inline void dcbtstct(Register s2, int ct); inline void dcbtstct(Register s2, int ct);
// Atomics: use ra0mem to disallow R0 as base. // Atomics: use ra0mem to disallow R0 as base.
inline void lbarx_unchecked(Register d, Register b, int eh1);
inline void lharx_unchecked(Register d, Register b, int eh1);
inline void lwarx_unchecked(Register d, Register b, int eh1); inline void lwarx_unchecked(Register d, Register b, int eh1);
inline void ldarx_unchecked(Register d, Register b, int eh1); inline void ldarx_unchecked(Register d, Register b, int eh1);
inline void lqarx_unchecked(Register d, Register b, int eh1); inline void lqarx_unchecked(Register d, Register b, int eh1);
inline void lbarx( Register d, Register b, bool hint_exclusive_access);
inline void lharx( Register d, Register b, bool hint_exclusive_access);
inline void lwarx( Register d, Register b, bool hint_exclusive_access); inline void lwarx( Register d, Register b, bool hint_exclusive_access);
inline void ldarx( Register d, Register b, bool hint_exclusive_access); inline void ldarx( Register d, Register b, bool hint_exclusive_access);
inline void lqarx( Register d, Register b, bool hint_exclusive_access); inline void lqarx( Register d, Register b, bool hint_exclusive_access);
inline void stbcx_(Register s, Register b);
inline void sthcx_(Register s, Register b);
inline void stwcx_(Register s, Register b); inline void stwcx_(Register s, Register b);
inline void stdcx_(Register s, Register b); inline void stdcx_(Register s, Register b);
inline void stqcx_(Register s, Register b); inline void stqcx_(Register s, Register b);

View File

@ -594,13 +594,19 @@ inline void Assembler::waitrsv() { emit_int32( WAIT_OPCODE | 1<<(31-10)); } // W
// atomics // atomics
// Use ra0mem to disallow R0 as base. // Use ra0mem to disallow R0 as base.
inline void Assembler::lbarx_unchecked(Register d, Register a, Register b, int eh1) { emit_int32( LBARX_OPCODE | rt(d) | ra0mem(a) | rb(b) | eh(eh1)); }
inline void Assembler::lharx_unchecked(Register d, Register a, Register b, int eh1) { emit_int32( LHARX_OPCODE | rt(d) | ra0mem(a) | rb(b) | eh(eh1)); }
inline void Assembler::lwarx_unchecked(Register d, Register a, Register b, int eh1) { emit_int32( LWARX_OPCODE | rt(d) | ra0mem(a) | rb(b) | eh(eh1)); } inline void Assembler::lwarx_unchecked(Register d, Register a, Register b, int eh1) { emit_int32( LWARX_OPCODE | rt(d) | ra0mem(a) | rb(b) | eh(eh1)); }
inline void Assembler::ldarx_unchecked(Register d, Register a, Register b, int eh1) { emit_int32( LDARX_OPCODE | rt(d) | ra0mem(a) | rb(b) | eh(eh1)); } inline void Assembler::ldarx_unchecked(Register d, Register a, Register b, int eh1) { emit_int32( LDARX_OPCODE | rt(d) | ra0mem(a) | rb(b) | eh(eh1)); }
inline void Assembler::lqarx_unchecked(Register d, Register a, Register b, int eh1) { emit_int32( LQARX_OPCODE | rt(d) | ra0mem(a) | rb(b) | eh(eh1)); } inline void Assembler::lqarx_unchecked(Register d, Register a, Register b, int eh1) { emit_int32( LQARX_OPCODE | rt(d) | ra0mem(a) | rb(b) | eh(eh1)); }
inline bool Assembler::lxarx_hint_exclusive_access() { return VM_Version::has_lxarxeh(); } inline bool Assembler::lxarx_hint_exclusive_access() { return VM_Version::has_lxarxeh(); }
inline void Assembler::lbarx( Register d, Register a, Register b, bool hint_exclusive_access) { lbarx_unchecked(d, a, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::lharx( Register d, Register a, Register b, bool hint_exclusive_access) { lharx_unchecked(d, a, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::lwarx( Register d, Register a, Register b, bool hint_exclusive_access) { lwarx_unchecked(d, a, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); } inline void Assembler::lwarx( Register d, Register a, Register b, bool hint_exclusive_access) { lwarx_unchecked(d, a, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::ldarx( Register d, Register a, Register b, bool hint_exclusive_access) { ldarx_unchecked(d, a, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); } inline void Assembler::ldarx( Register d, Register a, Register b, bool hint_exclusive_access) { ldarx_unchecked(d, a, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::lqarx( Register d, Register a, Register b, bool hint_exclusive_access) { lqarx_unchecked(d, a, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); } inline void Assembler::lqarx( Register d, Register a, Register b, bool hint_exclusive_access) { lqarx_unchecked(d, a, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::stbcx_(Register s, Register a, Register b) { emit_int32( STBCX_OPCODE | rs(s) | ra0mem(a) | rb(b) | rc(1)); }
inline void Assembler::sthcx_(Register s, Register a, Register b) { emit_int32( STHCX_OPCODE | rs(s) | ra0mem(a) | rb(b) | rc(1)); }
inline void Assembler::stwcx_(Register s, Register a, Register b) { emit_int32( STWCX_OPCODE | rs(s) | ra0mem(a) | rb(b) | rc(1)); } inline void Assembler::stwcx_(Register s, Register a, Register b) { emit_int32( STWCX_OPCODE | rs(s) | ra0mem(a) | rb(b) | rc(1)); }
inline void Assembler::stdcx_(Register s, Register a, Register b) { emit_int32( STDCX_OPCODE | rs(s) | ra0mem(a) | rb(b) | rc(1)); } inline void Assembler::stdcx_(Register s, Register a, Register b) { emit_int32( STDCX_OPCODE | rs(s) | ra0mem(a) | rb(b) | rc(1)); }
inline void Assembler::stqcx_(Register s, Register a, Register b) { emit_int32( STQCX_OPCODE | rs(s) | ra0mem(a) | rb(b) | rc(1)); } inline void Assembler::stqcx_(Register s, Register a, Register b) { emit_int32( STQCX_OPCODE | rs(s) | ra0mem(a) | rb(b) | rc(1)); }
@ -933,12 +939,18 @@ inline void Assembler::dcbtst( Register s2) { emit_int32( DCBTST_OPCOD
inline void Assembler::dcbtstct(Register s2, int ct) { emit_int32( DCBTST_OPCODE | rb(s2) | thct(ct)); } inline void Assembler::dcbtstct(Register s2, int ct) { emit_int32( DCBTST_OPCODE | rb(s2) | thct(ct)); }
// ra0 version // ra0 version
inline void Assembler::lbarx_unchecked(Register d, Register b, int eh1) { emit_int32( LBARX_OPCODE | rt(d) | rb(b) | eh(eh1)); }
inline void Assembler::lharx_unchecked(Register d, Register b, int eh1) { emit_int32( LHARX_OPCODE | rt(d) | rb(b) | eh(eh1)); }
inline void Assembler::lwarx_unchecked(Register d, Register b, int eh1) { emit_int32( LWARX_OPCODE | rt(d) | rb(b) | eh(eh1)); } inline void Assembler::lwarx_unchecked(Register d, Register b, int eh1) { emit_int32( LWARX_OPCODE | rt(d) | rb(b) | eh(eh1)); }
inline void Assembler::ldarx_unchecked(Register d, Register b, int eh1) { emit_int32( LDARX_OPCODE | rt(d) | rb(b) | eh(eh1)); } inline void Assembler::ldarx_unchecked(Register d, Register b, int eh1) { emit_int32( LDARX_OPCODE | rt(d) | rb(b) | eh(eh1)); }
inline void Assembler::lqarx_unchecked(Register d, Register b, int eh1) { emit_int32( LQARX_OPCODE | rt(d) | rb(b) | eh(eh1)); } inline void Assembler::lqarx_unchecked(Register d, Register b, int eh1) { emit_int32( LQARX_OPCODE | rt(d) | rb(b) | eh(eh1)); }
inline void Assembler::lbarx( Register d, Register b, bool hint_exclusive_access){ lbarx_unchecked(d, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::lharx( Register d, Register b, bool hint_exclusive_access){ lharx_unchecked(d, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::lwarx( Register d, Register b, bool hint_exclusive_access){ lwarx_unchecked(d, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); } inline void Assembler::lwarx( Register d, Register b, bool hint_exclusive_access){ lwarx_unchecked(d, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::ldarx( Register d, Register b, bool hint_exclusive_access){ ldarx_unchecked(d, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); } inline void Assembler::ldarx( Register d, Register b, bool hint_exclusive_access){ ldarx_unchecked(d, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::lqarx( Register d, Register b, bool hint_exclusive_access){ lqarx_unchecked(d, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); } inline void Assembler::lqarx( Register d, Register b, bool hint_exclusive_access){ lqarx_unchecked(d, b, (hint_exclusive_access && lxarx_hint_exclusive_access() && UseExtendedLoadAndReserveInstructionsPPC64) ? 1 : 0); }
inline void Assembler::stbcx_(Register s, Register b) { emit_int32( STBCX_OPCODE | rs(s) | rb(b) | rc(1)); }
inline void Assembler::sthcx_(Register s, Register b) { emit_int32( STHCX_OPCODE | rs(s) | rb(b) | rc(1)); }
inline void Assembler::stwcx_(Register s, Register b) { emit_int32( STWCX_OPCODE | rs(s) | rb(b) | rc(1)); } inline void Assembler::stwcx_(Register s, Register b) { emit_int32( STWCX_OPCODE | rs(s) | rb(b) | rc(1)); }
inline void Assembler::stdcx_(Register s, Register b) { emit_int32( STDCX_OPCODE | rs(s) | rb(b) | rc(1)); } inline void Assembler::stdcx_(Register s, Register b) { emit_int32( STDCX_OPCODE | rs(s) | rb(b) | rc(1)); }
inline void Assembler::stqcx_(Register s, Register b) { emit_int32( STQCX_OPCODE | rs(s) | rb(b) | rc(1)); } inline void Assembler::stqcx_(Register s, Register b) { emit_int32( STQCX_OPCODE | rs(s) | rb(b) | rc(1)); }

View File

@ -40,9 +40,9 @@ define_pd_global(bool, ImplicitNullChecks, true); // Generate code for impli
define_pd_global(bool, TrapBasedNullChecks, true); define_pd_global(bool, TrapBasedNullChecks, true);
define_pd_global(bool, UncommonNullCast, true); // Uncommon-trap NULLs passed to check cast. define_pd_global(bool, UncommonNullCast, true); // Uncommon-trap NULLs passed to check cast.
#define DEFAULT_STACK_YELLOW_PAGES (6) #define DEFAULT_STACK_YELLOW_PAGES (2)
#define DEFAULT_STACK_RED_PAGES (1) #define DEFAULT_STACK_RED_PAGES (1)
#define DEFAULT_STACK_SHADOW_PAGES (6 DEBUG_ONLY(+2)) #define DEFAULT_STACK_SHADOW_PAGES (20 DEBUG_ONLY(+2))
#define DEFAULT_STACK_RESERVED_PAGES (1) #define DEFAULT_STACK_RESERVED_PAGES (1)
#define MIN_STACK_YELLOW_PAGES DEFAULT_STACK_YELLOW_PAGES #define MIN_STACK_YELLOW_PAGES DEFAULT_STACK_YELLOW_PAGES

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2003, 2016, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2012, 2016 SAP SE. All rights reserved. * Copyright (c) 2012, 2016 SAP SE. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
@ -478,33 +478,6 @@ void InterpreterMacroAssembler::gen_subtype_check(Register Rsub_klass, Register
profile_typecheck_failed(Rtmp1, Rtmp2); profile_typecheck_failed(Rtmp1, Rtmp2);
} }
void InterpreterMacroAssembler::generate_stack_overflow_check_with_compare_and_throw(Register Rmem_frame_size, Register Rscratch1) {
Label done;
BLOCK_COMMENT("stack_overflow_check_with_compare_and_throw {");
sub(Rmem_frame_size, R1_SP, Rmem_frame_size);
ld(Rscratch1, thread_(stack_overflow_limit));
cmpld(CCR0/*is_stack_overflow*/, Rmem_frame_size, Rscratch1);
bgt(CCR0/*is_stack_overflow*/, done);
// Load target address of the runtime stub.
assert(StubRoutines::throw_StackOverflowError_entry() != NULL, "generated in wrong order");
load_const_optimized(Rscratch1, (StubRoutines::throw_StackOverflowError_entry()), R0);
mtctr(Rscratch1);
// Restore caller_sp.
#ifdef ASSERT
ld(Rscratch1, 0, R1_SP);
ld(R0, 0, R21_sender_SP);
cmpd(CCR0, R0, Rscratch1);
asm_assert_eq("backlink", 0x547);
#endif // ASSERT
mr(R1_SP, R21_sender_SP);
bctr();
align(32, 12);
bind(done);
BLOCK_COMMENT("} stack_overflow_check_with_compare_and_throw");
}
// Separate these two to allow for delay slot in middle. // Separate these two to allow for delay slot in middle.
// These are used to do a test and full jump to exception-throwing code. // These are used to do a test and full jump to exception-throwing code.

View File

@ -1,6 +1,6 @@
/* /*
* Copyright (c) 2002, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2002, 2016, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2012, 2015 SAP SE. All rights reserved. * Copyright (c) 2012, 2016 SAP SE. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -79,7 +79,6 @@ class InterpreterMacroAssembler: public MacroAssembler {
// Load object from cpool->resolved_references(index). // Load object from cpool->resolved_references(index).
void load_resolved_reference_at_index(Register result, Register index, Label *is_null = NULL); void load_resolved_reference_at_index(Register result, Register index, Label *is_null = NULL);
void generate_stack_overflow_check_with_compare_and_throw(Register Rmem_frame_size, Register Rscratch1);
void load_receiver(Register Rparam_count, Register Rrecv_dst); void load_receiver(Register Rparam_count, Register Rrecv_dst);
// helpers for expression stack // helpers for expression stack

View File

@ -1422,42 +1422,168 @@ void MacroAssembler::reserved_stack_check(Register return_pc) {
bind(no_reserved_zone_enabling); bind(no_reserved_zone_enabling);
} }
// CmpxchgX sets condition register to cmpX(current, compare). void MacroAssembler::getandsetd(Register dest_current_value, Register exchange_value, Register addr_base,
void MacroAssembler::cmpxchgw(ConditionRegister flag, Register dest_current_value, bool cmpxchgx_hint) {
Register compare_value, Register exchange_value,
Register addr_base, int semantics, bool cmpxchgx_hint,
Register int_flag_success, bool contention_hint, bool weak) {
Label retry; Label retry;
Label failed; bind(retry);
Label done; ldarx(dest_current_value, addr_base, cmpxchgx_hint);
stdcx_(exchange_value, addr_base);
// Save one branch if result is returned via register and if (UseStaticBranchPredictionInCompareAndSwapPPC64) {
// result register is different from the other ones. bne_predict_not_taken(CCR0, retry); // StXcx_ sets CCR0.
bool use_result_reg = (int_flag_success != noreg); } else {
bool preset_result_reg = (int_flag_success != dest_current_value && int_flag_success != compare_value && bne( CCR0, retry); // StXcx_ sets CCR0.
int_flag_success != exchange_value && int_flag_success != addr_base);
assert(!weak || flag == CCR0, "weak only supported with CCR0");
if (use_result_reg && preset_result_reg) {
li(int_flag_success, 0); // preset (assume cas failed)
} }
}
// Add simple guard in order to reduce risk of starving under high contention (recommended by IBM). void MacroAssembler::getandaddd(Register dest_current_value, Register inc_value, Register addr_base,
if (contention_hint) { // Don't try to reserve if cmp fails. Register tmp, bool cmpxchgx_hint) {
lwz(dest_current_value, 0, addr_base); Label retry;
cmpw(flag, dest_current_value, compare_value); bind(retry);
bne(flag, failed); ldarx(dest_current_value, addr_base, cmpxchgx_hint);
add(tmp, dest_current_value, inc_value);
stdcx_(tmp, addr_base);
if (UseStaticBranchPredictionInCompareAndSwapPPC64) {
bne_predict_not_taken(CCR0, retry); // StXcx_ sets CCR0.
} else {
bne( CCR0, retry); // StXcx_ sets CCR0.
} }
}
// release/fence semantics // Word/sub-word atomic helper functions
if (semantics & MemBarRel) {
release(); // Temps and addr_base are killed if size < 4 and processor does not support respective instructions.
// Only signed types are supported with size < 4.
// Atomic add always kills tmp1.
void MacroAssembler::atomic_get_and_modify_generic(Register dest_current_value, Register exchange_value,
Register addr_base, Register tmp1, Register tmp2, Register tmp3,
bool cmpxchgx_hint, bool is_add, int size) {
// Sub-word instructions are available since Power 8.
// For older processors, instruction_type != size holds, and we
// emulate the sub-word instructions by constructing a 4-byte value
// that leaves the other bytes unchanged.
const int instruction_type = VM_Version::has_lqarx() ? size : 4;
Label retry;
Register shift_amount = noreg,
val32 = dest_current_value,
modval = is_add ? tmp1 : exchange_value;
if (instruction_type != size) {
assert_different_registers(tmp1, tmp2, tmp3, dest_current_value, exchange_value, addr_base);
modval = tmp1;
shift_amount = tmp2;
val32 = tmp3;
// Need some preperation: Compute shift amount, align address. Note: shorts must be 2 byte aligned.
#ifdef VM_LITTLE_ENDIAN
rldic(shift_amount, addr_base, 3, 64-5); // (dest & 3) * 8;
clrrdi(addr_base, addr_base, 2);
#else
xori(shift_amount, addr_base, (size == 1) ? 3 : 2);
clrrdi(addr_base, addr_base, 2);
rldic(shift_amount, shift_amount, 3, 64-5); // byte: ((3-dest) & 3) * 8; short: ((1-dest/2) & 1) * 16;
#endif
} }
// atomic emulation loop // atomic emulation loop
bind(retry); bind(retry);
lwarx(dest_current_value, addr_base, cmpxchgx_hint); switch (instruction_type) {
case 4: lwarx(val32, addr_base, cmpxchgx_hint); break;
case 2: lharx(val32, addr_base, cmpxchgx_hint); break;
case 1: lbarx(val32, addr_base, cmpxchgx_hint); break;
default: ShouldNotReachHere();
}
if (instruction_type != size) {
srw(dest_current_value, val32, shift_amount);
}
if (is_add) { add(modval, dest_current_value, exchange_value); }
if (instruction_type != size) {
// Transform exchange value such that the replacement can be done by one xor instruction.
xorr(modval, dest_current_value, is_add ? modval : exchange_value);
clrldi(modval, modval, (size == 1) ? 56 : 48);
slw(modval, modval, shift_amount);
xorr(modval, val32, modval);
}
switch (instruction_type) {
case 4: stwcx_(modval, addr_base); break;
case 2: sthcx_(modval, addr_base); break;
case 1: stbcx_(modval, addr_base); break;
default: ShouldNotReachHere();
}
if (UseStaticBranchPredictionInCompareAndSwapPPC64) {
bne_predict_not_taken(CCR0, retry); // StXcx_ sets CCR0.
} else {
bne( CCR0, retry); // StXcx_ sets CCR0.
}
// l?arx zero-extends, but Java wants byte/short values sign-extended.
if (size == 1) {
extsb(dest_current_value, dest_current_value);
} else if (size == 2) {
extsh(dest_current_value, dest_current_value);
};
}
// Temps, addr_base and exchange_value are killed if size < 4 and processor does not support respective instructions.
// Only signed types are supported with size < 4.
void MacroAssembler::cmpxchg_loop_body(ConditionRegister flag, Register dest_current_value,
Register compare_value, Register exchange_value,
Register addr_base, Register tmp1, Register tmp2,
Label &retry, Label &failed, bool cmpxchgx_hint, int size) {
// Sub-word instructions are available since Power 8.
// For older processors, instruction_type != size holds, and we
// emulate the sub-word instructions by constructing a 4-byte value
// that leaves the other bytes unchanged.
const int instruction_type = VM_Version::has_lqarx() ? size : 4;
Register shift_amount = noreg,
val32 = dest_current_value,
modval = exchange_value;
if (instruction_type != size) {
assert_different_registers(tmp1, tmp2, dest_current_value, compare_value, exchange_value, addr_base);
shift_amount = tmp1;
val32 = tmp2;
modval = tmp2;
// Need some preperation: Compute shift amount, align address. Note: shorts must be 2 byte aligned.
#ifdef VM_LITTLE_ENDIAN
rldic(shift_amount, addr_base, 3, 64-5); // (dest & 3) * 8;
clrrdi(addr_base, addr_base, 2);
#else
xori(shift_amount, addr_base, (size == 1) ? 3 : 2);
clrrdi(addr_base, addr_base, 2);
rldic(shift_amount, shift_amount, 3, 64-5); // byte: ((3-dest) & 3) * 8; short: ((1-dest/2) & 1) * 16;
#endif
// Transform exchange value such that the replacement can be done by one xor instruction.
xorr(exchange_value, compare_value, exchange_value);
clrldi(exchange_value, exchange_value, (size == 1) ? 56 : 48);
slw(exchange_value, exchange_value, shift_amount);
}
// atomic emulation loop
bind(retry);
switch (instruction_type) {
case 4: lwarx(val32, addr_base, cmpxchgx_hint); break;
case 2: lharx(val32, addr_base, cmpxchgx_hint); break;
case 1: lbarx(val32, addr_base, cmpxchgx_hint); break;
default: ShouldNotReachHere();
}
if (instruction_type != size) {
srw(dest_current_value, val32, shift_amount);
}
if (size == 1) {
extsb(dest_current_value, dest_current_value);
} else if (size == 2) {
extsh(dest_current_value, dest_current_value);
};
cmpw(flag, dest_current_value, compare_value); cmpw(flag, dest_current_value, compare_value);
if (UseStaticBranchPredictionInCompareAndSwapPPC64) { if (UseStaticBranchPredictionInCompareAndSwapPPC64) {
bne_predict_not_taken(flag, failed); bne_predict_not_taken(flag, failed);
@ -1467,7 +1593,60 @@ void MacroAssembler::cmpxchgw(ConditionRegister flag, Register dest_current_valu
// branch to done => (flag == ne), (dest_current_value != compare_value) // branch to done => (flag == ne), (dest_current_value != compare_value)
// fall through => (flag == eq), (dest_current_value == compare_value) // fall through => (flag == eq), (dest_current_value == compare_value)
stwcx_(exchange_value, addr_base); if (instruction_type != size) {
xorr(modval, val32, exchange_value);
}
switch (instruction_type) {
case 4: stwcx_(modval, addr_base); break;
case 2: sthcx_(modval, addr_base); break;
case 1: stbcx_(modval, addr_base); break;
default: ShouldNotReachHere();
}
}
// CmpxchgX sets condition register to cmpX(current, compare).
void MacroAssembler::cmpxchg_generic(ConditionRegister flag, Register dest_current_value,
Register compare_value, Register exchange_value,
Register addr_base, Register tmp1, Register tmp2,
int semantics, bool cmpxchgx_hint,
Register int_flag_success, bool contention_hint, bool weak, int size) {
Label retry;
Label failed;
Label done;
// Save one branch if result is returned via register and
// result register is different from the other ones.
bool use_result_reg = (int_flag_success != noreg);
bool preset_result_reg = (int_flag_success != dest_current_value && int_flag_success != compare_value &&
int_flag_success != exchange_value && int_flag_success != addr_base &&
int_flag_success != tmp1 && int_flag_success != tmp2);
assert(!weak || flag == CCR0, "weak only supported with CCR0");
assert(size == 1 || size == 2 || size == 4, "unsupported");
if (use_result_reg && preset_result_reg) {
li(int_flag_success, 0); // preset (assume cas failed)
}
// Add simple guard in order to reduce risk of starving under high contention (recommended by IBM).
if (contention_hint) { // Don't try to reserve if cmp fails.
switch (size) {
case 1: lbz(dest_current_value, 0, addr_base); extsb(dest_current_value, dest_current_value); break;
case 2: lha(dest_current_value, 0, addr_base); break;
case 4: lwz(dest_current_value, 0, addr_base); break;
default: ShouldNotReachHere();
}
cmpw(flag, dest_current_value, compare_value);
bne(flag, failed);
}
// release/fence semantics
if (semantics & MemBarRel) {
release();
}
cmpxchg_loop_body(flag, dest_current_value, compare_value, exchange_value, addr_base, tmp1, tmp2,
retry, failed, cmpxchgx_hint, size);
if (!weak || use_result_reg) { if (!weak || use_result_reg) {
if (UseStaticBranchPredictionInCompareAndSwapPPC64) { if (UseStaticBranchPredictionInCompareAndSwapPPC64) {
bne_predict_not_taken(CCR0, weak ? failed : retry); // StXcx_ sets CCR0. bne_predict_not_taken(CCR0, weak ? failed : retry); // StXcx_ sets CCR0.
@ -3751,454 +3930,6 @@ void MacroAssembler::has_negatives(Register src, Register cnt, Register result,
bind(Ldone); bind(Ldone);
} }
// Intrinsics for non-CompactStrings
// Search for a single jchar in an jchar[].
//
// Assumes that result differs from all other registers.
//
// 'haystack' is the addresses of a jchar-array.
// 'needle' is either the character to search for or R0.
// 'needleChar' is the character to search for if 'needle' == R0..
// 'haycnt' is the length of the haystack. We assume 'haycnt' >=1.
//
// Preserves haystack, haycnt, needle and kills all other registers.
//
// If needle == R0, we search for the constant needleChar.
void MacroAssembler::string_indexof_1(Register result, Register haystack, Register haycnt,
Register needle, jchar needleChar,
Register tmp1, Register tmp2) {
assert_different_registers(result, haystack, haycnt, needle, tmp1, tmp2);
Label L_InnerLoop, L_FinalCheck, L_Found1, L_Found2, L_Found3, L_NotFound, L_End;
Register addr = tmp1,
ch1 = tmp2,
ch2 = R0;
//3:
dcbtct(haystack, 0x00); // Indicate R/O access to haystack.
srwi_(tmp2, haycnt, 1); // Shift right by exact_log2(UNROLL_FACTOR).
mr(addr, haystack);
beq(CCR0, L_FinalCheck);
mtctr(tmp2); // Move to count register.
//8:
bind(L_InnerLoop); // Main work horse (2x unrolled search loop).
lhz(ch1, 0, addr); // Load characters from haystack.
lhz(ch2, 2, addr);
(needle != R0) ? cmpw(CCR0, ch1, needle) : cmplwi(CCR0, ch1, needleChar);
(needle != R0) ? cmpw(CCR1, ch2, needle) : cmplwi(CCR1, ch2, needleChar);
beq(CCR0, L_Found1); // Did we find the needle?
beq(CCR1, L_Found2);
addi(addr, addr, 4);
bdnz(L_InnerLoop);
//16:
bind(L_FinalCheck);
andi_(R0, haycnt, 1);
beq(CCR0, L_NotFound);
lhz(ch1, 0, addr); // One position left at which we have to compare.
(needle != R0) ? cmpw(CCR1, ch1, needle) : cmplwi(CCR1, ch1, needleChar);
beq(CCR1, L_Found3);
//21:
bind(L_NotFound);
li(result, -1); // Not found.
b(L_End);
bind(L_Found2);
addi(addr, addr, 2);
//24:
bind(L_Found1);
bind(L_Found3); // Return index ...
subf(addr, haystack, addr); // relative to haystack,
srdi(result, addr, 1); // in characters.
bind(L_End);
}
// Implementation of IndexOf for jchar arrays.
//
// The length of haystack and needle are not constant, i.e. passed in a register.
//
// Preserves registers haystack, needle.
// Kills registers haycnt, needlecnt.
// Assumes that result differs from all other registers.
// Haystack, needle are the addresses of jchar-arrays.
// Haycnt, needlecnt are the lengths of them, respectively.
//
// Needlecntval must be zero or 15-bit unsigned immediate and > 1.
void MacroAssembler::string_indexof(Register result, Register haystack, Register haycnt,
Register needle, ciTypeArray* needle_values, Register needlecnt, int needlecntval,
Register tmp1, Register tmp2, Register tmp3, Register tmp4) {
// Ensure 0<needlecnt<=haycnt in ideal graph as prerequisite!
Label L_TooShort, L_Found, L_NotFound, L_End;
Register last_addr = haycnt, // Kill haycnt at the beginning.
addr = tmp1,
n_start = tmp2,
ch1 = tmp3,
ch2 = R0;
// **************************************************************************************************
// Prepare for main loop: optimized for needle count >=2, bail out otherwise.
// **************************************************************************************************
//1 (variable) or 3 (const):
dcbtct(needle, 0x00); // Indicate R/O access to str1.
dcbtct(haystack, 0x00); // Indicate R/O access to str2.
// Compute last haystack addr to use if no match gets found.
if (needlecntval == 0) { // variable needlecnt
//3:
subf(ch1, needlecnt, haycnt); // Last character index to compare is haycnt-needlecnt.
addi(addr, haystack, -2); // Accesses use pre-increment.
cmpwi(CCR6, needlecnt, 2);
blt(CCR6, L_TooShort); // Variable needlecnt: handle short needle separately.
slwi(ch1, ch1, 1); // Scale to number of bytes.
lwz(n_start, 0, needle); // Load first 2 characters of needle.
add(last_addr, haystack, ch1); // Point to last address to compare (haystack+2*(haycnt-needlecnt)).
addi(needlecnt, needlecnt, -2); // Rest of needle.
} else { // constant needlecnt
guarantee(needlecntval != 1, "IndexOf with single-character needle must be handled separately");
assert((needlecntval & 0x7fff) == needlecntval, "wrong immediate");
//5:
addi(ch1, haycnt, -needlecntval); // Last character index to compare is haycnt-needlecnt.
lwz(n_start, 0, needle); // Load first 2 characters of needle.
addi(addr, haystack, -2); // Accesses use pre-increment.
slwi(ch1, ch1, 1); // Scale to number of bytes.
add(last_addr, haystack, ch1); // Point to last address to compare (haystack+2*(haycnt-needlecnt)).
li(needlecnt, needlecntval-2); // Rest of needle.
}
// Main Loop (now we have at least 3 characters).
//11:
Label L_OuterLoop, L_InnerLoop, L_FinalCheck, L_Comp1, L_Comp2, L_Comp3;
bind(L_OuterLoop); // Search for 1st 2 characters.
Register addr_diff = tmp4;
subf(addr_diff, addr, last_addr); // Difference between already checked address and last address to check.
addi(addr, addr, 2); // This is the new address we want to use for comparing.
srdi_(ch2, addr_diff, 2);
beq(CCR0, L_FinalCheck); // 2 characters left?
mtctr(ch2); // addr_diff/4
//16:
bind(L_InnerLoop); // Main work horse (2x unrolled search loop)
lwz(ch1, 0, addr); // Load 2 characters of haystack (ignore alignment).
lwz(ch2, 2, addr);
cmpw(CCR0, ch1, n_start); // Compare 2 characters (1 would be sufficient but try to reduce branches to CompLoop).
cmpw(CCR1, ch2, n_start);
beq(CCR0, L_Comp1); // Did we find the needle start?
beq(CCR1, L_Comp2);
addi(addr, addr, 4);
bdnz(L_InnerLoop);
//24:
bind(L_FinalCheck);
rldicl_(addr_diff, addr_diff, 64-1, 63); // Remaining characters not covered by InnerLoop: (addr_diff>>1)&1.
beq(CCR0, L_NotFound);
lwz(ch1, 0, addr); // One position left at which we have to compare.
cmpw(CCR1, ch1, n_start);
beq(CCR1, L_Comp3);
//29:
bind(L_NotFound);
li(result, -1); // not found
b(L_End);
// **************************************************************************************************
// Special Case: unfortunately, the variable needle case can be called with needlecnt<2
// **************************************************************************************************
//31:
if ((needlecntval>>1) !=1 ) { // Const needlecnt is 2 or 3? Reduce code size.
int nopcnt = 5;
if (needlecntval !=0 ) ++nopcnt; // Balance alignment (other case: see below).
if (needlecntval == 0) { // We have to handle these cases separately.
Label L_OneCharLoop;
bind(L_TooShort);
mtctr(haycnt);
lhz(n_start, 0, needle); // First character of needle
bind(L_OneCharLoop);
lhzu(ch1, 2, addr);
cmpw(CCR1, ch1, n_start);
beq(CCR1, L_Found); // Did we find the one character needle?
bdnz(L_OneCharLoop);
li(result, -1); // Not found.
b(L_End);
} // 8 instructions, so no impact on alignment.
for (int x = 0; x < nopcnt; ++x) nop();
}
// **************************************************************************************************
// Regular Case Part II: compare rest of needle (first 2 characters have been compared already)
// **************************************************************************************************
// Compare the rest
//36 if needlecntval==0, else 37:
bind(L_Comp2);
addi(addr, addr, 2); // First comparison has failed, 2nd one hit.
bind(L_Comp1); // Addr points to possible needle start.
bind(L_Comp3); // Could have created a copy and use a different return address but saving code size here.
if (needlecntval != 2) { // Const needlecnt==2?
if (needlecntval != 3) {
if (needlecntval == 0) beq(CCR6, L_Found); // Variable needlecnt==2?
Register ind_reg = tmp4;
li(ind_reg, 2*2); // First 2 characters are already compared, use index 2.
mtctr(needlecnt); // Decremented by 2, still > 0.
//40:
Label L_CompLoop;
bind(L_CompLoop);
lhzx(ch2, needle, ind_reg);
lhzx(ch1, addr, ind_reg);
cmpw(CCR1, ch1, ch2);
bne(CCR1, L_OuterLoop);
addi(ind_reg, ind_reg, 2);
bdnz(L_CompLoop);
} else { // No loop required if there's only one needle character left.
lhz(ch2, 2*2, needle);
lhz(ch1, 2*2, addr);
cmpw(CCR1, ch1, ch2);
bne(CCR1, L_OuterLoop);
}
}
// Return index ...
//46:
bind(L_Found);
subf(addr, haystack, addr); // relative to haystack, ...
srdi(result, addr, 1); // in characters.
//48:
bind(L_End);
}
// Implementation of Compare for jchar arrays.
//
// Kills the registers str1, str2, cnt1, cnt2.
// Kills cr0, ctr.
// Assumes that result differes from the input registers.
void MacroAssembler::string_compare(Register str1_reg, Register str2_reg, Register cnt1_reg, Register cnt2_reg,
Register result_reg, Register tmp_reg) {
assert_different_registers(result_reg, str1_reg, str2_reg, cnt1_reg, cnt2_reg, tmp_reg);
Label Ldone, Lslow_case, Lslow_loop, Lfast_loop;
Register cnt_diff = R0,
limit_reg = cnt1_reg,
chr1_reg = result_reg,
chr2_reg = cnt2_reg,
addr_diff = str2_reg;
// 'cnt_reg' contains the number of characters in the string's character array for the
// pre-CompactStrings strings implementation and the number of bytes in the string's
// byte array for the CompactStrings strings implementation.
const int HAS_COMPACT_STRING = java_lang_String::has_coder_field() ? 1 : 0; // '1' = byte array, '0' = char array
// Offset 0 should be 32 byte aligned.
//-6:
srawi(cnt1_reg, cnt1_reg, HAS_COMPACT_STRING);
srawi(cnt2_reg, cnt2_reg, HAS_COMPACT_STRING);
//-4:
dcbtct(str1_reg, 0x00); // Indicate R/O access to str1.
dcbtct(str2_reg, 0x00); // Indicate R/O access to str2.
//-2:
// Compute min(cnt1, cnt2) and check if 0 (bail out if we don't need to compare characters).
subf(result_reg, cnt2_reg, cnt1_reg); // difference between cnt1/2
subf_(addr_diff, str1_reg, str2_reg); // alias?
beq(CCR0, Ldone); // return cnt difference if both ones are identical
srawi(limit_reg, result_reg, 31); // generate signmask (cnt1/2 must be non-negative so cnt_diff can't overflow)
mr(cnt_diff, result_reg);
andr(limit_reg, result_reg, limit_reg); // difference or zero (negative): cnt1<cnt2 ? cnt1-cnt2 : 0
add_(limit_reg, cnt2_reg, limit_reg); // min(cnt1, cnt2)==0?
beq(CCR0, Ldone); // return cnt difference if one has 0 length
lhz(chr1_reg, 0, str1_reg); // optional: early out if first characters mismatch
lhzx(chr2_reg, str1_reg, addr_diff); // optional: early out if first characters mismatch
addi(tmp_reg, limit_reg, -1); // min(cnt1, cnt2)-1
subf_(result_reg, chr2_reg, chr1_reg); // optional: early out if first characters mismatch
bne(CCR0, Ldone); // optional: early out if first characters mismatch
// Set loop counter by scaling down tmp_reg
srawi_(chr2_reg, tmp_reg, exact_log2(4)); // (min(cnt1, cnt2)-1)/4
ble(CCR0, Lslow_case); // need >4 characters for fast loop
andi(limit_reg, tmp_reg, 4-1); // remaining characters
// Adapt str1_reg str2_reg for the first loop iteration
mtctr(chr2_reg); // (min(cnt1, cnt2)-1)/4
addi(limit_reg, limit_reg, 4+1); // compare last 5-8 characters in slow_case if mismatch found in fast_loop
//16:
// Compare the rest of the characters
bind(Lfast_loop);
ld(chr1_reg, 0, str1_reg);
ldx(chr2_reg, str1_reg, addr_diff);
cmpd(CCR0, chr2_reg, chr1_reg);
bne(CCR0, Lslow_case); // return chr1_reg
addi(str1_reg, str1_reg, 4*2);
bdnz(Lfast_loop);
addi(limit_reg, limit_reg, -4); // no mismatch found in fast_loop, only 1-4 characters missing
//23:
bind(Lslow_case);
mtctr(limit_reg);
//24:
bind(Lslow_loop);
lhz(chr1_reg, 0, str1_reg);
lhzx(chr2_reg, str1_reg, addr_diff);
subf_(result_reg, chr2_reg, chr1_reg);
bne(CCR0, Ldone); // return chr1_reg
addi(str1_reg, str1_reg, 1*2);
bdnz(Lslow_loop);
//30:
// If strings are equal up to min length, return the length difference.
mr(result_reg, cnt_diff);
nop(); // alignment
//32:
// Otherwise, return the difference between the first mismatched chars.
bind(Ldone);
}
// Compare char[] arrays.
//
// str1_reg USE only
// str2_reg USE only
// cnt_reg USE_DEF, due to tmp reg shortage
// result_reg DEF only, might compromise USE only registers
void MacroAssembler::char_arrays_equals(Register str1_reg, Register str2_reg, Register cnt_reg, Register result_reg,
Register tmp1_reg, Register tmp2_reg, Register tmp3_reg, Register tmp4_reg,
Register tmp5_reg) {
// Str1 may be the same register as str2 which can occur e.g. after scalar replacement.
assert_different_registers(result_reg, str1_reg, cnt_reg, tmp1_reg, tmp2_reg, tmp3_reg, tmp4_reg, tmp5_reg);
assert_different_registers(result_reg, str2_reg, cnt_reg, tmp1_reg, tmp2_reg, tmp3_reg, tmp4_reg, tmp5_reg);
// Offset 0 should be 32 byte aligned.
Label Linit_cbc, Lcbc, Lloop, Ldone_true, Ldone_false;
Register index_reg = tmp5_reg;
Register cbc_iter = tmp4_reg;
// 'cnt_reg' contains the number of characters in the string's character array for the
// pre-CompactStrings strings implementation and the number of bytes in the string's
// byte array for the CompactStrings strings implementation.
const int HAS_COMPACT_STRING = java_lang_String::has_coder_field() ? 1 : 0; // '1' = byte array, '0' = char array
//-1:
dcbtct(str1_reg, 0x00); // Indicate R/O access to str1.
dcbtct(str2_reg, 0x00); // Indicate R/O access to str2.
//1:
// cbc_iter: remaining characters after the '4 java characters per iteration' loop.
rlwinm(cbc_iter, cnt_reg, 32 - HAS_COMPACT_STRING, 30, 31); // (cnt_reg % (HAS_COMPACT_STRING ? 8 : 4)) >> HAS_COMPACT_STRING
li(index_reg, 0); // init
li(result_reg, 0); // assume false
// tmp2_reg: units of 4 java characters (i.e. 8 bytes) per iteration (main loop).
srwi_(tmp2_reg, cnt_reg, exact_log2(4 << HAS_COMPACT_STRING)); // cnt_reg / (HAS_COMPACT_STRING ? 8 : 4)
cmpwi(CCR1, cbc_iter, 0); // CCR1 = (cbc_iter==0)
beq(CCR0, Linit_cbc); // too short
mtctr(tmp2_reg);
//8:
bind(Lloop);
ldx(tmp1_reg, str1_reg, index_reg);
ldx(tmp2_reg, str2_reg, index_reg);
cmpd(CCR0, tmp1_reg, tmp2_reg);
bne(CCR0, Ldone_false); // Unequal char pair found -> done.
addi(index_reg, index_reg, 4*sizeof(jchar));
bdnz(Lloop);
//14:
bind(Linit_cbc);
beq(CCR1, Ldone_true);
mtctr(cbc_iter);
//16:
bind(Lcbc);
lhzx(tmp1_reg, str1_reg, index_reg);
lhzx(tmp2_reg, str2_reg, index_reg);
cmpw(CCR0, tmp1_reg, tmp2_reg);
bne(CCR0, Ldone_false); // Unequal char pair found -> done.
addi(index_reg, index_reg, 1*sizeof(jchar));
bdnz(Lcbc);
nop();
bind(Ldone_true);
li(result_reg, 1);
//24:
bind(Ldone_false);
}
void MacroAssembler::char_arrays_equalsImm(Register str1_reg, Register str2_reg, int cntval, Register result_reg,
Register tmp1_reg, Register tmp2_reg) {
// Str1 may be the same register as str2 which can occur e.g. after scalar replacement.
assert_different_registers(result_reg, str1_reg, tmp1_reg, tmp2_reg);
assert_different_registers(result_reg, str2_reg, tmp1_reg, tmp2_reg);
assert(sizeof(jchar) == 2, "must be");
assert(cntval >= 0 && ((cntval & 0x7fff) == cntval), "wrong immediate");
// 'cntval' contains the number of characters in the string's character array for the
// pre-CompactStrings strings implementation and the number of bytes in the string's
// byte array for the CompactStrings strings implementation.
cntval >>= (java_lang_String::has_coder_field() ? 1 : 0); // '1' = byte array strings, '0' = char array strings
Label Ldone_false;
if (cntval < 16) { // short case
if (cntval != 0) li(result_reg, 0); // assume false
const int num_bytes = cntval*sizeof(jchar);
int index = 0;
for (int next_index; (next_index = index + 8) <= num_bytes; index = next_index) {
ld(tmp1_reg, index, str1_reg);
ld(tmp2_reg, index, str2_reg);
cmpd(CCR0, tmp1_reg, tmp2_reg);
bne(CCR0, Ldone_false);
}
if (cntval & 2) {
lwz(tmp1_reg, index, str1_reg);
lwz(tmp2_reg, index, str2_reg);
cmpw(CCR0, tmp1_reg, tmp2_reg);
bne(CCR0, Ldone_false);
index += 4;
}
if (cntval & 1) {
lhz(tmp1_reg, index, str1_reg);
lhz(tmp2_reg, index, str2_reg);
cmpw(CCR0, tmp1_reg, tmp2_reg);
bne(CCR0, Ldone_false);
}
// fallthrough: true
} else {
Label Lloop;
Register index_reg = tmp1_reg;
const int loopcnt = cntval/4;
assert(loopcnt > 0, "must be");
// Offset 0 should be 32 byte aligned.
//2:
dcbtct(str1_reg, 0x00); // Indicate R/O access to str1.
dcbtct(str2_reg, 0x00); // Indicate R/O access to str2.
li(tmp2_reg, loopcnt);
li(index_reg, 0); // init
li(result_reg, 0); // assume false
mtctr(tmp2_reg);
//8:
bind(Lloop);
ldx(R0, str1_reg, index_reg);
ldx(tmp2_reg, str2_reg, index_reg);
cmpd(CCR0, R0, tmp2_reg);
bne(CCR0, Ldone_false); // Unequal char pair found -> done.
addi(index_reg, index_reg, 4*sizeof(jchar));
bdnz(Lloop);
//14:
if (cntval & 2) {
lwzx(R0, str1_reg, index_reg);
lwzx(tmp2_reg, str2_reg, index_reg);
cmpw(CCR0, R0, tmp2_reg);
bne(CCR0, Ldone_false);
if (cntval & 1) addi(index_reg, index_reg, 2*sizeof(jchar));
}
if (cntval & 1) {
lhzx(R0, str1_reg, index_reg);
lhzx(tmp2_reg, str2_reg, index_reg);
cmpw(CCR0, R0, tmp2_reg);
bne(CCR0, Ldone_false);
}
// fallthru: true
}
li(result_reg, 1);
bind(Ldone_false);
}
#endif // Compiler2 #endif // Compiler2
// Helpers for Intrinsic Emitters // Helpers for Intrinsic Emitters

View File

@ -431,10 +431,81 @@ class MacroAssembler: public Assembler {
MemBarAcq = 2, MemBarAcq = 2,
MemBarFenceAfter = 4 // use powers of 2 MemBarFenceAfter = 4 // use powers of 2
}; };
private:
// Helper functions for word/sub-word atomics.
void atomic_get_and_modify_generic(Register dest_current_value, Register exchange_value,
Register addr_base, Register tmp1, Register tmp2, Register tmp3,
bool cmpxchgx_hint, bool is_add, int size);
void cmpxchg_loop_body(ConditionRegister flag, Register dest_current_value,
Register compare_value, Register exchange_value,
Register addr_base, Register tmp1, Register tmp2,
Label &retry, Label &failed, bool cmpxchgx_hint, int size);
void cmpxchg_generic(ConditionRegister flag,
Register dest_current_value, Register compare_value, Register exchange_value, Register addr_base,
Register tmp1, Register tmp2,
int semantics, bool cmpxchgx_hint, Register int_flag_success, bool contention_hint, bool weak, int size);
public:
// Temps and addr_base are killed if processor does not support Power 8 instructions.
// Result will be sign extended.
void getandsetb(Register dest_current_value, Register exchange_value, Register addr_base,
Register tmp1, Register tmp2, Register tmp3, bool cmpxchgx_hint) {
atomic_get_and_modify_generic(dest_current_value, exchange_value, addr_base, tmp1, tmp2, tmp3, cmpxchgx_hint, false, 1);
}
// Temps and addr_base are killed if processor does not support Power 8 instructions.
// Result will be sign extended.
void getandseth(Register dest_current_value, Register exchange_value, Register addr_base,
Register tmp1, Register tmp2, Register tmp3, bool cmpxchgx_hint) {
atomic_get_and_modify_generic(dest_current_value, exchange_value, addr_base, tmp1, tmp2, tmp3, cmpxchgx_hint, false, 2);
}
void getandsetw(Register dest_current_value, Register exchange_value, Register addr_base,
bool cmpxchgx_hint) {
atomic_get_and_modify_generic(dest_current_value, exchange_value, addr_base, noreg, noreg, noreg, cmpxchgx_hint, false, 4);
}
void getandsetd(Register dest_current_value, Register exchange_value, Register addr_base,
bool cmpxchgx_hint);
// tmp2/3 and addr_base are killed if processor does not support Power 8 instructions (tmp1 is always needed).
// Result will be sign extended.
void getandaddb(Register dest_current_value, Register inc_value, Register addr_base,
Register tmp1, Register tmp2, Register tmp3, bool cmpxchgx_hint) {
atomic_get_and_modify_generic(dest_current_value, inc_value, addr_base, tmp1, tmp2, tmp3, cmpxchgx_hint, true, 1);
}
// tmp2/3 and addr_base are killed if processor does not support Power 8 instructions (tmp1 is always needed).
// Result will be sign extended.
void getandaddh(Register dest_current_value, Register inc_value, Register addr_base,
Register tmp1, Register tmp2, Register tmp3, bool cmpxchgx_hint) {
atomic_get_and_modify_generic(dest_current_value, inc_value, addr_base, tmp1, tmp2, tmp3, cmpxchgx_hint, true, 2);
}
void getandaddw(Register dest_current_value, Register inc_value, Register addr_base,
Register tmp1, bool cmpxchgx_hint) {
atomic_get_and_modify_generic(dest_current_value, inc_value, addr_base, tmp1, noreg, noreg, cmpxchgx_hint, true, 4);
}
void getandaddd(Register dest_current_value, Register exchange_value, Register addr_base,
Register tmp, bool cmpxchgx_hint);
// Temps, addr_base and exchange_value are killed if processor does not support Power 8 instructions.
// compare_value must be at least 32 bit sign extended. Result will be sign extended.
void cmpxchgb(ConditionRegister flag,
Register dest_current_value, Register compare_value, Register exchange_value, Register addr_base,
Register tmp1, Register tmp2, int semantics, bool cmpxchgx_hint = false,
Register int_flag_success = noreg, bool contention_hint = false, bool weak = false) {
cmpxchg_generic(flag, dest_current_value, compare_value, exchange_value, addr_base, tmp1, tmp2,
semantics, cmpxchgx_hint, int_flag_success, contention_hint, weak, 1);
}
// Temps, addr_base and exchange_value are killed if processor does not support Power 8 instructions.
// compare_value must be at least 32 bit sign extended. Result will be sign extended.
void cmpxchgh(ConditionRegister flag,
Register dest_current_value, Register compare_value, Register exchange_value, Register addr_base,
Register tmp1, Register tmp2, int semantics, bool cmpxchgx_hint = false,
Register int_flag_success = noreg, bool contention_hint = false, bool weak = false) {
cmpxchg_generic(flag, dest_current_value, compare_value, exchange_value, addr_base, tmp1, tmp2,
semantics, cmpxchgx_hint, int_flag_success, contention_hint, weak, 2);
}
void cmpxchgw(ConditionRegister flag, void cmpxchgw(ConditionRegister flag,
Register dest_current_value, Register compare_value, Register exchange_value, Register addr_base, Register dest_current_value, Register compare_value, Register exchange_value, Register addr_base,
int semantics, bool cmpxchgx_hint = false, int semantics, bool cmpxchgx_hint = false,
Register int_flag_success = noreg, bool contention_hint = false, bool weak = false); Register int_flag_success = noreg, bool contention_hint = false, bool weak = false) {
cmpxchg_generic(flag, dest_current_value, compare_value, exchange_value, addr_base, noreg, noreg,
semantics, cmpxchgx_hint, int_flag_success, contention_hint, weak, 4);
}
void cmpxchgd(ConditionRegister flag, void cmpxchgd(ConditionRegister flag,
Register dest_current_value, RegisterOrConstant compare_value, Register exchange_value, Register dest_current_value, RegisterOrConstant compare_value, Register exchange_value,
Register addr_base, int semantics, bool cmpxchgx_hint = false, Register addr_base, int semantics, bool cmpxchgx_hint = false,
@ -717,23 +788,6 @@ class MacroAssembler: public Assembler {
Register needle, jchar needleChar, Register tmp1, Register tmp2, bool is_byte); Register needle, jchar needleChar, Register tmp1, Register tmp2, bool is_byte);
void has_negatives(Register src, Register cnt, Register result, Register tmp1, Register tmp2); void has_negatives(Register src, Register cnt, Register result, Register tmp1, Register tmp2);
// Intrinsics for non-CompactStrings
// Needle of length 1.
void string_indexof_1(Register result, Register haystack, Register haycnt,
Register needle, jchar needleChar,
Register tmp1, Register tmp2);
// General indexof, eventually with constant needle length.
void string_indexof(Register result, Register haystack, Register haycnt,
Register needle, ciTypeArray* needle_values, Register needlecnt, int needlecntval,
Register tmp1, Register tmp2, Register tmp3, Register tmp4);
void string_compare(Register str1_reg, Register str2_reg, Register cnt1_reg, Register cnt2_reg,
Register result_reg, Register tmp_reg);
void char_arrays_equals(Register str1_reg, Register str2_reg, Register cnt_reg, Register result_reg,
Register tmp1_reg, Register tmp2_reg, Register tmp3_reg, Register tmp4_reg,
Register tmp5_reg);
void char_arrays_equalsImm(Register str1_reg, Register str2_reg, int cntval, Register result_reg,
Register tmp1_reg, Register tmp2_reg);
#endif #endif
// Emitters for BigInteger.multiplyToLen intrinsic. // Emitters for BigInteger.multiplyToLen intrinsic.

File diff suppressed because it is too large Load Diff

View File

@ -1123,7 +1123,10 @@ class StubGenerator: public StubCodeGenerator {
Register tmp3 = R8_ARG6; Register tmp3 = R8_ARG6;
Register tmp4 = R9_ARG7; Register tmp4 = R9_ARG7;
Label l_1, l_2, l_3, l_4, l_5, l_6, l_7, l_8, l_9; VectorSRegister tmp_vsr1 = VSR1;
VectorSRegister tmp_vsr2 = VSR2;
Label l_1, l_2, l_3, l_4, l_5, l_6, l_7, l_8, l_9, l_10;
// Don't try anything fancy if arrays don't have many elements. // Don't try anything fancy if arrays don't have many elements.
__ li(tmp3, 0); __ li(tmp3, 0);
@ -1178,6 +1181,8 @@ class StubGenerator: public StubCodeGenerator {
__ andi_(R5_ARG3, R5_ARG3, 31); __ andi_(R5_ARG3, R5_ARG3, 31);
__ mtctr(tmp1); __ mtctr(tmp1);
if (!VM_Version::has_vsx()) {
__ bind(l_8); __ bind(l_8);
// Use unrolled version for mass copying (copy 32 elements a time) // Use unrolled version for mass copying (copy 32 elements a time)
// Load feeding store gets zero latency on Power6, however not on Power5. // Load feeding store gets zero latency on Power6, however not on Power5.
@ -1193,7 +1198,44 @@ class StubGenerator: public StubCodeGenerator {
__ addi(R3_ARG1, R3_ARG1, 32); __ addi(R3_ARG1, R3_ARG1, 32);
__ addi(R4_ARG2, R4_ARG2, 32); __ addi(R4_ARG2, R4_ARG2, 32);
__ bdnz(l_8); __ bdnz(l_8);
}
} else { // Processor supports VSX, so use it to mass copy.
// Prefetch the data into the L2 cache.
__ dcbt(R3_ARG1, 0);
// If supported set DSCR pre-fetch to deepest.
if (VM_Version::has_mfdscr()) {
__ load_const_optimized(tmp2, VM_Version::_dscr_val | 7);
__ mtdscr(tmp2);
}
__ li(tmp1, 16);
// Backbranch target aligned to 32-byte. Not 16-byte align as
// loop contains < 8 instructions that fit inside a single
// i-cache sector.
__ align(32);
__ bind(l_10);
// Use loop with VSX load/store instructions to
// copy 32 elements a time.
__ lxvd2x(tmp_vsr1, 0, R3_ARG1); // Load src
__ stxvd2x(tmp_vsr1, 0, R4_ARG2); // Store to dst
__ lxvd2x(tmp_vsr2, tmp1, R3_ARG1); // Load src + 16
__ stxvd2x(tmp_vsr2, tmp1, R4_ARG2); // Store to dst + 16
__ addi(R3_ARG1, R3_ARG1, 32); // Update src+=32
__ addi(R4_ARG2, R4_ARG2, 32); // Update dsc+=32
__ bdnz(l_10); // Dec CTR and loop if not zero.
// Restore DSCR pre-fetch value.
if (VM_Version::has_mfdscr()) {
__ load_const_optimized(tmp2, VM_Version::_dscr_val);
__ mtdscr(tmp2);
}
} // VSX
} // FasterArrayCopy
__ bind(l_6); __ bind(l_6);
@ -1557,7 +1599,10 @@ class StubGenerator: public StubCodeGenerator {
Register tmp3 = R8_ARG6; Register tmp3 = R8_ARG6;
Register tmp4 = R0; Register tmp4 = R0;
Label l_1, l_2, l_3, l_4, l_5, l_6; VectorSRegister tmp_vsr1 = VSR1;
VectorSRegister tmp_vsr2 = VSR2;
Label l_1, l_2, l_3, l_4, l_5, l_6, l_7;
// for short arrays, just do single element copy // for short arrays, just do single element copy
__ li(tmp3, 0); __ li(tmp3, 0);
@ -1593,6 +1638,8 @@ class StubGenerator: public StubCodeGenerator {
__ andi_(R5_ARG3, R5_ARG3, 7); __ andi_(R5_ARG3, R5_ARG3, 7);
__ mtctr(tmp1); __ mtctr(tmp1);
if (!VM_Version::has_vsx()) {
__ bind(l_6); __ bind(l_6);
// Use unrolled version for mass copying (copy 8 elements a time). // Use unrolled version for mass copying (copy 8 elements a time).
// Load feeding store gets zero latency on power6, however not on power 5. // Load feeding store gets zero latency on power6, however not on power 5.
@ -1608,7 +1655,44 @@ class StubGenerator: public StubCodeGenerator {
__ addi(R3_ARG1, R3_ARG1, 32); __ addi(R3_ARG1, R3_ARG1, 32);
__ addi(R4_ARG2, R4_ARG2, 32); __ addi(R4_ARG2, R4_ARG2, 32);
__ bdnz(l_6); __ bdnz(l_6);
}
} else { // Processor supports VSX, so use it to mass copy.
// Prefetch the data into the L2 cache.
__ dcbt(R3_ARG1, 0);
// If supported set DSCR pre-fetch to deepest.
if (VM_Version::has_mfdscr()) {
__ load_const_optimized(tmp2, VM_Version::_dscr_val | 7);
__ mtdscr(tmp2);
}
__ li(tmp1, 16);
// Backbranch target aligned to 32-byte. Not 16-byte align as
// loop contains < 8 instructions that fit inside a single
// i-cache sector.
__ align(32);
__ bind(l_7);
// Use loop with VSX load/store instructions to
// copy 8 elements a time.
__ lxvd2x(tmp_vsr1, 0, R3_ARG1); // Load src
__ stxvd2x(tmp_vsr1, 0, R4_ARG2); // Store to dst
__ lxvd2x(tmp_vsr2, tmp1, R3_ARG1); // Load src + 16
__ stxvd2x(tmp_vsr2, tmp1, R4_ARG2); // Store to dst + 16
__ addi(R3_ARG1, R3_ARG1, 32); // Update src+=32
__ addi(R4_ARG2, R4_ARG2, 32); // Update dsc+=32
__ bdnz(l_7); // Dec CTR and loop if not zero.
// Restore DSCR pre-fetch value.
if (VM_Version::has_mfdscr()) {
__ load_const_optimized(tmp2, VM_Version::_dscr_val);
__ mtdscr(tmp2);
}
} // VSX
} // FasterArrayCopy
// copy 1 element at a time // copy 1 element at a time
__ bind(l_2); __ bind(l_2);
@ -1757,7 +1841,10 @@ class StubGenerator: public StubCodeGenerator {
Register tmp3 = R8_ARG6; Register tmp3 = R8_ARG6;
Register tmp4 = R0; Register tmp4 = R0;
Label l_1, l_2, l_3, l_4; Label l_1, l_2, l_3, l_4, l_5;
VectorSRegister tmp_vsr1 = VSR1;
VectorSRegister tmp_vsr2 = VSR2;
{ // FasterArrayCopy { // FasterArrayCopy
__ cmpwi(CCR0, R5_ARG3, 3); __ cmpwi(CCR0, R5_ARG3, 3);
@ -1767,6 +1854,7 @@ class StubGenerator: public StubCodeGenerator {
__ andi_(R5_ARG3, R5_ARG3, 3); __ andi_(R5_ARG3, R5_ARG3, 3);
__ mtctr(tmp1); __ mtctr(tmp1);
if (!VM_Version::has_vsx()) {
__ bind(l_4); __ bind(l_4);
// Use unrolled version for mass copying (copy 4 elements a time). // Use unrolled version for mass copying (copy 4 elements a time).
// Load feeding store gets zero latency on Power6, however not on Power5. // Load feeding store gets zero latency on Power6, however not on Power5.
@ -1782,7 +1870,44 @@ class StubGenerator: public StubCodeGenerator {
__ addi(R3_ARG1, R3_ARG1, 32); __ addi(R3_ARG1, R3_ARG1, 32);
__ addi(R4_ARG2, R4_ARG2, 32); __ addi(R4_ARG2, R4_ARG2, 32);
__ bdnz(l_4); __ bdnz(l_4);
}
} else { // Processor supports VSX, so use it to mass copy.
// Prefetch the data into the L2 cache.
__ dcbt(R3_ARG1, 0);
// If supported set DSCR pre-fetch to deepest.
if (VM_Version::has_mfdscr()) {
__ load_const_optimized(tmp2, VM_Version::_dscr_val | 7);
__ mtdscr(tmp2);
}
__ li(tmp1, 16);
// Backbranch target aligned to 32-byte. Not 16-byte align as
// loop contains < 8 instructions that fit inside a single
// i-cache sector.
__ align(32);
__ bind(l_5);
// Use loop with VSX load/store instructions to
// copy 4 elements a time.
__ lxvd2x(tmp_vsr1, 0, R3_ARG1); // Load src
__ stxvd2x(tmp_vsr1, 0, R4_ARG2); // Store to dst
__ lxvd2x(tmp_vsr2, tmp1, R3_ARG1); // Load src + 16
__ stxvd2x(tmp_vsr2, tmp1, R4_ARG2); // Store to dst + 16
__ addi(R3_ARG1, R3_ARG1, 32); // Update src+=32
__ addi(R4_ARG2, R4_ARG2, 32); // Update dsc+=32
__ bdnz(l_5); // Dec CTR and loop if not zero.
// Restore DSCR pre-fetch value.
if (VM_Version::has_mfdscr()) {
__ load_const_optimized(tmp2, VM_Version::_dscr_val);
__ mtdscr(tmp2);
}
} // VSX
} // FasterArrayCopy
// copy 1 element at a time // copy 1 element at a time
__ bind(l_3); __ bind(l_3);

View File

@ -845,9 +845,40 @@ void TemplateInterpreterGenerator::generate_counter_overflow(Label& continue_ent
__ b(continue_entry); __ b(continue_entry);
} }
// See if we've got enough room on the stack for locals plus overhead below
// JavaThread::stack_overflow_limit(). If not, throw a StackOverflowError
// without going through the signal handler, i.e., reserved and yellow zones
// will not be made usable. The shadow zone must suffice to handle the
// overflow.
//
// Kills Rmem_frame_size, Rscratch1.
void TemplateInterpreterGenerator::generate_stack_overflow_check(Register Rmem_frame_size, Register Rscratch1) { void TemplateInterpreterGenerator::generate_stack_overflow_check(Register Rmem_frame_size, Register Rscratch1) {
Label done;
assert_different_registers(Rmem_frame_size, Rscratch1); assert_different_registers(Rmem_frame_size, Rscratch1);
__ generate_stack_overflow_check_with_compare_and_throw(Rmem_frame_size, Rscratch1);
BLOCK_COMMENT("stack_overflow_check_with_compare {");
__ sub(Rmem_frame_size, R1_SP, Rmem_frame_size);
__ ld(Rscratch1, thread_(stack_overflow_limit));
__ cmpld(CCR0/*is_stack_overflow*/, Rmem_frame_size, Rscratch1);
__ bgt(CCR0/*is_stack_overflow*/, done);
// The stack overflows. Load target address of the runtime stub and call it.
assert(StubRoutines::throw_StackOverflowError_entry() != NULL, "generated in wrong order");
__ load_const_optimized(Rscratch1, (StubRoutines::throw_StackOverflowError_entry()), R0);
__ mtctr(Rscratch1);
// Restore caller_sp.
#ifdef ASSERT
__ ld(Rscratch1, 0, R1_SP);
__ ld(R0, 0, R21_sender_SP);
__ cmpd(CCR0, R0, Rscratch1);
__ asm_assert_eq("backlink", 0x547);
#endif // ASSERT
__ mr(R1_SP, R21_sender_SP);
__ bctr();
__ align(32, 12);
__ bind(done);
BLOCK_COMMENT("} stack_overflow_check_with_compare");
} }
void TemplateInterpreterGenerator::unlock_method(bool check_exceptions) { void TemplateInterpreterGenerator::unlock_method(bool check_exceptions) {
@ -1014,10 +1045,10 @@ void TemplateInterpreterGenerator::generate_fixed_frame(bool native_call, Regist
// Enlarge by locals-parameters (not in case of native_call), shrink by ESP-SP-ABI48. // Enlarge by locals-parameters (not in case of native_call), shrink by ESP-SP-ABI48.
if (!native_call) { if (!native_call) {
// -------------------------------------------------------------------------- // Stack overflow check.
// Stack overflow check // Native calls don't need the stack size check since they have no
// expression stack and the arguments are already on the stack and
Label cont; // we only add a handful of words to the stack.
__ add(R11_scratch1, parent_frame_resize, top_frame_size); __ add(R11_scratch1, parent_frame_resize, top_frame_size);
generate_stack_overflow_check(R11_scratch1, R12_scratch2); generate_stack_overflow_check(R11_scratch1, R12_scratch2);
} }

View File

@ -1,6 +1,6 @@
/* /*
* Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2013, 2015 SAP SE. All rights reserved. * Copyright (c) 2013, 2016 SAP SE. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -4093,20 +4093,8 @@ void TemplateTable::monitorenter() {
__ lock_object(Rcurrent_monitor, Robj_to_lock); __ lock_object(Rcurrent_monitor, Robj_to_lock);
// Check if there's enough space on the stack for the monitors after locking. // Check if there's enough space on the stack for the monitors after locking.
Label Lskip_stack_check; // This emits a single store.
// Optimization: If the monitors stack section is less then a std page size (4K) don't run __ generate_stack_overflow_check(0);
// the stack check. There should be enough shadow pages to fit that in.
__ ld(Rscratch3, 0, R1_SP);
__ sub(Rscratch3, Rscratch3, R26_monitor);
__ cmpdi(CCR0, Rscratch3, 4*K);
__ blt(CCR0, Lskip_stack_check);
DEBUG_ONLY(__ untested("stack overflow check during monitor enter");)
__ li(Rscratch1, 0);
__ generate_stack_overflow_check_with_compare_and_throw(Rscratch1, Rscratch2);
__ align(32, 12);
__ bind(Lskip_stack_check);
// The bcp has already been incremented. Just need to dispatch to next instruction. // The bcp has already been incremented. Just need to dispatch to next instruction.
__ dispatch_next(vtos); __ dispatch_next(vtos);

View File

@ -59,11 +59,11 @@ define_pd_global(intx, InlineSmallCode, 1500);
// Stack slots are 2X larger in LP64 than in the 32 bit VM. // Stack slots are 2X larger in LP64 than in the 32 bit VM.
define_pd_global(intx, ThreadStackSize, 1024); define_pd_global(intx, ThreadStackSize, 1024);
define_pd_global(intx, VMThreadStackSize, 1024); define_pd_global(intx, VMThreadStackSize, 1024);
#define DEFAULT_STACK_SHADOW_PAGES (10 DEBUG_ONLY(+1)) #define DEFAULT_STACK_SHADOW_PAGES (20 DEBUG_ONLY(+2))
#else #else
define_pd_global(intx, ThreadStackSize, 512); define_pd_global(intx, ThreadStackSize, 512);
define_pd_global(intx, VMThreadStackSize, 512); define_pd_global(intx, VMThreadStackSize, 512);
#define DEFAULT_STACK_SHADOW_PAGES (3 DEBUG_ONLY(+1)) #define DEFAULT_STACK_SHADOW_PAGES (6 DEBUG_ONLY(+2))
#endif // _LP64 #endif // _LP64
#define MIN_STACK_YELLOW_PAGES DEFAULT_STACK_YELLOW_PAGES #define MIN_STACK_YELLOW_PAGES DEFAULT_STACK_YELLOW_PAGES

View File

@ -578,51 +578,39 @@ void TemplateInterpreterGenerator::lock_method() {
__ lock_object(Lmonitors, O0); __ lock_object(Lmonitors, O0);
} }
// See if we've got enough room on the stack for locals plus overhead below
// JavaThread::stack_overflow_limit(). If not, throw a StackOverflowError
// without going through the signal handler, i.e., reserved and yellow zones
// will not be made usable. The shadow zone must suffice to handle the
// overflow.
void TemplateInterpreterGenerator::generate_stack_overflow_check(Register Rframe_size, void TemplateInterpreterGenerator::generate_stack_overflow_check(Register Rframe_size,
Register Rscratch, Register Rscratch) {
Register Rscratch2) {
const int page_size = os::vm_page_size(); const int page_size = os::vm_page_size();
Label after_frame_check; Label after_frame_check;
assert_different_registers(Rframe_size, Rscratch, Rscratch2); assert_different_registers(Rframe_size, Rscratch);
__ set(page_size, Rscratch); __ set(page_size, Rscratch);
__ cmp_and_br_short(Rframe_size, Rscratch, Assembler::lessEqual, Assembler::pt, after_frame_check); __ cmp_and_br_short(Rframe_size, Rscratch, Assembler::lessEqual, Assembler::pt, after_frame_check);
// get the stack base, and in debug, verify it is non-zero // Get the stack overflow limit, and in debug, verify it is non-zero.
__ ld_ptr( G2_thread, Thread::stack_base_offset(), Rscratch ); __ ld_ptr(G2_thread, JavaThread::stack_overflow_limit_offset(), Rscratch);
#ifdef ASSERT #ifdef ASSERT
Label base_not_zero; Label limit_ok;
__ br_notnull_short(Rscratch, Assembler::pn, base_not_zero); __ br_notnull_short(Rscratch, Assembler::pn, limit_ok);
__ stop("stack base is zero in generate_stack_overflow_check"); __ stop("stack overflow limit is zero in generate_stack_overflow_check");
__ bind(base_not_zero); __ bind(limit_ok);
#endif #endif
// get the stack size, and in debug, verify it is non-zero
assert( sizeof(size_t) == sizeof(intptr_t), "wrong load size" );
__ ld_ptr( G2_thread, Thread::stack_size_offset(), Rscratch2 );
#ifdef ASSERT
Label size_not_zero;
__ br_notnull_short(Rscratch2, Assembler::pn, size_not_zero);
__ stop("stack size is zero in generate_stack_overflow_check");
__ bind(size_not_zero);
#endif
// compute the beginning of the protected zone minus the requested frame size
__ sub( Rscratch, Rscratch2, Rscratch );
__ set(MAX2(JavaThread::stack_shadow_zone_size(), JavaThread::stack_guard_zone_size()), Rscratch2 );
__ add( Rscratch, Rscratch2, Rscratch );
// Add in the size of the frame (which is the same as subtracting it from the // Add in the size of the frame (which is the same as subtracting it from the
// SP, which would take another register // SP, which would take another register.
__ add( Rscratch, Rframe_size, Rscratch ); __ add(Rscratch, Rframe_size, Rscratch);
// the frame is greater than one page in size, so check against // The frame is greater than one page in size, so check against
// the bottom of the stack // the bottom of the stack.
__ cmp_and_brx_short(SP, Rscratch, Assembler::greaterUnsigned, Assembler::pt, after_frame_check); __ cmp_and_brx_short(SP, Rscratch, Assembler::greaterUnsigned, Assembler::pt, after_frame_check);
// the stack will overflow, throw an exception // The stack will overflow, throw an exception.
// Note that SP is restored to sender's sp (in the delay slot). This // Note that SP is restored to sender's sp (in the delay slot). This
// is necessary if the sender's frame is an extended compiled frame // is necessary if the sender's frame is an extended compiled frame
@ -636,8 +624,8 @@ void TemplateInterpreterGenerator::generate_stack_overflow_check(Register Rframe
__ jump_to(stub, Rscratch); __ jump_to(stub, Rscratch);
__ delayed()->mov(O5_savedSP, SP); __ delayed()->mov(O5_savedSP, SP);
// if you get to here, then there is enough stack space // If you get to here, then there is enough stack space.
__ bind( after_frame_check ); __ bind(after_frame_check);
} }
@ -821,40 +809,44 @@ void TemplateInterpreterGenerator::generate_fixed_frame(bool native_call) {
__ add( Gframe_size, extra_space, Gframe_size); __ add( Gframe_size, extra_space, Gframe_size);
__ round_to( Gframe_size, WordsPerLong ); __ round_to( Gframe_size, WordsPerLong );
__ sll( Gframe_size, LogBytesPerWord, Gframe_size ); __ sll( Gframe_size, LogBytesPerWord, Gframe_size );
// Native calls don't need the stack size check since they have no
// expression stack and the arguments are already on the stack and
// we only add a handful of words to the stack.
} else { } else {
// //
// Compute number of locals in method apart from incoming parameters // Compute number of locals in method apart from incoming parameters
// //
const Address size_of_locals (Otmp1, ConstMethod::size_of_locals_offset()); const Address size_of_locals(Otmp1, ConstMethod::size_of_locals_offset());
__ ld_ptr( constMethod, Otmp1 ); __ ld_ptr(constMethod, Otmp1);
__ lduh( size_of_locals, Otmp1 ); __ lduh(size_of_locals, Otmp1);
__ sub( Otmp1, Glocals_size, Glocals_size ); __ sub(Otmp1, Glocals_size, Glocals_size);
__ round_to( Glocals_size, WordsPerLong ); __ round_to(Glocals_size, WordsPerLong);
__ sll( Glocals_size, Interpreter::logStackElementSize, Glocals_size ); __ sll(Glocals_size, Interpreter::logStackElementSize, Glocals_size);
// see if the frame is greater than one page in size. If so, // See if the frame is greater than one page in size. If so,
// then we need to verify there is enough stack space remaining // then we need to verify there is enough stack space remaining.
// Frame_size = (max_stack + extra_space) * BytesPerWord; // Frame_size = (max_stack + extra_space) * BytesPerWord;
__ ld_ptr( constMethod, Gframe_size ); __ ld_ptr(constMethod, Gframe_size);
__ lduh( Gframe_size, in_bytes(ConstMethod::max_stack_offset()), Gframe_size ); __ lduh(Gframe_size, in_bytes(ConstMethod::max_stack_offset()), Gframe_size);
__ add( Gframe_size, extra_space, Gframe_size ); __ add(Gframe_size, extra_space, Gframe_size);
__ round_to( Gframe_size, WordsPerLong ); __ round_to(Gframe_size, WordsPerLong);
__ sll( Gframe_size, Interpreter::logStackElementSize, Gframe_size); __ sll(Gframe_size, Interpreter::logStackElementSize, Gframe_size);
// Add in java locals size for stack overflow check only // Add in java locals size for stack overflow check only
__ add( Gframe_size, Glocals_size, Gframe_size ); __ add(Gframe_size, Glocals_size, Gframe_size);
const Register Otmp2 = O4; const Register Otmp2 = O4;
assert_different_registers(Otmp1, Otmp2, O5_savedSP); assert_different_registers(Otmp1, Otmp2, O5_savedSP);
generate_stack_overflow_check(Gframe_size, Otmp1, Otmp2); generate_stack_overflow_check(Gframe_size, Otmp1);
__ sub( Gframe_size, Glocals_size, Gframe_size); __ sub(Gframe_size, Glocals_size, Gframe_size);
// //
// bump SP to accomodate the extra locals // bump SP to accomodate the extra locals
// //
__ sub( SP, Glocals_size, SP ); __ sub(SP, Glocals_size, SP);
} }
// //

View File

@ -1173,6 +1173,23 @@ void Assembler::addl(Address dst, int32_t imm32) {
emit_arith_operand(0x81, rax, dst, imm32); emit_arith_operand(0x81, rax, dst, imm32);
} }
void Assembler::addb(Address dst, int imm8) {
InstructionMark im(this);
prefix(dst);
emit_int8((unsigned char)0x80);
emit_operand(rax, dst, 1);
emit_int8(imm8);
}
void Assembler::addw(Address dst, int imm16) {
InstructionMark im(this);
emit_int8(0x66);
prefix(dst);
emit_int8((unsigned char)0x81);
emit_operand(rax, dst, 2);
emit_int16(imm16);
}
void Assembler::addl(Address dst, Register src) { void Assembler::addl(Address dst, Register src) {
InstructionMark im(this); InstructionMark im(this);
prefix(dst, src); prefix(dst, src);
@ -4567,6 +4584,23 @@ void Assembler::xabort(int8_t imm8) {
emit_int8((unsigned char)(imm8 & 0xFF)); emit_int8((unsigned char)(imm8 & 0xFF));
} }
void Assembler::xaddb(Address dst, Register src) {
InstructionMark im(this);
prefix(dst, src, true);
emit_int8(0x0F);
emit_int8((unsigned char)0xC0);
emit_operand(src, dst);
}
void Assembler::xaddw(Address dst, Register src) {
InstructionMark im(this);
emit_int8(0x66);
prefix(dst, src);
emit_int8(0x0F);
emit_int8((unsigned char)0xC1);
emit_operand(src, dst);
}
void Assembler::xaddl(Address dst, Register src) { void Assembler::xaddl(Address dst, Register src) {
InstructionMark im(this); InstructionMark im(this);
prefix(dst, src); prefix(dst, src);
@ -4593,6 +4627,21 @@ void Assembler::xbegin(Label& abort, relocInfo::relocType rtype) {
} }
} }
void Assembler::xchgb(Register dst, Address src) { // xchg
InstructionMark im(this);
prefix(src, dst, true);
emit_int8((unsigned char)0x86);
emit_operand(dst, src);
}
void Assembler::xchgw(Register dst, Address src) { // xchg
InstructionMark im(this);
emit_int8(0x66);
prefix(src, dst);
emit_int8((unsigned char)0x87);
emit_operand(dst, src);
}
void Assembler::xchgl(Register dst, Address src) { // xchg void Assembler::xchgl(Register dst, Address src) { // xchg
InstructionMark im(this); InstructionMark im(this);
prefix(src, dst); prefix(src, dst);

View File

@ -881,6 +881,9 @@ private:
void adcq(Register dst, Address src); void adcq(Register dst, Address src);
void adcq(Register dst, Register src); void adcq(Register dst, Register src);
void addb(Address dst, int imm8);
void addw(Address dst, int imm16);
void addl(Address dst, int32_t imm32); void addl(Address dst, int32_t imm32);
void addl(Address dst, Register src); void addl(Address dst, Register src);
void addl(Register dst, int32_t imm32); void addl(Register dst, int32_t imm32);
@ -1816,12 +1819,15 @@ private:
void xabort(int8_t imm8); void xabort(int8_t imm8);
void xaddb(Address dst, Register src);
void xaddw(Address dst, Register src);
void xaddl(Address dst, Register src); void xaddl(Address dst, Register src);
void xaddq(Address dst, Register src); void xaddq(Address dst, Register src);
void xbegin(Label& abort, relocInfo::relocType rtype = relocInfo::none); void xbegin(Label& abort, relocInfo::relocType rtype = relocInfo::none);
void xchgb(Register reg, Address adr);
void xchgw(Register reg, Address adr);
void xchgl(Register reg, Address adr); void xchgl(Register reg, Address adr);
void xchgl(Register dst, Register src); void xchgl(Register dst, Register src);

View File

@ -473,7 +473,11 @@ void TemplateInterpreterGenerator::generate_counter_overflow(Label& do_continue)
__ jmp(do_continue, relocInfo::none); __ jmp(do_continue, relocInfo::none);
} }
// See if we've got enough room on the stack for locals plus overhead. // See if we've got enough room on the stack for locals plus overhead below
// JavaThread::stack_overflow_limit(). If not, throw a StackOverflowError
// without going through the signal handler, i.e., reserved and yellow zones
// will not be made usable. The shadow zone must suffice to handle the
// overflow.
// The expression stack grows down incrementally, so the normal guard // The expression stack grows down incrementally, so the normal guard
// page mechanism will work for that. // page mechanism will work for that.
// //
@ -518,40 +522,26 @@ void TemplateInterpreterGenerator::generate_stack_overflow_check(void) {
__ get_thread(thread); __ get_thread(thread);
#endif #endif
const Address stack_base(thread, Thread::stack_base_offset()); const Address stack_limit(thread, JavaThread::stack_overflow_limit_offset());
const Address stack_size(thread, Thread::stack_size_offset());
// locals + overhead, in bytes // locals + overhead, in bytes
__ mov(rax, rdx); __ mov(rax, rdx);
__ shlptr(rax, Interpreter::logStackElementSize); // 2 slots per parameter. __ shlptr(rax, Interpreter::logStackElementSize); // Convert parameter count to bytes.
__ addptr(rax, overhead_size); __ addptr(rax, overhead_size);
#ifdef ASSERT #ifdef ASSERT
Label stack_base_okay, stack_size_okay; Label limit_okay;
// verify that thread stack base is non-zero // Verify that thread stack overflow limit is non-zero.
__ cmpptr(stack_base, (int32_t)NULL_WORD); __ cmpptr(stack_limit, (int32_t)NULL_WORD);
__ jcc(Assembler::notEqual, stack_base_okay); __ jcc(Assembler::notEqual, limit_okay);
__ stop("stack base is zero"); __ stop("stack overflow limit is zero");
__ bind(stack_base_okay); __ bind(limit_okay);
// verify that thread stack size is non-zero
__ cmpptr(stack_size, 0);
__ jcc(Assembler::notEqual, stack_size_okay);
__ stop("stack size is zero");
__ bind(stack_size_okay);
#endif #endif
// Add stack base to locals and subtract stack size // Add locals/frame size to stack limit.
__ addptr(rax, stack_base); __ addptr(rax, stack_limit);
__ subptr(rax, stack_size);
// Use the bigger size for banging. // Check against the current stack bottom.
const int max_bang_size = (int)MAX2(JavaThread::stack_shadow_zone_size(),
JavaThread::stack_guard_zone_size());
// add in the red and yellow zone sizes
__ addptr(rax, max_bang_size);
// check against the current stack bottom
__ cmpptr(rsp, rax); __ cmpptr(rsp, rax);
__ jcc(Assembler::above, after_frame_check_pop); __ jcc(Assembler::above, after_frame_check_pop);
@ -782,8 +772,6 @@ address TemplateInterpreterGenerator::generate_Reference_get_entry(void) {
return NULL; return NULL;
} }
// TODO: rather than touching all pages, check against stack_overflow_limit and bang yellow page to
// generate exception. Windows might need this to map the shadow pages though.
void TemplateInterpreterGenerator::bang_stack_shadow_pages(bool native_call) { void TemplateInterpreterGenerator::bang_stack_shadow_pages(bool native_call) {
// Quick & dirty stack overflow checking: bang the stack & handle trap. // Quick & dirty stack overflow checking: bang the stack & handle trap.
// Note that we do the banging after the frame is setup, since the exception // Note that we do the banging after the frame is setup, since the exception
@ -945,7 +933,7 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
__ load_unsigned_short(t, Address(t, ConstMethod::size_of_parameters_offset())); __ load_unsigned_short(t, Address(t, ConstMethod::size_of_parameters_offset()));
#ifndef _LP64 #ifndef _LP64
__ shlptr(t, Interpreter::logStackElementSize); __ shlptr(t, Interpreter::logStackElementSize); // Convert parameter count to bytes.
__ addptr(t, 2*wordSize); // allocate two more slots for JNIEnv and possible mirror __ addptr(t, 2*wordSize); // allocate two more slots for JNIEnv and possible mirror
__ subptr(rsp, t); __ subptr(rsp, t);
__ andptr(rsp, -(StackAlignmentInBytes)); // gcc needs 16 byte aligned stacks to do XMM intrinsics __ andptr(rsp, -(StackAlignmentInBytes)); // gcc needs 16 byte aligned stacks to do XMM intrinsics

View File

@ -2127,6 +2127,31 @@ encode %{
emit_rm( cbuf, 0x0, 1, $mem_ptr$$reg ); emit_rm( cbuf, 0x0, 1, $mem_ptr$$reg );
%} %}
enc_class enc_cmpxchgb(eSIRegP mem_ptr) %{
// [Lock]
if( os::is_MP() )
emit_opcode(cbuf,0xF0);
// CMPXCHGB [Eptr]
emit_opcode(cbuf,0x0F);
emit_opcode(cbuf,0xB0);
emit_rm( cbuf, 0x0, 1, $mem_ptr$$reg );
%}
enc_class enc_cmpxchgw(eSIRegP mem_ptr) %{
// [Lock]
if( os::is_MP() )
emit_opcode(cbuf,0xF0);
// 16-bit mode
emit_opcode(cbuf, 0x66);
// CMPXCHGW [Eptr]
emit_opcode(cbuf,0x0F);
emit_opcode(cbuf,0xB1);
emit_rm( cbuf, 0x0, 1, $mem_ptr$$reg );
%}
enc_class enc_flags_ne_to_boolean( iRegI res ) %{ enc_class enc_flags_ne_to_boolean( iRegI res ) %{
int res_encoding = $res$$reg; int res_encoding = $res$$reg;
@ -7262,6 +7287,34 @@ instruct compareAndSwapP( rRegI res, pRegP mem_ptr, eAXRegP oldval, eCXRegP new
ins_pipe( pipe_cmpxchg ); ins_pipe( pipe_cmpxchg );
%} %}
instruct compareAndSwapB( rRegI res, pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFlagsReg cr ) %{
match(Set res (CompareAndSwapB mem_ptr (Binary oldval newval)));
match(Set res (WeakCompareAndSwapB mem_ptr (Binary oldval newval)));
effect(KILL cr, KILL oldval);
format %{ "CMPXCHGB [$mem_ptr],$newval\t# If EAX==[$mem_ptr] Then store $newval into [$mem_ptr]\n\t"
"MOV $res,0\n\t"
"JNE,s fail\n\t"
"MOV $res,1\n"
"fail:" %}
ins_encode( enc_cmpxchgb(mem_ptr),
enc_flags_ne_to_boolean(res) );
ins_pipe( pipe_cmpxchg );
%}
instruct compareAndSwapS( rRegI res, pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFlagsReg cr ) %{
match(Set res (CompareAndSwapS mem_ptr (Binary oldval newval)));
match(Set res (WeakCompareAndSwapS mem_ptr (Binary oldval newval)));
effect(KILL cr, KILL oldval);
format %{ "CMPXCHGW [$mem_ptr],$newval\t# If EAX==[$mem_ptr] Then store $newval into [$mem_ptr]\n\t"
"MOV $res,0\n\t"
"JNE,s fail\n\t"
"MOV $res,1\n"
"fail:" %}
ins_encode( enc_cmpxchgw(mem_ptr),
enc_flags_ne_to_boolean(res) );
ins_pipe( pipe_cmpxchg );
%}
instruct compareAndSwapI( rRegI res, pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFlagsReg cr) %{ instruct compareAndSwapI( rRegI res, pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFlagsReg cr) %{
match(Set res (CompareAndSwapI mem_ptr (Binary oldval newval))); match(Set res (CompareAndSwapI mem_ptr (Binary oldval newval)));
match(Set res (WeakCompareAndSwapI mem_ptr (Binary oldval newval))); match(Set res (WeakCompareAndSwapI mem_ptr (Binary oldval newval)));
@ -7292,6 +7345,22 @@ instruct compareAndExchangeP( pRegP mem_ptr, eAXRegP oldval, eCXRegP newval, eFl
ins_pipe( pipe_cmpxchg ); ins_pipe( pipe_cmpxchg );
%} %}
instruct compareAndExchangeB( pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFlagsReg cr) %{
match(Set oldval (CompareAndExchangeB mem_ptr (Binary oldval newval)));
effect(KILL cr);
format %{ "CMPXCHGB [$mem_ptr],$newval\t# If EAX==[$mem_ptr] Then store $newval into [$mem_ptr]\n\t" %}
ins_encode( enc_cmpxchgb(mem_ptr) );
ins_pipe( pipe_cmpxchg );
%}
instruct compareAndExchangeS( pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFlagsReg cr) %{
match(Set oldval (CompareAndExchangeS mem_ptr (Binary oldval newval)));
effect(KILL cr);
format %{ "CMPXCHGW [$mem_ptr],$newval\t# If EAX==[$mem_ptr] Then store $newval into [$mem_ptr]\n\t" %}
ins_encode( enc_cmpxchgw(mem_ptr) );
ins_pipe( pipe_cmpxchg );
%}
instruct compareAndExchangeI( pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFlagsReg cr) %{ instruct compareAndExchangeI( pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFlagsReg cr) %{
match(Set oldval (CompareAndExchangeI mem_ptr (Binary oldval newval))); match(Set oldval (CompareAndExchangeI mem_ptr (Binary oldval newval)));
effect(KILL cr); effect(KILL cr);
@ -7300,6 +7369,53 @@ instruct compareAndExchangeI( pRegP mem_ptr, eAXRegI oldval, eCXRegI newval, eFl
ins_pipe( pipe_cmpxchg ); ins_pipe( pipe_cmpxchg );
%} %}
instruct xaddB_no_res( memory mem, Universe dummy, immI add, eFlagsReg cr) %{
predicate(n->as_LoadStore()->result_not_used());
match(Set dummy (GetAndAddB mem add));
effect(KILL cr);
format %{ "ADDB [$mem],$add" %}
ins_encode %{
if (os::is_MP()) { __ lock(); }
__ addb($mem$$Address, $add$$constant);
%}
ins_pipe( pipe_cmpxchg );
%}
// Important to match to xRegI: only 8-bit regs.
instruct xaddB( memory mem, xRegI newval, eFlagsReg cr) %{
match(Set newval (GetAndAddB mem newval));
effect(KILL cr);
format %{ "XADDB [$mem],$newval" %}
ins_encode %{
if (os::is_MP()) { __ lock(); }
__ xaddb($mem$$Address, $newval$$Register);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xaddS_no_res( memory mem, Universe dummy, immI add, eFlagsReg cr) %{
predicate(n->as_LoadStore()->result_not_used());
match(Set dummy (GetAndAddS mem add));
effect(KILL cr);
format %{ "ADDS [$mem],$add" %}
ins_encode %{
if (os::is_MP()) { __ lock(); }
__ addw($mem$$Address, $add$$constant);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xaddS( memory mem, rRegI newval, eFlagsReg cr) %{
match(Set newval (GetAndAddS mem newval));
effect(KILL cr);
format %{ "XADDS [$mem],$newval" %}
ins_encode %{
if (os::is_MP()) { __ lock(); }
__ xaddw($mem$$Address, $newval$$Register);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xaddI_no_res( memory mem, Universe dummy, immI add, eFlagsReg cr) %{ instruct xaddI_no_res( memory mem, Universe dummy, immI add, eFlagsReg cr) %{
predicate(n->as_LoadStore()->result_not_used()); predicate(n->as_LoadStore()->result_not_used());
match(Set dummy (GetAndAddI mem add)); match(Set dummy (GetAndAddI mem add));
@ -7323,6 +7439,25 @@ instruct xaddI( memory mem, rRegI newval, eFlagsReg cr) %{
ins_pipe( pipe_cmpxchg ); ins_pipe( pipe_cmpxchg );
%} %}
// Important to match to xRegI: only 8-bit regs.
instruct xchgB( memory mem, xRegI newval) %{
match(Set newval (GetAndSetB mem newval));
format %{ "XCHGB $newval,[$mem]" %}
ins_encode %{
__ xchgb($newval$$Register, $mem$$Address);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xchgS( memory mem, rRegI newval) %{
match(Set newval (GetAndSetS mem newval));
format %{ "XCHGW $newval,[$mem]" %}
ins_encode %{
__ xchgw($newval$$Register, $mem$$Address);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xchgI( memory mem, rRegI newval) %{ instruct xchgI( memory mem, rRegI newval) %{
match(Set newval (GetAndSetI mem newval)); match(Set newval (GetAndSetI mem newval));
format %{ "XCHGL $newval,[$mem]" %} format %{ "XCHGL $newval,[$mem]" %}

View File

@ -7340,6 +7340,54 @@ instruct compareAndSwapI(rRegI res,
ins_pipe( pipe_cmpxchg ); ins_pipe( pipe_cmpxchg );
%} %}
instruct compareAndSwapB(rRegI res,
memory mem_ptr,
rax_RegI oldval, rRegI newval,
rFlagsReg cr)
%{
match(Set res (CompareAndSwapB mem_ptr (Binary oldval newval)));
match(Set res (WeakCompareAndSwapB mem_ptr (Binary oldval newval)));
effect(KILL cr, KILL oldval);
format %{ "cmpxchgb $mem_ptr,$newval\t# "
"If rax == $mem_ptr then store $newval into $mem_ptr\n\t"
"sete $res\n\t"
"movzbl $res, $res" %}
opcode(0x0F, 0xB0);
ins_encode(lock_prefix,
REX_reg_mem(newval, mem_ptr),
OpcP, OpcS,
reg_mem(newval, mem_ptr),
REX_breg(res), Opcode(0x0F), Opcode(0x94), reg(res), // sete
REX_reg_breg(res, res), // movzbl
Opcode(0xF), Opcode(0xB6), reg_reg(res, res));
ins_pipe( pipe_cmpxchg );
%}
instruct compareAndSwapS(rRegI res,
memory mem_ptr,
rax_RegI oldval, rRegI newval,
rFlagsReg cr)
%{
match(Set res (CompareAndSwapS mem_ptr (Binary oldval newval)));
match(Set res (WeakCompareAndSwapS mem_ptr (Binary oldval newval)));
effect(KILL cr, KILL oldval);
format %{ "cmpxchgw $mem_ptr,$newval\t# "
"If rax == $mem_ptr then store $newval into $mem_ptr\n\t"
"sete $res\n\t"
"movzbl $res, $res" %}
opcode(0x0F, 0xB1);
ins_encode(lock_prefix,
SizePrefix,
REX_reg_mem(newval, mem_ptr),
OpcP, OpcS,
reg_mem(newval, mem_ptr),
REX_breg(res), Opcode(0x0F), Opcode(0x94), reg(res), // sete
REX_reg_breg(res, res), // movzbl
Opcode(0xF), Opcode(0xB6), reg_reg(res, res));
ins_pipe( pipe_cmpxchg );
%}
instruct compareAndSwapN(rRegI res, instruct compareAndSwapN(rRegI res,
memory mem_ptr, memory mem_ptr,
@ -7364,6 +7412,45 @@ instruct compareAndSwapN(rRegI res,
ins_pipe( pipe_cmpxchg ); ins_pipe( pipe_cmpxchg );
%} %}
instruct compareAndExchangeB(
memory mem_ptr,
rax_RegI oldval, rRegI newval,
rFlagsReg cr)
%{
match(Set oldval (CompareAndExchangeB mem_ptr (Binary oldval newval)));
effect(KILL cr);
format %{ "cmpxchgb $mem_ptr,$newval\t# "
"If rax == $mem_ptr then store $newval into $mem_ptr\n\t" %}
opcode(0x0F, 0xB0);
ins_encode(lock_prefix,
REX_reg_mem(newval, mem_ptr),
OpcP, OpcS,
reg_mem(newval, mem_ptr) // lock cmpxchg
);
ins_pipe( pipe_cmpxchg );
%}
instruct compareAndExchangeS(
memory mem_ptr,
rax_RegI oldval, rRegI newval,
rFlagsReg cr)
%{
match(Set oldval (CompareAndExchangeS mem_ptr (Binary oldval newval)));
effect(KILL cr);
format %{ "cmpxchgw $mem_ptr,$newval\t# "
"If rax == $mem_ptr then store $newval into $mem_ptr\n\t" %}
opcode(0x0F, 0xB1);
ins_encode(lock_prefix,
SizePrefix,
REX_reg_mem(newval, mem_ptr),
OpcP, OpcS,
reg_mem(newval, mem_ptr) // lock cmpxchg
);
ins_pipe( pipe_cmpxchg );
%}
instruct compareAndExchangeI( instruct compareAndExchangeI(
memory mem_ptr, memory mem_ptr,
rax_RegI oldval, rRegI newval, rax_RegI oldval, rRegI newval,
@ -7441,6 +7528,52 @@ instruct compareAndExchangeP(
ins_pipe( pipe_cmpxchg ); ins_pipe( pipe_cmpxchg );
%} %}
instruct xaddB_no_res( memory mem, Universe dummy, immI add, rFlagsReg cr) %{
predicate(n->as_LoadStore()->result_not_used());
match(Set dummy (GetAndAddB mem add));
effect(KILL cr);
format %{ "ADDB [$mem],$add" %}
ins_encode %{
if (os::is_MP()) { __ lock(); }
__ addb($mem$$Address, $add$$constant);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xaddB( memory mem, rRegI newval, rFlagsReg cr) %{
match(Set newval (GetAndAddB mem newval));
effect(KILL cr);
format %{ "XADDB [$mem],$newval" %}
ins_encode %{
if (os::is_MP()) { __ lock(); }
__ xaddb($mem$$Address, $newval$$Register);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xaddS_no_res( memory mem, Universe dummy, immI add, rFlagsReg cr) %{
predicate(n->as_LoadStore()->result_not_used());
match(Set dummy (GetAndAddS mem add));
effect(KILL cr);
format %{ "ADDW [$mem],$add" %}
ins_encode %{
if (os::is_MP()) { __ lock(); }
__ addw($mem$$Address, $add$$constant);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xaddS( memory mem, rRegI newval, rFlagsReg cr) %{
match(Set newval (GetAndAddS mem newval));
effect(KILL cr);
format %{ "XADDW [$mem],$newval" %}
ins_encode %{
if (os::is_MP()) { __ lock(); }
__ xaddw($mem$$Address, $newval$$Register);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xaddI_no_res( memory mem, Universe dummy, immI add, rFlagsReg cr) %{ instruct xaddI_no_res( memory mem, Universe dummy, immI add, rFlagsReg cr) %{
predicate(n->as_LoadStore()->result_not_used()); predicate(n->as_LoadStore()->result_not_used());
match(Set dummy (GetAndAddI mem add)); match(Set dummy (GetAndAddI mem add));
@ -7487,6 +7620,24 @@ instruct xaddL( memory mem, rRegL newval, rFlagsReg cr) %{
ins_pipe( pipe_cmpxchg ); ins_pipe( pipe_cmpxchg );
%} %}
instruct xchgB( memory mem, rRegI newval) %{
match(Set newval (GetAndSetB mem newval));
format %{ "XCHGB $newval,[$mem]" %}
ins_encode %{
__ xchgb($newval$$Register, $mem$$Address);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xchgS( memory mem, rRegI newval) %{
match(Set newval (GetAndSetS mem newval));
format %{ "XCHGW $newval,[$mem]" %}
ins_encode %{
__ xchgw($newval$$Register, $mem$$Address);
%}
ins_pipe( pipe_cmpxchg );
%}
instruct xchgI( memory mem, rRegI newval) %{ instruct xchgI( memory mem, rRegI newval) %{
match(Set newval (GetAndSetI mem newval)); match(Set newval (GetAndSetI mem newval));
format %{ "XCHGL $newval,[$mem]" %} format %{ "XCHGL $newval,[$mem]" %}

View File

@ -28,6 +28,7 @@ import java.util.EnumSet;
import jdk.vm.ci.code.Architecture; import jdk.vm.ci.code.Architecture;
import jdk.vm.ci.code.Register; import jdk.vm.ci.code.Register;
import jdk.vm.ci.code.Register.RegisterCategory; import jdk.vm.ci.code.Register.RegisterCategory;
import jdk.vm.ci.code.RegisterArray;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
import jdk.vm.ci.meta.PlatformKind; import jdk.vm.ci.meta.PlatformKind;
@ -84,13 +85,13 @@ public class AArch64 extends Architecture {
public static final Register lr = r30; public static final Register lr = r30;
// @formatter:off // @formatter:off
public static final Register[] cpuRegisters = { public static final RegisterArray cpuRegisters = new RegisterArray(
r0, r1, r2, r3, r4, r5, r6, r7, r0, r1, r2, r3, r4, r5, r6, r7,
r8, r9, r10, r11, r12, r13, r14, r15, r8, r9, r10, r11, r12, r13, r14, r15,
r16, r17, r18, r19, r20, r21, r22, r23, r16, r17, r18, r19, r20, r21, r22, r23,
r24, r25, r26, r27, r28, r29, r30, r31, r24, r25, r26, r27, r28, r29, r30, r31,
zr, sp zr, sp
}; );
// @formatter:on // @formatter:on
public static final RegisterCategory SIMD = new RegisterCategory("SIMD"); public static final RegisterCategory SIMD = new RegisterCategory("SIMD");
@ -130,16 +131,16 @@ public class AArch64 extends Architecture {
public static final Register v31 = new Register(65, 31, "v31", SIMD); public static final Register v31 = new Register(65, 31, "v31", SIMD);
// @formatter:off // @formatter:off
public static final Register[] simdRegisters = { public static final RegisterArray simdRegisters = new RegisterArray(
v0, v1, v2, v3, v4, v5, v6, v7, v0, v1, v2, v3, v4, v5, v6, v7,
v8, v9, v10, v11, v12, v13, v14, v15, v8, v9, v10, v11, v12, v13, v14, v15,
v16, v17, v18, v19, v20, v21, v22, v23, v16, v17, v18, v19, v20, v21, v22, v23,
v24, v25, v26, v27, v28, v29, v30, v31 v24, v25, v26, v27, v28, v29, v30, v31
}; );
// @formatter:on // @formatter:on
// @formatter:off // @formatter:off
public static final Register[] allRegisters = { public static final RegisterArray allRegisters = new RegisterArray(
r0, r1, r2, r3, r4, r5, r6, r7, r0, r1, r2, r3, r4, r5, r6, r7,
r8, r9, r10, r11, r12, r13, r14, r15, r8, r9, r10, r11, r12, r13, r14, r15,
r16, r17, r18, r19, r20, r21, r22, r23, r16, r17, r18, r19, r20, r21, r22, r23,
@ -150,7 +151,7 @@ public class AArch64 extends Architecture {
v8, v9, v10, v11, v12, v13, v14, v15, v8, v9, v10, v11, v12, v13, v14, v15,
v16, v17, v18, v19, v20, v21, v22, v23, v16, v17, v18, v19, v20, v21, v22, v23,
v24, v25, v26, v27, v28, v29, v30, v31 v24, v25, v26, v27, v28, v29, v30, v31
}; );
// @formatter:on // @formatter:on
/** /**

View File

@ -33,6 +33,7 @@ import java.util.EnumSet;
import jdk.vm.ci.code.Architecture; import jdk.vm.ci.code.Architecture;
import jdk.vm.ci.code.Register; import jdk.vm.ci.code.Register;
import jdk.vm.ci.code.Register.RegisterCategory; import jdk.vm.ci.code.Register.RegisterCategory;
import jdk.vm.ci.code.RegisterArray;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
import jdk.vm.ci.meta.PlatformKind; import jdk.vm.ci.meta.PlatformKind;
@ -131,14 +132,14 @@ public class AMD64 extends Architecture {
public static final Register k6 = new Register(54, 6, "k6", MASK); public static final Register k6 = new Register(54, 6, "k6", MASK);
public static final Register k7 = new Register(55, 7, "k7", MASK); public static final Register k7 = new Register(55, 7, "k7", MASK);
public static final Register[] valueRegistersSSE = { public static final RegisterArray valueRegistersSSE = new RegisterArray(
rax, rcx, rdx, rbx, rsp, rbp, rsi, rdi, rax, rcx, rdx, rbx, rsp, rbp, rsi, rdi,
r8, r9, r10, r11, r12, r13, r14, r15, r8, r9, r10, r11, r12, r13, r14, r15,
xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7, xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7,
xmm8, xmm9, xmm10, xmm11, xmm12, xmm13, xmm14, xmm15 xmm8, xmm9, xmm10, xmm11, xmm12, xmm13, xmm14, xmm15
}; );
public static final Register[] valueRegistersAVX512 = { public static final RegisterArray valueRegistersAVX512 = new RegisterArray(
rax, rcx, rdx, rbx, rsp, rbp, rsi, rdi, rax, rcx, rdx, rbx, rsp, rbp, rsi, rdi,
r8, r9, r10, r11, r12, r13, r14, r15, r8, r9, r10, r11, r12, r13, r14, r15,
xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7, xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7,
@ -146,14 +147,14 @@ public class AMD64 extends Architecture {
xmm16, xmm17, xmm18, xmm19, xmm20, xmm21, xmm22, xmm23, xmm16, xmm17, xmm18, xmm19, xmm20, xmm21, xmm22, xmm23,
xmm24, xmm25, xmm26, xmm27, xmm28, xmm29, xmm30, xmm31, xmm24, xmm25, xmm26, xmm27, xmm28, xmm29, xmm30, xmm31,
k0, k1, k2, k3, k4, k5, k6, k7 k0, k1, k2, k3, k4, k5, k6, k7
}; );
/** /**
* Register used to construct an instruction-relative address. * Register used to construct an instruction-relative address.
*/ */
public static final Register rip = new Register(56, -1, "rip", SPECIAL); public static final Register rip = new Register(56, -1, "rip", SPECIAL);
public static final Register[] allRegisters = { public static final RegisterArray allRegisters = new RegisterArray(
rax, rcx, rdx, rbx, rsp, rbp, rsi, rdi, rax, rcx, rdx, rbx, rsp, rbp, rsi, rdi,
r8, r9, r10, r11, r12, r13, r14, r15, r8, r9, r10, r11, r12, r13, r14, r15,
xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7, xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7,
@ -162,7 +163,7 @@ public class AMD64 extends Architecture {
xmm24, xmm25, xmm26, xmm27, xmm28, xmm29, xmm30, xmm31, xmm24, xmm25, xmm26, xmm27, xmm28, xmm29, xmm30, xmm31,
k0, k1, k2, k3, k4, k5, k6, k7, k0, k1, k2, k3, k4, k5, k6, k7,
rip rip
}; );
// @formatter:on // @formatter:on
@ -245,7 +246,7 @@ public class AMD64 extends Architecture {
} }
@Override @Override
public Register[] getAvailableValueRegisters() { public RegisterArray getAvailableValueRegisters() {
if (features.contains(CPUFeature.AVX512F)) { if (features.contains(CPUFeature.AVX512F)) {
return valueRegistersAVX512; return valueRegistersAVX512;
} else { } else {

View File

@ -23,7 +23,6 @@
package jdk.vm.ci.code; package jdk.vm.ci.code;
import java.nio.ByteOrder; import java.nio.ByteOrder;
import java.util.Arrays;
import jdk.vm.ci.code.Register.RegisterCategory; import jdk.vm.ci.code.Register.RegisterCategory;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
@ -46,10 +45,10 @@ public abstract class Architecture {
private final String name; private final String name;
/** /**
* Array of all available registers on this architecture. The index of each register in this * List of all available registers on this architecture. The index of each register in this list
* array is equal to its {@linkplain Register#number number}. * is equal to its {@linkplain Register#number number}.
*/ */
private final Register[] registers; private final RegisterArray registers;
/** /**
* The byte ordering can be either little or big endian. * The byte ordering can be either little or big endian.
@ -78,7 +77,8 @@ public abstract class Architecture {
*/ */
private final int returnAddressSize; private final int returnAddressSize;
protected Architecture(String name, PlatformKind wordKind, ByteOrder byteOrder, boolean unalignedMemoryAccess, Register[] registers, int implicitMemoryBarriers, int nativeCallDisplacementOffset, protected Architecture(String name, PlatformKind wordKind, ByteOrder byteOrder, boolean unalignedMemoryAccess, RegisterArray registers, int implicitMemoryBarriers,
int nativeCallDisplacementOffset,
int returnAddressSize) { int returnAddressSize) {
this.name = name; this.name = name;
this.registers = registers; this.registers = registers;
@ -120,20 +120,20 @@ public abstract class Architecture {
} }
/** /**
* Gets an array of all registers that exist on this architecture. This contains all registers * Gets the list of all registers that exist on this architecture. This contains all registers
* that exist in the specification of this architecture. Not all of them may be available on * that exist in the specification of this architecture. Not all of them may be available on
* this particular architecture instance. The index of each register in this array is equal to * this particular architecture instance. The index of each register in this list is equal to
* its {@linkplain Register#number number}. * its {@linkplain Register#number number}.
*/ */
public Register[] getRegisters() { public RegisterArray getRegisters() {
return registers.clone(); return registers;
} }
/** /**
* Gets an array of all registers available for storing values on this architecture. This may be * Gets a list of all registers available for storing values on this architecture. This may be a
* a subset of {@link #getRegisters()}, depending on the capabilities of this particular CPU. * subset of {@link #getRegisters()}, depending on the capabilities of this particular CPU.
*/ */
public Register[] getAvailableValueRegisters() { public RegisterArray getAvailableValueRegisters() {
return getRegisters(); return getRegisters();
} }
@ -206,7 +206,7 @@ public abstract class Architecture {
assert this.byteOrder.equals(that.byteOrder); assert this.byteOrder.equals(that.byteOrder);
assert this.implicitMemoryBarriers == that.implicitMemoryBarriers; assert this.implicitMemoryBarriers == that.implicitMemoryBarriers;
assert this.machineCodeCallDisplacementOffset == that.machineCodeCallDisplacementOffset; assert this.machineCodeCallDisplacementOffset == that.machineCodeCallDisplacementOffset;
assert Arrays.equals(this.registers, that.registers); assert this.registers.equals(that.registers);
assert this.returnAddressSize == that.returnAddressSize; assert this.returnAddressSize == that.returnAddressSize;
assert this.unalignedMemoryAccess == that.unalignedMemoryAccess; assert this.unalignedMemoryAccess == that.unalignedMemoryAccess;
assert this.wordKind == that.wordKind; assert this.wordKind == that.wordKind;

View File

@ -34,7 +34,7 @@ import jdk.vm.ci.meta.Value;
* where to find the local variables, operand stack values and locked objects of the bytecode * where to find the local variables, operand stack values and locked objects of the bytecode
* frame(s). * frame(s).
*/ */
public class BytecodeFrame extends BytecodePosition { public final class BytecodeFrame extends BytecodePosition {
/** /**
* An array of values representing how to reconstruct the state of the Java frame. This is array * An array of values representing how to reconstruct the state of the Java frame. This is array
@ -65,14 +65,18 @@ public class BytecodeFrame extends BytecodePosition {
* <p> * <p>
* Note that the number of locals and the number of stack slots may be smaller than the maximum * Note that the number of locals and the number of stack slots may be smaller than the maximum
* number of locals and stack slots as specified in the compiled method. * number of locals and stack slots as specified in the compiled method.
*
* This field is intentionally exposed as a mutable array that a compiler may modify (e.g.
* during register allocation).
*/ */
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "field is intentionally mutable")//
public final JavaValue[] values; public final JavaValue[] values;
/** /**
* An array describing the Java kind of the {@link #values}. It records a kind for the locals * An array describing the Java kinds in {@link #values}. It records a kind for the locals and
* and the operand stack. * the operand stack.
*/ */
public final JavaKind[] slotKinds; private final JavaKind[] slotKinds;
/** /**
* The number of locals in the values array. * The number of locals in the values array.
@ -99,8 +103,8 @@ public class BytecodeFrame extends BytecodePosition {
public final boolean rethrowException; public final boolean rethrowException;
/** /**
* Specifies if this object represents a frame state in the middle of executing a call. If * Specifies if this object represents a frame state in the middle of executing a call. If true,
* true, the arguments to the call have been popped from the stack and the return value (for a * the arguments to the call have been popped from the stack and the return value (for a
* non-void call) has not yet been pushed. * non-void call) has not yet been pushed.
*/ */
public final boolean duringCall; public final boolean duringCall;
@ -178,11 +182,14 @@ public class BytecodeFrame extends BytecodePosition {
* @param bci a BCI within the method * @param bci a BCI within the method
* @param rethrowException specifies if the VM should re-throw the pending exception when * @param rethrowException specifies if the VM should re-throw the pending exception when
* deopt'ing using this frame * deopt'ing using this frame
* @param values the frame state {@link #values} * @param values the frame state {@link #values}.
* @param slotKinds the kinds in {@code values}. This array is now owned by this object and must
* not be mutated by the caller.
* @param numLocals the number of local variables * @param numLocals the number of local variables
* @param numStack the depth of the stack * @param numStack the depth of the stack
* @param numLocks the number of locked objects * @param numLocks the number of locked objects
*/ */
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `slotKinds`")
public BytecodeFrame(BytecodeFrame caller, ResolvedJavaMethod method, int bci, boolean rethrowException, boolean duringCall, JavaValue[] values, JavaKind[] slotKinds, int numLocals, int numStack, public BytecodeFrame(BytecodeFrame caller, ResolvedJavaMethod method, int bci, boolean rethrowException, boolean duringCall, JavaValue[] values, JavaKind[] slotKinds, int numLocals, int numStack,
int numLocks) { int numLocks) {
super(caller, method, bci); super(caller, method, bci);
@ -218,13 +225,45 @@ public class BytecodeFrame extends BytecodePosition {
return true; return true;
} }
/**
* Gets the kind of a local variable.
*
* @param i the local variable to query
* @return the kind of local variable {@code i}
* @throw {@link IndexOutOfBoundsException} if {@code i < 0 || i >= this.numLocals}
*/
public JavaKind getLocalValueKind(int i) {
if (i < 0 || i >= numLocals) {
throw new IndexOutOfBoundsException();
}
return slotKinds[i];
}
/**
* Gets the kind of a stack slot.
*
* @param i the local variable to query
* @return the kind of stack slot {@code i}
* @throw {@link IndexOutOfBoundsException} if {@code i < 0 || i >= this.numStack}
*/
public JavaKind getStackValueKind(int i) {
if (i < 0 || i >= numStack) {
throw new IndexOutOfBoundsException();
}
return slotKinds[i + numLocals];
}
/** /**
* Gets the value representing the specified local variable. * Gets the value representing the specified local variable.
* *
* @param i the local variable index * @param i the local variable index
* @return the value that can be used to reconstruct the local's current value * @return the value that can be used to reconstruct the local's current value
* @throw {@link IndexOutOfBoundsException} if {@code i < 0 || i >= this.numLocals}
*/ */
public JavaValue getLocalValue(int i) { public JavaValue getLocalValue(int i) {
if (i < 0 || i >= numLocals) {
throw new IndexOutOfBoundsException();
}
return values[i]; return values[i];
} }
@ -233,8 +272,12 @@ public class BytecodeFrame extends BytecodePosition {
* *
* @param i the stack index * @param i the stack index
* @return the value that can be used to reconstruct the stack slot's current value * @return the value that can be used to reconstruct the stack slot's current value
* @throw {@link IndexOutOfBoundsException} if {@code i < 0 || i >= this.numStack}
*/ */
public JavaValue getStackValue(int i) { public JavaValue getStackValue(int i) {
if (i < 0 || i >= numStack) {
throw new IndexOutOfBoundsException();
}
return values[i + numLocals]; return values[i + numLocals];
} }
@ -243,8 +286,12 @@ public class BytecodeFrame extends BytecodePosition {
* *
* @param i the lock index * @param i the lock index
* @return the value that can be used to reconstruct the lock's current value * @return the value that can be used to reconstruct the lock's current value
* @throw {@link IndexOutOfBoundsException} if {@code i < 0 || i >= this.numLocks}
*/ */
public JavaValue getLockValue(int i) { public JavaValue getLockValue(int i) {
if (i < 0 || i >= numLocks) {
throw new IndexOutOfBoundsException();
}
return values[i + numLocals + numStack]; return values[i + numLocals + numStack];
} }
@ -257,6 +304,11 @@ public class BytecodeFrame extends BytecodePosition {
return (BytecodeFrame) getCaller(); return (BytecodeFrame) getCaller();
} }
@Override
public int hashCode() {
return (numLocals + 1) ^ (numStack + 11) ^ (numLocks + 7);
}
@Override @Override
public boolean equals(Object obj) { public boolean equals(Object obj) {
if (this == obj) { if (this == obj) {

View File

@ -100,6 +100,7 @@ public class CallingConvention {
/** /**
* Gets the locations required for the arguments. * Gets the locations required for the arguments.
*/ */
@SuppressFBWarnings(value = "EI_EXPOSE_REP", justification = "FB false positive")
public AllocatableValue[] getArguments() { public AllocatableValue[] getArguments() {
if (argumentLocations.length == 0) { if (argumentLocations.length == 0) {
return argumentLocations; return argumentLocations;

View File

@ -48,8 +48,10 @@ public final class DebugInfo {
* *
* @param codePos the {@linkplain BytecodePosition code position} or {@linkplain BytecodeFrame * @param codePos the {@linkplain BytecodePosition code position} or {@linkplain BytecodeFrame
* frame} info * frame} info
* @param virtualObjectMapping the mapping of {@link VirtualObject}s to their real values * @param virtualObjectMapping the mapping of {@link VirtualObject}s to their real values. This
* array is now owned by this object and must not be mutated by the caller.
*/ */
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `virtualObjectMapping`")
public DebugInfo(BytecodePosition codePos, VirtualObject[] virtualObjectMapping) { public DebugInfo(BytecodePosition codePos, VirtualObject[] virtualObjectMapping) {
this.bytecodePosition = codePos; this.bytecodePosition = codePos;
this.virtualObjectMapping = virtualObjectMapping; this.virtualObjectMapping = virtualObjectMapping;

View File

@ -0,0 +1,105 @@
/*
* Copyright (c) 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 jdk.vm.ci.code;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
/**
* An immutable ordered list of registers. Only required because Java lacks immutable arrays.
*/
public final class RegisterArray implements Iterable<Register> {
private final Register[] registers;
private int hash;
public RegisterArray(Register... registers) {
this.registers = registers;
}
public RegisterArray(Collection<Register> registers) {
this.registers = registers.toArray(new Register[registers.size()]);
}
/**
* Gets the number of registers.
*/
public int size() {
return registers.length;
}
/**
* Gets the register at a given index.
*
* @param index the index of the register to retrieve
*/
public Register get(int index) {
return registers[index];
}
public void addTo(Collection<Register> collection) {
collection.addAll(Arrays.asList(registers));
}
/**
* Gets an immutable view of the registers as a list.
*/
public List<Register> asList() {
return Collections.unmodifiableList(Arrays.asList(registers));
}
/**
* Gets a copy of the registers as an array.
*/
public Register[] toArray() {
return registers.clone();
}
public Iterator<Register> iterator() {
return Arrays.asList(registers).iterator();
}
@Override
public int hashCode() {
if (hash == 0 && registers.length > 0) {
hash = Arrays.hashCode(registers);
}
return hash;
}
@Override
public boolean equals(Object obj) {
if (obj instanceof RegisterArray) {
return Arrays.equals(registers, ((RegisterArray) obj).registers);
}
return false;
}
@Override
public String toString() {
return Arrays.toString(registers);
}
}

View File

@ -23,6 +23,8 @@
package jdk.vm.ci.code; package jdk.vm.ci.code;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collections;
import java.util.List;
/** /**
* A collection of register attributes. The specific attribute values for a register may be local to * A collection of register attributes. The specific attribute values for a register may be local to
@ -53,13 +55,14 @@ public class RegisterAttributes {
* @return an array whose length is the max register number in {@code registers} plus 1. An * @return an array whose length is the max register number in {@code registers} plus 1. An
* element at index i holds the attributes of the register whose number is i. * element at index i holds the attributes of the register whose number is i.
*/ */
public static RegisterAttributes[] createMap(RegisterConfig registerConfig, Register[] registers) { public static RegisterAttributes[] createMap(RegisterConfig registerConfig, RegisterArray registers) {
RegisterAttributes[] map = new RegisterAttributes[registers.length]; RegisterAttributes[] map = new RegisterAttributes[registers.size()];
List<Register> callerSaveRegisters = registerConfig.getCallerSaveRegisters().asList();
List<Register> calleeSaveRegisters = registerConfig.getCalleeSaveRegisters() == null ? Collections.emptyList() : registerConfig.getCalleeSaveRegisters().asList();
List<Register> allocatableRegisters = registerConfig.getAllocatableRegisters().asList();
for (Register reg : registers) { for (Register reg : registers) {
if (reg != null) { if (reg != null) {
Register[] csr = registerConfig.getCalleeSaveRegisters(); RegisterAttributes attr = new RegisterAttributes(callerSaveRegisters.contains(reg), calleeSaveRegisters.contains(reg), allocatableRegisters.contains(reg));
RegisterAttributes attr = new RegisterAttributes(Arrays.asList(registerConfig.getCallerSaveRegisters()).contains(reg), csr == null ? false : Arrays.asList(csr).contains(reg),
Arrays.asList(registerConfig.getAllocatableRegisters()).contains(reg));
if (map.length <= reg.number) { if (map.length <= reg.number) {
map = Arrays.copyOf(map, reg.number + 1); map = Arrays.copyOf(map, reg.number + 1);
} }
@ -75,23 +78,24 @@ public class RegisterAttributes {
} }
/** /**
* @return Denotes a register that is available for use by a register allocator. * @return {@code true} if a register is available for use by a register allocator otherwise
* {@code false}
*/ */
public boolean isAllocatable() { public boolean isAllocatable() {
return allocatable; return allocatable;
} }
/** /**
* @return Denotes a register whose value preservation (if required) across a call is the * @return {@code true} if a register whose value preservation (if required) across a call is
* responsibility of the callee. * the responsibility of the callee otherwise {@code false}
*/ */
public boolean isCalleeSave() { public boolean isCalleeSave() {
return calleeSave; return calleeSave;
} }
/** /**
* @return Denotes a register whose value preservation (if required) across a call is the * @return {@code true} if a register whose value preservation (if required) across a call is
* responsibility of the caller. * the responsibility of the caller otherwise {@code false}
*/ */
public boolean isCallerSave() { public boolean isCallerSave() {
return callerSave; return callerSave;

View File

@ -71,7 +71,7 @@ public interface RegisterConfig {
* @return the ordered set of registers that may be used to pass parameters in a call conforming * @return the ordered set of registers that may be used to pass parameters in a call conforming
* to {@code type} * to {@code type}
*/ */
Register[] getCallingConventionRegisters(Type type, JavaKind kind); RegisterArray getCallingConventionRegisters(Type type, JavaKind kind);
/** /**
* Gets the set of all registers that might be used by the register allocator. * Gets the set of all registers that might be used by the register allocator.
@ -80,23 +80,23 @@ public interface RegisterConfig {
* {@link RegisterAllocationConfig#getAllocatableRegisters()} * {@link RegisterAllocationConfig#getAllocatableRegisters()}
*/ */
@SuppressWarnings("javadoc") @SuppressWarnings("javadoc")
Register[] getAllocatableRegisters(); RegisterArray getAllocatableRegisters();
/** /**
* Filters a set of registers and returns only those that can be used by the register allocator * Filters a set of registers and returns only those that can be used by the register allocator
* for a value of a particular kind. * for a value of a particular kind.
*/ */
Register[] filterAllocatableRegisters(PlatformKind kind, Register[] registers); RegisterArray filterAllocatableRegisters(PlatformKind kind, RegisterArray registers);
/** /**
* Gets the registers whose values must be preserved by a method across any call it makes. * Gets the registers whose values must be preserved by a method across any call it makes.
*/ */
Register[] getCallerSaveRegisters(); RegisterArray getCallerSaveRegisters();
/** /**
* Gets the registers whose values must be preserved by the callee. * Gets the registers whose values must be preserved by the callee.
*/ */
Register[] getCalleeSaveRegisters(); RegisterArray getCalleeSaveRegisters();
/** /**
* Gets a map from register {@linkplain Register#number numbers} to register * Gets a map from register {@linkplain Register#number numbers} to register

View File

@ -50,6 +50,7 @@ public final class RegisterSaveLayout {
* @param registers the keys in the map * @param registers the keys in the map
* @param slots frame slot index for each register in {@code registers} * @param slots frame slot index for each register in {@code registers}
*/ */
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `registers` and `slots`")
public RegisterSaveLayout(Register[] registers, int[] slots) { public RegisterSaveLayout(Register[] registers, int[] slots) {
assert registers.length == slots.length; assert registers.length == slots.length;
this.registers = registers; this.registers = registers;

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -20,25 +20,21 @@
* or visit www.oracle.com if you need additional information or have any * or visit www.oracle.com if you need additional information or have any
* questions. * questions.
*/ */
package jdk.vm.ci.hotspotvmconfig; package jdk.vm.ci.code;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/** /**
* Refers to a entry in {@code gHotSpotVMData}. * Used to suppress <a href="http://findbugs.sourceforge.net">FindBugs</a> warnings.
*/ */
@Target(ElementType.FIELD) @interface SuppressFBWarnings {
@Retention(RetentionPolicy.RUNTIME) /**
public @interface HotSpotVMData { * The set of FindBugs
* <a href="http://findbugs.sourceforge.net/bugDescriptions.html">warnings</a> that are to be
* suppressed in annotated element. The value can be a bug category, kind or pattern.
*/
String[] value();
/** /**
* Returns the array index of this field. * Reason why the warning is suppressed.
*
* @return array index of field
*/ */
int index(); String justification();
} }

View File

@ -119,17 +119,20 @@ public final class VirtualObject implements JavaValue {
} }
/** /**
* Returns an array containing all the values to be stored into the object when it is recreated. * Returns the array containing all the values to be stored into the object when it is
* recreated. This field is intentional exposed as a mutable array that a compiler may modify
* (e.g. during register allocation).
*/ */
@SuppressFBWarnings(value = "EI_EXPOSE_REP", justification = "`values` is intentional mutable")//
public JavaValue[] getValues() { public JavaValue[] getValues() {
return values; return values;
} }
/** /**
* Returns an array containing the Java kind of all values in the object. * Returns the kind of the value at {@code index}.
*/ */
public JavaKind[] getSlotKinds() { public JavaKind getSlotKind(int index) {
return slotKinds; return slotKinds[index];
} }
/** /**
@ -145,9 +148,13 @@ public final class VirtualObject implements JavaValue {
* *
* @param values an array containing all the values to be stored into the object when it is * @param values an array containing all the values to be stored into the object when it is
* recreated. * recreated.
* @param slotKinds an array containing the Java kinds of the values. * @param slotKinds an array containing the Java kinds of the values. This must have the same
* length as {@code values}. This array is now owned by this object and must not be
* mutated by the caller.
*/ */
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `slotKinds`")
public void setValues(JavaValue[] values, JavaKind[] slotKinds) { public void setValues(JavaValue[] values, JavaKind[] slotKinds) {
assert values.length == slotKinds.length;
this.values = values; this.values = values;
this.slotKinds = slotKinds; this.slotKinds = slotKinds;
} }

View File

@ -30,8 +30,8 @@ import java.util.concurrent.atomic.AtomicInteger;
* {@code "jvmci.inittimer"} system property to {@code "true"}. * {@code "jvmci.inittimer"} system property to {@code "true"}.
*/ */
public final class InitTimer implements AutoCloseable { public final class InitTimer implements AutoCloseable {
final String name; private final String name;
final long start; private final long start;
private InitTimer(String name) { private InitTimer(String name) {
int n = nesting.getAndIncrement(); int n = nesting.getAndIncrement();
@ -76,5 +76,5 @@ public final class InitTimer implements AutoCloseable {
/** /**
* Used to assert the invariant that all related initialization happens on the same thread. * Used to assert the invariant that all related initialization happens on the same thread.
*/ */
public static Thread initializingThread; static Thread initializingThread;
} }

View File

@ -38,24 +38,23 @@ import jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory;
import jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider; import jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider;
import jdk.vm.ci.hotspot.HotSpotMetaAccessProvider; import jdk.vm.ci.hotspot.HotSpotMetaAccessProvider;
import jdk.vm.ci.hotspot.HotSpotStackIntrospection; import jdk.vm.ci.hotspot.HotSpotStackIntrospection;
import jdk.vm.ci.hotspot.HotSpotVMConfig;
import jdk.vm.ci.meta.ConstantReflectionProvider; import jdk.vm.ci.meta.ConstantReflectionProvider;
import jdk.vm.ci.runtime.JVMCIBackend; import jdk.vm.ci.runtime.JVMCIBackend;
public class AArch64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFactory { public class AArch64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFactory {
protected EnumSet<AArch64.CPUFeature> computeFeatures(@SuppressWarnings("unused") HotSpotVMConfig config) { protected EnumSet<AArch64.CPUFeature> computeFeatures(@SuppressWarnings("unused") AArch64HotSpotVMConfig config) {
// Configure the feature set using the HotSpot flag settings. // Configure the feature set using the HotSpot flag settings.
EnumSet<AArch64.CPUFeature> features = EnumSet.noneOf(AArch64.CPUFeature.class); EnumSet<AArch64.CPUFeature> features = EnumSet.noneOf(AArch64.CPUFeature.class);
return features; return features;
} }
protected EnumSet<AArch64.Flag> computeFlags(@SuppressWarnings("unused") HotSpotVMConfig config) { protected EnumSet<AArch64.Flag> computeFlags(@SuppressWarnings("unused") AArch64HotSpotVMConfig config) {
EnumSet<AArch64.Flag> flags = EnumSet.noneOf(AArch64.Flag.class); EnumSet<AArch64.Flag> flags = EnumSet.noneOf(AArch64.Flag.class);
return flags; return flags;
} }
protected TargetDescription createTarget(HotSpotVMConfig config) { protected TargetDescription createTarget(AArch64HotSpotVMConfig config) {
final int stackFrameAlignment = 16; final int stackFrameAlignment = 16;
final int implicitNullCheckLimit = 4096; final int implicitNullCheckLimit = 4096;
final boolean inlineObjects = true; final boolean inlineObjects = true;
@ -67,8 +66,8 @@ public class AArch64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFac
return new HotSpotConstantReflectionProvider(runtime); return new HotSpotConstantReflectionProvider(runtime);
} }
protected RegisterConfig createRegisterConfig(HotSpotJVMCIRuntimeProvider runtime, TargetDescription target) { protected RegisterConfig createRegisterConfig(AArch64HotSpotVMConfig config, TargetDescription target) {
return new AArch64HotSpotRegisterConfig(target, runtime.getConfig()); return new AArch64HotSpotRegisterConfig(target, config.useCompressedOops);
} }
protected HotSpotCodeCacheProvider createCodeCache(HotSpotJVMCIRuntimeProvider runtime, TargetDescription target, RegisterConfig regConfig) { protected HotSpotCodeCacheProvider createCodeCache(HotSpotJVMCIRuntimeProvider runtime, TargetDescription target, RegisterConfig regConfig) {
@ -93,7 +92,8 @@ public class AArch64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFac
public JVMCIBackend createJVMCIBackend(HotSpotJVMCIRuntimeProvider runtime, JVMCIBackend host) { public JVMCIBackend createJVMCIBackend(HotSpotJVMCIRuntimeProvider runtime, JVMCIBackend host) {
assert host == null; assert host == null;
TargetDescription target = createTarget(runtime.getConfig()); AArch64HotSpotVMConfig config = new AArch64HotSpotVMConfig(runtime.getConfigStore());
TargetDescription target = createTarget(config);
RegisterConfig regConfig; RegisterConfig regConfig;
HotSpotCodeCacheProvider codeCache; HotSpotCodeCacheProvider codeCache;
@ -105,7 +105,7 @@ public class AArch64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFac
metaAccess = createMetaAccess(runtime); metaAccess = createMetaAccess(runtime);
} }
try (InitTimer rt = timer("create RegisterConfig")) { try (InitTimer rt = timer("create RegisterConfig")) {
regConfig = createRegisterConfig(runtime, target); regConfig = createRegisterConfig(config, target);
} }
try (InitTimer rt = timer("create CodeCache provider")) { try (InitTimer rt = timer("create CodeCache provider")) {
codeCache = createCodeCache(runtime, target, regConfig); codeCache = createCodeCache(runtime, target, regConfig);

View File

@ -49,8 +49,6 @@ import static jdk.vm.ci.aarch64.AArch64.v7;
import static jdk.vm.ci.aarch64.AArch64.zr; import static jdk.vm.ci.aarch64.AArch64.zr;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet; import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Set; import java.util.Set;
@ -60,6 +58,7 @@ import jdk.vm.ci.code.Architecture;
import jdk.vm.ci.code.CallingConvention; import jdk.vm.ci.code.CallingConvention;
import jdk.vm.ci.code.CallingConvention.Type; import jdk.vm.ci.code.CallingConvention.Type;
import jdk.vm.ci.code.Register; import jdk.vm.ci.code.Register;
import jdk.vm.ci.code.RegisterArray;
import jdk.vm.ci.code.RegisterAttributes; import jdk.vm.ci.code.RegisterAttributes;
import jdk.vm.ci.code.RegisterConfig; import jdk.vm.ci.code.RegisterConfig;
import jdk.vm.ci.code.StackSlot; import jdk.vm.ci.code.StackSlot;
@ -67,7 +66,6 @@ import jdk.vm.ci.code.TargetDescription;
import jdk.vm.ci.code.ValueKindFactory; import jdk.vm.ci.code.ValueKindFactory;
import jdk.vm.ci.common.JVMCIError; import jdk.vm.ci.common.JVMCIError;
import jdk.vm.ci.hotspot.HotSpotCallingConventionType; import jdk.vm.ci.hotspot.HotSpotCallingConventionType;
import jdk.vm.ci.hotspot.HotSpotVMConfig;
import jdk.vm.ci.meta.AllocatableValue; import jdk.vm.ci.meta.AllocatableValue;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
import jdk.vm.ci.meta.JavaType; import jdk.vm.ci.meta.JavaType;
@ -79,30 +77,24 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
private final TargetDescription target; private final TargetDescription target;
private final Register[] allocatable; private final RegisterArray allocatable;
private final int maxFrameSize;
/** /**
* The caller saved registers always include all parameter registers. * The caller saved registers always include all parameter registers.
*/ */
private final Register[] callerSaved; private final RegisterArray callerSaved;
private final boolean allAllocatableAreCallerSaved; private final boolean allAllocatableAreCallerSaved;
private final RegisterAttributes[] attributesMap; private final RegisterAttributes[] attributesMap;
public int getMaximumFrameSize() { @Override
return maxFrameSize; public RegisterArray getAllocatableRegisters() {
return allocatable;
} }
@Override @Override
public Register[] getAllocatableRegisters() { public RegisterArray filterAllocatableRegisters(PlatformKind kind, RegisterArray registers) {
return allocatable.clone();
}
@Override
public Register[] filterAllocatableRegisters(PlatformKind kind, Register[] registers) {
ArrayList<Register> list = new ArrayList<>(); ArrayList<Register> list = new ArrayList<>();
for (Register reg : registers) { for (Register reg : registers) {
if (target.arch.canStoreValue(reg.getRegisterCategory(), kind)) { if (target.arch.canStoreValue(reg.getRegisterCategory(), kind)) {
@ -110,8 +102,7 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
} }
} }
Register[] ret = list.toArray(new Register[list.size()]); return new RegisterArray(list);
return ret;
} }
@Override @Override
@ -119,9 +110,9 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
return attributesMap.clone(); return attributesMap.clone();
} }
private final Register[] javaGeneralParameterRegisters = {r1, r2, r3, r4, r5, r6, r7, r0}; private final RegisterArray javaGeneralParameterRegisters = new RegisterArray(r1, r2, r3, r4, r5, r6, r7, r0);
private final Register[] nativeGeneralParameterRegisters = {r0, r1, r2, r3, r4, r5, r6, r7}; private final RegisterArray nativeGeneralParameterRegisters = new RegisterArray(r0, r1, r2, r3, r4, r5, r6, r7);
private final Register[] simdParameterRegisters = {v0, v1, v2, v3, v4, v5, v6, v7}; private final RegisterArray simdParameterRegisters = new RegisterArray(v0, v1, v2, v3, v4, v5, v6, v7);
public static final Register inlineCacheRegister = r9; public static final Register inlineCacheRegister = r9;
@ -134,12 +125,12 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
public static final Register threadRegister = r28; public static final Register threadRegister = r28;
public static final Register fp = r29; public static final Register fp = r29;
private static final Register[] reservedRegisters = {threadRegister, fp, lr, r31, zr, sp}; private static final RegisterArray reservedRegisters = new RegisterArray(threadRegister, fp, lr, r31, zr, sp);
private static Register[] initAllocatable(Architecture arch, boolean reserveForHeapBase) { private static RegisterArray initAllocatable(Architecture arch, boolean reserveForHeapBase) {
Register[] allRegisters = arch.getAvailableValueRegisters(); RegisterArray allRegisters = arch.getAvailableValueRegisters();
Register[] registers = new Register[allRegisters.length - reservedRegisters.length - (reserveForHeapBase ? 1 : 0)]; Register[] registers = new Register[allRegisters.size() - reservedRegisters.size() - (reserveForHeapBase ? 1 : 0)];
List<Register> reservedRegistersList = Arrays.asList(reservedRegisters); List<Register> reservedRegistersList = reservedRegisters.asList();
int idx = 0; int idx = 0;
for (Register reg : allRegisters) { for (Register reg : allRegisters) {
@ -157,36 +148,35 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
} }
assert idx == registers.length; assert idx == registers.length;
return registers; return new RegisterArray(registers);
} }
public AArch64HotSpotRegisterConfig(TargetDescription target, HotSpotVMConfig config) { public AArch64HotSpotRegisterConfig(TargetDescription target, boolean useCompressedOops) {
this(target, config, initAllocatable(target.arch, config.useCompressedOops)); this(target, initAllocatable(target.arch, useCompressedOops));
assert callerSaved.length >= allocatable.length; assert callerSaved.size() >= allocatable.size();
} }
public AArch64HotSpotRegisterConfig(TargetDescription target, HotSpotVMConfig config, Register[] allocatable) { public AArch64HotSpotRegisterConfig(TargetDescription target, RegisterArray allocatable) {
this.target = target; this.target = target;
this.maxFrameSize = config.maxFrameSize;
this.allocatable = allocatable.clone(); this.allocatable = allocatable;
Set<Register> callerSaveSet = new HashSet<>(); Set<Register> callerSaveSet = new HashSet<>();
Collections.addAll(callerSaveSet, allocatable); allocatable.addTo(callerSaveSet);
Collections.addAll(callerSaveSet, simdParameterRegisters); simdParameterRegisters.addTo(callerSaveSet);
Collections.addAll(callerSaveSet, javaGeneralParameterRegisters); javaGeneralParameterRegisters.addTo(callerSaveSet);
Collections.addAll(callerSaveSet, nativeGeneralParameterRegisters); nativeGeneralParameterRegisters.addTo(callerSaveSet);
callerSaved = callerSaveSet.toArray(new Register[callerSaveSet.size()]); callerSaved = new RegisterArray(callerSaveSet);
allAllocatableAreCallerSaved = true; allAllocatableAreCallerSaved = true;
attributesMap = RegisterAttributes.createMap(this, AArch64.allRegisters); attributesMap = RegisterAttributes.createMap(this, AArch64.allRegisters);
} }
@Override @Override
public Register[] getCallerSaveRegisters() { public RegisterArray getCallerSaveRegisters() {
return callerSaved; return callerSaved;
} }
public Register[] getCalleeSaveRegisters() { public RegisterArray getCalleeSaveRegisters() {
return null; return null;
} }
@ -207,7 +197,7 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
} }
@Override @Override
public Register[] getCallingConventionRegisters(Type type, JavaKind kind) { public RegisterArray getCallingConventionRegisters(Type type, JavaKind kind) {
HotSpotCallingConventionType hotspotType = (HotSpotCallingConventionType) type; HotSpotCallingConventionType hotspotType = (HotSpotCallingConventionType) type;
switch (kind) { switch (kind) {
case Boolean: case Boolean:
@ -226,7 +216,7 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
} }
} }
private CallingConvention callingConvention(Register[] generalParameterRegisters, JavaType returnType, JavaType[] parameterTypes, HotSpotCallingConventionType type, private CallingConvention callingConvention(RegisterArray generalParameterRegisters, JavaType returnType, JavaType[] parameterTypes, HotSpotCallingConventionType type,
ValueKindFactory<?> valueKindFactory) { ValueKindFactory<?> valueKindFactory) {
AllocatableValue[] locations = new AllocatableValue[parameterTypes.length]; AllocatableValue[] locations = new AllocatableValue[parameterTypes.length];
@ -245,15 +235,15 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
case Int: case Int:
case Long: case Long:
case Object: case Object:
if (currentGeneral < generalParameterRegisters.length) { if (currentGeneral < generalParameterRegisters.size()) {
Register register = generalParameterRegisters[currentGeneral++]; Register register = generalParameterRegisters.get(currentGeneral++);
locations[i] = register.asValue(valueKindFactory.getValueKind(kind)); locations[i] = register.asValue(valueKindFactory.getValueKind(kind));
} }
break; break;
case Float: case Float:
case Double: case Double:
if (currentSIMD < simdParameterRegisters.length) { if (currentSIMD < simdParameterRegisters.size()) {
Register register = simdParameterRegisters[currentSIMD++]; Register register = simdParameterRegisters.get(currentSIMD++);
locations[i] = register.asValue(valueKindFactory.getValueKind(kind)); locations[i] = register.asValue(valueKindFactory.getValueKind(kind));
} }
break; break;
@ -302,6 +292,6 @@ public class AArch64HotSpotRegisterConfig implements RegisterConfig {
@Override @Override
public String toString() { public String toString() {
return String.format("Allocatable: " + Arrays.toString(getAllocatableRegisters()) + "%n" + "CallerSave: " + Arrays.toString(getCallerSaveRegisters()) + "%n"); return String.format("Allocatable: " + getAllocatableRegisters() + "%n" + "CallerSave: " + getCallerSaveRegisters() + "%n");
} }
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2013, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -20,34 +20,23 @@
* or visit www.oracle.com if you need additional information or have any * or visit www.oracle.com if you need additional information or have any
* questions. * questions.
*/ */
package jdk.vm.ci.hotspotvmconfig; package jdk.vm.ci.hotspot.aarch64;
import java.lang.annotation.ElementType; import jdk.vm.ci.hotspot.HotSpotVMConfigAccess;
import java.lang.annotation.Retention; import jdk.vm.ci.hotspot.HotSpotVMConfigStore;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/** /**
* Refers to a C++ flag in the VM. * Used to access native configuration details.
*
* All non-static, public fields in this class are so that they can be compiled as constants.
*/ */
@Target(ElementType.FIELD) class AArch64HotSpotVMConfig extends HotSpotVMConfigAccess {
@Retention(RetentionPolicy.RUNTIME)
public @interface HotSpotVMFlag {
/** AArch64HotSpotVMConfig(HotSpotVMConfigStore config) {
* Returns the name of this flag. super(config);
* }
* @return name of flag.
*/
String name();
/** final boolean linuxOs = System.getProperty("os.name", "").startsWith("Linux");
* List of architectures where this constant is required. Names are derived from
* {@link HotSpotVMConfig#getHostArchitectureName()}. An empty list implies that the constant is
* required on all architectures.
*/
@SuppressWarnings("javadoc")
String[] archs() default {};
boolean optional() default false; final boolean useCompressedOops = getFlag("UseCompressedOops", Boolean.class);
} }

View File

@ -38,13 +38,12 @@ import jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory;
import jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider; import jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider;
import jdk.vm.ci.hotspot.HotSpotMetaAccessProvider; import jdk.vm.ci.hotspot.HotSpotMetaAccessProvider;
import jdk.vm.ci.hotspot.HotSpotStackIntrospection; import jdk.vm.ci.hotspot.HotSpotStackIntrospection;
import jdk.vm.ci.hotspot.HotSpotVMConfig;
import jdk.vm.ci.meta.ConstantReflectionProvider; import jdk.vm.ci.meta.ConstantReflectionProvider;
import jdk.vm.ci.runtime.JVMCIBackend; import jdk.vm.ci.runtime.JVMCIBackend;
public class AMD64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFactory { public class AMD64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFactory {
protected EnumSet<AMD64.CPUFeature> computeFeatures(HotSpotVMConfig config) { protected EnumSet<AMD64.CPUFeature> computeFeatures(AMD64HotSpotVMConfig config) {
// Configure the feature set using the HotSpot flag settings. // Configure the feature set using the HotSpot flag settings.
EnumSet<AMD64.CPUFeature> features = EnumSet.noneOf(AMD64.CPUFeature.class); EnumSet<AMD64.CPUFeature> features = EnumSet.noneOf(AMD64.CPUFeature.class);
if ((config.vmVersionFeatures & config.amd643DNOWPREFETCH) != 0) { if ((config.vmVersionFeatures & config.amd643DNOWPREFETCH) != 0) {
@ -128,7 +127,7 @@ public class AMD64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFacto
return features; return features;
} }
protected EnumSet<AMD64.Flag> computeFlags(HotSpotVMConfig config) { protected EnumSet<AMD64.Flag> computeFlags(AMD64HotSpotVMConfig config) {
EnumSet<AMD64.Flag> flags = EnumSet.noneOf(AMD64.Flag.class); EnumSet<AMD64.Flag> flags = EnumSet.noneOf(AMD64.Flag.class);
if (config.useCountLeadingZerosInstruction) { if (config.useCountLeadingZerosInstruction) {
flags.add(AMD64.Flag.UseCountLeadingZerosInstruction); flags.add(AMD64.Flag.UseCountLeadingZerosInstruction);
@ -139,7 +138,7 @@ public class AMD64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFacto
return flags; return flags;
} }
protected TargetDescription createTarget(HotSpotVMConfig config) { protected TargetDescription createTarget(AMD64HotSpotVMConfig config) {
final int stackFrameAlignment = 16; final int stackFrameAlignment = 16;
final int implicitNullCheckLimit = 4096; final int implicitNullCheckLimit = 4096;
final boolean inlineObjects = true; final boolean inlineObjects = true;
@ -151,8 +150,8 @@ public class AMD64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFacto
return new HotSpotConstantReflectionProvider(runtime); return new HotSpotConstantReflectionProvider(runtime);
} }
protected RegisterConfig createRegisterConfig(HotSpotJVMCIRuntimeProvider runtime, TargetDescription target) { protected RegisterConfig createRegisterConfig(AMD64HotSpotVMConfig config, TargetDescription target) {
return new AMD64HotSpotRegisterConfig(target, runtime.getConfig()); return new AMD64HotSpotRegisterConfig(target, config.useCompressedOops, config.windowsOs);
} }
protected HotSpotCodeCacheProvider createCodeCache(HotSpotJVMCIRuntimeProvider runtime, TargetDescription target, RegisterConfig regConfig) { protected HotSpotCodeCacheProvider createCodeCache(HotSpotJVMCIRuntimeProvider runtime, TargetDescription target, RegisterConfig regConfig) {
@ -175,9 +174,9 @@ public class AMD64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFacto
@SuppressWarnings("try") @SuppressWarnings("try")
public JVMCIBackend createJVMCIBackend(HotSpotJVMCIRuntimeProvider runtime, JVMCIBackend host) { public JVMCIBackend createJVMCIBackend(HotSpotJVMCIRuntimeProvider runtime, JVMCIBackend host) {
assert host == null; assert host == null;
TargetDescription target = createTarget(runtime.getConfig()); AMD64HotSpotVMConfig config = new AMD64HotSpotVMConfig(runtime.getConfigStore());
TargetDescription target = createTarget(config);
RegisterConfig regConfig; RegisterConfig regConfig;
HotSpotCodeCacheProvider codeCache; HotSpotCodeCacheProvider codeCache;
@ -189,7 +188,7 @@ public class AMD64HotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFacto
metaAccess = createMetaAccess(runtime); metaAccess = createMetaAccess(runtime);
} }
try (InitTimer rt = timer("create RegisterConfig")) { try (InitTimer rt = timer("create RegisterConfig")) {
regConfig = createRegisterConfig(runtime, target); regConfig = createRegisterConfig(config, target);
} }
try (InitTimer rt = timer("create CodeCache provider")) { try (InitTimer rt = timer("create CodeCache provider")) {
codeCache = createCodeCache(runtime, target, regConfig); codeCache = createCodeCache(runtime, target, regConfig);

View File

@ -42,8 +42,6 @@ import static jdk.vm.ci.amd64.AMD64.xmm6;
import static jdk.vm.ci.amd64.AMD64.xmm7; import static jdk.vm.ci.amd64.AMD64.xmm7;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet; import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Set; import java.util.Set;
@ -52,6 +50,7 @@ import jdk.vm.ci.code.Architecture;
import jdk.vm.ci.code.CallingConvention; import jdk.vm.ci.code.CallingConvention;
import jdk.vm.ci.code.CallingConvention.Type; import jdk.vm.ci.code.CallingConvention.Type;
import jdk.vm.ci.code.Register; import jdk.vm.ci.code.Register;
import jdk.vm.ci.code.RegisterArray;
import jdk.vm.ci.code.RegisterAttributes; import jdk.vm.ci.code.RegisterAttributes;
import jdk.vm.ci.code.RegisterConfig; import jdk.vm.ci.code.RegisterConfig;
import jdk.vm.ci.code.StackSlot; import jdk.vm.ci.code.StackSlot;
@ -59,7 +58,6 @@ import jdk.vm.ci.code.TargetDescription;
import jdk.vm.ci.code.ValueKindFactory; import jdk.vm.ci.code.ValueKindFactory;
import jdk.vm.ci.common.JVMCIError; import jdk.vm.ci.common.JVMCIError;
import jdk.vm.ci.hotspot.HotSpotCallingConventionType; import jdk.vm.ci.hotspot.HotSpotCallingConventionType;
import jdk.vm.ci.hotspot.HotSpotVMConfig;
import jdk.vm.ci.meta.AllocatableValue; import jdk.vm.ci.meta.AllocatableValue;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
import jdk.vm.ci.meta.JavaType; import jdk.vm.ci.meta.JavaType;
@ -71,30 +69,24 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
private final TargetDescription target; private final TargetDescription target;
private final Register[] allocatable; private final RegisterArray allocatable;
private final int maxFrameSize;
/** /**
* The caller saved registers always include all parameter registers. * The caller saved registers always include all parameter registers.
*/ */
private final Register[] callerSaved; private final RegisterArray callerSaved;
private final boolean allAllocatableAreCallerSaved; private final boolean allAllocatableAreCallerSaved;
private final RegisterAttributes[] attributesMap; private final RegisterAttributes[] attributesMap;
public int getMaximumFrameSize() { @Override
return maxFrameSize; public RegisterArray getAllocatableRegisters() {
return allocatable;
} }
@Override @Override
public Register[] getAllocatableRegisters() { public RegisterArray filterAllocatableRegisters(PlatformKind kind, RegisterArray registers) {
return allocatable.clone();
}
@Override
public Register[] filterAllocatableRegisters(PlatformKind kind, Register[] registers) {
ArrayList<Register> list = new ArrayList<>(); ArrayList<Register> list = new ArrayList<>();
for (Register reg : registers) { for (Register reg : registers) {
if (target.arch.canStoreValue(reg.getRegisterCategory(), kind)) { if (target.arch.canStoreValue(reg.getRegisterCategory(), kind)) {
@ -102,7 +94,7 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
} }
} }
Register[] ret = list.toArray(new Register[list.size()]); RegisterArray ret = new RegisterArray(list);
return ret; return ret;
} }
@ -111,9 +103,9 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
return attributesMap.clone(); return attributesMap.clone();
} }
private final Register[] javaGeneralParameterRegisters; private final RegisterArray javaGeneralParameterRegisters;
private final Register[] nativeGeneralParameterRegisters; private final RegisterArray nativeGeneralParameterRegisters;
private final Register[] xmmParameterRegisters = {xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7}; private final RegisterArray xmmParameterRegisters = new RegisterArray(xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7);
/* /*
* Some ABIs (e.g. Windows) require a so-called "home space", that is a save area on the stack * Some ABIs (e.g. Windows) require a so-called "home space", that is a save area on the stack
@ -121,12 +113,12 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
*/ */
private final boolean needsNativeStackHomeSpace; private final boolean needsNativeStackHomeSpace;
private static final Register[] reservedRegisters = {rsp, r15}; private static final RegisterArray reservedRegisters = new RegisterArray(rsp, r15);
private static Register[] initAllocatable(Architecture arch, boolean reserveForHeapBase) { private static RegisterArray initAllocatable(Architecture arch, boolean reserveForHeapBase) {
Register[] allRegisters = arch.getAvailableValueRegisters(); RegisterArray allRegisters = arch.getAvailableValueRegisters();
Register[] registers = new Register[allRegisters.length - reservedRegisters.length - (reserveForHeapBase ? 1 : 0)]; Register[] registers = new Register[allRegisters.size() - reservedRegisters.size() - (reserveForHeapBase ? 1 : 0)];
List<Register> reservedRegistersList = Arrays.asList(reservedRegisters); List<Register> reservedRegistersList = reservedRegisters.asList();
int idx = 0; int idx = 0;
for (Register reg : allRegisters) { for (Register reg : allRegisters) {
@ -143,47 +135,46 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
} }
assert idx == registers.length; assert idx == registers.length;
return registers; return new RegisterArray(registers);
} }
public AMD64HotSpotRegisterConfig(TargetDescription target, HotSpotVMConfig config) { public AMD64HotSpotRegisterConfig(TargetDescription target, boolean useCompressedOops, boolean windowsOs) {
this(target, config, initAllocatable(target.arch, config.useCompressedOops)); this(target, initAllocatable(target.arch, useCompressedOops), windowsOs);
assert callerSaved.length >= allocatable.length; assert callerSaved.size() >= allocatable.size();
} }
public AMD64HotSpotRegisterConfig(TargetDescription target, HotSpotVMConfig config, Register[] allocatable) { public AMD64HotSpotRegisterConfig(TargetDescription target, RegisterArray allocatable, boolean windowsOs) {
this.target = target; this.target = target;
this.maxFrameSize = config.maxFrameSize;
if (config.windowsOs) { if (windowsOs) {
javaGeneralParameterRegisters = new Register[]{rdx, r8, r9, rdi, rsi, rcx}; javaGeneralParameterRegisters = new RegisterArray(rdx, r8, r9, rdi, rsi, rcx);
nativeGeneralParameterRegisters = new Register[]{rcx, rdx, r8, r9}; nativeGeneralParameterRegisters = new RegisterArray(rcx, rdx, r8, r9);
this.needsNativeStackHomeSpace = true; this.needsNativeStackHomeSpace = true;
} else { } else {
javaGeneralParameterRegisters = new Register[]{rsi, rdx, rcx, r8, r9, rdi}; javaGeneralParameterRegisters = new RegisterArray(rsi, rdx, rcx, r8, r9, rdi);
nativeGeneralParameterRegisters = new Register[]{rdi, rsi, rdx, rcx, r8, r9}; nativeGeneralParameterRegisters = new RegisterArray(rdi, rsi, rdx, rcx, r8, r9);
this.needsNativeStackHomeSpace = false; this.needsNativeStackHomeSpace = false;
} }
this.allocatable = allocatable; this.allocatable = allocatable;
Set<Register> callerSaveSet = new HashSet<>(); Set<Register> callerSaveSet = new HashSet<>();
Collections.addAll(callerSaveSet, allocatable); allocatable.addTo(callerSaveSet);
Collections.addAll(callerSaveSet, xmmParameterRegisters); xmmParameterRegisters.addTo(callerSaveSet);
Collections.addAll(callerSaveSet, javaGeneralParameterRegisters); callerSaveSet.addAll(javaGeneralParameterRegisters.asList());
Collections.addAll(callerSaveSet, nativeGeneralParameterRegisters); nativeGeneralParameterRegisters.addTo(callerSaveSet);
callerSaved = callerSaveSet.toArray(new Register[callerSaveSet.size()]); callerSaved = new RegisterArray(callerSaveSet);
allAllocatableAreCallerSaved = true; allAllocatableAreCallerSaved = true;
attributesMap = RegisterAttributes.createMap(this, target.arch.getRegisters()); attributesMap = RegisterAttributes.createMap(this, target.arch.getRegisters());
} }
@Override @Override
public Register[] getCallerSaveRegisters() { public RegisterArray getCallerSaveRegisters() {
return callerSaved; return callerSaved;
} }
@Override @Override
public Register[] getCalleeSaveRegisters() { public RegisterArray getCalleeSaveRegisters() {
return null; return null;
} }
@ -204,7 +195,7 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
} }
@Override @Override
public Register[] getCallingConventionRegisters(Type type, JavaKind kind) { public RegisterArray getCallingConventionRegisters(Type type, JavaKind kind) {
HotSpotCallingConventionType hotspotType = (HotSpotCallingConventionType) type; HotSpotCallingConventionType hotspotType = (HotSpotCallingConventionType) type;
switch (kind) { switch (kind) {
case Boolean: case Boolean:
@ -223,13 +214,13 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
} }
} }
private CallingConvention callingConvention(Register[] generalParameterRegisters, JavaType returnType, JavaType[] parameterTypes, HotSpotCallingConventionType type, private CallingConvention callingConvention(RegisterArray generalParameterRegisters, JavaType returnType, JavaType[] parameterTypes, HotSpotCallingConventionType type,
ValueKindFactory<?> valueKindFactory) { ValueKindFactory<?> valueKindFactory) {
AllocatableValue[] locations = new AllocatableValue[parameterTypes.length]; AllocatableValue[] locations = new AllocatableValue[parameterTypes.length];
int currentGeneral = 0; int currentGeneral = 0;
int currentXMM = 0; int currentXMM = 0;
int currentStackOffset = type == HotSpotCallingConventionType.NativeCall && needsNativeStackHomeSpace ? generalParameterRegisters.length * target.wordSize : 0; int currentStackOffset = type == HotSpotCallingConventionType.NativeCall && needsNativeStackHomeSpace ? generalParameterRegisters.size() * target.wordSize : 0;
for (int i = 0; i < parameterTypes.length; i++) { for (int i = 0; i < parameterTypes.length; i++) {
final JavaKind kind = parameterTypes[i].getJavaKind().getStackKind(); final JavaKind kind = parameterTypes[i].getJavaKind().getStackKind();
@ -242,15 +233,15 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
case Int: case Int:
case Long: case Long:
case Object: case Object:
if (currentGeneral < generalParameterRegisters.length) { if (currentGeneral < generalParameterRegisters.size()) {
Register register = generalParameterRegisters[currentGeneral++]; Register register = generalParameterRegisters.get(currentGeneral++);
locations[i] = register.asValue(valueKindFactory.getValueKind(kind)); locations[i] = register.asValue(valueKindFactory.getValueKind(kind));
} }
break; break;
case Float: case Float:
case Double: case Double:
if (currentXMM < xmmParameterRegisters.length) { if (currentXMM < xmmParameterRegisters.size()) {
Register register = xmmParameterRegisters[currentXMM++]; Register register = xmmParameterRegisters.get(currentXMM++);
locations[i] = register.asValue(valueKindFactory.getValueKind(kind)); locations[i] = register.asValue(valueKindFactory.getValueKind(kind));
} }
break; break;
@ -299,6 +290,6 @@ public class AMD64HotSpotRegisterConfig implements RegisterConfig {
@Override @Override
public String toString() { public String toString() {
return String.format("Allocatable: " + Arrays.toString(getAllocatableRegisters()) + "%n" + "CallerSave: " + Arrays.toString(getCallerSaveRegisters()) + "%n"); return String.format("Allocatable: " + getAllocatableRegisters() + "%n" + "CallerSave: " + getCallerSaveRegisters() + "%n");
} }
} }

View File

@ -0,0 +1,81 @@
/*
* Copyright (c) 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 jdk.vm.ci.hotspot.amd64;
import jdk.vm.ci.hotspot.HotSpotVMConfigAccess;
import jdk.vm.ci.hotspot.HotSpotVMConfigStore;
class AMD64HotSpotVMConfig extends HotSpotVMConfigAccess {
AMD64HotSpotVMConfig(HotSpotVMConfigStore config) {
super(config);
}
final boolean windowsOs = System.getProperty("os.name", "").startsWith("Windows");
final boolean useCountLeadingZerosInstruction = getFlag("UseCountLeadingZerosInstruction", Boolean.class);
final boolean useCountTrailingZerosInstruction = getFlag("UseCountTrailingZerosInstruction", Boolean.class);
final boolean useCompressedOops = getFlag("UseCompressedOops", Boolean.class);
// CPU capabilities
final int useSSE = getFlag("UseSSE", Integer.class);
final int useAVX = getFlag("UseAVX", Integer.class);
final long vmVersionFeatures = getFieldValue("Abstract_VM_Version::_features", Long.class, "uint64_t");
// CPU feature flags
final long amd64CX8 = getConstant("VM_Version::CPU_CX8", Long.class);
final long amd64CMOV = getConstant("VM_Version::CPU_CMOV", Long.class);
final long amd64FXSR = getConstant("VM_Version::CPU_FXSR", Long.class);
final long amd64HT = getConstant("VM_Version::CPU_HT", Long.class);
final long amd64MMX = getConstant("VM_Version::CPU_MMX", Long.class);
final long amd643DNOWPREFETCH = getConstant("VM_Version::CPU_3DNOW_PREFETCH", Long.class);
final long amd64SSE = getConstant("VM_Version::CPU_SSE", Long.class);
final long amd64SSE2 = getConstant("VM_Version::CPU_SSE2", Long.class);
final long amd64SSE3 = getConstant("VM_Version::CPU_SSE3", Long.class);
final long amd64SSSE3 = getConstant("VM_Version::CPU_SSSE3", Long.class);
final long amd64SSE4A = getConstant("VM_Version::CPU_SSE4A", Long.class);
final long amd64SSE41 = getConstant("VM_Version::CPU_SSE4_1", Long.class);
final long amd64SSE42 = getConstant("VM_Version::CPU_SSE4_2", Long.class);
final long amd64POPCNT = getConstant("VM_Version::CPU_POPCNT", Long.class);
final long amd64LZCNT = getConstant("VM_Version::CPU_LZCNT", Long.class);
final long amd64TSC = getConstant("VM_Version::CPU_TSC", Long.class);
final long amd64TSCINV = getConstant("VM_Version::CPU_TSCINV", Long.class);
final long amd64AVX = getConstant("VM_Version::CPU_AVX", Long.class);
final long amd64AVX2 = getConstant("VM_Version::CPU_AVX2", Long.class);
final long amd64AES = getConstant("VM_Version::CPU_AES", Long.class);
final long amd64ERMS = getConstant("VM_Version::CPU_ERMS", Long.class);
final long amd64CLMUL = getConstant("VM_Version::CPU_CLMUL", Long.class);
final long amd64BMI1 = getConstant("VM_Version::CPU_BMI1", Long.class);
final long amd64BMI2 = getConstant("VM_Version::CPU_BMI2", Long.class);
final long amd64RTM = getConstant("VM_Version::CPU_RTM", Long.class);
final long amd64ADX = getConstant("VM_Version::CPU_ADX", Long.class);
final long amd64AVX512F = getConstant("VM_Version::CPU_AVX512F", Long.class);
final long amd64AVX512DQ = getConstant("VM_Version::CPU_AVX512DQ", Long.class);
final long amd64AVX512PF = getConstant("VM_Version::CPU_AVX512PF", Long.class);
final long amd64AVX512ER = getConstant("VM_Version::CPU_AVX512ER", Long.class);
final long amd64AVX512CD = getConstant("VM_Version::CPU_AVX512CD", Long.class);
final long amd64AVX512BW = getConstant("VM_Version::CPU_AVX512BW", Long.class);
final long amd64AVX512VL = getConstant("VM_Version::CPU_AVX512VL", Long.class);
final long amd64SHA = getConstant("VM_Version::CPU_SHA", Long.class);
}

View File

@ -37,14 +37,13 @@ import jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory;
import jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider; import jdk.vm.ci.hotspot.HotSpotJVMCIRuntimeProvider;
import jdk.vm.ci.hotspot.HotSpotMetaAccessProvider; import jdk.vm.ci.hotspot.HotSpotMetaAccessProvider;
import jdk.vm.ci.hotspot.HotSpotStackIntrospection; import jdk.vm.ci.hotspot.HotSpotStackIntrospection;
import jdk.vm.ci.hotspot.HotSpotVMConfig;
import jdk.vm.ci.runtime.JVMCIBackend; import jdk.vm.ci.runtime.JVMCIBackend;
import jdk.vm.ci.sparc.SPARC; import jdk.vm.ci.sparc.SPARC;
import jdk.vm.ci.sparc.SPARC.CPUFeature; import jdk.vm.ci.sparc.SPARC.CPUFeature;
public class SPARCHotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFactory { public class SPARCHotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFactory {
protected TargetDescription createTarget(HotSpotVMConfig config) { protected TargetDescription createTarget(SPARCHotSpotVMConfig config) {
final int stackFrameAlignment = 16; final int stackFrameAlignment = 16;
final int implicitNullCheckLimit = 4096; final int implicitNullCheckLimit = 4096;
final boolean inlineObjects = false; final boolean inlineObjects = false;
@ -56,7 +55,7 @@ public class SPARCHotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFacto
return new HotSpotCodeCacheProvider(runtime, runtime.getConfig(), target, regConfig); return new HotSpotCodeCacheProvider(runtime, runtime.getConfig(), target, regConfig);
} }
protected EnumSet<CPUFeature> computeFeatures(HotSpotVMConfig config) { protected EnumSet<CPUFeature> computeFeatures(SPARCHotSpotVMConfig config) {
EnumSet<CPUFeature> features = EnumSet.noneOf(CPUFeature.class); EnumSet<CPUFeature> features = EnumSet.noneOf(CPUFeature.class);
if ((config.vmVersionFeatures & config.sparcVis1Instructions) != 0) { if ((config.vmVersionFeatures & config.sparcVis1Instructions) != 0) {
features.add(CPUFeature.VIS1); features.add(CPUFeature.VIS1);
@ -143,10 +142,11 @@ public class SPARCHotSpotJVMCIBackendFactory implements HotSpotJVMCIBackendFacto
@SuppressWarnings("try") @SuppressWarnings("try")
public JVMCIBackend createJVMCIBackend(HotSpotJVMCIRuntimeProvider runtime, JVMCIBackend host) { public JVMCIBackend createJVMCIBackend(HotSpotJVMCIRuntimeProvider runtime, JVMCIBackend host) {
assert host == null; assert host == null;
TargetDescription target = createTarget(runtime.getConfig()); SPARCHotSpotVMConfig config = new SPARCHotSpotVMConfig(runtime.getConfigStore());
TargetDescription target = createTarget(config);
HotSpotMetaAccessProvider metaAccess = new HotSpotMetaAccessProvider(runtime); HotSpotMetaAccessProvider metaAccess = new HotSpotMetaAccessProvider(runtime);
RegisterConfig regConfig = new SPARCHotSpotRegisterConfig(target, runtime.getConfig()); RegisterConfig regConfig = new SPARCHotSpotRegisterConfig(target, config.useCompressedOops);
HotSpotCodeCacheProvider codeCache = createCodeCache(runtime, target, regConfig); HotSpotCodeCacheProvider codeCache = createCodeCache(runtime, target, regConfig);
HotSpotConstantReflectionProvider constantReflection = new HotSpotConstantReflectionProvider(runtime); HotSpotConstantReflectionProvider constantReflection = new HotSpotConstantReflectionProvider(runtime);
StackIntrospection stackIntrospection = new HotSpotStackIntrospection(runtime); StackIntrospection stackIntrospection = new HotSpotStackIntrospection(runtime);

View File

@ -65,8 +65,6 @@ import static jdk.vm.ci.sparc.SPARC.o5;
import static jdk.vm.ci.sparc.SPARC.sp; import static jdk.vm.ci.sparc.SPARC.sp;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet; import java.util.HashSet;
import java.util.List; import java.util.List;
@ -74,6 +72,7 @@ import jdk.vm.ci.code.Architecture;
import jdk.vm.ci.code.CallingConvention; import jdk.vm.ci.code.CallingConvention;
import jdk.vm.ci.code.CallingConvention.Type; import jdk.vm.ci.code.CallingConvention.Type;
import jdk.vm.ci.code.Register; import jdk.vm.ci.code.Register;
import jdk.vm.ci.code.RegisterArray;
import jdk.vm.ci.code.RegisterAttributes; import jdk.vm.ci.code.RegisterAttributes;
import jdk.vm.ci.code.RegisterConfig; import jdk.vm.ci.code.RegisterConfig;
import jdk.vm.ci.code.StackSlot; import jdk.vm.ci.code.StackSlot;
@ -81,7 +80,6 @@ import jdk.vm.ci.code.TargetDescription;
import jdk.vm.ci.code.ValueKindFactory; import jdk.vm.ci.code.ValueKindFactory;
import jdk.vm.ci.common.JVMCIError; import jdk.vm.ci.common.JVMCIError;
import jdk.vm.ci.hotspot.HotSpotCallingConventionType; import jdk.vm.ci.hotspot.HotSpotCallingConventionType;
import jdk.vm.ci.hotspot.HotSpotVMConfig;
import jdk.vm.ci.meta.AllocatableValue; import jdk.vm.ci.meta.AllocatableValue;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
import jdk.vm.ci.meta.JavaType; import jdk.vm.ci.meta.JavaType;
@ -93,7 +91,7 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
private final TargetDescription target; private final TargetDescription target;
private final Register[] allocatable; private final RegisterArray allocatable;
private final RegisterAttributes[] attributesMap; private final RegisterAttributes[] attributesMap;
@ -103,20 +101,19 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
private final boolean addNativeRegisterArgumentSlots; private final boolean addNativeRegisterArgumentSlots;
@Override @Override
public Register[] getAllocatableRegisters() { public RegisterArray getAllocatableRegisters() {
return allocatable.clone(); return allocatable;
} }
@Override @Override
public Register[] filterAllocatableRegisters(PlatformKind kind, Register[] registers) { public RegisterArray filterAllocatableRegisters(PlatformKind kind, RegisterArray registers) {
ArrayList<Register> list = new ArrayList<>(); ArrayList<Register> list = new ArrayList<>();
for (Register reg : registers) { for (Register reg : registers) {
if (target.arch.canStoreValue(reg.getRegisterCategory(), kind)) { if (target.arch.canStoreValue(reg.getRegisterCategory(), kind)) {
list.add(reg); list.add(reg);
} }
} }
Register[] ret = list.toArray(new Register[list.size()]); return new RegisterArray(list);
return ret;
} }
@Override @Override
@ -124,30 +121,29 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
return attributesMap.clone(); return attributesMap.clone();
} }
private final Register[] cpuCallerParameterRegisters = {o0, o1, o2, o3, o4, o5}; private final RegisterArray cpuCallerParameterRegisters = new RegisterArray(o0, o1, o2, o3, o4, o5);
private final Register[] cpuCalleeParameterRegisters = {i0, i1, i2, i3, i4, i5}; private final RegisterArray cpuCalleeParameterRegisters = new RegisterArray(i0, i1, i2, i3, i4, i5);
private final Register[] fpuFloatParameterRegisters = {f0, f1, f2, f3, f4, f5, f6, f7}; private final RegisterArray fpuFloatParameterRegisters = new RegisterArray(f0, f1, f2, f3, f4, f5, f6, f7);
private final Register[] fpuDoubleParameterRegisters = {d0, null, d2, null, d4, null, d6, null}; private final RegisterArray fpuDoubleParameterRegisters = new RegisterArray(d0, null, d2, null, d4, null, d6, null);
// @formatter:off // @formatter:off
private final Register[] callerSaveRegisters; private final RegisterArray callerSaveRegisters;
/** /**
* Registers saved by the callee. This lists all L and I registers which are saved in the * This lists all L and I registers which are saved in the register window.
* register window.
*/ */
private final Register[] calleeSaveRegisters = { private final RegisterArray windowSaveRegisters = new RegisterArray(
l0, l1, l2, l3, l4, l5, l6, l7, l0, l1, l2, l3, l4, l5, l6, l7,
i0, i1, i2, i3, i4, i5, i6, i7}; i0, i1, i2, i3, i4, i5, i6, i7);
// @formatter:on // @formatter:on
private static final Register[] reservedRegisters = {sp, g0, g2}; private static final RegisterArray reservedRegisters = new RegisterArray(sp, g0, g2);
private static Register[] initAllocatable(Architecture arch, boolean reserveForHeapBase) { private static RegisterArray initAllocatable(Architecture arch, boolean reserveForHeapBase) {
Register[] allRegisters = arch.getAvailableValueRegisters(); RegisterArray allRegisters = arch.getAvailableValueRegisters();
Register[] registers = new Register[allRegisters.length - reservedRegisters.length - (reserveForHeapBase ? 1 : 0)]; Register[] registers = new Register[allRegisters.size() - reservedRegisters.size() - (reserveForHeapBase ? 1 : 0)];
List<Register> reservedRegistersList = Arrays.asList(reservedRegisters); List<Register> reservedRegistersList = reservedRegisters.asList();
int idx = 0; int idx = 0;
for (Register reg : allRegisters) { for (Register reg : allRegisters) {
@ -164,33 +160,33 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
} }
assert idx == registers.length; assert idx == registers.length;
return registers; return new RegisterArray(registers);
} }
public SPARCHotSpotRegisterConfig(TargetDescription target, HotSpotVMConfig config) { public SPARCHotSpotRegisterConfig(TargetDescription target, boolean useCompressedOops) {
this(target, initAllocatable(target.arch, config.useCompressedOops), config); this(target, initAllocatable(target.arch, useCompressedOops));
} }
public SPARCHotSpotRegisterConfig(TargetDescription target, Register[] allocatable, HotSpotVMConfig config) { public SPARCHotSpotRegisterConfig(TargetDescription target, RegisterArray allocatable) {
this.target = target; this.target = target;
this.allocatable = allocatable.clone(); this.allocatable = allocatable;
this.addNativeRegisterArgumentSlots = config.linuxOs; this.addNativeRegisterArgumentSlots = false;
HashSet<Register> callerSaveSet = new HashSet<>(); HashSet<Register> callerSaveSet = new HashSet<>(target.arch.getAvailableValueRegisters().asList());
Collections.addAll(callerSaveSet, target.arch.getAvailableValueRegisters()); for (Register cs : windowSaveRegisters) {
for (Register cs : calleeSaveRegisters) {
callerSaveSet.remove(cs); callerSaveSet.remove(cs);
} }
this.callerSaveRegisters = callerSaveSet.toArray(new Register[callerSaveSet.size()]); this.callerSaveRegisters = new RegisterArray(callerSaveSet);
attributesMap = RegisterAttributes.createMap(this, SPARC.allRegisters); attributesMap = RegisterAttributes.createMap(this, SPARC.allRegisters);
} }
@Override @Override
public Register[] getCallerSaveRegisters() { public RegisterArray getCallerSaveRegisters() {
return callerSaveRegisters; return callerSaveRegisters;
} }
public Register[] getCalleeSaveRegisters() { @Override
return calleeSaveRegisters; public RegisterArray getCalleeSaveRegisters() {
return null;
} }
@Override @Override
@ -211,7 +207,7 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
} }
@Override @Override
public Register[] getCallingConventionRegisters(Type type, JavaKind kind) { public RegisterArray getCallingConventionRegisters(Type type, JavaKind kind) {
HotSpotCallingConventionType hotspotType = (HotSpotCallingConventionType) type; HotSpotCallingConventionType hotspotType = (HotSpotCallingConventionType) type;
switch (kind) { switch (kind) {
case Boolean: case Boolean:
@ -230,7 +226,7 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
} }
} }
private CallingConvention callingConvention(Register[] generalParameterRegisters, JavaType returnType, JavaType[] parameterTypes, HotSpotCallingConventionType type, private CallingConvention callingConvention(RegisterArray generalParameterRegisters, JavaType returnType, JavaType[] parameterTypes, HotSpotCallingConventionType type,
ValueKindFactory<?> valueKindFactory) { ValueKindFactory<?> valueKindFactory) {
AllocatableValue[] locations = new AllocatableValue[parameterTypes.length]; AllocatableValue[] locations = new AllocatableValue[parameterTypes.length];
@ -249,25 +245,25 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
case Int: case Int:
case Long: case Long:
case Object: case Object:
if (currentGeneral < generalParameterRegisters.length) { if (currentGeneral < generalParameterRegisters.size()) {
Register register = generalParameterRegisters[currentGeneral++]; Register register = generalParameterRegisters.get(currentGeneral++);
locations[i] = register.asValue(valueKindFactory.getValueKind(kind)); locations[i] = register.asValue(valueKindFactory.getValueKind(kind));
} }
break; break;
case Double: case Double:
if (currentFloating < fpuFloatParameterRegisters.length) { if (currentFloating < fpuFloatParameterRegisters.size()) {
if (currentFloating % 2 != 0) { if (currentFloating % 2 != 0) {
// Make register number even to be a double reg // Make register number even to be a double reg
currentFloating++; currentFloating++;
} }
Register register = fpuDoubleParameterRegisters[currentFloating]; Register register = fpuDoubleParameterRegisters.get(currentFloating);
currentFloating += 2; // Only every second is a double register currentFloating += 2; // Only every second is a double register
locations[i] = register.asValue(valueKindFactory.getValueKind(kind)); locations[i] = register.asValue(valueKindFactory.getValueKind(kind));
} }
break; break;
case Float: case Float:
if (currentFloating < fpuFloatParameterRegisters.length) { if (currentFloating < fpuFloatParameterRegisters.size()) {
Register register = fpuFloatParameterRegisters[currentFloating++]; Register register = fpuFloatParameterRegisters.get(currentFloating++);
locations[i] = register.asValue(valueKindFactory.getValueKind(kind)); locations[i] = register.asValue(valueKindFactory.getValueKind(kind));
} }
break; break;
@ -292,7 +288,7 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
int outArgSpillArea; int outArgSpillArea;
if (type == HotSpotCallingConventionType.NativeCall && addNativeRegisterArgumentSlots) { if (type == HotSpotCallingConventionType.NativeCall && addNativeRegisterArgumentSlots) {
// Space for native callee which may spill our outgoing arguments // Space for native callee which may spill our outgoing arguments
outArgSpillArea = Math.min(locations.length, generalParameterRegisters.length) * target.wordSize; outArgSpillArea = Math.min(locations.length, generalParameterRegisters.size()) * target.wordSize;
} else { } else {
outArgSpillArea = 0; outArgSpillArea = 0;
} }
@ -337,6 +333,6 @@ public class SPARCHotSpotRegisterConfig implements RegisterConfig {
@Override @Override
public String toString() { public String toString() {
return String.format("Allocatable: " + Arrays.toString(getAllocatableRegisters()) + "%n" + "CallerSave: " + Arrays.toString(getCallerSaveRegisters()) + "%n"); return String.format("Allocatable: " + getAllocatableRegisters() + "%n" + "CallerSave: " + getCallerSaveRegisters() + "%n");
} }
} }

View File

@ -0,0 +1,71 @@
/*
* Copyright (c) 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 jdk.vm.ci.hotspot.sparc;
import jdk.vm.ci.hotspot.HotSpotVMConfigAccess;
import jdk.vm.ci.hotspot.HotSpotVMConfigStore;
/**
* Used to access native configuration details.
*
* All non-static, public fields in this class are so that they can be compiled as constants.
*/
class SPARCHotSpotVMConfig extends HotSpotVMConfigAccess {
SPARCHotSpotVMConfig(HotSpotVMConfigStore config) {
super(config);
}
final boolean useCompressedOops = getFlag("UseCompressedOops", Boolean.class);
// CPU capabilities
final long vmVersionFeatures = getFieldValue("Abstract_VM_Version::_features", Long.class, "uint64_t");
// SPARC specific values
final int sparcVis3Instructions = getConstant("VM_Version::vis3_instructions_m", Integer.class);
final int sparcVis2Instructions = getConstant("VM_Version::vis2_instructions_m", Integer.class);
final int sparcVis1Instructions = getConstant("VM_Version::vis1_instructions_m", Integer.class);
final int sparcCbcondInstructions = getConstant("VM_Version::cbcond_instructions_m", Integer.class);
final int sparcV8Instructions = getConstant("VM_Version::v8_instructions_m", Integer.class);
final int sparcHardwareMul32 = getConstant("VM_Version::hardware_mul32_m", Integer.class);
final int sparcHardwareDiv32 = getConstant("VM_Version::hardware_div32_m", Integer.class);
final int sparcHardwareFsmuld = getConstant("VM_Version::hardware_fsmuld_m", Integer.class);
final int sparcHardwarePopc = getConstant("VM_Version::hardware_popc_m", Integer.class);
final int sparcV9Instructions = getConstant("VM_Version::v9_instructions_m", Integer.class);
final int sparcSun4v = getConstant("VM_Version::sun4v_m", Integer.class);
final int sparcBlkInitInstructions = getConstant("VM_Version::blk_init_instructions_m", Integer.class);
final int sparcFmafInstructions = getConstant("VM_Version::fmaf_instructions_m", Integer.class);
final int sparcFmauInstructions = getConstant("VM_Version::fmau_instructions_m", Integer.class);
final int sparcSparc64Family = getConstant("VM_Version::sparc64_family_m", Integer.class);
final int sparcMFamily = getConstant("VM_Version::M_family_m", Integer.class);
final int sparcTFamily = getConstant("VM_Version::T_family_m", Integer.class);
final int sparcT1Model = getConstant("VM_Version::T1_model_m", Integer.class);
final int sparcSparc5Instructions = getConstant("VM_Version::sparc5_instructions_m", Integer.class);
final int sparcAesInstructions = getConstant("VM_Version::aes_instructions_m", Integer.class);
final int sparcSha1Instruction = getConstant("VM_Version::sha1_instruction_m", Integer.class);
final int sparcSha256Instruction = getConstant("VM_Version::sha256_instruction_m", Integer.class);
final int sparcSha512Instruction = getConstant("VM_Version::sha512_instruction_m", Integer.class);
final boolean useBlockZeroing = getFlag("UseBlockZeroing", Boolean.class);
final int blockZeroingLowLimit = getFlag("BlockZeroingLowLimit", Integer.class);
}

View File

@ -35,11 +35,9 @@ import jdk.vm.ci.code.InvalidInstalledCodeException;
import jdk.vm.ci.code.TargetDescription; import jdk.vm.ci.code.TargetDescription;
import jdk.vm.ci.common.InitTimer; import jdk.vm.ci.common.InitTimer;
import jdk.vm.ci.common.JVMCIError; import jdk.vm.ci.common.JVMCIError;
import jdk.vm.ci.hotspotvmconfig.HotSpotVMField;
import jdk.vm.ci.meta.JavaType; import jdk.vm.ci.meta.JavaType;
import jdk.vm.ci.meta.ResolvedJavaMethod; import jdk.vm.ci.meta.ResolvedJavaMethod;
import jdk.vm.ci.meta.ResolvedJavaType; import jdk.vm.ci.meta.ResolvedJavaType;
import jdk.internal.misc.Unsafe;
/** /**
* Calls from Java into HotSpot. The behavior of all the methods in this class that take a native * Calls from Java into HotSpot. The behavior of all the methods in this class that take a native
@ -267,8 +265,9 @@ final class CompilerToVM {
native HotSpotResolvedObjectTypeImpl resolveTypeInPool(HotSpotConstantPool constantPool, int cpi) throws LinkageError; native HotSpotResolvedObjectTypeImpl resolveTypeInPool(HotSpotConstantPool constantPool, int cpi) throws LinkageError;
/** /**
* Looks up and attempts to resolve the {@code JVM_CONSTANT_Field} entry at index {@code cpi} in * Looks up and attempts to resolve the {@code JVM_CONSTANT_Field} entry for at index {@code cpi} in
* {@code constantPool}. The values returned in {@code info} are: * {@code constantPool}. For some opcodes, checks are performed that require the {@code method}
* that contains {@code opcode} to be specified. The values returned in {@code info} are:
* *
* <pre> * <pre>
* [(int) flags, // only valid if field is resolved * [(int) flags, // only valid if field is resolved
@ -281,7 +280,7 @@ final class CompilerToVM {
* @param info an array in which the details of the field are returned * @param info an array in which the details of the field are returned
* @return the type defining the field if resolution is successful, 0 otherwise * @return the type defining the field if resolution is successful, 0 otherwise
*/ */
native HotSpotResolvedObjectTypeImpl resolveFieldInPool(HotSpotConstantPool constantPool, int cpi, byte opcode, long[] info); native HotSpotResolvedObjectTypeImpl resolveFieldInPool(HotSpotConstantPool constantPool, int cpi, HotSpotResolvedJavaMethodImpl method, byte opcode, long[] info);
/** /**
* Converts {@code cpci} from an index into the cache for {@code constantPool} to an index * Converts {@code cpci} from an index into the cache for {@code constantPool} to an index
@ -338,9 +337,22 @@ final class CompilerToVM {
native void resetCompilationStatistics(); native void resetCompilationStatistics();
/** /**
* Initializes the fields of {@code config}. * Reads the database of VM info. The return value encodes the info in a nested object array
* that is described by the pseudo Java object {@code info} below:
*
* <pre>
* info = [
* VMField[] vmFields,
* [String name, Long size, ...] vmTypeSizes,
* [String name, Long value, ...] vmConstants,
* [String name, Long value, ...] vmAddresses,
* VMFlag[] vmFlags
* ]
* </pre>
*
* @return VM info as encoded above
*/ */
native long initializeConfiguration(HotSpotVMConfig config); native Object[] readConfiguration();
/** /**
* Resolves the implementation of {@code method} for virtual dispatches on objects of dynamic * Resolves the implementation of {@code method} for virtual dispatches on objects of dynamic
@ -428,7 +440,6 @@ final class CompilerToVM {
* <li>{@link HotSpotVMConfig#localVariableTableElementLengthOffset}</li> * <li>{@link HotSpotVMConfig#localVariableTableElementLengthOffset}</li>
* <li>{@link HotSpotVMConfig#localVariableTableElementNameCpIndexOffset}</li> * <li>{@link HotSpotVMConfig#localVariableTableElementNameCpIndexOffset}</li>
* <li>{@link HotSpotVMConfig#localVariableTableElementDescriptorCpIndexOffset}</li> * <li>{@link HotSpotVMConfig#localVariableTableElementDescriptorCpIndexOffset}</li>
* <li>{@link HotSpotVMConfig#localVariableTableElementSignatureCpIndexOffset}
* <li>{@link HotSpotVMConfig#localVariableTableElementSlotOffset} * <li>{@link HotSpotVMConfig#localVariableTableElementSlotOffset}
* <li>{@link HotSpotVMConfig#localVariableTableElementStartBciOffset} * <li>{@link HotSpotVMConfig#localVariableTableElementStartBciOffset}
* </ul> * </ul>

View File

@ -42,7 +42,7 @@ import jdk.vm.ci.meta.SpeculationLog;
public class HotSpotCodeCacheProvider implements CodeCacheProvider { public class HotSpotCodeCacheProvider implements CodeCacheProvider {
protected final HotSpotJVMCIRuntimeProvider runtime; protected final HotSpotJVMCIRuntimeProvider runtime;
public final HotSpotVMConfig config; protected final HotSpotVMConfig config;
protected final TargetDescription target; protected final TargetDescription target;
protected final RegisterConfig regConfig; protected final RegisterConfig regConfig;
@ -80,12 +80,13 @@ public class HotSpotCodeCacheProvider implements CodeCacheProvider {
for (Field f : fields) { for (Field f : fields) {
if (f.getName().endsWith("Stub")) { if (f.getName().endsWith("Stub")) {
f.setAccessible(true); f.setAccessible(true);
Object address;
try { try {
Object address = f.get(runtime.getConfig()); address = f.get(runtime.getConfig());
if (address.equals(call.target)) { if (address.equals(call.target)) {
return f.getName() + ":0x" + Long.toHexString((Long) address); return f.getName() + ":0x" + Long.toHexString((Long) address);
} }
} catch (Exception e) { } catch (IllegalArgumentException | IllegalAccessException e) {
} }
} }
} }

View File

@ -115,6 +115,7 @@ public class HotSpotCompiledCode implements CompiledCode {
} }
} }
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `sites`, `targetCode`, `comments`, `methods`, `dataSection`, `dataSectionPatches` and `assumptions`")
public HotSpotCompiledCode(String name, byte[] targetCode, int targetCodeSize, Site[] sites, Assumption[] assumptions, ResolvedJavaMethod[] methods, Comment[] comments, byte[] dataSection, public HotSpotCompiledCode(String name, byte[] targetCode, int targetCodeSize, Site[] sites, Assumption[] assumptions, ResolvedJavaMethod[] methods, Comment[] comments, byte[] dataSection,
int dataSectionAlignment, DataPatch[] dataSectionPatches, boolean isImmutablePIC, int totalFrameSize, StackSlot deoptRescueSlot) { int dataSectionAlignment, DataPatch[] dataSectionPatches, boolean isImmutablePIC, int totalFrameSize, StackSlot deoptRescueSlot) {
this.name = name; this.name = name;

View File

@ -196,7 +196,9 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
*/ */
@SuppressWarnings("unused") @SuppressWarnings("unused")
private static HotSpotConstantPool fromMetaspace(long metaspaceConstantPool) { private static HotSpotConstantPool fromMetaspace(long metaspaceConstantPool) {
return new HotSpotConstantPool(metaspaceConstantPool); HotSpotConstantPool cp = new HotSpotConstantPool(metaspaceConstantPool);
runtime().metaAccessContext.add(cp);
return cp;
} }
private HotSpotConstantPool(long metaspaceConstantPool) { private HotSpotConstantPool(long metaspaceConstantPool) {
@ -298,7 +300,8 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
*/ */
private long getEntryAt(int index) { private long getEntryAt(int index) {
assertBounds(index); assertBounds(index);
return UNSAFE.getAddress(getMetaspaceConstantPool() + config().constantPoolSize + index * runtime().getHostJVMCIBackend().getTarget().wordSize); int offset = index * runtime().getHostJVMCIBackend().getTarget().wordSize;
return UNSAFE.getAddress(getMetaspaceConstantPool() + config().constantPoolSize + offset);
} }
/** /**
@ -309,7 +312,8 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
*/ */
private int getIntAt(int index) { private int getIntAt(int index) {
assertTag(index, JVM_CONSTANT.Integer); assertTag(index, JVM_CONSTANT.Integer);
return UNSAFE.getInt(getMetaspaceConstantPool() + config().constantPoolSize + index * runtime().getHostJVMCIBackend().getTarget().wordSize); int offset = index * runtime().getHostJVMCIBackend().getTarget().wordSize;
return UNSAFE.getInt(getMetaspaceConstantPool() + config().constantPoolSize + offset);
} }
/** /**
@ -320,7 +324,8 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
*/ */
private long getLongAt(int index) { private long getLongAt(int index) {
assertTag(index, JVM_CONSTANT.Long); assertTag(index, JVM_CONSTANT.Long);
return UNSAFE.getLong(getMetaspaceConstantPool() + config().constantPoolSize + index * runtime().getHostJVMCIBackend().getTarget().wordSize); int offset = index * runtime().getHostJVMCIBackend().getTarget().wordSize;
return UNSAFE.getLong(getMetaspaceConstantPool() + config().constantPoolSize + offset);
} }
/** /**
@ -331,7 +336,8 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
*/ */
private float getFloatAt(int index) { private float getFloatAt(int index) {
assertTag(index, JVM_CONSTANT.Float); assertTag(index, JVM_CONSTANT.Float);
return UNSAFE.getFloat(getMetaspaceConstantPool() + config().constantPoolSize + index * runtime().getHostJVMCIBackend().getTarget().wordSize); int offset = index * runtime().getHostJVMCIBackend().getTarget().wordSize;
return UNSAFE.getFloat(getMetaspaceConstantPool() + config().constantPoolSize + offset);
} }
/** /**
@ -342,7 +348,8 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
*/ */
private double getDoubleAt(int index) { private double getDoubleAt(int index) {
assertTag(index, JVM_CONSTANT.Double); assertTag(index, JVM_CONSTANT.Double);
return UNSAFE.getDouble(getMetaspaceConstantPool() + config().constantPoolSize + index * runtime().getHostJVMCIBackend().getTarget().wordSize); int offset = index * runtime().getHostJVMCIBackend().getTarget().wordSize;
return UNSAFE.getDouble(getMetaspaceConstantPool() + config().constantPoolSize + offset);
} }
/** /**
@ -353,7 +360,8 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
*/ */
private int getNameAndTypeAt(int index) { private int getNameAndTypeAt(int index) {
assertTag(index, JVM_CONSTANT.NameAndType); assertTag(index, JVM_CONSTANT.NameAndType);
return UNSAFE.getInt(getMetaspaceConstantPool() + config().constantPoolSize + index * runtime().getHostJVMCIBackend().getTarget().wordSize); int offset = index * runtime().getHostJVMCIBackend().getTarget().wordSize;
return UNSAFE.getInt(getMetaspaceConstantPool() + config().constantPoolSize + offset);
} }
/** /**
@ -434,7 +442,8 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
*/ */
private int getUncachedKlassRefIndexAt(int index) { private int getUncachedKlassRefIndexAt(int index) {
assertTagIsFieldOrMethod(index); assertTagIsFieldOrMethod(index);
final int refIndex = UNSAFE.getInt(getMetaspaceConstantPool() + config().constantPoolSize + index * runtime().getHostJVMCIBackend().getTarget().wordSize); int offset = index * runtime().getHostJVMCIBackend().getTarget().wordSize;
final int refIndex = UNSAFE.getInt(getMetaspaceConstantPool() + config().constantPoolSize + offset);
// klass ref index is in the low 16-bits. // klass ref index is in the low 16-bits.
return refIndex & 0xFFFF; return refIndex & 0xFFFF;
} }
@ -587,7 +596,7 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
} }
@Override @Override
public JavaField lookupField(int cpi, int opcode) { public JavaField lookupField(int cpi, ResolvedJavaMethod method, int opcode) {
final int index = rawIndexToConstantPoolIndex(cpi, opcode); final int index = rawIndexToConstantPoolIndex(cpi, opcode);
final int nameAndTypeIndex = getNameAndTypeRefIndexAt(index); final int nameAndTypeIndex = getNameAndTypeRefIndexAt(index);
final int nameIndex = getNameRefIndexAt(nameAndTypeIndex); final int nameIndex = getNameRefIndexAt(nameAndTypeIndex);
@ -603,7 +612,7 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
long[] info = new long[2]; long[] info = new long[2];
HotSpotResolvedObjectTypeImpl resolvedHolder; HotSpotResolvedObjectTypeImpl resolvedHolder;
try { try {
resolvedHolder = compilerToVM().resolveFieldInPool(this, index, (byte) opcode, info); resolvedHolder = compilerToVM().resolveFieldInPool(this, index, (HotSpotResolvedJavaMethodImpl) method, (byte) opcode, info);
} catch (Throwable t) { } catch (Throwable t) {
/* /*
* If there was an exception resolving the field we give up and return an unresolved * If there was an exception resolving the field we give up and return an unresolved
@ -680,15 +689,15 @@ final class HotSpotConstantPool implements ConstantPool, MetaspaceWrapperObject
if (!klass.isPrimitive() && !klass.isArray()) { if (!klass.isPrimitive() && !klass.isArray()) {
UNSAFE.ensureClassInitialized(klass); UNSAFE.ensureClassInitialized(klass);
} }
switch (tag) { if (tag == JVM_CONSTANT.MethodRef) {
case MethodRef: if (Bytecodes.isInvokeHandleAlias(opcode)) {
if (Bytecodes.isInvokeHandleAlias(opcode)) { final int methodRefCacheIndex = rawIndexToConstantPoolIndex(cpi, opcode);
final int methodRefCacheIndex = rawIndexToConstantPoolIndex(cpi, opcode); if (isInvokeHandle(methodRefCacheIndex, type)) {
if (isInvokeHandle(methodRefCacheIndex, type)) { compilerToVM().resolveInvokeHandleInPool(this, methodRefCacheIndex);
compilerToVM().resolveInvokeHandleInPool(this, methodRefCacheIndex);
}
} }
}
} }
break; break;
case InvokeDynamic: case InvokeDynamic:
if (isInvokedynamicIndex(cpi)) { if (isInvokedynamicIndex(cpi)) {

View File

@ -25,7 +25,6 @@ package jdk.vm.ci.hotspot;
import java.lang.reflect.Array; import java.lang.reflect.Array;
import java.util.Objects; import java.util.Objects;
import jdk.internal.vm.annotation.Stable;
import jdk.vm.ci.common.JVMCIError; import jdk.vm.ci.common.JVMCIError;
import jdk.vm.ci.meta.Constant; import jdk.vm.ci.meta.Constant;
import jdk.vm.ci.meta.ConstantReflectionProvider; import jdk.vm.ci.meta.ConstantReflectionProvider;

View File

@ -29,6 +29,7 @@ public class HotSpotForeignCallTarget implements InvokeTarget {
/** /**
* The entry point address of this call's target. * The entry point address of this call's target.
*/ */
@SuppressFBWarnings(value = "URF_UNREAD_PUBLIC_OR_PROTECTED_FIELD", justification = "accessed by subclasses")//
protected long address; protected long address;
public HotSpotForeignCallTarget(long address) { public HotSpotForeignCallTarget(long address) {

View File

@ -22,8 +22,6 @@
*/ */
package jdk.vm.ci.hotspot; package jdk.vm.ci.hotspot;
import java.lang.reflect.Module;
import jdk.vm.ci.code.CompilationRequest; import jdk.vm.ci.code.CompilationRequest;
import jdk.vm.ci.common.JVMCIError; import jdk.vm.ci.common.JVMCIError;
import jdk.vm.ci.hotspot.HotSpotJVMCIRuntime.Option; import jdk.vm.ci.hotspot.HotSpotJVMCIRuntime.Option;
@ -68,7 +66,6 @@ final class HotSpotJVMCICompilerConfig {
if (compilerName != null) { if (compilerName != null) {
for (JVMCICompilerFactory f : Services.load(JVMCICompilerFactory.class)) { for (JVMCICompilerFactory f : Services.load(JVMCICompilerFactory.class)) {
if (f.getCompilerName().equals(compilerName)) { if (f.getCompilerName().equals(compilerName)) {
Module jvmciModule = JVMCICompilerFactory.class.getModule();
Services.exportJVMCITo(f.getClass()); Services.exportJVMCITo(f.getClass());
f.onSelection(); f.onSelection();
factory = f; factory = f;

View File

@ -135,6 +135,7 @@ public class HotSpotJVMCIMetaAccessContext {
*/ */
metadataRoots = list.getHead(); metadataRoots = list.getHead();
} }
assert isRegistered(metaspaceObject);
} }
protected ResolvedJavaType createClass(Class<?> javaClass) { protected ResolvedJavaType createClass(Class<?> javaClass) {
@ -208,7 +209,7 @@ public class HotSpotJVMCIMetaAccessContext {
ChunkIterator() { ChunkIterator() {
currentChunk = head; currentChunk = head;
currentIndex = -1; currentIndex = -1;
findNext(); next = findNext();
} }
Object[] currentChunk; Object[] currentChunk;
@ -245,4 +246,13 @@ public class HotSpotJVMCIMetaAccessContext {
} }
} }
synchronized boolean isRegistered(MetaspaceWrapperObject wrapper) {
for (WeakReference<MetaspaceWrapperObject> m : list) {
if (m != null && m.get() == wrapper) {
return true;
}
}
return false;
}
} }

View File

@ -27,10 +27,6 @@ import static jdk.vm.ci.common.InitTimer.timer;
import java.io.IOException; import java.io.IOException;
import java.io.OutputStream; import java.io.OutputStream;
import java.io.PrintStream; import java.io.PrintStream;
import java.lang.reflect.Array;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
@ -48,6 +44,7 @@ import jdk.vm.ci.code.InstalledCode;
import jdk.vm.ci.common.InitTimer; import jdk.vm.ci.common.InitTimer;
import jdk.vm.ci.common.JVMCIError; import jdk.vm.ci.common.JVMCIError;
import jdk.vm.ci.hotspot.services.HotSpotJVMCICompilerFactory; import jdk.vm.ci.hotspot.services.HotSpotJVMCICompilerFactory;
import jdk.vm.ci.hotspot.services.HotSpotJVMCICompilerFactory.CompilationLevel;
import jdk.vm.ci.hotspot.services.HotSpotVMEventListener; import jdk.vm.ci.hotspot.services.HotSpotVMEventListener;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
import jdk.vm.ci.meta.JavaType; import jdk.vm.ci.meta.JavaType;
@ -94,9 +91,10 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
*/ */
public enum Option { public enum Option {
Compiler(String.class, null, "Selects the system compiler."), Compiler(String.class, null, "Selects the system compiler."),
// Note: The following one is not used (see InitTimer.ENABLED). // Note: The following one is not used (see InitTimer.ENABLED). It is added here
// so that -Djvmci.PrintFlags=true shows the option.
InitTimer(boolean.class, false, "Specifies if initialization timing is enabled."), InitTimer(boolean.class, false, "Specifies if initialization timing is enabled."),
PrintConfig(boolean.class, false, "Prints all HotSpotVMConfig fields."), PrintConfig(boolean.class, false, "Prints VM configuration available via JVMCI and exits."),
PrintFlags(boolean.class, false, "Prints all JVMCI flags and exits."), PrintFlags(boolean.class, false, "Prints all JVMCI flags and exits."),
ShowFlags(boolean.class, false, "Prints all JVMCI flags and continues."), ShowFlags(boolean.class, false, "Prints all JVMCI flags and continues."),
TraceMethodDataFilter(String.class, null, ""); TraceMethodDataFilter(String.class, null, "");
@ -200,6 +198,7 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
protected final CompilerToVM compilerToVm; protected final CompilerToVM compilerToVm;
protected final HotSpotVMConfigStore configStore;
protected final HotSpotVMConfig config; protected final HotSpotVMConfig config;
private final JVMCIBackend hostBackend; private final JVMCIBackend hostBackend;
@ -240,11 +239,13 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
@SuppressWarnings("unused") private final String[] trivialPrefixes; @SuppressWarnings("unused") private final String[] trivialPrefixes;
@SuppressWarnings("try") @SuppressWarnings("try")
@SuppressFBWarnings(value = "DM_EXIT", justification = "PrintFlags is meant to exit the VM")
private HotSpotJVMCIRuntime() { private HotSpotJVMCIRuntime() {
compilerToVm = new CompilerToVM(); compilerToVm = new CompilerToVM();
try (InitTimer t = timer("HotSpotVMConfig<init>")) { try (InitTimer t = timer("HotSpotVMConfig<init>")) {
config = new HotSpotVMConfig(compilerToVm); configStore = new HotSpotVMConfigStore(compilerToVm);
config = new HotSpotVMConfig(configStore);
} }
String hostArchitecture = config.getHostArchitectureName(); String hostArchitecture = config.getHostArchitectureName();
@ -270,18 +271,32 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
} }
if (Option.PrintConfig.getBoolean()) { if (Option.PrintConfig.getBoolean()) {
printConfig(config, compilerToVm); printConfig(configStore, compilerToVm);
System.exit(0);
} }
compilerFactory = HotSpotJVMCICompilerConfig.getCompilerFactory(); compilerFactory = HotSpotJVMCICompilerConfig.getCompilerFactory();
if (compilerFactory instanceof HotSpotJVMCICompilerFactory) { if (compilerFactory instanceof HotSpotJVMCICompilerFactory) {
hsCompilerFactory = (HotSpotJVMCICompilerFactory) compilerFactory; hsCompilerFactory = (HotSpotJVMCICompilerFactory) compilerFactory;
trivialPrefixes = hsCompilerFactory.getTrivialPrefixes(); trivialPrefixes = hsCompilerFactory.getTrivialPrefixes();
compilationLevelAdjustment = hsCompilerFactory.getCompilationLevelAdjustment(config); switch (hsCompilerFactory.getCompilationLevelAdjustment()) {
case None:
compilationLevelAdjustment = config.compLevelAdjustmentNone;
break;
case ByHolder:
compilationLevelAdjustment = config.compLevelAdjustmentByHolder;
break;
case ByFullSignature:
compilationLevelAdjustment = config.compLevelAdjustmentByFullSignature;
break;
default:
compilationLevelAdjustment = config.compLevelAdjustmentNone;
break;
}
} else { } else {
hsCompilerFactory = null; hsCompilerFactory = null;
trivialPrefixes = null; trivialPrefixes = null;
compilationLevelAdjustment = 0; compilationLevelAdjustment = config.compLevelAdjustmentNone;
} }
} }
@ -296,6 +311,10 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
return metaAccessContext.fromClass(javaClass); return metaAccessContext.fromClass(javaClass);
} }
public HotSpotVMConfigStore getConfigStore() {
return configStore;
}
public HotSpotVMConfig getConfig() { public HotSpotVMConfig getConfig() {
return config; return config;
} }
@ -352,7 +371,35 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
*/ */
@SuppressWarnings({"unused"}) @SuppressWarnings({"unused"})
private int adjustCompilationLevel(Class<?> declaringClass, String name, String signature, boolean isOsr, int level) { private int adjustCompilationLevel(Class<?> declaringClass, String name, String signature, boolean isOsr, int level) {
return hsCompilerFactory.adjustCompilationLevel(config, declaringClass, name, signature, isOsr, level); CompilationLevel curLevel;
if (level == config.compilationLevelNone) {
curLevel = CompilationLevel.None;
} else if (level == config.compilationLevelSimple) {
curLevel = CompilationLevel.Simple;
} else if (level == config.compilationLevelLimitedProfile) {
curLevel = CompilationLevel.LimitedProfile;
} else if (level == config.compilationLevelFullProfile) {
curLevel = CompilationLevel.FullProfile;
} else if (level == config.compilationLevelFullOptimization) {
curLevel = CompilationLevel.FullOptimization;
} else {
throw JVMCIError.shouldNotReachHere();
}
switch (hsCompilerFactory.adjustCompilationLevel(declaringClass, name, signature, isOsr, curLevel)) {
case None:
return config.compilationLevelNone;
case Simple:
return config.compilationLevelSimple;
case LimitedProfile:
return config.compilationLevelLimitedProfile;
case FullProfile:
return config.compilationLevelFullProfile;
case FullOptimization:
return config.compilationLevelFullOptimization;
default:
return level;
}
} }
/** /**
@ -416,71 +463,40 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
} }
} }
private static void printConfig(HotSpotVMConfig config, CompilerToVM vm) { @SuppressFBWarnings(value = "DM_DEFAULT_ENCODING", justification = "no localization here please!")
Field[] fields = config.getClass().getDeclaredFields(); private static void printConfigLine(CompilerToVM vm, String format, Object... args) {
Map<String, Field> sortedFields = new TreeMap<>(); String line = String.format(format, args);
for (Field f : fields) { byte[] lineBytes = line.getBytes();
if (!f.isSynthetic() && !Modifier.isStatic(f.getModifiers())) { vm.writeDebugOutput(lineBytes, 0, lineBytes.length);
f.setAccessible(true); vm.flushDebugOutput();
sortedFields.put(f.getName(), f);
}
}
for (Field f : sortedFields.values()) {
try {
String line = String.format("%9s %-40s = %s%n", f.getType().getSimpleName(), f.getName(), pretty(f.get(config)));
byte[] lineBytes = line.getBytes();
vm.writeDebugOutput(lineBytes, 0, lineBytes.length);
vm.flushDebugOutput();
} catch (Exception e) {
}
}
} }
private static String pretty(Object value) { private static void printConfig(HotSpotVMConfigStore store, CompilerToVM vm) {
if (value == null) { TreeMap<String, VMField> fields = new TreeMap<>(store.getFields());
return "null"; for (VMField field : fields.values()) {
if (!field.isStatic()) {
printConfigLine(vm, "[vmconfig:instance field] %s %s {offset=%d[0x%x]}%n", field.type, field.name, field.offset, field.offset);
} else {
String value = field.value == null ? "null" : String.format("%d[0x%x]", field.value, field.value);
printConfigLine(vm, "[vmconfig:static field] %s %s = %s {address=0x%x}%n", field.type, field.name, value, field.address);
}
} }
TreeMap<String, VMFlag> flags = new TreeMap<>(store.getFlags());
Class<?> klass = value.getClass(); for (VMFlag flag : flags.values()) {
if (value instanceof String) { printConfigLine(vm, "[vmconfig:flag] %s %s = %s%n", flag.type, flag.name, flag.value);
return "\"" + value + "\""; }
} else if (value instanceof Method) { TreeMap<String, Long> addresses = new TreeMap<>(store.getAddresses());
return "method \"" + ((Method) value).getName() + "\""; for (Map.Entry<String, Long> e : addresses.entrySet()) {
} else if (value instanceof Class<?>) { printConfigLine(vm, "[vmconfig:address] %s = %d[0x%x]%n", e.getKey(), e.getValue(), e.getValue());
return "class \"" + ((Class<?>) value).getSimpleName() + "\""; }
} else if (value instanceof Integer) { TreeMap<String, Long> constants = new TreeMap<>(store.getConstants());
if ((Integer) value < 10) { for (Map.Entry<String, Long> e : constants.entrySet()) {
return value.toString(); printConfigLine(vm, "[vmconfig:constant] %s = %d[0x%x]%n", e.getKey(), e.getValue(), e.getValue());
} }
return value + " (0x" + Integer.toHexString((Integer) value) + ")"; TreeMap<String, Long> typeSizes = new TreeMap<>(store.getTypeSizes());
} else if (value instanceof Long) { for (Map.Entry<String, Long> e : typeSizes.entrySet()) {
if ((Long) value < 10 && (Long) value > -10) { printConfigLine(vm, "[vmconfig:type size] %s = %d%n", e.getKey(), e.getValue());
return value + "l";
}
return value + "l (0x" + Long.toHexString((Long) value) + "l)";
} else if (klass.isArray()) {
StringBuilder str = new StringBuilder();
int dimensions = 0;
while (klass.isArray()) {
dimensions++;
klass = klass.getComponentType();
}
int length = Array.getLength(value);
str.append(klass.getSimpleName()).append('[').append(length).append(']');
for (int i = 1; i < dimensions; i++) {
str.append("[]");
}
str.append(" {");
for (int i = 0; i < length; i++) {
str.append(pretty(Array.get(value, i)));
if (i < length - 1) {
str.append(", ");
}
}
str.append('}');
return str.toString();
} }
return value.toString();
} }
public OutputStream getLogStream() { public OutputStream getLogStream() {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -36,6 +36,8 @@ import jdk.vm.ci.runtime.JVMCIRuntime;
*/ */
public interface HotSpotJVMCIRuntimeProvider extends JVMCIRuntime { public interface HotSpotJVMCIRuntimeProvider extends JVMCIRuntime {
HotSpotVMConfigStore getConfigStore();
HotSpotVMConfig getConfig(); HotSpotVMConfig getConfig();
CompilerToVM getCompilerToVM(); CompilerToVM getCompilerToVM();

View File

@ -22,7 +22,6 @@
*/ */
package jdk.vm.ci.hotspot; package jdk.vm.ci.hotspot;
import jdk.vm.ci.hotspot.HotSpotVMConfig.CompressEncoding;
import jdk.vm.ci.meta.Constant; import jdk.vm.ci.meta.Constant;
import jdk.vm.ci.meta.JavaConstant; import jdk.vm.ci.meta.JavaConstant;
import jdk.vm.ci.meta.MemoryAccessProvider; import jdk.vm.ci.meta.MemoryAccessProvider;
@ -32,11 +31,11 @@ import jdk.vm.ci.meta.MemoryAccessProvider;
*/ */
public interface HotSpotMemoryAccessProvider extends MemoryAccessProvider { public interface HotSpotMemoryAccessProvider extends MemoryAccessProvider {
JavaConstant readNarrowOopConstant(Constant base, long displacement, CompressEncoding encoding); JavaConstant readNarrowOopConstant(Constant base, long displacement);
Constant readKlassPointerConstant(Constant base, long displacement); Constant readKlassPointerConstant(Constant base, long displacement);
Constant readNarrowKlassPointerConstant(Constant base, long displacement, CompressEncoding encoding); Constant readNarrowKlassPointerConstant(Constant base, long displacement);
Constant readMethodPointerConstant(Constant base, long displacement); Constant readMethodPointerConstant(Constant base, long displacement);
} }

View File

@ -23,7 +23,7 @@
package jdk.vm.ci.hotspot; package jdk.vm.ci.hotspot;
import static jdk.vm.ci.hotspot.UnsafeAccess.UNSAFE; import static jdk.vm.ci.hotspot.UnsafeAccess.UNSAFE;
import jdk.vm.ci.hotspot.HotSpotVMConfig.CompressEncoding;
import jdk.vm.ci.meta.Constant; import jdk.vm.ci.meta.Constant;
import jdk.vm.ci.meta.JavaConstant; import jdk.vm.ci.meta.JavaConstant;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
@ -206,8 +206,7 @@ class HotSpotMemoryAccessProviderImpl implements HotSpotMemoryAccessProvider {
} }
@Override @Override
public JavaConstant readNarrowOopConstant(Constant base, long displacement, CompressEncoding encoding) { public JavaConstant readNarrowOopConstant(Constant base, long displacement) {
assert encoding.equals(runtime.getConfig().getOopEncoding()) : "unexpected oop encoding: " + encoding + " != " + runtime.getConfig().getOopEncoding();
return HotSpotObjectConstantImpl.forObject(readRawObject(base, displacement, true), true); return HotSpotObjectConstantImpl.forObject(readRawObject(base, displacement, true), true);
} }
@ -227,7 +226,7 @@ class HotSpotMemoryAccessProviderImpl implements HotSpotMemoryAccessProvider {
} }
@Override @Override
public Constant readNarrowKlassPointerConstant(Constant base, long displacement, CompressEncoding encoding) { public Constant readNarrowKlassPointerConstant(Constant base, long displacement) {
HotSpotResolvedObjectTypeImpl klass = readKlass(base, displacement, true); HotSpotResolvedObjectTypeImpl klass = readKlass(base, displacement, true);
if (klass == null) { if (klass == null) {
return HotSpotCompressedNullConstant.COMPRESSED_NULL; return HotSpotCompressedNullConstant.COMPRESSED_NULL;

View File

@ -30,7 +30,7 @@ import static jdk.vm.ci.hotspot.UnsafeAccess.UNSAFE;
import java.util.Arrays; import java.util.Arrays;
import jdk.vm.ci.hotspot.HotSpotMethodDataAccessor.Tag; import jdk.internal.misc.Unsafe;
import jdk.vm.ci.meta.DeoptimizationReason; import jdk.vm.ci.meta.DeoptimizationReason;
import jdk.vm.ci.meta.JavaMethodProfile; import jdk.vm.ci.meta.JavaMethodProfile;
import jdk.vm.ci.meta.JavaMethodProfile.ProfiledMethod; import jdk.vm.ci.meta.JavaMethodProfile.ProfiledMethod;
@ -39,41 +39,20 @@ import jdk.vm.ci.meta.JavaTypeProfile.ProfiledType;
import jdk.vm.ci.meta.ResolvedJavaMethod; import jdk.vm.ci.meta.ResolvedJavaMethod;
import jdk.vm.ci.meta.ResolvedJavaType; import jdk.vm.ci.meta.ResolvedJavaType;
import jdk.vm.ci.meta.TriState; import jdk.vm.ci.meta.TriState;
import jdk.internal.misc.Unsafe;
/** /**
* Access to a HotSpot MethodData structure (defined in methodData.hpp). * Access to a HotSpot {@code MethodData} structure (defined in methodData.hpp).
*/ */
public final class HotSpotMethodData { final class HotSpotMethodData {
private static final HotSpotVMConfig config = config(); static final HotSpotVMConfig config = config();
private static final HotSpotMethodDataAccessor NO_DATA_NO_EXCEPTION_ACCESSOR = new NoMethodData(TriState.FALSE); static final HotSpotMethodDataAccessor NO_DATA_NO_EXCEPTION_ACCESSOR = new NoMethodData(config, config.dataLayoutNoTag, TriState.FALSE);
private static final HotSpotMethodDataAccessor NO_DATA_EXCEPTION_POSSIBLY_NOT_RECORDED_ACCESSOR = new NoMethodData(TriState.UNKNOWN); static final HotSpotMethodDataAccessor NO_DATA_EXCEPTION_POSSIBLY_NOT_RECORDED_ACCESSOR = new NoMethodData(config, config.dataLayoutNoTag, TriState.UNKNOWN);
// sorted by tag
// @formatter:off
private static final HotSpotMethodDataAccessor[] PROFILE_DATA_ACCESSORS = {
null,
new BitData(),
new CounterData(),
new JumpData(),
new ReceiverTypeData(),
new VirtualCallData(),
new RetData(),
new BranchData(),
new MultiBranchData(),
new ArgInfoData(),
new UnknownProfileData(Tag.CallTypeData),
new VirtualCallTypeData(),
new UnknownProfileData(Tag.ParametersTypeData),
new UnknownProfileData(Tag.SpeculativeTrapData),
};
// @formatter:on
/** /**
* Reference to the C++ MethodData object. * Reference to the C++ MethodData object.
*/ */
private final long metaspaceMethodData; final long metaspaceMethodData;
@SuppressWarnings("unused") private final HotSpotResolvedJavaMethodImpl method; @SuppressWarnings("unused") private final HotSpotResolvedJavaMethodImpl method;
public HotSpotMethodData(long metaspaceMethodData, HotSpotResolvedJavaMethodImpl method) { public HotSpotMethodData(long metaspaceMethodData, HotSpotResolvedJavaMethodImpl method) {
@ -133,10 +112,7 @@ public final class HotSpotMethodData {
return null; return null;
} }
HotSpotMethodDataAccessor result = getData(position); return getData(position);
final Tag tag = AbstractMethodData.readTag(this, position);
assert result != null : "NO_DATA tag is not allowed " + tag;
return result;
} }
public HotSpotMethodDataAccessor getExtraData(int position) { public HotSpotMethodDataAccessor getExtraData(int position) {
@ -160,18 +136,18 @@ public final class HotSpotMethodData {
private HotSpotMethodDataAccessor getData(int position) { private HotSpotMethodDataAccessor getData(int position) {
assert position >= 0 : "out of bounds"; assert position >= 0 : "out of bounds";
final Tag tag = AbstractMethodData.readTag(this, position); final int tag = HotSpotMethodDataAccessor.readTag(config, this, position);
HotSpotMethodDataAccessor accessor = PROFILE_DATA_ACCESSORS[tag.getValue()]; HotSpotMethodDataAccessor accessor = PROFILE_DATA_ACCESSORS[tag];
assert accessor == null || accessor.getTag() == tag : "wrong data accessor " + accessor + " for tag " + tag; assert accessor == null || accessor.getTag() == tag : "wrong data accessor " + accessor + " for tag " + tag;
return accessor; return accessor;
} }
private int readUnsignedByte(int position, int offsetInBytes) { int readUnsignedByte(int position, int offsetInBytes) {
long fullOffsetInBytes = computeFullOffset(position, offsetInBytes); long fullOffsetInBytes = computeFullOffset(position, offsetInBytes);
return UNSAFE.getByte(metaspaceMethodData + fullOffsetInBytes) & 0xFF; return UNSAFE.getByte(metaspaceMethodData + fullOffsetInBytes) & 0xFF;
} }
private int readUnsignedShort(int position, int offsetInBytes) { int readUnsignedShort(int position, int offsetInBytes) {
long fullOffsetInBytes = computeFullOffset(position, offsetInBytes); long fullOffsetInBytes = computeFullOffset(position, offsetInBytes);
return UNSAFE.getShort(metaspaceMethodData + fullOffsetInBytes) & 0xFFFF; return UNSAFE.getShort(metaspaceMethodData + fullOffsetInBytes) & 0xFFFF;
} }
@ -269,102 +245,14 @@ public final class HotSpotMethodData {
return sb.toString(); return sb.toString();
} }
private abstract static class AbstractMethodData implements HotSpotMethodDataAccessor { static final int NO_DATA_SIZE = cellIndexToOffset(0);
/** static class NoMethodData extends HotSpotMethodDataAccessor {
* Corresponds to {@code exception_seen_flag}.
*/
private static final int EXCEPTIONS_MASK = 1 << config.bitDataExceptionSeenFlag;
private final Tag tag;
protected final int staticSize;
protected AbstractMethodData(Tag tag, int staticSize) {
this.tag = tag;
this.staticSize = staticSize;
}
public Tag getTag() {
return tag;
}
public static Tag readTag(HotSpotMethodData data, int position) {
final int tag = data.readUnsignedByte(position, config.dataLayoutTagOffset);
return Tag.getEnum(tag);
}
@Override
public int getBCI(HotSpotMethodData data, int position) {
return data.readUnsignedShort(position, config.dataLayoutBCIOffset);
}
@Override
public final int getSize(HotSpotMethodData data, int position) {
int size = staticSize + getDynamicSize(data, position);
// Sanity check against VM
int vmSize = HotSpotJVMCIRuntime.runtime().compilerToVm.methodDataProfileDataSize(data.metaspaceMethodData, position);
assert size == vmSize : size + " != " + vmSize;
return size;
}
@Override
public TriState getExceptionSeen(HotSpotMethodData data, int position) {
return TriState.get((getFlags(data, position) & EXCEPTIONS_MASK) != 0);
}
@Override
public JavaTypeProfile getTypeProfile(HotSpotMethodData data, int position) {
return null;
}
@Override
public JavaMethodProfile getMethodProfile(HotSpotMethodData data, int position) {
return null;
}
@Override
public double getBranchTakenProbability(HotSpotMethodData data, int position) {
return -1;
}
@Override
public double[] getSwitchProbabilities(HotSpotMethodData data, int position) {
return null;
}
@Override
public int getExecutionCount(HotSpotMethodData data, int position) {
return -1;
}
@Override
public TriState getNullSeen(HotSpotMethodData data, int position) {
return TriState.UNKNOWN;
}
protected int getFlags(HotSpotMethodData data, int position) {
return data.readUnsignedByte(position, config.dataLayoutFlagsOffset);
}
/**
* @param data
* @param position
*/
protected int getDynamicSize(HotSpotMethodData data, int position) {
return 0;
}
public abstract StringBuilder appendTo(StringBuilder sb, HotSpotMethodData data, int pos);
}
private static class NoMethodData extends AbstractMethodData {
private static final int NO_DATA_SIZE = cellIndexToOffset(0);
private final TriState exceptionSeen; private final TriState exceptionSeen;
protected NoMethodData(TriState exceptionSeen) { protected NoMethodData(HotSpotVMConfig config, int tag, TriState exceptionSeen) {
super(Tag.No, NO_DATA_SIZE); super(config, tag, NO_DATA_SIZE);
this.exceptionSeen = exceptionSeen; this.exceptionSeen = exceptionSeen;
} }
@ -384,17 +272,17 @@ public final class HotSpotMethodData {
} }
} }
private static class BitData extends AbstractMethodData { static final int BIT_DATA_SIZE = cellIndexToOffset(0);
static final int BIT_DATA_NULL_SEEN_FLAG = 1 << config.bitDataNullSeenFlag;
private static final int BIT_DATA_SIZE = cellIndexToOffset(0); static class BitData extends HotSpotMethodDataAccessor {
private static final int BIT_DATA_NULL_SEEN_FLAG = 1 << config.bitDataNullSeenFlag;
private BitData() { private BitData(HotSpotVMConfig config, int tag) {
super(Tag.BitData, BIT_DATA_SIZE); super(config, tag, BIT_DATA_SIZE);
} }
protected BitData(Tag tag, int staticSize) { protected BitData(HotSpotVMConfig config, int tag, int staticSize) {
super(tag, staticSize); super(config, tag, staticSize);
} }
@Override @Override
@ -408,17 +296,17 @@ public final class HotSpotMethodData {
} }
} }
private static class CounterData extends BitData { static final int COUNTER_DATA_SIZE = cellIndexToOffset(1);
static final int COUNTER_DATA_COUNT_OFFSET = cellIndexToOffset(config.methodDataCountOffset);
private static final int COUNTER_DATA_SIZE = cellIndexToOffset(1); static class CounterData extends BitData {
private static final int COUNTER_DATA_COUNT_OFFSET = cellIndexToOffset(config.methodDataCountOffset);
CounterData() { CounterData(HotSpotVMConfig config, int tag) {
super(Tag.CounterData, COUNTER_DATA_SIZE); super(config, tag, COUNTER_DATA_SIZE);
} }
protected CounterData(Tag tag, int staticSize) { protected CounterData(HotSpotVMConfig config, int tag, int staticSize) {
super(tag, staticSize); super(config, tag, staticSize);
} }
@Override @Override
@ -436,18 +324,18 @@ public final class HotSpotMethodData {
} }
} }
private static class JumpData extends AbstractMethodData { static final int JUMP_DATA_SIZE = cellIndexToOffset(2);
static final int TAKEN_COUNT_OFFSET = cellIndexToOffset(config.jumpDataTakenOffset);
static final int TAKEN_DISPLACEMENT_OFFSET = cellIndexToOffset(config.jumpDataDisplacementOffset);
private static final int JUMP_DATA_SIZE = cellIndexToOffset(2); static class JumpData extends HotSpotMethodDataAccessor {
protected static final int TAKEN_COUNT_OFFSET = cellIndexToOffset(config.jumpDataTakenOffset);
protected static final int TAKEN_DISPLACEMENT_OFFSET = cellIndexToOffset(config.jumpDataDisplacementOffset);
JumpData() { JumpData(HotSpotVMConfig config, int tag) {
super(Tag.JumpData, JUMP_DATA_SIZE); super(config, tag, JUMP_DATA_SIZE);
} }
protected JumpData(Tag tag, int staticSize) { protected JumpData(HotSpotVMConfig config, int tag, int staticSize) {
super(tag, staticSize); super(config, tag, staticSize);
} }
@Override @Override
@ -484,16 +372,16 @@ public final class HotSpotMethodData {
} }
} }
private abstract static class AbstractTypeData extends CounterData { static final int TYPE_DATA_ROW_SIZE = cellsToBytes(config.receiverTypeDataReceiverTypeRowCellCount);
protected static final int TYPE_DATA_ROW_SIZE = cellsToBytes(config.receiverTypeDataReceiverTypeRowCellCount); static final int NONPROFILED_COUNT_OFFSET = cellIndexToOffset(config.receiverTypeDataNonprofiledCountOffset);
static final int TYPE_DATA_FIRST_TYPE_OFFSET = cellIndexToOffset(config.receiverTypeDataReceiver0Offset);
static final int TYPE_DATA_FIRST_TYPE_COUNT_OFFSET = cellIndexToOffset(config.receiverTypeDataCount0Offset);
protected static final int NONPROFILED_COUNT_OFFSET = cellIndexToOffset(config.receiverTypeDataNonprofiledCountOffset); abstract static class AbstractTypeData extends CounterData {
protected static final int TYPE_DATA_FIRST_TYPE_OFFSET = cellIndexToOffset(config.receiverTypeDataReceiver0Offset);
protected static final int TYPE_DATA_FIRST_TYPE_COUNT_OFFSET = cellIndexToOffset(config.receiverTypeDataCount0Offset);
protected AbstractTypeData(Tag tag, int staticSize) { protected AbstractTypeData(HotSpotVMConfig config, int tag, int staticSize) {
super(tag, staticSize); super(config, tag, staticSize);
} }
@Override @Override
@ -539,7 +427,7 @@ public final class HotSpotMethodData {
protected abstract long getTypesNotRecordedExecutionCount(HotSpotMethodData data, int position); protected abstract long getTypesNotRecordedExecutionCount(HotSpotMethodData data, int position);
private static JavaTypeProfile createTypeProfile(TriState nullSeen, RawItemProfile<ResolvedJavaType> profile) { private JavaTypeProfile createTypeProfile(TriState nullSeen, RawItemProfile<ResolvedJavaType> profile) {
if (profile.entries <= 0 || profile.totalCount <= 0) { if (profile.entries <= 0 || profile.totalCount <= 0) {
return null; return null;
} }
@ -583,16 +471,16 @@ public final class HotSpotMethodData {
} }
} }
private static class ReceiverTypeData extends AbstractTypeData { static final int TYPE_CHECK_DATA_SIZE = cellIndexToOffset(2) + TYPE_DATA_ROW_SIZE * config.typeProfileWidth;
private static final int TYPE_CHECK_DATA_SIZE = cellIndexToOffset(2) + TYPE_DATA_ROW_SIZE * config.typeProfileWidth; static class ReceiverTypeData extends AbstractTypeData {
ReceiverTypeData() { ReceiverTypeData(HotSpotVMConfig config, int tag) {
super(Tag.ReceiverTypeData, TYPE_CHECK_DATA_SIZE); super(config, tag, TYPE_CHECK_DATA_SIZE);
} }
protected ReceiverTypeData(Tag tag, int staticSize) { protected ReceiverTypeData(HotSpotVMConfig config, int tag, int staticSize) {
super(tag, staticSize); super(config, tag, staticSize);
} }
@Override @Override
@ -606,18 +494,18 @@ public final class HotSpotMethodData {
} }
} }
private static class VirtualCallData extends ReceiverTypeData { static final int VIRTUAL_CALL_DATA_SIZE = cellIndexToOffset(2) + TYPE_DATA_ROW_SIZE * (config.typeProfileWidth + config.methodProfileWidth);
static final int VIRTUAL_CALL_DATA_FIRST_METHOD_OFFSET = TYPE_DATA_FIRST_TYPE_OFFSET + TYPE_DATA_ROW_SIZE * config.typeProfileWidth;
static final int VIRTUAL_CALL_DATA_FIRST_METHOD_COUNT_OFFSET = TYPE_DATA_FIRST_TYPE_COUNT_OFFSET + TYPE_DATA_ROW_SIZE * config.typeProfileWidth;
private static final int VIRTUAL_CALL_DATA_SIZE = cellIndexToOffset(2) + TYPE_DATA_ROW_SIZE * (config.typeProfileWidth + config.methodProfileWidth); static class VirtualCallData extends ReceiverTypeData {
private static final int VIRTUAL_CALL_DATA_FIRST_METHOD_OFFSET = TYPE_DATA_FIRST_TYPE_OFFSET + TYPE_DATA_ROW_SIZE * config.typeProfileWidth;
private static final int VIRTUAL_CALL_DATA_FIRST_METHOD_COUNT_OFFSET = TYPE_DATA_FIRST_TYPE_COUNT_OFFSET + TYPE_DATA_ROW_SIZE * config.typeProfileWidth;
VirtualCallData() { VirtualCallData(HotSpotVMConfig config, int tag) {
super(Tag.VirtualCallData, VIRTUAL_CALL_DATA_SIZE); super(config, tag, VIRTUAL_CALL_DATA_SIZE);
} }
protected VirtualCallData(Tag tag, int staticSize) { protected VirtualCallData(HotSpotVMConfig config, int tag, int staticSize) {
super(tag, staticSize); super(config, tag, staticSize);
} }
@Override @Override
@ -647,7 +535,7 @@ public final class HotSpotMethodData {
return createMethodProfile(getRawMethodProfile(data, position)); return createMethodProfile(getRawMethodProfile(data, position));
} }
private static RawItemProfile<ResolvedJavaMethod> getRawMethodProfile(HotSpotMethodData data, int position) { private RawItemProfile<ResolvedJavaMethod> getRawMethodProfile(HotSpotMethodData data, int position) {
int profileWidth = config.methodProfileWidth; int profileWidth = config.methodProfileWidth;
ResolvedJavaMethod[] methods = new ResolvedJavaMethod[profileWidth]; ResolvedJavaMethod[] methods = new ResolvedJavaMethod[profileWidth];
@ -671,7 +559,7 @@ public final class HotSpotMethodData {
return new RawItemProfile<>(entries, methods, counts, totalCount); return new RawItemProfile<>(entries, methods, counts, totalCount);
} }
private static JavaMethodProfile createMethodProfile(RawItemProfile<ResolvedJavaMethod> profile) { private JavaMethodProfile createMethodProfile(RawItemProfile<ResolvedJavaMethod> profile) {
if (profile.entries <= 0 || profile.totalCount <= 0) { if (profile.entries <= 0 || profile.totalCount <= 0) {
return null; return null;
} }
@ -712,10 +600,10 @@ public final class HotSpotMethodData {
} }
} }
private static class VirtualCallTypeData extends VirtualCallData { static class VirtualCallTypeData extends VirtualCallData {
VirtualCallTypeData() { VirtualCallTypeData(HotSpotVMConfig config, int tag) {
super(Tag.VirtualCallTypeData, 0); super(config, tag, 0);
} }
@Override @Override
@ -725,23 +613,23 @@ public final class HotSpotMethodData {
} }
} }
private static class RetData extends CounterData { static final int RET_DATA_ROW_SIZE = cellsToBytes(3);
static final int RET_DATA_SIZE = cellIndexToOffset(1) + RET_DATA_ROW_SIZE * config.bciProfileWidth;
private static final int RET_DATA_ROW_SIZE = cellsToBytes(3); static class RetData extends CounterData {
private static final int RET_DATA_SIZE = cellIndexToOffset(1) + RET_DATA_ROW_SIZE * config.bciProfileWidth;
RetData() { RetData(HotSpotVMConfig config, int tag) {
super(Tag.RetData, RET_DATA_SIZE); super(config, tag, RET_DATA_SIZE);
} }
} }
private static class BranchData extends JumpData { static final int BRANCH_DATA_SIZE = cellIndexToOffset(3);
static final int NOT_TAKEN_COUNT_OFFSET = cellIndexToOffset(config.branchDataNotTakenOffset);
private static final int BRANCH_DATA_SIZE = cellIndexToOffset(3); static class BranchData extends JumpData {
private static final int NOT_TAKEN_COUNT_OFFSET = cellIndexToOffset(config.branchDataNotTakenOffset);
BranchData() { BranchData(HotSpotVMConfig config, int tag) {
super(Tag.BranchData, BRANCH_DATA_SIZE); super(config, tag, BRANCH_DATA_SIZE);
} }
@Override @Override
@ -768,13 +656,13 @@ public final class HotSpotMethodData {
} }
} }
private static class ArrayData extends AbstractMethodData { static final int ARRAY_DATA_LENGTH_OFFSET = cellIndexToOffset(config.arrayDataArrayLenOffset);
static final int ARRAY_DATA_START_OFFSET = cellIndexToOffset(config.arrayDataArrayStartOffset);
private static final int ARRAY_DATA_LENGTH_OFFSET = cellIndexToOffset(config.arrayDataArrayLenOffset); static class ArrayData extends HotSpotMethodDataAccessor {
protected static final int ARRAY_DATA_START_OFFSET = cellIndexToOffset(config.arrayDataArrayStartOffset);
ArrayData(Tag tag, int staticSize) { ArrayData(HotSpotVMConfig config, int tag, int staticSize) {
super(tag, staticSize); super(config, tag, staticSize);
} }
@Override @Override
@ -792,16 +680,16 @@ public final class HotSpotMethodData {
} }
} }
private static class MultiBranchData extends ArrayData { static final int MULTI_BRANCH_DATA_SIZE = cellIndexToOffset(1);
static final int MULTI_BRANCH_DATA_ROW_SIZE_IN_CELLS = config.multiBranchDataPerCaseCellCount;
static final int MULTI_BRANCH_DATA_ROW_SIZE = cellsToBytes(MULTI_BRANCH_DATA_ROW_SIZE_IN_CELLS);
static final int MULTI_BRANCH_DATA_FIRST_COUNT_OFFSET = ARRAY_DATA_START_OFFSET + cellsToBytes(0);
static final int MULTI_BRANCH_DATA_FIRST_DISPLACEMENT_OFFSET = ARRAY_DATA_START_OFFSET + cellsToBytes(1);
private static final int MULTI_BRANCH_DATA_SIZE = cellIndexToOffset(1); static class MultiBranchData extends ArrayData {
private static final int MULTI_BRANCH_DATA_ROW_SIZE_IN_CELLS = config.multiBranchDataPerCaseCellCount;
private static final int MULTI_BRANCH_DATA_ROW_SIZE = cellsToBytes(MULTI_BRANCH_DATA_ROW_SIZE_IN_CELLS);
private static final int MULTI_BRANCH_DATA_FIRST_COUNT_OFFSET = ARRAY_DATA_START_OFFSET + cellsToBytes(0);
private static final int MULTI_BRANCH_DATA_FIRST_DISPLACEMENT_OFFSET = ARRAY_DATA_START_OFFSET + cellsToBytes(1);
MultiBranchData() { MultiBranchData(HotSpotVMConfig config, int tag) {
super(Tag.MultiBranchData, MULTI_BRANCH_DATA_SIZE); super(config, tag, MULTI_BRANCH_DATA_SIZE);
} }
@Override @Override
@ -878,18 +766,18 @@ public final class HotSpotMethodData {
} }
} }
private static class ArgInfoData extends ArrayData { static final int ARG_INFO_DATA_SIZE = cellIndexToOffset(1);
private static final int ARG_INFO_DATA_SIZE = cellIndexToOffset(1); static class ArgInfoData extends ArrayData {
ArgInfoData() { ArgInfoData(HotSpotVMConfig config, int tag) {
super(Tag.ArgInfoData, ARG_INFO_DATA_SIZE); super(config, tag, ARG_INFO_DATA_SIZE);
} }
} }
private static class UnknownProfileData extends AbstractMethodData { static class UnknownProfileData extends HotSpotMethodDataAccessor {
UnknownProfileData(Tag tag) { UnknownProfileData(HotSpotVMConfig config, int tag) {
super(tag, 0); super(config, tag, 0);
} }
@Override @Override
@ -900,7 +788,6 @@ public final class HotSpotMethodData {
@Override @Override
public StringBuilder appendTo(StringBuilder sb, HotSpotMethodData data, int pos) { public StringBuilder appendTo(StringBuilder sb, HotSpotMethodData data, int pos) {
// TODO Auto-generated method stub
return null; return null;
} }
} }
@ -912,4 +799,41 @@ public final class HotSpotMethodData {
public int getCompiledIRSize() { public int getCompiledIRSize() {
return UNSAFE.getInt(metaspaceMethodData + config.methodDataIRSizeOffset); return UNSAFE.getInt(metaspaceMethodData + config.methodDataIRSizeOffset);
} }
// sorted by tag
// @formatter:off
static final HotSpotMethodDataAccessor[] PROFILE_DATA_ACCESSORS = {
null,
new BitData(config, config.dataLayoutBitDataTag),
new CounterData(config, config.dataLayoutCounterDataTag),
new JumpData(config, config.dataLayoutJumpDataTag),
new ReceiverTypeData(config, config.dataLayoutReceiverTypeDataTag),
new VirtualCallData(config, config.dataLayoutVirtualCallDataTag),
new RetData(config, config.dataLayoutRetDataTag),
new BranchData(config, config.dataLayoutBranchDataTag),
new MultiBranchData(config, config.dataLayoutMultiBranchDataTag),
new ArgInfoData(config, config.dataLayoutArgInfoDataTag),
new UnknownProfileData(config, config.dataLayoutCallTypeDataTag),
new VirtualCallTypeData(config, config.dataLayoutVirtualCallTypeDataTag),
new UnknownProfileData(config, config.dataLayoutParametersTypeDataTag),
new UnknownProfileData(config, config.dataLayoutSpeculativeTrapDataTag),
};
private static boolean checkAccessorTags() {
int expectedTag = 0;
for (HotSpotMethodDataAccessor accessor : PROFILE_DATA_ACCESSORS) {
if (expectedTag ==0 ) {
assert accessor == null;
} else {
assert accessor.tag == expectedTag: expectedTag + " != " + accessor.tag + " " + accessor;
}
expectedTag++;
}
return true;
}
static {
assert checkAccessorTags();
}
// @formatter:on
} }

View File

@ -1,110 +1,129 @@
/*
* 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 jdk.vm.ci.hotspot; package jdk.vm.ci.hotspot;
import static jdk.vm.ci.hotspot.HotSpotVMConfig.config;
import jdk.vm.ci.meta.JavaMethodProfile; import jdk.vm.ci.meta.JavaMethodProfile;
import jdk.vm.ci.meta.JavaTypeProfile; import jdk.vm.ci.meta.JavaTypeProfile;
import jdk.vm.ci.meta.ProfilingInfo; import jdk.vm.ci.meta.ProfilingInfo;
import jdk.vm.ci.meta.TriState; import jdk.vm.ci.meta.TriState;
/** /**
* Interface for accessor objects that encapsulate the logic for accessing the different kinds of * Base class for accessing the different kinds of data in a HotSpot {@code MethodData}. This is
* data in a HotSpot methodDataOop. This interface is similar to the interface {@link ProfilingInfo} * similar to {@link ProfilingInfo}, but most methods require a {@link HotSpotMethodData} and the
* , but most methods require a MethodDataObject and the exact position within the methodData. * exact position within the method data.
*/ */
public interface HotSpotMethodDataAccessor { abstract class HotSpotMethodDataAccessor {
/** final int tag;
* {@code DataLayout} tag values. final int staticSize;
*/ final HotSpotVMConfig config;
enum Tag {
No(config().dataLayoutNoTag),
BitData(config().dataLayoutBitDataTag),
CounterData(config().dataLayoutCounterDataTag),
JumpData(config().dataLayoutJumpDataTag),
ReceiverTypeData(config().dataLayoutReceiverTypeDataTag),
VirtualCallData(config().dataLayoutVirtualCallDataTag),
RetData(config().dataLayoutRetDataTag),
BranchData(config().dataLayoutBranchDataTag),
MultiBranchData(config().dataLayoutMultiBranchDataTag),
ArgInfoData(config().dataLayoutArgInfoDataTag),
CallTypeData(config().dataLayoutCallTypeDataTag),
VirtualCallTypeData(config().dataLayoutVirtualCallTypeDataTag),
ParametersTypeData(config().dataLayoutParametersTypeDataTag),
SpeculativeTrapData(config().dataLayoutSpeculativeTrapDataTag);
private final int value; protected HotSpotMethodDataAccessor(HotSpotVMConfig config, int tag, int staticSize) {
this.config = config;
Tag(int value) { this.tag = tag;
this.value = value; this.staticSize = staticSize;
}
public int getValue() {
return value;
}
public static Tag getEnum(int value) {
Tag result = values()[value];
assert value == result.value;
return result;
}
} }
/** /**
* Returns the {@link Tag} stored in the LayoutData header. * Returns the tag stored in the LayoutData header.
* *
* @return tag stored in the LayoutData header * @return tag stored in the LayoutData header
*/ */
Tag getTag(); int getTag() {
return tag;
}
static int readTag(HotSpotVMConfig config, HotSpotMethodData data, int position) {
final int tag = data.readUnsignedByte(position, config.dataLayoutTagOffset);
assert tag >= config.dataLayoutNoTag && tag <= config.dataLayoutSpeculativeTrapDataTag : "profile data tag out of bounds: " + tag;
return tag;
}
/** /**
* Returns the BCI stored in the LayoutData header. * Returns the BCI stored in the LayoutData header.
* *
* @return An integer &ge; 0 and &le; Short.MAX_VALUE, or -1 if not supported. * @return an integer between 0 and {@link Short#MAX_VALUE} inclusive, or -1 if not supported
*/ */
int getBCI(HotSpotMethodData data, int position); int getBCI(HotSpotMethodData data, int position) {
return data.readUnsignedShort(position, config.dataLayoutBCIOffset);
}
/** /**
* Computes the size for the specific data at the given position. * Computes the size for the specific data at the given position.
* *
* @return An integer &gt; 0. * @return a value greater than 0
*/ */
int getSize(HotSpotMethodData data, int position); final int getSize(HotSpotMethodData data, int position) {
int size = staticSize + getDynamicSize(data, position);
// Sanity check against VM
int vmSize = HotSpotJVMCIRuntime.runtime().compilerToVm.methodDataProfileDataSize(data.metaspaceMethodData, position);
assert size == vmSize : size + " != " + vmSize;
return size;
}
JavaTypeProfile getTypeProfile(HotSpotMethodData data, int position); TriState getExceptionSeen(HotSpotMethodData data, int position) {
final int EXCEPTIONS_MASK = 1 << config.bitDataExceptionSeenFlag;
return TriState.get((getFlags(data, position) & EXCEPTIONS_MASK) != 0);
}
JavaMethodProfile getMethodProfile(HotSpotMethodData data, int position); /**
* @param data
* @param position
*/
JavaTypeProfile getTypeProfile(HotSpotMethodData data, int position) {
return null;
}
double getBranchTakenProbability(HotSpotMethodData data, int position); /**
* @param data
* @param position
*/
JavaMethodProfile getMethodProfile(HotSpotMethodData data, int position) {
return null;
}
double[] getSwitchProbabilities(HotSpotMethodData data, int position); /**
* @param data
* @param position
*/
double getBranchTakenProbability(HotSpotMethodData data, int position) {
return -1;
}
TriState getExceptionSeen(HotSpotMethodData data, int position); /**
* @param data
* @param position
*/
double[] getSwitchProbabilities(HotSpotMethodData data, int position) {
return null;
}
TriState getNullSeen(HotSpotMethodData data, int position); /**
* @param data
* @param position
*/
int getExecutionCount(HotSpotMethodData data, int position) {
return -1;
}
int getExecutionCount(HotSpotMethodData data, int position); /**
* @param data
* @param position
*/
TriState getNullSeen(HotSpotMethodData data, int position) {
return TriState.UNKNOWN;
}
protected int getFlags(HotSpotMethodData data, int position) {
return data.readUnsignedByte(position, config.dataLayoutFlagsOffset);
}
/**
* @param data
* @param position
*/
protected int getDynamicSize(HotSpotMethodData data, int position) {
return 0;
}
abstract StringBuilder appendTo(StringBuilder sb, HotSpotMethodData data, int pos);
StringBuilder appendTo(StringBuilder sb, HotSpotMethodData data, int pos);
} }

View File

@ -27,13 +27,25 @@ import java.util.Arrays;
import jdk.vm.ci.code.Location; import jdk.vm.ci.code.Location;
import jdk.vm.ci.code.ReferenceMap; import jdk.vm.ci.code.ReferenceMap;
/**
* Describes where the object references are in machine state, compliant with what HotSpot expects.
*/
public final class HotSpotReferenceMap extends ReferenceMap { public final class HotSpotReferenceMap extends ReferenceMap {
final Location[] objects; private final Location[] objects;
final Location[] derivedBase; private final Location[] derivedBase;
final int[] sizeInBytes; private final int[] sizeInBytes;
final int maxRegisterSize; private final int maxRegisterSize;
/**
*
* @param objects This array is now owned by this object and must not be mutated by the caller.
* @param derivedBase This array is now owned by this object and must not be mutated by the
* caller.
* @param sizeInBytes This array is now owned by this object and must not be mutated by the
* caller.
*/
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `objects`, `derivedBase` and `sizeInBytes`")
public HotSpotReferenceMap(Location[] objects, Location[] derivedBase, int[] sizeInBytes, int maxRegisterSize) { public HotSpotReferenceMap(Location[] objects, Location[] derivedBase, int[] sizeInBytes, int maxRegisterSize) {
this.objects = objects; this.objects = objects;
this.derivedBase = derivedBase; this.derivedBase = derivedBase;

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2011, 2015, 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. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -28,12 +28,8 @@ import java.lang.annotation.Annotation;
import java.lang.reflect.Field; import java.lang.reflect.Field;
import jdk.internal.vm.annotation.Stable; import jdk.internal.vm.annotation.Stable;
import jdk.vm.ci.common.JVMCIError;
import jdk.vm.ci.hotspot.HotSpotJVMCIRuntime.Option;
import jdk.vm.ci.meta.JavaType; import jdk.vm.ci.meta.JavaType;
import jdk.vm.ci.meta.MetaAccessProvider;
import jdk.vm.ci.meta.ModifiersProvider; import jdk.vm.ci.meta.ModifiersProvider;
import jdk.vm.ci.meta.ResolvedJavaField;
import jdk.vm.ci.meta.ResolvedJavaType; import jdk.vm.ci.meta.ResolvedJavaType;
/** /**

View File

@ -132,10 +132,20 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
return UNSAFE.getInt(javaClass, config().klassOffset) & 0xFFFFFFFFL; return UNSAFE.getInt(javaClass, config().klassOffset) & 0xFFFFFFFFL;
} }
@Override
public long getMetaspacePointer() { public long getMetaspacePointer() {
return getMetaspaceKlass(); return getMetaspaceKlass();
} }
/**
* The Klass* for this object is kept alive by the direct reference to {@link #javaClass} so no
* extra work is required.
*/
@Override
public boolean isRegistered() {
return true;
}
@Override @Override
public int getModifiers() { public int getModifiers() {
if (isArray()) { if (isArray()) {
@ -428,7 +438,13 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
} }
public HotSpotConstantPool getConstantPool() { public HotSpotConstantPool getConstantPool() {
if (constantPool == null) { if (constantPool == null || !isArray() && UNSAFE.getAddress(getMetaspaceKlass() + config().instanceKlassConstantsOffset) != constantPool.getMetaspaceConstantPool()) {
/*
* If the pointer to the ConstantPool has changed since this was last read refresh the
* HotSpotConstantPool wrapper object. This ensures that uses of the constant pool are
* operating on the latest one and that HotSpotResolvedJavaMethodImpls will be able to
* use the shared copy instead of creating their own instance.
*/
constantPool = compilerToVM().getConstantPool(this, config().instanceKlassConstantsOffset); constantPool = compilerToVM().getConstantPool(this, config().instanceKlassConstantsOffset);
} }
return constantPool; return constantPool;
@ -575,7 +591,8 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
// Get Klass::_fields // Get Klass::_fields
final long metaspaceFields = UNSAFE.getAddress(getMetaspaceKlass() + config.instanceKlassFieldsOffset); final long metaspaceFields = UNSAFE.getAddress(getMetaspaceKlass() + config.instanceKlassFieldsOffset);
assert config.fieldInfoFieldSlots == 6 : "revisit the field parsing code"; assert config.fieldInfoFieldSlots == 6 : "revisit the field parsing code";
metaspaceData = metaspaceFields + config.arrayU2DataOffset + config.fieldInfoFieldSlots * Short.BYTES * index; int offset = config.fieldInfoFieldSlots * Short.BYTES * index;
metaspaceData = metaspaceFields + config.arrayU2DataOffset + offset;
} }
private int getAccessFlags() { private int getAccessFlags() {
@ -603,7 +620,8 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
* on top an array of Java shorts. * on top an array of Java shorts.
*/ */
private int readFieldSlot(int index) { private int readFieldSlot(int index) {
return UNSAFE.getChar(metaspaceData + Short.BYTES * index); int offset = Short.BYTES * index;
return UNSAFE.getChar(metaspaceData + offset);
} }
/** /**
@ -612,7 +630,7 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
*/ */
public String getName() { public String getName() {
final int nameIndex = getNameIndex(); final int nameIndex = getNameIndex();
return isInternal() ? HotSpotVmSymbols.symbolAt(nameIndex) : getConstantPool().lookupUtf8(nameIndex); return isInternal() ? config().symbolAt(nameIndex) : getConstantPool().lookupUtf8(nameIndex);
} }
/** /**
@ -621,7 +639,7 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
*/ */
public String getSignature() { public String getSignature() {
final int signatureIndex = getSignatureIndex(); final int signatureIndex = getSignatureIndex();
return isInternal() ? HotSpotVmSymbols.symbolAt(signatureIndex) : getConstantPool().lookupUtf8(signatureIndex); return isInternal() ? config().symbolAt(signatureIndex) : getConstantPool().lookupUtf8(signatureIndex);
} }
public JavaType getType() { public JavaType getType() {
@ -642,6 +660,7 @@ final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implem
} }
} }
@SuppressFBWarnings(value = "SE_COMPARATOR_SHOULD_BE_SERIALIZABLE", justification = "comparator is only used transiently")
private static class OffsetComparator implements java.util.Comparator<HotSpotResolvedJavaField> { private static class OffsetComparator implements java.util.Comparator<HotSpotResolvedJavaField> {
@Override @Override
public int compare(HotSpotResolvedJavaField o1, HotSpotResolvedJavaField o2) { public int compare(HotSpotResolvedJavaField o1, HotSpotResolvedJavaField o2) {

View File

@ -35,7 +35,7 @@ public class HotSpotSpeculationLog implements SpeculationLog {
/** Written by the C++ code that performs deoptimization. */ /** Written by the C++ code that performs deoptimization. */
private volatile Object lastFailed; private volatile Object lastFailed;
/** All speculations that have been a deoptimization reason. */ /** All speculations that have caused a deoptimization. */
private Set<SpeculationReason> failedSpeculations; private Set<SpeculationReason> failedSpeculations;
/** Strong references to all reasons embedded in the current nmethod. */ /** Strong references to all reasons embedded in the current nmethod. */
@ -54,7 +54,7 @@ public class HotSpotSpeculationLog implements SpeculationLog {
} }
@Override @Override
public boolean maySpeculate(SpeculationReason reason) { public synchronized boolean maySpeculate(SpeculationReason reason) {
if (failedSpeculations != null && failedSpeculations.contains(reason)) { if (failedSpeculations != null && failedSpeculations.contains(reason)) {
return false; return false;
} }

View File

@ -0,0 +1,327 @@
/*
* Copyright (c) 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 jdk.vm.ci.hotspot;
import jdk.vm.ci.common.JVMCIError;
/**
* Access to VM configuration data.
*/
public class HotSpotVMConfigAccess {
/**
* Gets the address of a C++ symbol.
*
* @param name name of C++ symbol
* @param notPresent if non-null and the symbol is not present then this value is returned
* @return the address of the symbol
* @throws JVMCIError if the symbol is not present and {@code notPresent == null}
*/
public long getAddress(String name, Long notPresent) {
Long entry = store.vmAddresses.get(name);
if (entry == null) {
if (notPresent != null) {
return notPresent;
}
throw new JVMCIError("expected VM symbol not found: " + name);
}
return entry;
}
/**
* Gets the address of a C++ symbol.
*
* @param name name of C++ symbol
* @return the address of the symbol
* @throws JVMCIError if the symbol is not present
*/
public long getAddress(String name) {
return getAddress(name, null);
}
/**
* Gets the size of a C++ type.
*
* @param name name of the type
* @return the size in bytes of the requested field
* @throws JVMCIError if the field is not present and {@code notPresent} is null
*/
public int getTypeSize(String name) {
Long entry = store.vmTypeSizes.get(name);
if (entry == null) {
throw new JVMCIError("expected VM type not found: " + name);
}
return (int) (long) entry;
}
/**
* Gets the value of a C++ constant.
*
* @param name name of the constant (e.g., {@code "frame::arg_reg_save_area_bytes"})
* @param type the boxed type to which the constant value will be converted
* @param notPresent if non-null and the constant is not present then this value is returned
* @return the constant value converted to {@code type}
* @throws JVMCIError if the constant is not present and {@code notPresent == null}
*/
public <T> T getConstant(String name, Class<T> type, T notPresent) {
Long c = store.vmConstants.get(name);
if (c == null) {
if (notPresent != null) {
return notPresent;
}
throw new JVMCIError("expected VM constant not found: " + name);
}
return type.cast(convertValue(name, type, c, null));
}
/**
* Gets the value of a C++ constant.
*
* @param name name of the constant (e.g., {@code "frame::arg_reg_save_area_bytes"})
* @param type the boxed type to which the constant value will be converted
* @return the constant value converted to {@code type}
* @throws JVMCIError if the constant is not present
*/
public <T> T getConstant(String name, Class<T> type) {
return getConstant(name, type, null);
}
/**
* Gets the offset of a non-static C++ field.
*
* @param name fully qualified name of the field
* @param type the boxed type to which the offset value will be converted (must be
* {@link Integer} or {@link Long})
* @param cppType if non-null, the expected C++ type of the field (e.g., {@code "HeapWord*"})
* @param notPresent if non-null and the field is not present then this value is returned
* @return the offset in bytes of the requested field
* @throws JVMCIError if the field is static or not present and {@code notPresent} is null
*/
public <T> T getFieldOffset(String name, Class<T> type, String cppType, T notPresent) {
assert type == Integer.class || type == Long.class;
VMField entry = getField(name, cppType, notPresent == null);
if (entry == null) {
return notPresent;
}
if (entry.address != 0) {
throw new JVMCIError("cannot get offset of static field " + name);
}
return type.cast(convertValue(name, type, entry.offset, cppType));
}
/**
* Gets the offset of a non-static C++ field.
*
* @param name fully qualified name of the field
* @param type the boxed type to which the offset value will be converted (must be
* {@link Integer} or {@link Long})
* @param cppType if non-null, the expected C++ type of the field (e.g., {@code "HeapWord*"})
* @return the offset in bytes of the requested field
* @throws JVMCIError if the field is static or not present
*/
public <T> T getFieldOffset(String name, Class<T> type, String cppType) {
return getFieldOffset(name, type, cppType, null);
}
/**
* Gets the offset of a non-static C++ field.
*
* @param name fully qualified name of the field
* @param type the boxed type to which the offset value will be converted (must be
* {@link Integer} or {@link Long})
* @return the offset in bytes of the requested field
* @throws JVMCIError if the field is static or not present
*/
public <T> T getFieldOffset(String name, Class<T> type) {
return getFieldOffset(name, type, null, null);
}
/**
* Gets the address of a static C++ field.
*
* @param name fully qualified name of the field
* @param cppType if non-null, the expected C++ type of the field (e.g., {@code "HeapWord*"})
* @param notPresent if non-null and the field is not present then this value is returned
* @return the address of the requested field
* @throws JVMCIError if the field is not static or not present and {@code notPresent} is null
*/
public long getFieldAddress(String name, String cppType, Long notPresent) {
VMField entry = getField(name, cppType, notPresent == null);
if (entry == null) {
return notPresent;
}
if (entry.address == 0) {
throw new JVMCIError(name + " is not a static field");
}
return entry.address;
}
/**
* Gets the address of a static C++ field.
*
* @param name fully qualified name of the field
* @param cppType if non-null, the expected C++ type of the field (e.g., {@code "HeapWord*"})
* @return the address of the requested field
* @throws JVMCIError if the field is not static or not present
*/
public long getFieldAddress(String name, String cppType) {
return getFieldAddress(name, cppType, null);
}
/**
* Gets the value of a static C++ field.
*
* @param name fully qualified name of the field
* @param type the boxed type to which the constant value will be converted
* @param cppType if non-null, the expected C++ type of the field (e.g., {@code "HeapWord*"})
* @param notPresent if non-null and the field is not present then this value is returned
* @return the value of the requested field
* @throws JVMCIError if the field is not static or not present and {@code notPresent} is null
*/
public <T> T getFieldValue(String name, Class<T> type, String cppType, T notPresent) {
VMField entry = getField(name, cppType, notPresent == null);
if (entry == null) {
return notPresent;
}
if (entry.value == null) {
throw new JVMCIError(name + " is not a static field");
}
return type.cast(convertValue(name, type, entry.value, cppType));
}
/**
* Gets the value of a static C++ field.
*
* @param name fully qualified name of the field
* @param type the boxed type to which the constant value will be converted
* @param cppType if non-null, the expected C++ type of the field (e.g., {@code "HeapWord*"})
* @return the value of the requested field
* @throws JVMCIError if the field is not static or not present
*/
public <T> T getFieldValue(String name, Class<T> type, String cppType) {
return getFieldValue(name, type, cppType, null);
}
/**
* Gets the value of a static C++ field.
*
* @param name fully qualified name of the field
* @param type the boxed type to which the constant value will be converted
* @return the value of the requested field
* @throws JVMCIError if the field is not static or not present
*/
public <T> T getFieldValue(String name, Class<T> type) {
return getFieldValue(name, type, null, null);
}
/**
* Gets a C++ field.
*
* @param name fully qualified name of the field
* @param cppType if non-null, the expected C++ type of the field (e.g., {@code "HeapWord*"})
* @param required specifies if the field must be present
* @return the field
* @throws JVMCIError if the field is not present and {@code required == true}
*/
private VMField getField(String name, String cppType, boolean required) {
VMField entry = store.vmFields.get(name);
if (entry == null) {
if (!required) {
return null;
}
throw new JVMCIError("expected VM field not found: " + name);
}
// Make sure the native type is still the type we expect.
if (cppType != null && !cppType.equals(entry.type)) {
throw new JVMCIError("expected type " + cppType + " but VM field " + name + " is of type " + entry.type);
}
return entry;
}
/**
* Gets a VM flag value.
*
* @param name name of the flag (e.g., {@code "CompileTheWorldStartAt"})
* @param type the boxed type to which the flag's value will be converted
* @return the flag's value converted to {@code type} or {@code notPresent} if the flag is not
* present
* @throws JVMCIError if the flag is not present
*/
public <T> T getFlag(String name, Class<T> type) {
return getFlag(name, type, null);
}
/**
* Gets a VM flag value.
*
* @param name name of the flag (e.g., {@code "CompileTheWorldStartAt"})
* @param type the boxed type to which the flag's value will be converted
* @param notPresent if non-null and the flag is not present then this value is returned
* @return the flag's value converted to {@code type} or {@code notPresent} if the flag is not
* present
* @throws JVMCIError if the flag is not present and {@code notPresent == null}
*/
public <T> T getFlag(String name, Class<T> type, T notPresent) {
VMFlag entry = store.vmFlags.get(name);
if (entry == null) {
if (notPresent != null) {
return notPresent;
}
throw new JVMCIError("expected VM flag not found: " + name);
}
return type.cast(convertValue(name, type, entry.value, entry.type));
}
private static <T> Object convertValue(String name, Class<T> toType, Object value, String cppType) throws JVMCIError {
if (toType == Boolean.class) {
if (value instanceof String) {
return Boolean.valueOf((String) value);
} else if (value instanceof Boolean) {
return value;
} else if (value instanceof Long) {
return ((long) value) != 0;
}
} else if (toType == Byte.class) {
if (value instanceof Long) {
return (byte) (long) value;
}
} else if (toType == Integer.class) {
if (value instanceof Integer) {
return value;
} else if (value instanceof Long) {
return (int) (long) value;
}
} else if (toType == Long.class) {
return value;
}
throw new JVMCIError("cannot convert " + name + " of type " + value.getClass().getSimpleName() + (cppType == null ? "" : " [" + cppType + "]") + " to " + toType.getSimpleName());
}
private final HotSpotVMConfigStore store;
public HotSpotVMConfigAccess(HotSpotVMConfigStore store) {
this.store = store;
}
}

View File

@ -0,0 +1,153 @@
/*
* Copyright (c) 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 jdk.vm.ci.hotspot;
import static jdk.vm.ci.common.InitTimer.timer;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import jdk.vm.ci.common.InitTimer;
/**
* Access to VM configuration data.
*/
public final class HotSpotVMConfigStore {
/**
* Gets the C++ symbols whose addresses are exposed by this object.
*
* @return an unmodifiable map from the symbol names to their addresses
*/
public Map<String, Long> getAddresses() {
return Collections.unmodifiableMap(vmAddresses);
}
/**
* Gets the C++ type sizes exposed by this object.
*
* @return an unmodifiable map from C++ type names to their sizes in bytes
*/
public Map<String, Long> getTypeSizes() {
return Collections.unmodifiableMap(vmTypeSizes);
}
/**
* Gets the C++ constants exposed by this object.
*
* @return an unmodifiable map from the names of C++ constants to their values
*/
public Map<String, Long> getConstants() {
return Collections.unmodifiableMap(vmConstants);
}
/**
* Gets the VM flags exposed by this object.
*
* @return an unmodifiable map from VM flag names to {@link VMFlag} objects
*/
public Map<String, VMFlag> getFlags() {
return Collections.unmodifiableMap(vmFlags);
}
/**
* Gets the C++ fields exposed by this object.
*
* @return an unmodifiable map from VM field names to {@link VMField} objects
*/
public Map<String, VMField> getFields() {
return Collections.unmodifiableMap(vmFields);
}
final HashMap<String, VMField> vmFields;
final HashMap<String, Long> vmTypeSizes;
final HashMap<String, Long> vmConstants;
final HashMap<String, Long> vmAddresses;
final HashMap<String, VMFlag> vmFlags;
/**
* Reads the database of VM info. The return value encodes the info in a nested object array
* that is described by the pseudo Java object {@code info} below:
*
* <pre>
* info = [
* VMField[] vmFields,
* [String name, Long size, ...] vmTypeSizes,
* [String name, Long value, ...] vmConstants,
* [String name, Long value, ...] vmAddresses,
* VMFlag[] vmFlags
* ]
* </pre>
*/
@SuppressWarnings("try")
HotSpotVMConfigStore(CompilerToVM compilerToVm) {
Object[] data;
try (InitTimer t = timer("CompilerToVm readConfiguration")) {
data = compilerToVm.readConfiguration();
}
assert data.length == 5 : data.length;
// @formatter:off
VMField[] vmFieldsInfo = (VMField[]) data[0];
Object[] vmTypesSizesInfo = (Object[]) data[1];
Object[] vmConstantsInfo = (Object[]) data[2];
Object[] vmAddressesInfo = (Object[]) data[3];
VMFlag[] vmFlagsInfo = (VMFlag[]) data[4];
vmFields = new HashMap<>(vmFieldsInfo.length);
vmTypeSizes = new HashMap<>(vmTypesSizesInfo.length);
vmConstants = new HashMap<>(vmConstantsInfo.length);
vmAddresses = new HashMap<>(vmAddressesInfo.length);
vmFlags = new HashMap<>(vmFlagsInfo.length);
// @formatter:on
try (InitTimer t = timer("HotSpotVMConfigStore<init> fill maps")) {
for (VMField vmField : vmFieldsInfo) {
vmFields.put(vmField.name, vmField);
}
for (int i = 0; i < vmTypesSizesInfo.length / 2; i++) {
String name = (String) vmTypesSizesInfo[i * 2];
Long size = (Long) vmTypesSizesInfo[i * 2 + 1];
vmTypeSizes.put(name, size);
}
for (int i = 0; i < vmConstantsInfo.length / 2; i++) {
String name = (String) vmConstantsInfo[i * 2];
Long value = (Long) vmConstantsInfo[i * 2 + 1];
vmConstants.put(name, value);
}
for (int i = 0; i < vmAddressesInfo.length / 2; i++) {
String name = (String) vmAddressesInfo[i * 2];
Long value = (Long) vmAddressesInfo[i * 2 + 1];
vmAddresses.put(name, value);
}
for (VMFlag vmFlag : vmFlagsInfo) {
vmFlags.put(vmFlag.name, vmFlag);
}
}
}
}

View File

@ -1,164 +0,0 @@
/*
* Copyright (c) 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.
*/
package jdk.vm.ci.hotspot;
import static java.lang.String.format;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.Executable;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Arrays;
import java.util.Objects;
import jdk.vm.ci.common.JVMCIError;
import jdk.internal.org.objectweb.asm.ClassReader;
import jdk.internal.org.objectweb.asm.ClassVisitor;
import jdk.internal.org.objectweb.asm.Label;
import jdk.internal.org.objectweb.asm.MethodVisitor;
import jdk.internal.org.objectweb.asm.Opcodes;
import jdk.internal.org.objectweb.asm.Type;
import jdk.internal.misc.Unsafe;
/**
* A {@link ClassVisitor} that verifies {@link HotSpotVMConfig} does not access {@link Unsafe} from
* any of its non-static, non-constructor methods. This ensures that a deserialized
* {@link HotSpotVMConfig} object does not perform any unsafe reads on addresses that are only valid
* in the context in which the object was serialized. Note that this does not catch cases where a
* client uses an address stored in a {@link HotSpotVMConfig} field.
*/
final class HotSpotVMConfigVerifier extends ClassVisitor {
public static boolean check() {
Class<?> cls = HotSpotVMConfig.class;
String classFilePath = "/" + cls.getName().replace('.', '/') + ".class";
try {
InputStream classfile = cls.getResourceAsStream(classFilePath);
ClassReader cr = new ClassReader(Objects.requireNonNull(classfile, "Could not find class file for " + cls.getName()));
ClassVisitor cv = new HotSpotVMConfigVerifier();
cr.accept(cv, 0);
return true;
} catch (IOException e) {
throw new JVMCIError(e);
}
}
/**
* Source file context for error reporting.
*/
String sourceFile = null;
/**
* Line number for error reporting.
*/
int lineNo = -1;
private static Class<?> resolve(String name) {
try {
return Class.forName(name.replace('/', '.'));
} catch (ClassNotFoundException e) {
throw new JVMCIError(e);
}
}
HotSpotVMConfigVerifier() {
super(Opcodes.ASM5);
}
@Override
public void visitSource(String source, String debug) {
this.sourceFile = source;
}
void verify(boolean condition, String message) {
if (!condition) {
error(message);
}
}
void error(String message) {
String errorMessage = format("%s:%d: %s is not allowed in the context of compilation replay. The unsafe access should be moved into the %s constructor and the result cached in a field",
sourceFile, lineNo, message, HotSpotVMConfig.class.getSimpleName());
throw new JVMCIError(errorMessage);
}
@Override
public MethodVisitor visitMethod(int access, String name, String d, String signature, String[] exceptions) {
if (!Modifier.isStatic(access) && Modifier.isPublic(access) && !name.equals("<init>")) {
return new MethodVisitor(Opcodes.ASM5) {
@Override
public void visitLineNumber(int line, Label start) {
lineNo = line;
}
private Executable resolveMethod(String owner, String methodName, String methodDesc) {
Class<?> declaringClass = resolve(owner);
while (declaringClass != null) {
if (methodName.equals("<init>")) {
for (Constructor<?> c : declaringClass.getDeclaredConstructors()) {
if (methodDesc.equals(Type.getConstructorDescriptor(c))) {
return c;
}
}
} else {
Type[] argumentTypes = Type.getArgumentTypes(methodDesc);
for (Method m : declaringClass.getDeclaredMethods()) {
if (m.getName().equals(methodName)) {
if (Arrays.equals(argumentTypes, Type.getArgumentTypes(m))) {
if (Type.getReturnType(methodDesc).equals(Type.getReturnType(m))) {
return m;
}
}
}
}
}
declaringClass = declaringClass.getSuperclass();
}
throw new NoSuchMethodError(owner + "." + methodName + methodDesc);
}
/**
* Checks whether a given method is allowed to be called.
*/
private boolean checkInvokeTarget(Executable method) {
if (method.getDeclaringClass().equals(Unsafe.class)) {
return false;
}
return true;
}
@Override
public void visitMethodInsn(int opcode, String owner, String methodName, String methodDesc, boolean itf) {
Executable callee = resolveMethod(owner, methodName, methodDesc);
verify(checkInvokeTarget(callee), "invocation of " + callee);
}
};
} else {
return null;
}
}
}

View File

@ -1,48 +0,0 @@
/*
* Copyright (c) 2014, 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 jdk.vm.ci.hotspot;
import static jdk.vm.ci.hotspot.HotSpotJVMCIRuntime.runtime;
import static jdk.vm.ci.hotspot.UnsafeAccess.UNSAFE;
import jdk.internal.misc.Unsafe;
/**
* Class to access the C++ {@code vmSymbols} table.
*/
final class HotSpotVmSymbols {
/**
* Returns the symbol in the {@code vmSymbols} table at position {@code index} as {@link String}
* .
*
* @param index position in the symbol table
* @return the symbol at position id
*/
static String symbolAt(int index) {
HotSpotJVMCIRuntimeProvider runtime = runtime();
HotSpotVMConfig config = runtime.getConfig();
assert config.vmSymbolsFirstSID <= index && index < config.vmSymbolsSIDLimit : "index " + index + " is out of bounds";
assert config.symbolPointerSize == Unsafe.ADDRESS_SIZE : "the following address read is broken";
return runtime.getCompilerToVM().getSymbol(UNSAFE.getAddress(config.vmSymbolsSymbols + index * config.symbolPointerSize));
}
}

View File

@ -23,7 +23,8 @@
package jdk.vm.ci.hotspot; package jdk.vm.ci.hotspot;
/** /**
* A tag interface indicating that this type is a wrapper around a HotSpot metaspace object. * A tag interface indicating that this type is a wrapper around a HotSpot metaspace object that
* requires GC interaction to keep alive.
* *
* It would preferable if this were the base class containing the pointer but that would require * It would preferable if this were the base class containing the pointer but that would require
* mixins since most of the wrapper types have complex supertype hierarchies. * mixins since most of the wrapper types have complex supertype hierarchies.
@ -31,4 +32,18 @@ package jdk.vm.ci.hotspot;
interface MetaspaceWrapperObject { interface MetaspaceWrapperObject {
long getMetaspacePointer(); long getMetaspacePointer();
/**
* Check if this object is properly registered for metadata tracking. All classes which
* implement this interface must be registered with the
* {@link HotSpotJVMCIMetaAccessContext#add} unless they are kept alive through other means.
* Currently the only type which doesn't require explicit registration is
* {@link HotSpotResolvedObjectTypeImpl} since it's kept alive by references to the
* {@link Class}.
*
* @return true if this object is properly registered for meta data tracking.
*/
default boolean isRegistered() {
return HotSpotJVMCIRuntime.runtime().metaAccessContext.isRegistered(this);
}
} }

View File

@ -0,0 +1,92 @@
/*
* Copyright (c) 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 jdk.vm.ci.hotspot;
/**
* Describes a C++ field exposed via {@link HotSpotVMConfigAccess}.
*/
public final class VMField {
/**
* Fully qualified name of the represented field (e.g., "Klass::_name").
*/
public final String name;
/**
* The represented field's type (e.g., "Symbol*"). This may be {@code null}.
*/
public final String type;
/**
* If represented field is non-static, this is its offset within the containing structure.
*/
public final long offset;
/**
* If represented field is static, this is its address. Otherwise, this field is 0.
*/
public final long address;
/**
* Value of the field represented as a boxed long; only valid for non-oop static fields. This
* value is only captured once, during JVMCI initialization. If {@link #type} cannot be
* meaningfully (e.g., a struct) or safely (e.g., an oop) expressed as a boxed long, this is
* {@code null}.
*/
public final Long value;
/**
* Determines if the represented field is static.
*/
public boolean isStatic() {
return address != 0;
}
/**
* Creates a description of a non-static field.
*/
public VMField(String name, String type, long offset) {
this.name = name;
this.type = type;
this.offset = offset;
this.address = 0;
this.value = null;
}
/**
* Creates a description of a static field.
*/
public VMField(String name, String type, long address, Long value) {
this.name = name;
this.type = type;
this.offset = 0;
this.address = address;
this.value = value;
}
@Override
public String toString() {
String val = value == null ? "null" : String.format("0x%x", value);
return String.format("Field[name=%s, type=%s, offset=%d, address=0x%x, value=%s]", name, type, offset, address, val);
}
}

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2013, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -20,41 +20,36 @@
* or visit www.oracle.com if you need additional information or have any * or visit www.oracle.com if you need additional information or have any
* questions. * questions.
*/ */
package jdk.vm.ci.hotspotvmconfig; package jdk.vm.ci.hotspot;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/** /**
* Refers to a C++ type in the VM. * Describes a VM flag exposed via {@link HotSpotVMConfigAccess}.
*/ */
@Target(ElementType.FIELD) public final class VMFlag {
@Retention(RetentionPolicy.RUNTIME)
public @interface HotSpotVMType {
/** /**
* Types of information this annotation can return. * The name of the flag.
*/ */
enum Type { public final String name;
/**
* Returns the size of the type (C++ {@code sizeof()}). /**
*/ * The C++ type of the flag.
SIZE; */
public final String type;
/**
* The flag's value.
*/
public final Object value;
VMFlag(String name, String type, Object value) {
this.name = name;
this.type = type;
this.value = value;
} }
/** @Override
* Specifies what type of information to return. public String toString() {
* return String.format("Flag[type=%s, name=%s, value=%s]", type, name, value);
* @see Type }
*/
Type get();
/**
* Returns the name of the type.
*
* @return name of type
*/
String name();
} }

View File

@ -22,7 +22,6 @@
*/ */
package jdk.vm.ci.hotspot.services; package jdk.vm.ci.hotspot.services;
import jdk.vm.ci.hotspot.HotSpotVMConfig;
import jdk.vm.ci.runtime.services.JVMCICompilerFactory; import jdk.vm.ci.runtime.services.JVMCICompilerFactory;
/** /**
@ -42,35 +41,54 @@ public abstract class HotSpotJVMCICompilerFactory extends JVMCICompilerFactory {
return null; return null;
} }
public enum CompilationLevelAdjustment {
/**
* No adjustment.
*/
None,
/**
* Adjust based on declaring class of method.
*/
ByHolder,
/**
* Adjust based on declaring class, name and signature of method.
*/
ByFullSignature
}
/** /**
* Determines if this object may want to adjust the compilation level for a method that is being * Determines if this object may want to adjust the compilation level for a method that is being
* scheduled by the VM for compilation. The legal return values and their meanings are: * scheduled by the VM for compilation.
* <ul>
* <li>0 - no adjustment</li>
* <li>1 - adjust based on declaring class of method</li>
* <li>2 - adjust based on declaring class, name and signature of method</li>
* </ul>
*/ */
public int getCompilationLevelAdjustment(HotSpotVMConfig config) { public CompilationLevelAdjustment getCompilationLevelAdjustment() {
return config.compLevelAdjustmentNone; return CompilationLevelAdjustment.None;
}
public enum CompilationLevel {
None,
Simple,
LimitedProfile,
FullProfile,
FullOptimization
} }
/** /**
* Potentially modifies the compilation level currently selected by the VM compilation policy * Potentially modifies the compilation level currently selected by the VM compilation policy
* for a method. * for a method.
* *
* @param config object for reading HotSpot {@code CompLevel} enum values
* @param declaringClass the class in which the method is declared * @param declaringClass the class in which the method is declared
* @param name the name of the method or {@code null} depending on the value that was returned * @param name the name of the method or {@code null} depending on the value that was returned
* by {@link #getCompilationLevelAdjustment(HotSpotVMConfig)} * by {@link #getCompilationLevelAdjustment()}
* @param signature the signature of the method or {@code null} depending on the value that was * @param signature the signature of the method or {@code null} depending on the value that was
* returned by {@link #getCompilationLevelAdjustment(HotSpotVMConfig)} * returned by {@link #getCompilationLevelAdjustment()}
* @param isOsr specifies if the compilation being scheduled in an OSR compilation * @param isOsr specifies if the compilation being scheduled in an OSR compilation
* @param level the compilation level currently selected by the VM compilation policy * @param level the compilation level currently selected by the VM compilation policy
* @return the compilation level to use for the compilation being scheduled (must be a valid * @return the compilation level to use for the compilation being scheduled (must be a valid
* {@code CompLevel} enum value) * {@code CompLevel} enum value)
*/ */
public int adjustCompilationLevel(HotSpotVMConfig config, Class<?> declaringClass, String name, String signature, boolean isOsr, int level) { public CompilationLevel adjustCompilationLevel(Class<?> declaringClass, String name, String signature, boolean isOsr, CompilationLevel level) {
throw new InternalError("Should not reach here"); throw new InternalError("Should not reach here");
} }
} }

View File

@ -1,60 +0,0 @@
/*
* Copyright (c) 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 jdk.vm.ci.hotspotvmconfig;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* Refers to a C++ address in the VM.
*/
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface HotSpotVMAddress {
/**
* Returns the name of the symbol this address is referring to.
*
* @return name of symbol of this address
*/
String name();
/**
* List of architectures where this constant is required. Names are derived from
* {@link HotSpotVMConfig#getHostArchitectureName()}. An empty list implies that the constant is
* required on all architectures.
*/
@SuppressWarnings("javadoc")
String[] archs() default {};
/**
* List of operating systems where this constant is required. Names are derived from
* {@link HotSpotVMConfig#getHostOSName()}. An empty list implies that the constant is required
* on all operating systems.
*/
@SuppressWarnings("javadoc")
String[] os() default {};
}

View File

@ -1,85 +0,0 @@
/*
* Copyright (c) 2013, 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.
*/
package jdk.vm.ci.hotspotvmconfig;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* Refers to a C++ field in the VM.
*/
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface HotSpotVMField {
/**
* Types of information this annotation can return.
*/
enum Type {
/**
* Returns the offset of this field within the type. Only valid for instance fields.
*/
OFFSET,
/**
* Returns the absolute address of this field. Only valid for static fields.
*/
ADDRESS,
/**
* Returns the value of this field. Only valid for static fields.
*/
VALUE;
}
/**
* Specifies what type of information to return.
*
* @see Type
*/
Type get();
/**
* Returns the type name containing this field.
*
* @return name of containing type
*/
String type();
/**
* Returns the name of this field.
*
* @return name of field
*/
String name();
/**
* List of architectures where this constant is required. Names are derived from
* {@link HotSpotVMConfig#getHostArchitectureName()}. An empty list implies that the constant is
* required on all architectures.
*/
@SuppressWarnings("javadoc")
String[] archs() default {};
}

View File

@ -37,6 +37,12 @@ public abstract class AbstractJavaProfile<T extends AbstractProfiledItem<U>, U>
private final double notRecordedProbability; private final double notRecordedProbability;
private final T[] pitems; private final T[] pitems;
/**
*
* @param notRecordedProbability
* @param pitems
*/
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of the `pitems` array parameter")
public AbstractJavaProfile(double notRecordedProbability, T[] pitems) { public AbstractJavaProfile(double notRecordedProbability, T[] pitems) {
this.pitems = pitems; this.pitems = pitems;
assert !Double.isNaN(notRecordedProbability); assert !Double.isNaN(notRecordedProbability);

View File

@ -51,14 +51,14 @@ public abstract class AbstractProfiledItem<T> implements Comparable<AbstractProf
return probability; return probability;
} }
/**
* Returns -1 if the {@linkplain #getProbability() probability} of this item is greater than
* {@code o}'s probability, 0 if there are equal otherwise 1.
*/
@Override @Override
public int compareTo(AbstractProfiledItem<?> o) { public int compareTo(AbstractProfiledItem<?> o) {
if (getProbability() > o.getProbability()) { // Need to swap the order of operands so that higher probabilities are sorted first
return -1; return Double.compare(o.getProbability(), getProbability());
} else if (getProbability() < o.getProbability()) {
return 1;
}
return 0;
} }
@Override @Override

View File

@ -48,16 +48,18 @@ public interface ConstantPool {
/** /**
* Looks up a reference to a field. If {@code opcode} is non-negative, then resolution checks * Looks up a reference to a field. If {@code opcode} is non-negative, then resolution checks
* specific to the bytecode it denotes are performed if the field is already resolved. Should * specific to the bytecode it denotes are performed if the field is already resolved. Checks
* for some bytecodes require the method that contains the bytecode to be specified. Should
* any of these checks fail, an unresolved field reference is returned. * any of these checks fail, an unresolved field reference is returned.
* *
* @param cpi the constant pool index * @param cpi the constant pool index
* @param opcode the opcode of the instruction for which the lookup is being performed or * @param opcode the opcode of the instruction for which the lookup is being performed or
* {@code -1} * {@code -1}
* @param method the method for which the lookup is being performed
* @return a reference to the field at {@code cpi} in this pool * @return a reference to the field at {@code cpi} in this pool
* @throws ClassFormatError if the entry at {@code cpi} is not a field * @throws ClassFormatError if the entry at {@code cpi} is not a field
*/ */
JavaField lookupField(int cpi, int opcode); JavaField lookupField(int cpi, ResolvedJavaMethod method, int opcode);
/** /**
* Looks up a reference to a method. If {@code opcode} is non-negative, then resolution checks * Looks up a reference to a method. If {@code opcode} is non-negative, then resolution checks

View File

@ -22,24 +22,34 @@
*/ */
package jdk.vm.ci.meta; package jdk.vm.ci.meta;
/**
* Maps bytecode indexes to source line numbers.
*
* @see "https://docs.oracle.com/javase/specs/jvms/se8/html/jvms-4.html#jvms-4.7.12"
*/
public class LineNumberTable { public class LineNumberTable {
private final int[] lineNumbers; private final int[] lineNumbers;
private final int[] bci; private final int[] bci;
/**
*
* @param lineNumbers an array or source line numbers. This array is now owned by this object
* and should not be mutated by the caller.
* @param bci an array of bytecode indexes the same length at {@code lineNumbers} whose entries
* are sorted in ascending order. This array is now owned by this object and must not
* be mutated by the caller.
*/
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `lineNumbers` and `bcis`")
public LineNumberTable(int[] lineNumbers, int[] bci) { public LineNumberTable(int[] lineNumbers, int[] bci) {
assert bci.length == lineNumbers.length;
this.lineNumbers = lineNumbers; this.lineNumbers = lineNumbers;
this.bci = bci; this.bci = bci;
} }
public int[] getLineNumberEntries() { /**
return lineNumbers; * Gets a source line number for {@code atBci}.
} */
public int[] getBciEntries() {
return bci;
}
public int getLineNumber(int atBci) { public int getLineNumber(int atBci) {
for (int i = 0; i < this.bci.length - 1; i++) { for (int i = 0; i < this.bci.length - 1; i++) {
if (this.bci[i] <= atBci && atBci < this.bci[i + 1]) { if (this.bci[i] <= atBci && atBci < this.bci[i + 1]) {

View File

@ -22,6 +22,9 @@
*/ */
package jdk.vm.ci.meta; package jdk.vm.ci.meta;
/**
* Describes the type and bytecode index range in which a local variable is live.
*/
public class Local { public class Local {
private final String name; private final String name;

View File

@ -25,10 +25,19 @@ package jdk.vm.ci.meta;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
/**
* @see "https://docs.oracle.com/javase/specs/jvms/se8/html/jvms-4.html#jvms-4.7.13"
*/
public class LocalVariableTable { public class LocalVariableTable {
private final Local[] locals; private final Local[] locals;
/**
*
* @param locals array of objects describing local variables. This array is now owned by this
* object and must not be mutated by the caller.
*/
@SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `locals`")
public LocalVariableTable(Local[] locals) { public LocalVariableTable(Local[] locals) {
this.locals = locals; this.locals = locals;
} }
@ -47,10 +56,6 @@ public class LocalVariableTable {
return result; return result;
} }
public Local[] getLocals() {
return locals;
}
public Local[] getLocalsAt(int bci) { public Local[] getLocalsAt(int bci) {
List<Local> result = new ArrayList<>(); List<Local> result = new ArrayList<>();
for (Local l : locals) { for (Local l : locals) {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2013, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -20,32 +20,21 @@
* or visit www.oracle.com if you need additional information or have any * or visit www.oracle.com if you need additional information or have any
* questions. * questions.
*/ */
package jdk.vm.ci.hotspotvmconfig; package jdk.vm.ci.meta;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/** /**
* Refers to a C++ constant in the VM. * Used to suppress <a href="http://findbugs.sourceforge.net">FindBugs</a> warnings.
*/ */
@Target(ElementType.FIELD) @interface SuppressFBWarnings {
@Retention(RetentionPolicy.RUNTIME) /**
public @interface HotSpotVMConstant { * The set of FindBugs
* <a href="http://findbugs.sourceforge.net/bugDescriptions.html">warnings</a> that are to be
* suppressed in annotated element. The value can be a bug category, kind or pattern.
*/
String[] value();
/** /**
* Returns the name of the constant. * Reason why the warning is suppressed.
*
* @return name of constant
*/ */
String name(); String justification();
/**
* List of architectures where this constant is required. Names are derived from
* {@link HotSpotVMConfig#getHostArchitectureName()}. An empty list implies that the constant is
* required on all architectures.
*/
@SuppressWarnings("javadoc")
String[] archs() default {};
} }

View File

@ -65,7 +65,7 @@ public abstract class Value {
* {@link #toString()} implementation of subclasses. * {@link #toString()} implementation of subclasses.
*/ */
protected final String getKindSuffix() { protected final String getKindSuffix() {
return "|" + getPlatformKind().getTypeChar(); return "|" + valueKind.getKindSuffix();
} }
public final ValueKind<?> getValueKind() { public final ValueKind<?> getValueKind() {

View File

@ -60,6 +60,11 @@ public abstract class ValueKind<K extends ValueKind<K>> {
public IllegalValueKind changeType(PlatformKind newPlatformKind) { public IllegalValueKind changeType(PlatformKind newPlatformKind) {
return this; return this;
} }
@Override
public String toString() {
return "ILLEGAL";
}
} }
/** /**
@ -82,4 +87,13 @@ public abstract class ValueKind<K extends ValueKind<K>> {
* override this to preserve the additional information added by the compiler. * override this to preserve the additional information added by the compiler.
*/ */
public abstract K changeType(PlatformKind newPlatformKind); public abstract K changeType(PlatformKind newPlatformKind);
/**
* Returns a String representation of the kind, which will be included at the end of
* {@link Value#toString()} implementation. Defaults to {@link #toString()} but can be
* overridden to provide something more specific.
*/
public String getKindSuffix() {
return toString();
}
} }

View File

@ -32,6 +32,7 @@ import java.util.Set;
import jdk.vm.ci.code.Architecture; import jdk.vm.ci.code.Architecture;
import jdk.vm.ci.code.Register; import jdk.vm.ci.code.Register;
import jdk.vm.ci.code.Register.RegisterCategory; import jdk.vm.ci.code.Register.RegisterCategory;
import jdk.vm.ci.code.RegisterArray;
import jdk.vm.ci.meta.JavaKind; import jdk.vm.ci.meta.JavaKind;
import jdk.vm.ci.meta.PlatformKind; import jdk.vm.ci.meta.PlatformKind;
@ -179,35 +180,35 @@ public class SPARC extends Architecture {
public static final Register q60 = new Register(111, getQuadncoding(60), "q60", FPUq); public static final Register q60 = new Register(111, getQuadncoding(60), "q60", FPUq);
// @formatter:off // @formatter:off
public static final Register[] cpuRegisters = { public static final RegisterArray cpuRegisters = new RegisterArray(
g0, g1, g2, g3, g4, g5, g6, g7, g0, g1, g2, g3, g4, g5, g6, g7,
o0, o1, o2, o3, o4, o5, o6, o7, o0, o1, o2, o3, o4, o5, o6, o7,
l0, l1, l2, l3, l4, l5, l6, l7, l0, l1, l2, l3, l4, l5, l6, l7,
i0, i1, i2, i3, i4, i5, i6, i7 i0, i1, i2, i3, i4, i5, i6, i7
}; );
public static final Register[] fpusRegisters = { public static final RegisterArray fpusRegisters = new RegisterArray(
f0, f1, f2, f3, f4, f5, f6, f7, f0, f1, f2, f3, f4, f5, f6, f7,
f8, f9, f10, f11, f12, f13, f14, f15, f8, f9, f10, f11, f12, f13, f14, f15,
f16, f17, f18, f19, f20, f21, f22, f23, f16, f17, f18, f19, f20, f21, f22, f23,
f24, f25, f26, f27, f28, f29, f30, f31 f24, f25, f26, f27, f28, f29, f30, f31
}; );
public static final Register[] fpudRegisters = { public static final RegisterArray fpudRegisters = new RegisterArray(
d0, d2, d4, d6, d8, d10, d12, d14, d0, d2, d4, d6, d8, d10, d12, d14,
d16, d18, d20, d22, d24, d26, d28, d30, d16, d18, d20, d22, d24, d26, d28, d30,
d32, d34, d36, d38, d40, d42, d44, d46, d32, d34, d36, d38, d40, d42, d44, d46,
d48, d50, d52, d54, d56, d58, d60, d62 d48, d50, d52, d54, d56, d58, d60, d62
}; );
public static final Register[] fpuqRegisters = { public static final RegisterArray fpuqRegisters = new RegisterArray(
q0, q4, q8, q12, q0, q4, q8, q12,
q16, q20, q24, q28, q16, q20, q24, q28,
q32, q36, q40, q44, q32, q36, q40, q44,
q48, q52, q56, q60, q48, q52, q56, q60
}; );
public static final Register[] allRegisters = { public static final RegisterArray allRegisters = new RegisterArray(
g0, g1, g2, g3, g4, g5, g6, g7, g0, g1, g2, g3, g4, g5, g6, g7,
o0, o1, o2, o3, o4, o5, o6, o7, o0, o1, o2, o3, o4, o5, o6, o7,
l0, l1, l2, l3, l4, l5, l6, l7, l0, l1, l2, l3, l4, l5, l6, l7,
@ -226,8 +227,8 @@ public class SPARC extends Architecture {
q0, q4, q8, q12, q0, q4, q8, q12,
q16, q20, q24, q28, q16, q20, q24, q28,
q32, q36, q40, q44, q32, q36, q40, q44,
q48, q52, q56, q60, q48, q52, q56, q60
}; );
// @formatter:on // @formatter:on
/** /**
@ -248,7 +249,7 @@ public class SPARC extends Architecture {
} }
@Override @Override
public Register[] getAvailableValueRegisters() { public RegisterArray getAvailableValueRegisters() {
return allRegisters; return allRegisters;
} }

View File

@ -156,8 +156,6 @@ int os::Aix::_on_pase = -1;
// SS - service pack, if known, 0 otherwise // SS - service pack, if known, 0 otherwise
uint32_t os::Aix::_os_version = 0; uint32_t os::Aix::_os_version = 0;
int os::Aix::_stack_page_size = -1;
// -1 = uninitialized, 0 - no, 1 - yes // -1 = uninitialized, 0 - no, 1 - yes
int os::Aix::_xpg_sus_mode = -1; int os::Aix::_xpg_sus_mode = -1;
@ -1499,7 +1497,6 @@ void os::print_memory_info(outputStream* st) {
g_multipage_support.error); g_multipage_support.error);
st->cr(); st->cr();
st->print_cr(" os::vm_page_size: %s", describe_pagesize(os::vm_page_size())); st->print_cr(" os::vm_page_size: %s", describe_pagesize(os::vm_page_size()));
// not used in OpenJDK st->print_cr(" os::stack_page_size: %s", describe_pagesize(os::stack_page_size()));
// print out LDR_CNTRL because it affects the default page sizes // print out LDR_CNTRL because it affects the default page sizes
const char* const ldr_cntrl = ::getenv("LDR_CNTRL"); const char* const ldr_cntrl = ::getenv("LDR_CNTRL");
@ -3451,10 +3448,6 @@ void os::init(void) {
FLAG_SET_ERGO(bool, Use64KPages, true); FLAG_SET_ERGO(bool, Use64KPages, true);
} }
// Short-wire stack page size to base page size; if that works, we just remove
// that stack page size altogether.
Aix::_stack_page_size = Aix::_page_size;
// For now UseLargePages is just ignored. // For now UseLargePages is just ignored.
FLAG_SET_ERGO(bool, UseLargePages, false); FLAG_SET_ERGO(bool, UseLargePages, false);
_page_sizes[0] = 0; _page_sizes[0] = 0;
@ -3589,15 +3582,15 @@ jint os::init_2(void) {
// Check minimum allowable stack size for thread creation and to initialize // Check minimum allowable stack size for thread creation and to initialize
// the java system classes, including StackOverflowError - depends on page // the java system classes, including StackOverflowError - depends on page
// size. Add a page for compiler2 recursion in main thread. // size. Add two 4K pages for compiler2 recursion in main thread.
// Add in 2*BytesPerWord times page size to account for VM stack during // Add in 4*BytesPerWord 4K pages to account for VM stack during
// class initialization depending on 32 or 64 bit VM. // class initialization depending on 32 or 64 bit VM.
os::Aix::min_stack_allowed = MAX2(os::Aix::min_stack_allowed, os::Aix::min_stack_allowed = MAX2(os::Aix::min_stack_allowed,
JavaThread::stack_guard_zone_size() + JavaThread::stack_guard_zone_size() +
JavaThread::stack_shadow_zone_size() + JavaThread::stack_shadow_zone_size() +
(2*BytesPerWord COMPILER2_PRESENT(+1)) * Aix::vm_default_page_size()); (4*BytesPerWord COMPILER2_PRESENT(+2)) * 4 * K);
os::Aix::min_stack_allowed = align_size_up(os::Aix::min_stack_allowed, os::Aix::page_size()); os::Aix::min_stack_allowed = align_size_up(os::Aix::min_stack_allowed, os::vm_page_size());
size_t threadStackSizeInBytes = ThreadStackSize * K; size_t threadStackSizeInBytes = ThreadStackSize * K;
if (threadStackSizeInBytes != 0 && if (threadStackSizeInBytes != 0 &&

View File

@ -49,9 +49,6 @@ class Aix {
static Mutex* _createThread_lock; static Mutex* _createThread_lock;
static int _page_size; static int _page_size;
// Page size of newly created pthreads.
static int _stack_page_size;
// -1 = uninitialized, 0 = AIX, 1 = OS/400 (PASE) // -1 = uninitialized, 0 = AIX, 1 = OS/400 (PASE)
static int _on_pase; static int _on_pase;
@ -113,15 +110,6 @@ class Aix {
return _page_size; return _page_size;
} }
// Accessor methods for stack page size which may be different from usual page size.
static int stack_page_size(void) {
assert(_stack_page_size != -1, "not initialized");
return _stack_page_size;
}
// This is used to scale stack space (guard pages etc.). The name is somehow misleading.
static int vm_default_page_size(void ) { return 8*K; }
static address ucontext_get_pc(const ucontext_t* uc); static address ucontext_get_pc(const ucontext_t* uc);
static intptr_t* ucontext_get_sp(const ucontext_t* uc); static intptr_t* ucontext_get_sp(const ucontext_t* uc);
static intptr_t* ucontext_get_fp(const ucontext_t* uc); static intptr_t* ucontext_get_fp(const ucontext_t* uc);

View File

@ -1,6 +1,6 @@
/* /*
* Copyright (c) 1999, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1999, 2016, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2012, 2015 SAP SE. All rights reserved. * Copyright (c) 2012, 2016 SAP SE. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -52,7 +52,7 @@ inline bool os::uses_stack_guard_pages() {
// Whether or not calling code should/can commit/uncommit stack pages // Whether or not calling code should/can commit/uncommit stack pages
// before guarding them. Answer for AIX is definitly no, because memory // before guarding them. Answer for AIX is definitly no, because memory
// is automatically committed on touch. // is automatically committed on touch.
inline bool os::allocate_stack_guard_pages() { inline bool os::must_commit_stack_guard_pages() {
assert(uses_stack_guard_pages(), "sanity check"); assert(uses_stack_guard_pages(), "sanity check");
return false; return false;
} }
@ -65,7 +65,7 @@ inline void os::pd_split_reserved_memory(char *base, size_t size,
} }
// Bang the shadow pages if they need to be touched to be mapped. // Bang the shadow pages if they need to be touched to be mapped.
inline void os::map_stack_shadow_pages() { inline void os::map_stack_shadow_pages(address sp) {
} }
inline void os::dll_unload(void *lib) { inline void os::dll_unload(void *lib) {

View File

@ -3491,13 +3491,15 @@ jint os::init_2(void) {
// Check minimum allowable stack size for thread creation and to initialize // Check minimum allowable stack size for thread creation and to initialize
// the java system classes, including StackOverflowError - depends on page // the java system classes, including StackOverflowError - depends on page
// size. Add a page for compiler2 recursion in main thread. // size. Add two 4K pages for compiler2 recursion in main thread.
// Add in 2*BytesPerWord times page size to account for VM stack during // Add in 4*BytesPerWord 4K pages to account for VM stack during
// class initialization depending on 32 or 64 bit VM. // class initialization depending on 32 or 64 bit VM.
os::Bsd::min_stack_allowed = MAX2(os::Bsd::min_stack_allowed, os::Bsd::min_stack_allowed = MAX2(os::Bsd::min_stack_allowed,
JavaThread::stack_guard_zone_size() + JavaThread::stack_guard_zone_size() +
JavaThread::stack_shadow_zone_size() + JavaThread::stack_shadow_zone_size() +
(2*BytesPerWord COMPILER2_PRESENT(+1)) * Bsd::page_size()); (4*BytesPerWord COMPILER2_PRESENT(+2)) * 4 * K);
os::Bsd::min_stack_allowed = align_size_up(os::Bsd::min_stack_allowed, os::vm_page_size());
size_t threadStackSizeInBytes = ThreadStackSize * K; size_t threadStackSizeInBytes = ThreadStackSize * K;
if (threadStackSizeInBytes != 0 && if (threadStackSizeInBytes != 0 &&

View File

@ -47,7 +47,7 @@ inline bool os::uses_stack_guard_pages() {
return true; return true;
} }
inline bool os::allocate_stack_guard_pages() { inline bool os::must_commit_stack_guard_pages() {
assert(uses_stack_guard_pages(), "sanity check"); assert(uses_stack_guard_pages(), "sanity check");
#if !defined(__FreeBSD__) || __FreeBSD__ < 5 #if !defined(__FreeBSD__) || __FreeBSD__ < 5
// Since FreeBSD 4 uses malloc() for allocating the thread stack // Since FreeBSD 4 uses malloc() for allocating the thread stack
@ -68,7 +68,7 @@ inline void os::pd_split_reserved_memory(char *base, size_t size,
// Bang the shadow pages if they need to be touched to be mapped. // Bang the shadow pages if they need to be touched to be mapped.
inline void os::map_stack_shadow_pages() { inline void os::map_stack_shadow_pages(address sp) {
} }
inline void os::dll_unload(void *lib) { inline void os::dll_unload(void *lib) {

View File

@ -142,7 +142,6 @@ int (*os::Linux::_pthread_setname_np)(pthread_t, const char*) = NULL;
Mutex* os::Linux::_createThread_lock = NULL; Mutex* os::Linux::_createThread_lock = NULL;
pthread_t os::Linux::_main_thread; pthread_t os::Linux::_main_thread;
int os::Linux::_page_size = -1; int os::Linux::_page_size = -1;
const int os::Linux::_vm_default_page_size = (8 * K);
bool os::Linux::_supports_fast_thread_cpu_time = false; bool os::Linux::_supports_fast_thread_cpu_time = false;
uint32_t os::Linux::_os_version = 0; uint32_t os::Linux::_os_version = 0;
const char * os::Linux::_glibc_version = NULL; const char * os::Linux::_glibc_version = NULL;
@ -4784,13 +4783,15 @@ jint os::init_2(void) {
// Check minimum allowable stack size for thread creation and to initialize // Check minimum allowable stack size for thread creation and to initialize
// the java system classes, including StackOverflowError - depends on page // the java system classes, including StackOverflowError - depends on page
// size. Add a page for compiler2 recursion in main thread. // size. Add two 4K pages for compiler2 recursion in main thread.
// Add in 2*BytesPerWord times page size to account for VM stack during // Add in 4*BytesPerWord 4K pages to account for VM stack during
// class initialization depending on 32 or 64 bit VM. // class initialization depending on 32 or 64 bit VM.
os::Linux::min_stack_allowed = MAX2(os::Linux::min_stack_allowed, os::Linux::min_stack_allowed = MAX2(os::Linux::min_stack_allowed,
JavaThread::stack_guard_zone_size() + JavaThread::stack_guard_zone_size() +
JavaThread::stack_shadow_zone_size() + JavaThread::stack_shadow_zone_size() +
(2*BytesPerWord COMPILER2_PRESENT(+1)) * Linux::vm_default_page_size()); (4*BytesPerWord COMPILER2_PRESENT(+2)) * 4 * K);
os::Linux::min_stack_allowed = align_size_up(os::Linux::min_stack_allowed, os::vm_page_size());
size_t threadStackSizeInBytes = ThreadStackSize * K; size_t threadStackSizeInBytes = ThreadStackSize * K;
if (threadStackSizeInBytes != 0 && if (threadStackSizeInBytes != 0 &&

View File

@ -71,7 +71,6 @@ class Linux {
static pthread_t _main_thread; static pthread_t _main_thread;
static Mutex* _createThread_lock; static Mutex* _createThread_lock;
static int _page_size; static int _page_size;
static const int _vm_default_page_size;
static julong available_memory(); static julong available_memory();
static julong physical_memory() { return _physical_memory; } static julong physical_memory() { return _physical_memory; }
@ -130,8 +129,6 @@ class Linux {
static int page_size(void) { return _page_size; } static int page_size(void) { return _page_size; }
static void set_page_size(int val) { _page_size = val; } static void set_page_size(int val) { _page_size = val; }
static int vm_default_page_size(void) { return _vm_default_page_size; }
static address ucontext_get_pc(const ucontext_t* uc); static address ucontext_get_pc(const ucontext_t* uc);
static void ucontext_set_pc(ucontext_t* uc, address pc); static void ucontext_set_pc(ucontext_t* uc, address pc);
static intptr_t* ucontext_get_sp(const ucontext_t* uc); static intptr_t* ucontext_get_sp(const ucontext_t* uc);

View File

@ -47,7 +47,7 @@ inline bool os::uses_stack_guard_pages() {
return true; return true;
} }
inline bool os::allocate_stack_guard_pages() { inline bool os::must_commit_stack_guard_pages() {
assert(uses_stack_guard_pages(), "sanity check"); assert(uses_stack_guard_pages(), "sanity check");
return true; return true;
} }
@ -60,7 +60,7 @@ inline void os::pd_split_reserved_memory(char *base, size_t size,
// Bang the shadow pages if they need to be touched to be mapped. // Bang the shadow pages if they need to be touched to be mapped.
inline void os::map_stack_shadow_pages() { inline void os::map_stack_shadow_pages(address sp) {
} }
inline void os::dll_unload(void *lib) { inline void os::dll_unload(void *lib) {

View File

@ -4477,13 +4477,15 @@ jint os::init_2(void) {
// Check minimum allowable stack size for thread creation and to initialize // Check minimum allowable stack size for thread creation and to initialize
// the java system classes, including StackOverflowError - depends on page // the java system classes, including StackOverflowError - depends on page
// size. Add a page for compiler2 recursion in main thread. // size. Add two 4K pages for compiler2 recursion in main thread.
// Add in 2*BytesPerWord times page size to account for VM stack during // Add in 4*BytesPerWord 4K pages to account for VM stack during
// class initialization depending on 32 or 64 bit VM. // class initialization depending on 32 or 64 bit VM.
os::Solaris::min_stack_allowed = MAX2(os::Solaris::min_stack_allowed, os::Solaris::min_stack_allowed = MAX2(os::Solaris::min_stack_allowed,
JavaThread::stack_guard_zone_size() + JavaThread::stack_guard_zone_size() +
JavaThread::stack_shadow_zone_size() + JavaThread::stack_shadow_zone_size() +
(2*BytesPerWord COMPILER2_PRESENT(+1)) * page_size); (4*BytesPerWord COMPILER2_PRESENT(+2)) * 4 * K);
os::Solaris::min_stack_allowed = align_size_up(os::Solaris::min_stack_allowed, os::vm_page_size());
size_t threadStackSizeInBytes = ThreadStackSize * K; size_t threadStackSizeInBytes = ThreadStackSize * K;
if (threadStackSizeInBytes != 0 && if (threadStackSizeInBytes != 0 &&

View File

@ -46,7 +46,7 @@ inline bool os::uses_stack_guard_pages() {
return true; return true;
} }
inline bool os::allocate_stack_guard_pages() { inline bool os::must_commit_stack_guard_pages() {
assert(uses_stack_guard_pages(), "sanity check"); assert(uses_stack_guard_pages(), "sanity check");
int r = thr_main() ; int r = thr_main() ;
guarantee (r == 0 || r == 1, "CR6501650 or CR6493689") ; guarantee (r == 0 || r == 1, "CR6501650 or CR6493689") ;
@ -61,8 +61,9 @@ inline void os::pd_split_reserved_memory(char *base, size_t size,
// Bang the shadow pages if they need to be touched to be mapped. // Bang the shadow pages if they need to be touched to be mapped.
inline void os::map_stack_shadow_pages() { inline void os::map_stack_shadow_pages(address sp) {
} }
inline void os::dll_unload(void *lib) { ::dlclose(lib); } inline void os::dll_unload(void *lib) { ::dlclose(lib); }
inline const int os::default_file_open_flags() { return 0;} inline const int os::default_file_open_flags() { return 0;}

View File

@ -4183,13 +4183,16 @@ jint os::init_2(void) {
// Check minimum allowable stack size for thread creation and to initialize // Check minimum allowable stack size for thread creation and to initialize
// the java system classes, including StackOverflowError - depends on page // the java system classes, including StackOverflowError - depends on page
// size. Add a page for compiler2 recursion in main thread. // size. Add two 4K pages for compiler2 recursion in main thread.
// Add in 2*BytesPerWord times page size to account for VM stack during // Add in 4*BytesPerWord 4K pages to account for VM stack during
// class initialization depending on 32 or 64 bit VM. // class initialization depending on 32 or 64 bit VM.
size_t min_stack_allowed = size_t min_stack_allowed =
(size_t)(JavaThread::stack_yellow_zone_size() + JavaThread::stack_red_zone_size() + (size_t)(JavaThread::stack_guard_zone_size() +
JavaThread::stack_shadow_zone_size() + JavaThread::stack_shadow_zone_size() +
(2*BytesPerWord COMPILER2_PRESENT(+1)) * os::vm_page_size()); (4*BytesPerWord COMPILER2_PRESENT(+2)) * 4 * K);
min_stack_allowed = align_size_up(min_stack_allowed, os::vm_page_size());
if (actual_reserve_size < min_stack_allowed) { if (actual_reserve_size < min_stack_allowed) {
tty->print_cr("\nThe stack size specified is too small, " tty->print_cr("\nThe stack size specified is too small, "
"Specify at least %dk", "Specify at least %dk",
@ -5247,6 +5250,12 @@ int os::fork_and_exec(char* cmd) {
static int mallocDebugIntervalCounter = 0; static int mallocDebugIntervalCounter = 0;
static int mallocDebugCounter = 0; static int mallocDebugCounter = 0;
// For debugging possible bugs inside HeapWalk (a ring buffer)
#define SAVE_COUNT 8
static PROCESS_HEAP_ENTRY saved_heap_entries[SAVE_COUNT];
static int saved_heap_entry_index;
bool os::check_heap(bool force) { bool os::check_heap(bool force) {
if (++mallocDebugCounter < MallocVerifyStart && !force) return true; if (++mallocDebugCounter < MallocVerifyStart && !force) return true;
if (++mallocDebugIntervalCounter >= MallocVerifyInterval || force) { if (++mallocDebugIntervalCounter >= MallocVerifyInterval || force) {
@ -5267,13 +5276,28 @@ bool os::check_heap(bool force) {
if (HeapLock(heap) != 0) { if (HeapLock(heap) != 0) {
PROCESS_HEAP_ENTRY phe; PROCESS_HEAP_ENTRY phe;
phe.lpData = NULL; phe.lpData = NULL;
memset(saved_heap_entries, 0, sizeof(saved_heap_entries));
saved_heap_entry_index = 0;
int count = 0;
while (HeapWalk(heap, &phe) != 0) { while (HeapWalk(heap, &phe) != 0) {
count ++;
if ((phe.wFlags & PROCESS_HEAP_ENTRY_BUSY) && if ((phe.wFlags & PROCESS_HEAP_ENTRY_BUSY) &&
!HeapValidate(heap, 0, phe.lpData)) { !HeapValidate(heap, 0, phe.lpData)) {
tty->print_cr("C heap has been corrupted (time: %d allocations)", mallocDebugCounter); tty->print_cr("C heap has been corrupted (time: %d allocations)", mallocDebugCounter);
tty->print_cr("corrupted block near address %#x, length %d", phe.lpData, phe.cbData); tty->print_cr("corrupted block near address %#x, length %d, count %d", phe.lpData, phe.cbData, count);
HeapUnlock(heap); HeapUnlock(heap);
fatal("corrupted C heap"); fatal("corrupted C heap");
} else {
// Save previous seen entries in a ring buffer. We have seen strange
// heap corruption fatal errors that produced mdmp files, but when we load
// these mdmp files in WinDBG, "!heap -triage" shows no error.
// We can examine the saved_heap_entries[] array in the mdmp file to
// diagnose such seemingly spurious errors reported by HeapWalk.
saved_heap_entries[saved_heap_entry_index++] = phe;
if (saved_heap_entry_index >= SAVE_COUNT) {
saved_heap_entry_index = 0;
}
} }
} }
DWORD err = GetLastError(); DWORD err = GetLastError();

View File

@ -53,7 +53,7 @@ inline bool os::uses_stack_guard_pages() {
return true; return true;
} }
inline bool os::allocate_stack_guard_pages() { inline bool os::must_commit_stack_guard_pages() {
return true; return true;
} }
@ -66,14 +66,16 @@ inline int os::readdir_buf_size(const char *path)
} }
// Bang the shadow pages if they need to be touched to be mapped. // Bang the shadow pages if they need to be touched to be mapped.
inline void os::map_stack_shadow_pages() { inline void os::map_stack_shadow_pages(address sp) {
// Write to each page of our new frame to force OS mapping. // Write to each page of our new frame to force OS mapping.
// If we decrement stack pointer more than one page // If we decrement stack pointer more than one page
// the OS may not map an intervening page into our space // the OS may not map an intervening page into our space
// and may fault on a memory access to interior of our frame. // and may fault on a memory access to interior of our frame.
address sp = current_stack_pointer(); const int page_size = os::win32::vm_page_size();
for (size_t pages = 1; pages <= (JavaThread::stack_shadow_zone_size() / os::vm_page_size()); pages++) { const size_t n_pages = JavaThread::stack_shadow_zone_size() / page_size;
*((int *)(sp - (pages * vm_page_size()))) = 0; for (size_t pages = 1; pages <= n_pages; pages++) {
sp -= page_size;
*sp = 0;
} }
} }

View File

@ -291,12 +291,30 @@ inline void* Atomic::xchg_ptr(void* exchange_value, volatile void* dest) {
return (void*)xchg_ptr((intptr_t)exchange_value, (volatile intptr_t*)dest); return (void*)xchg_ptr((intptr_t)exchange_value, (volatile intptr_t*)dest);
} }
inline void cmpxchg_pre_membar(cmpxchg_memory_order order) {
if (order != memory_order_relaxed) {
__asm__ __volatile__ (
/* fence */
strasm_sync
);
}
}
inline void cmpxchg_post_membar(cmpxchg_memory_order order) {
if (order != memory_order_relaxed) {
__asm__ __volatile__ (
/* fence */
strasm_sync
);
}
}
#define VM_HAS_SPECIALIZED_CMPXCHG_BYTE #define VM_HAS_SPECIALIZED_CMPXCHG_BYTE
inline jbyte Atomic::cmpxchg(jbyte exchange_value, volatile jbyte* dest, jbyte compare_value) { inline jbyte Atomic::cmpxchg(jbyte exchange_value, volatile jbyte* dest, jbyte compare_value, cmpxchg_memory_order order) {
// Note that cmpxchg guarantees a two-way memory barrier across // Note that cmpxchg guarantees a two-way memory barrier across
// the cmpxchg, so it's really a a 'fence_cmpxchg_acquire' // the cmpxchg, so it's really a a 'fence_cmpxchg_fence' if not
// (see atomic.hpp). // specified otherwise (see atomic.hpp).
// Using 32 bit internally. // Using 32 bit internally.
volatile int *dest_base = (volatile int*)((uintptr_t)dest & ~3); volatile int *dest_base = (volatile int*)((uintptr_t)dest & ~3);
@ -312,9 +330,9 @@ inline jbyte Atomic::cmpxchg(jbyte exchange_value, volatile jbyte* dest, jbyte c
unsigned int old_value, value32; unsigned int old_value, value32;
cmpxchg_pre_membar(order);
__asm__ __volatile__ ( __asm__ __volatile__ (
/* fence */
strasm_sync
/* simple guard */ /* simple guard */
" lbz %[old_value], 0(%[dest]) \n" " lbz %[old_value], 0(%[dest]) \n"
" cmpw %[masked_compare_val], %[old_value] \n" " cmpw %[masked_compare_val], %[old_value] \n"
@ -331,8 +349,6 @@ inline jbyte Atomic::cmpxchg(jbyte exchange_value, volatile jbyte* dest, jbyte c
" xor %[value32], %[xor_value], %[value32] \n" " xor %[value32], %[xor_value], %[value32] \n"
" stwcx. %[value32], 0, %[dest_base] \n" " stwcx. %[value32], 0, %[dest_base] \n"
" bne- 1b \n" " bne- 1b \n"
/* acquire */
strasm_sync
/* exit */ /* exit */
"2: \n" "2: \n"
/* out */ /* out */
@ -353,21 +369,23 @@ inline jbyte Atomic::cmpxchg(jbyte exchange_value, volatile jbyte* dest, jbyte c
"memory" "memory"
); );
cmpxchg_post_membar(order);
return (jbyte)(unsigned char)old_value; return (jbyte)(unsigned char)old_value;
} }
inline jint Atomic::cmpxchg(jint exchange_value, volatile jint* dest, jint compare_value) { inline jint Atomic::cmpxchg(jint exchange_value, volatile jint* dest, jint compare_value, cmpxchg_memory_order order) {
// Note that cmpxchg guarantees a two-way memory barrier across // Note that cmpxchg guarantees a two-way memory barrier across
// the cmpxchg, so it's really a a 'fence_cmpxchg_acquire' // the cmpxchg, so it's really a a 'fence_cmpxchg_fence' if not
// (see atomic.hpp). // specified otherwise (see atomic.hpp).
unsigned int old_value; unsigned int old_value;
const uint64_t zero = 0; const uint64_t zero = 0;
cmpxchg_pre_membar(order);
__asm__ __volatile__ ( __asm__ __volatile__ (
/* fence */
strasm_sync
/* simple guard */ /* simple guard */
" lwz %[old_value], 0(%[dest]) \n" " lwz %[old_value], 0(%[dest]) \n"
" cmpw %[compare_value], %[old_value] \n" " cmpw %[compare_value], %[old_value] \n"
@ -379,8 +397,6 @@ inline jint Atomic::cmpxchg(jint exchange_value, volatile jint* dest, jint compa
" bne- 2f \n" " bne- 2f \n"
" stwcx. %[exchange_value], %[dest], %[zero] \n" " stwcx. %[exchange_value], %[dest], %[zero] \n"
" bne- 1b \n" " bne- 1b \n"
/* acquire */
strasm_sync
/* exit */ /* exit */
"2: \n" "2: \n"
/* out */ /* out */
@ -397,21 +413,23 @@ inline jint Atomic::cmpxchg(jint exchange_value, volatile jint* dest, jint compa
"memory" "memory"
); );
cmpxchg_post_membar(order);
return (jint) old_value; return (jint) old_value;
} }
inline jlong Atomic::cmpxchg(jlong exchange_value, volatile jlong* dest, jlong compare_value) { inline jlong Atomic::cmpxchg(jlong exchange_value, volatile jlong* dest, jlong compare_value, cmpxchg_memory_order order) {
// Note that cmpxchg guarantees a two-way memory barrier across // Note that cmpxchg guarantees a two-way memory barrier across
// the cmpxchg, so it's really a a 'fence_cmpxchg_acquire' // the cmpxchg, so it's really a a 'fence_cmpxchg_fence' if not
// (see atomic.hpp). // specified otherwise (see atomic.hpp).
long old_value; long old_value;
const uint64_t zero = 0; const uint64_t zero = 0;
cmpxchg_pre_membar(order);
__asm__ __volatile__ ( __asm__ __volatile__ (
/* fence */
strasm_sync
/* simple guard */ /* simple guard */
" ld %[old_value], 0(%[dest]) \n" " ld %[old_value], 0(%[dest]) \n"
" cmpd %[compare_value], %[old_value] \n" " cmpd %[compare_value], %[old_value] \n"
@ -423,8 +441,6 @@ inline jlong Atomic::cmpxchg(jlong exchange_value, volatile jlong* dest, jlong c
" bne- 2f \n" " bne- 2f \n"
" stdcx. %[exchange_value], %[dest], %[zero] \n" " stdcx. %[exchange_value], %[dest], %[zero] \n"
" bne- 1b \n" " bne- 1b \n"
/* acquire */
strasm_sync
/* exit */ /* exit */
"2: \n" "2: \n"
/* out */ /* out */
@ -441,15 +457,17 @@ inline jlong Atomic::cmpxchg(jlong exchange_value, volatile jlong* dest, jlong c
"memory" "memory"
); );
cmpxchg_post_membar(order);
return (jlong) old_value; return (jlong) old_value;
} }
inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value, volatile intptr_t* dest, intptr_t compare_value) { inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value, volatile intptr_t* dest, intptr_t compare_value, cmpxchg_memory_order order) {
return (intptr_t)cmpxchg((jlong)exchange_value, (volatile jlong*)dest, (jlong)compare_value); return (intptr_t)cmpxchg((jlong)exchange_value, (volatile jlong*)dest, (jlong)compare_value, order);
} }
inline void* Atomic::cmpxchg_ptr(void* exchange_value, volatile void* dest, void* compare_value) { inline void* Atomic::cmpxchg_ptr(void* exchange_value, volatile void* dest, void* compare_value, cmpxchg_memory_order order) {
return (void*)cmpxchg((jlong)exchange_value, (volatile jlong*)dest, (jlong)compare_value); return (void*)cmpxchg((jlong)exchange_value, (volatile jlong*)dest, (jlong)compare_value, order);
} }
#undef strasm_sync #undef strasm_sync

View File

@ -89,7 +89,7 @@ inline void* Atomic::xchg_ptr(void* exchange_value, volatile void* des
} }
#define VM_HAS_SPECIALIZED_CMPXCHG_BYTE #define VM_HAS_SPECIALIZED_CMPXCHG_BYTE
inline jbyte Atomic::cmpxchg (jbyte exchange_value, volatile jbyte* dest, jbyte compare_value) { inline jbyte Atomic::cmpxchg (jbyte exchange_value, volatile jbyte* dest, jbyte compare_value, cmpxchg_memory_order order) {
int mp = os::is_MP(); int mp = os::is_MP();
__asm__ volatile (LOCK_IF_MP(%4) "cmpxchgb %1,(%3)" __asm__ volatile (LOCK_IF_MP(%4) "cmpxchgb %1,(%3)"
: "=a" (exchange_value) : "=a" (exchange_value)
@ -98,7 +98,7 @@ inline jbyte Atomic::cmpxchg (jbyte exchange_value, volatile jbyte*
return exchange_value; return exchange_value;
} }
inline jint Atomic::cmpxchg (jint exchange_value, volatile jint* dest, jint compare_value) { inline jint Atomic::cmpxchg (jint exchange_value, volatile jint* dest, jint compare_value, cmpxchg_memory_order order) {
int mp = os::is_MP(); int mp = os::is_MP();
__asm__ volatile (LOCK_IF_MP(%4) "cmpxchgl %1,(%3)" __asm__ volatile (LOCK_IF_MP(%4) "cmpxchgl %1,(%3)"
: "=a" (exchange_value) : "=a" (exchange_value)
@ -149,7 +149,7 @@ inline intptr_t Atomic::xchg_ptr(intptr_t exchange_value, volatile intptr_t* des
return exchange_value; return exchange_value;
} }
inline jlong Atomic::cmpxchg (jlong exchange_value, volatile jlong* dest, jlong compare_value) { inline jlong Atomic::cmpxchg (jlong exchange_value, volatile jlong* dest, jlong compare_value, cmpxchg_memory_order order) {
bool mp = os::is_MP(); bool mp = os::is_MP();
__asm__ __volatile__ (LOCK_IF_MP(%4) "cmpxchgq %1,(%3)" __asm__ __volatile__ (LOCK_IF_MP(%4) "cmpxchgq %1,(%3)"
: "=a" (exchange_value) : "=a" (exchange_value)
@ -158,12 +158,12 @@ inline jlong Atomic::cmpxchg (jlong exchange_value, volatile jlong*
return exchange_value; return exchange_value;
} }
inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value, volatile intptr_t* dest, intptr_t compare_value) { inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value, volatile intptr_t* dest, intptr_t compare_value, cmpxchg_memory_order order) {
return (intptr_t)cmpxchg((jlong)exchange_value, (volatile jlong*)dest, (jlong)compare_value); return (intptr_t)cmpxchg((jlong)exchange_value, (volatile jlong*)dest, (jlong)compare_value, order);
} }
inline void* Atomic::cmpxchg_ptr(void* exchange_value, volatile void* dest, void* compare_value) { inline void* Atomic::cmpxchg_ptr(void* exchange_value, volatile void* dest, void* compare_value, cmpxchg_memory_order order) {
return (void*)cmpxchg((jlong)exchange_value, (volatile jlong*)dest, (jlong)compare_value); return (void*)cmpxchg((jlong)exchange_value, (volatile jlong*)dest, (jlong)compare_value, order);
} }
inline jlong Atomic::load(volatile jlong* src) { return *src; } inline jlong Atomic::load(volatile jlong* src) { return *src; }
@ -197,16 +197,16 @@ extern "C" {
void _Atomic_move_long(volatile jlong* src, volatile jlong* dst); void _Atomic_move_long(volatile jlong* src, volatile jlong* dst);
} }
inline jlong Atomic::cmpxchg (jlong exchange_value, volatile jlong* dest, jlong compare_value) { inline jlong Atomic::cmpxchg (jlong exchange_value, volatile jlong* dest, jlong compare_value, cmpxchg_memory_order order) {
return _Atomic_cmpxchg_long(exchange_value, dest, compare_value, os::is_MP()); return _Atomic_cmpxchg_long(exchange_value, dest, compare_value, os::is_MP());
} }
inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value, volatile intptr_t* dest, intptr_t compare_value) { inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value, volatile intptr_t* dest, intptr_t compare_value, cmpxchg_memory_order order) {
return (intptr_t)cmpxchg((jint)exchange_value, (volatile jint*)dest, (jint)compare_value); return (intptr_t)cmpxchg((jint)exchange_value, (volatile jint*)dest, (jint)compare_value, order);
} }
inline void* Atomic::cmpxchg_ptr(void* exchange_value, volatile void* dest, void* compare_value) { inline void* Atomic::cmpxchg_ptr(void* exchange_value, volatile void* dest, void* compare_value, cmpxchg_memory_order order) {
return (void*)cmpxchg((jint)exchange_value, (volatile jint*)dest, (jint)compare_value); return (void*)cmpxchg((jint)exchange_value, (volatile jint*)dest, (jint)compare_value, order);
} }
inline jlong Atomic::load(volatile jlong* src) { inline jlong Atomic::load(volatile jlong* src) {

View File

@ -270,7 +270,8 @@ inline void* Atomic::xchg_ptr(void* exchange_value, volatile void* dest) {
inline jint Atomic::cmpxchg(jint exchange_value, inline jint Atomic::cmpxchg(jint exchange_value,
volatile jint* dest, volatile jint* dest,
jint compare_value) { jint compare_value,
cmpxchg_memory_order order) {
#ifdef ARM #ifdef ARM
return arm_compare_and_swap(dest, compare_value, exchange_value); return arm_compare_and_swap(dest, compare_value, exchange_value);
#else #else
@ -284,14 +285,16 @@ inline jint Atomic::cmpxchg(jint exchange_value,
inline jlong Atomic::cmpxchg(jlong exchange_value, inline jlong Atomic::cmpxchg(jlong exchange_value,
volatile jlong* dest, volatile jlong* dest,
jlong compare_value) { jlong compare_value,
cmpxchg_memory_order order) {
return __sync_val_compare_and_swap(dest, compare_value, exchange_value); return __sync_val_compare_and_swap(dest, compare_value, exchange_value);
} }
inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value, inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value,
volatile intptr_t* dest, volatile intptr_t* dest,
intptr_t compare_value) { intptr_t compare_value,
cmpxchg_memory_order order) {
#ifdef ARM #ifdef ARM
return arm_compare_and_swap(dest, compare_value, exchange_value); return arm_compare_and_swap(dest, compare_value, exchange_value);
#else #else
@ -305,11 +308,13 @@ inline intptr_t Atomic::cmpxchg_ptr(intptr_t exchange_value,
inline void* Atomic::cmpxchg_ptr(void* exchange_value, inline void* Atomic::cmpxchg_ptr(void* exchange_value,
volatile void* dest, volatile void* dest,
void* compare_value) { void* compare_value,
cmpxchg_memory_order order) {
return (void *) cmpxchg_ptr((intptr_t) exchange_value, return (void *) cmpxchg_ptr((intptr_t) exchange_value,
(volatile intptr_t*) dest, (volatile intptr_t*) dest,
(intptr_t) compare_value); (intptr_t) compare_value,
order);
} }
inline jlong Atomic::load(volatile jlong* src) { inline jlong Atomic::load(volatile jlong* src) {

Some files were not shown because too many files have changed in this diff Show More