From 512604605cd64840690d3f19806d487568131e10 Mon Sep 17 00:00:00 2001 From: Lana Steuck Date: Tue, 15 Mar 2016 13:48:18 -0700 Subject: [PATCH 1/4] Added tag jdk-9+110 for changeset 428cc83bfc1e --- .hgtags-top-repo | 1 + 1 file changed, 1 insertion(+) diff --git a/.hgtags-top-repo b/.hgtags-top-repo index 831d5d6213f..623af343bef 100644 --- a/.hgtags-top-repo +++ b/.hgtags-top-repo @@ -352,3 +352,4 @@ be58b02c11f90b88c67e4d0e2cb5e4cf2d9b3c57 jdk-9+105 4d65eba233a8730f913734a6804910b842d2cb54 jdk-9+107 c7be2a78c31b3b6132f2f5e9e4b3d3bb1c20245c jdk-9+108 1787bdaabb2b6f4193406e25a50cb0419ea8e8f3 jdk-9+109 +925be13b3740d07a5958ccb5ab3c0ae1baba7055 jdk-9+110 From 007b0fa3db17be62e6a3153e7336c5aba558dc44 Mon Sep 17 00:00:00 2001 From: Lana Steuck Date: Tue, 15 Mar 2016 13:48:21 -0700 Subject: [PATCH 2/4] Added tag jdk-9+110 for changeset 8045e63439dd --- hotspot/.hgtags | 1 + 1 file changed, 1 insertion(+) diff --git a/hotspot/.hgtags b/hotspot/.hgtags index cbe0b715f78..e1c8a7dd0a3 100644 --- a/hotspot/.hgtags +++ b/hotspot/.hgtags @@ -512,3 +512,4 @@ c5f55130b1b69510d9a6f4a3105b58e21cd7ffe1 jdk-9+103 c5146d4da417f76edfc43097d2e2ced042a65b4e jdk-9+107 934f6793f5f7dca44f69b4559d525fa64b31840d jdk-9+108 7e7e50ac4faf19899fc811569e32cfa478759ebb jdk-9+109 +2f5d1578b24060ea06bd1f340a124db95d1475b2 jdk-9+110 From cc661dd1c69398f6fec82695881e64408941a974 Mon Sep 17 00:00:00 2001 From: Alan Bateman Date: Thu, 17 Mar 2016 19:03:53 +0000 Subject: [PATCH 3/4] 8142968: Module System implementation Initial integration of JEP 200, JEP 260, JEP 261, and JEP 282 Co-authored-by: Alex Buckley Co-authored-by: Jonathan Gibbons Co-authored-by: Karen Kinnear Co-authored-by: Mandy Chung Co-authored-by: Mark Reinhold Co-authored-by: Erik Joelsson Co-authored-by: Chris Hegarty Co-authored-by: Christian Tornqvist Co-authored-by: Harold Seigel Co-authored-by: Igor Ignatyev Co-authored-by: James Laskey Co-authored-by: Jean-Francois Denise Co-authored-by: Sundararajan Athijegannathan Reviewed-by: alanb, mchung, tbell --- common/autoconf/basics.m4 | 2 + common/autoconf/boot-jdk.m4 | 109 ++ common/autoconf/bootcycle-spec.gmk.in | 2 + common/autoconf/buildjdk-spec.gmk.in | 148 ++ common/autoconf/configure.ac | 4 + common/autoconf/flags.m4 | 59 +- common/autoconf/generated-configure.sh | 1626 +++++++++++++++++- common/autoconf/jdk-options.m4 | 28 + common/autoconf/platform.m4 | 54 + common/autoconf/source-dirs.m4 | 53 + common/autoconf/spec.gmk.in | 50 +- common/autoconf/toolchain.m4 | 6 + common/bin/compare.sh | 14 +- common/conf/jib-profiles.js | 6 +- make/CheckModules.gmk | 46 - make/CompileJavaModules.gmk | 118 +- make/CopyImportModules.gmk | 86 + make/CreateBuildJdkCopy.gmk | 86 + make/CreateJmods.gmk | 95 ++ make/GenerateModuleDeps.gmk | 64 - make/GensrcModuleInfo.gmk | 154 ++ make/HotspotWrapper.gmk | 7 +- make/Images.gmk | 268 ++- make/Javadoc.gmk | 224 ++- make/Jprt.gmk | 17 +- make/JrtfsJar.gmk | 29 +- make/Main.gmk | 197 ++- make/MainSupport.gmk | 1 - make/StripBinaries.gmk | 38 +- make/common/CORE_PKGS.gmk | 3 +- make/common/JavaCompilation.gmk | 44 +- make/common/MakeBase.gmk | 17 +- make/common/Modules.gmk | 88 +- make/common/NativeCompilation.gmk | 13 +- make/common/SetupJavaCompilers.gmk | 8 +- modules.xml | 2139 ------------------------ test/lib/sun/hotspot/WhiteBox.java | 20 +- 37 files changed, 3301 insertions(+), 2622 deletions(-) create mode 100644 common/autoconf/buildjdk-spec.gmk.in delete mode 100644 make/CheckModules.gmk create mode 100644 make/CopyImportModules.gmk create mode 100644 make/CreateBuildJdkCopy.gmk create mode 100644 make/CreateJmods.gmk delete mode 100644 make/GenerateModuleDeps.gmk create mode 100644 make/GensrcModuleInfo.gmk delete mode 100644 modules.xml diff --git a/common/autoconf/basics.m4 b/common/autoconf/basics.m4 index d748c4f89a1..753d170beb2 100644 --- a/common/autoconf/basics.m4 +++ b/common/autoconf/basics.m4 @@ -843,6 +843,8 @@ AC_DEFUN_ONCE([BASIC_SETUP_OUTPUT_DIR], AC_CONFIG_FILES([$OUTPUT_ROOT/hotspot-spec.gmk:$AUTOCONF_DIR/hotspot-spec.gmk.in]) # The bootcycle-spec.gmk file contains support for boot cycle builds. AC_CONFIG_FILES([$OUTPUT_ROOT/bootcycle-spec.gmk:$AUTOCONF_DIR/bootcycle-spec.gmk.in]) + # The buildjdk-spec.gmk file contains support for building a buildjdk when cross compiling. + AC_CONFIG_FILES([$OUTPUT_ROOT/buildjdk-spec.gmk:$AUTOCONF_DIR/buildjdk-spec.gmk.in]) # The compare.sh is used to compare the build output to other builds. AC_CONFIG_FILES([$OUTPUT_ROOT/compare.sh:$AUTOCONF_DIR/compare.sh.in]) # The generated Makefile knows where the spec.gmk is and where the source is. diff --git a/common/autoconf/boot-jdk.m4 b/common/autoconf/boot-jdk.m4 index 7864ea956d4..91bf548fbf8 100644 --- a/common/autoconf/boot-jdk.m4 +++ b/common/autoconf/boot-jdk.m4 @@ -304,6 +304,18 @@ AC_DEFUN_ONCE([BOOTJDK_SETUP_BOOT_JDK], # When compiling code to be executed by the Boot JDK, force jdk8 compatibility. BOOT_JDK_SOURCETARGET="-source 8 -target 8" AC_SUBST(BOOT_JDK_SOURCETARGET) + + ADD_JVM_ARG_IF_OK([-Xpatch:], dummy, [$JAVA]) + AC_MSG_CHECKING([if Boot JDK supports modules]) + if test "x$JVM_ARG_OK" = "xtrue"; then + AC_MSG_RESULT([yes]) + BOOT_JDK_MODULAR="true" + else + AC_MSG_RESULT([no]) + BOOT_JDK_MODULAR="false" + fi + AC_SUBST(BOOT_JDK_MODULAR) + AC_SUBST(JAVAC_FLAGS) # Check if the boot jdk is 32 or 64 bit @@ -397,3 +409,100 @@ AC_DEFUN_ONCE([BOOTJDK_SETUP_BOOT_JDK_ARGUMENTS], done AC_SUBST(JAVA_TOOL_FLAGS_SMALL) ]) + +# BUILD_JDK: the location of the latest JDK that can run +# on the host system and supports the target class file version +# generated in this JDK build. This variable should only be +# used after the launchers are built. +# + +# Execute the check given as argument, and verify the result. +# If the JDK was previously found, do nothing. +# $1 A command line (typically autoconf macro) to execute +AC_DEFUN([BOOTJDK_CHECK_BUILD_JDK], +[ + if test "x$BUILD_JDK_FOUND" = xno; then + # Execute the test + $1 + + # If previous step claimed to have found a JDK, check it to see if it seems to be valid. + if test "x$BUILD_JDK_FOUND" = xmaybe; then + # Do we have a bin/java? + if test ! -x "$BUILD_JDK/bin/java"; then + AC_MSG_NOTICE([Potential Build JDK found at $BUILD_JDK did not contain bin/java; ignoring]) + BUILD_JDK_FOUND=no + elif test ! -x "$BUILD_JDK/bin/jlink"; then + AC_MSG_NOTICE([Potential Build JDK found at $BUILD_JDK did not contain bin/jlink; ignoring]) + BUILD_JDK_FOUND=no + elif test ! -x "$BUILD_JDK/bin/javac"; then + # Do we have a bin/javac? + AC_MSG_NOTICE([Potential Build JDK found at $BUILD_JDK did not contain bin/javac; ignoring]) + AC_MSG_NOTICE([(This might be a JRE instead of an JDK)]) + BUILD_JDK_FOUND=no + else + # Oh, this is looking good! We probably have found a proper JDK. Is it the correct version? + BUILD_JDK_VERSION=`"$BUILD_JDK/bin/java" -version 2>&1 | head -n 1` + + # Extra M4 quote needed to protect [] in grep expression. + [FOUND_CORRECT_VERSION=`echo $BUILD_JDK_VERSION | grep '\"1\.[9]\.'`] + if test "x$FOUND_CORRECT_VERSION" = x; then + AC_MSG_NOTICE([Potential Boot JDK found at $BUILD_JDK is incorrect JDK version ($BUILD_JDK_VERSION); ignoring]) + AC_MSG_NOTICE([(Your Build JDK must be version 9)]) + BUILD_JDK_FOUND=no + else + # We're done! + BUILD_JDK_FOUND=yes + BASIC_FIXUP_PATH(BUILD_JDK) + AC_MSG_CHECKING([for Build JDK]) + AC_MSG_RESULT([$BUILD_JDK]) + AC_MSG_CHECKING([Build JDK version]) + BUILD_JDK_VERSION=`"$BUILD_JDK/bin/java" -version 2>&1 | $TR '\n\r' ' '` + AC_MSG_RESULT([$BUILD_JDK_VERSION]) + fi # end check jdk version + fi # end check java + fi # end check build jdk found + fi +]) + +# By default the BUILD_JDK is the JDK_OUTPUTDIR. If the target architecture +# is different than the host system doing the build (e.g. cross-compilation), +# a special BUILD_JDK is built as part of the build process. An external +# prebuilt BUILD_JDK can also be supplied. +AC_DEFUN([BOOTJDK_SETUP_BUILD_JDK], +[ + AC_ARG_WITH(build-jdk, [AS_HELP_STRING([--with-build-jdk], + [path to JDK of same version as is being built@<:@the newly built JDK@:>@])]) + + CREATE_BUILDJDK_FOR_HOST=false + BUILD_JDK_FOUND="no" + if test "x$with_build_jdk" != "x"; then + BOOTJDK_CHECK_BUILD_JDK([ + if test "x$with_build_jdk" != x; then + BUILD_JDK=$with_build_jdk + BUILD_JDK_FOUND=maybe + AC_MSG_NOTICE([Found potential Build JDK using configure arguments]) + fi]) + else + if test "x$COMPILE_TYPE" = "xcross"; then + BUILD_JDK="\$(BUILDJDK_OUTPUTDIR)/jdk" + BUILD_JDK_FOUND=yes + CREATE_BUILDJDK=true + AC_MSG_CHECKING([for Build JDK]) + AC_MSG_RESULT([yes, will build it for the host platform]) + else + BUILD_JDK="\$(JDK_OUTPUTDIR)" + BUILD_JDK_FOUND=yes + AC_MSG_CHECKING([for Build JDK]) + AC_MSG_RESULT([yes, will use output dir]) + fi + fi + + if test "x$BUILD_JDK_FOUND" != "xyes"; then + AC_MSG_CHECKING([for Build JDK]) + AC_MSG_RESULT([no]) + AC_MSG_ERROR([Could not find a suitable Build JDK]) + fi + + AC_SUBST(CREATE_BUILDJDK) + AC_SUBST(BUILD_JDK) +]) diff --git a/common/autoconf/bootcycle-spec.gmk.in b/common/autoconf/bootcycle-spec.gmk.in index 0335615795f..c37097d1b1d 100644 --- a/common/autoconf/bootcycle-spec.gmk.in +++ b/common/autoconf/bootcycle-spec.gmk.in @@ -25,6 +25,8 @@ # Support for building boot cycle builds +BOOT_JDK_MODULAR := true + # First include the real base spec.gmk file include @SPEC@ diff --git a/common/autoconf/buildjdk-spec.gmk.in b/common/autoconf/buildjdk-spec.gmk.in new file mode 100644 index 00000000000..f92602edc34 --- /dev/null +++ b/common/autoconf/buildjdk-spec.gmk.in @@ -0,0 +1,148 @@ +# +# 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. Oracle designates this +# particular file as subject to the "Classpath" exception as provided +# by Oracle in the LICENSE file that accompanied this code. +# +# This code is distributed in the hope that it will be useful, but WITHOUT +# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +# version 2 for more details (a copy is included in the LICENSE file that +# accompanied this code). +# +# You should have received a copy of the GNU General Public License version +# 2 along with this work; if not, write to the Free Software Foundation, +# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. +# +# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA +# or visit www.oracle.com if you need additional information or have any +# questions. +# + +# This spec file is used to compile a BUILD_JDK while cross compiling. The +# BUILD_JDK runs on the build/host platform and is of the same version as +# the main build. + +# First include the real base spec.gmk file +include @SPEC@ + +CC := @BUILD_CC@ +CXX := @BUILD_CXX@ +LD := @BUILD_LD@ +AS := @BUILD_AS@ +NM := @BUILD_NM@ +AR := @BUILD_AR@ +OBJCOPY := @BUILD_OBJCOPY@ +STRIP := @BUILD_STRIP@ +SYSROOT_CFLAGS := @BUILD_SYSROOT_CFLAGS@ +SYSROOT_LDFLAGS := @BUILD_SYSROOT_LDFLAGS@ + +# These directories should not be moved to BUILDJDK_OUTPUTDIR +HOTSPOT_OUTPUTDIR := $(patsubst $(BUILD_OUTPUT)%,$(BUILDJDK_OUTPUTDIR)%,$(HOTSPOT_OUTPUTDIR)) +HOTSPOT_DIST := $(patsubst $(BUILD_OUTPUT)%,$(BUILDJDK_OUTPUTDIR)%,$(HOTSPOT_DIST)) +SUPPORT_OUTPUTDIR := $(patsubst $(BUILD_OUTPUT)%,$(BUILDJDK_OUTPUTDIR)%,$(SUPPORT_OUTPUTDIR)) +JDK_OUTPUTDIR := $(patsubst $(BUILD_OUTPUT)%,$(BUILDJDK_OUTPUTDIR)%,$(JDK_OUTPUTDIR)) + +OPENJDK_BUILD_CPU_LEGACY := @OPENJDK_BUILD_CPU_LEGACY@ +OPENJDK_BUILD_CPU_LEGACY_LIB := @OPENJDK_BUILD_CPU_LEGACY_LIB@ +OPENJDK_BUILD_CPU_LIBDIR := @OPENJDK_BUILD_CPU_LIBDIR@ +OPENJDK_TARGET_CPU_LIBDIR := @OPENJDK_BUILD_CPU_LIBDIR@ +OPENJDK_TARGET_CPU := @OPENJDK_BUILD_CPU@ +OPENJDK_TARGET_CPU_ARCH := @OPENJDK_BUILD_CPU_ARCH@ +OPENJDK_TARGET_CPU_BITS := @OPENJDK_BUILD_CPU_BITS@ +OPENJDK_TARGET_CPU_ENDIAN := @OPENJDK_BUILD_CPU_ENDIAN@ +OPENJDK_TARGET_CPU_LEGACY := @OPENJDK_BUILD_CPU_LEGACY@ + +CFLAGS_JDKLIB := @OPENJDK_BUILD_CFLAGS_JDKLIB@ +CXXFLAGS_JDKLIB := @OPENJDK_BUILD_CXXFLAGS_JDKLIB@ +LDFLAGS_JDKLIB := @OPENJDK_BUILD_LDFLAGS_JDKLIB@ +CFLAGS_JDKEXE := @OPENJDK_BUILD_CFLAGS_JDKEXE@ +CXXFLAGS_JDKEXE := @OPENJDK_BUILD_CXXFLAGS_JDKEXE@ +LDFLAGS_JDKEXE := @OPENJDK_BUILD_LDFLAGS_JDKEXE@ +OPENJDK_TARGET_CPU_JLI_CFLAGS := @OPENJDK_BUILD_CPU_JLI_CFLAGS@ + +# The compiler for the build platform is likely not warning compatible with the official +# compiler. +WARNINGS_AS_ERRORS := false +DISABLE_WARNING_PREFIX := @BUILD_CC_DISABLE_WARNING_PREFIX@ + +# Save speed and disk space by not enabling debug symbols for the buildjdk +ENABLE_DEBUG_SYMBOLS := false + +#################################################### +# +# Legacy Hotspot support + +# Legacy setting: OPT or DBG +VARIANT := OPT +# Legacy setting: true or false +FASTDEBUG := false +# Legacy setting: debugging the class files? +DEBUG_CLASSFILES := false + +# Some users still set EXTRA_*FLAGS on the make command line. Must +# make sure to override that when building buildjdk. +override EXTRA_CFLAGS := +override EXTRA_CXXFLAGS := +override EXTRA_LDFLAGS := + +# The HOSTCC/HOSTCXX is Hotspot terminology for the BUILD_CC/BUILD_CXX, i.e. the +# compiler that produces code that can be run on the build platform. +HOSTCC := $(BUILD_CC) +HOSTCXX := $(BUILD_CXX) + +# Old name for OPENJDK_TARGET_OS (aix,bsd,hpux,linux,macosx,solaris,windows etc) +PLATFORM := $(OPENJDK_BUILD_OS) +# 32 or 64 bit +ARCH_DATA_MODEL := $(OPENJDK_BUILD_CPU_BITS) + +ALT_BOOTDIR := $(BOOT_JDK) +# Yet another name for arch used for an extra subdir below the jvm lib. +# Uses i386 and amd64, instead of x86 and x86_64. +LIBARCH := @OPENJDK_BUILD_CPU_LEGACY_LIB@ +# Set the cpu architecture. Some users still set ARCH on the make command line. Must +# make sure to override that when building buildjdk. +override ARCH := $(OPENJDK_BUILD_CPU_ARCH) +# Legacy setting for building for a 64 bit machine. +# If yes then this expands to _LP64 := 1 +ifeq ($(OPENJDK_BUILD_CPU_BITS), 64) + _LP64 := 1 +endif + +ALT_OUTPUTDIR := $(HOTSPOT_OUTPUTDIR) +ALT_EXPORT_PATH := $(HOTSPOT_DIST) + +JVM_INTERPRETER := @JVM_INTERPRETER@ +ifeq ($(JVM_INTERPRETER), cpp) + CC_INTERP=true +endif + +HOTSPOT_MAKE_ARGS := product docs export_product +# Control wether Hotspot runs Queens test after building +TEST_IN_BUILD := false + +USE_PRECOMPILED_HEADER := @USE_PRECOMPILED_HEADER@ + +# Hotspot expects the variable FULL_DEBUG_SYMBOLS=1/0 to control debug symbols +# creation. +FULL_DEBUG_SYMBOLS := 0 +ZIP_DEBUGINFO_FILES := 0 +# Disable stripping +STRIP_POLICY := none + +JVM_VARIANTS := server +JVM_VARIANT_SERVER := true +JVM_VARIANT_CLIENT := false +JVM_VARIANT_MINIMAL1 := false +JVM_VARIANT_KERNEL := false +JVM_VARIANT_ZERO := false +JVM_VARIANT_ZEROSHARK := false +JVM_VARIANT_CORE := false + +# Sneak this in via the spec.gmk file, since we don't want to mess around too much with the Hotspot make files. +# This is needed to get the LOG setting to work properly. +include $(SRC_ROOT)/make/common/MakeBase.gmk diff --git a/common/autoconf/configure.ac b/common/autoconf/configure.ac index 127d9088555..c7bab7c53a8 100644 --- a/common/autoconf/configure.ac +++ b/common/autoconf/configure.ac @@ -134,6 +134,7 @@ BASIC_SETUP_DEFAULT_MAKE_TARGET # We need build & target for this. JDKOPT_SETUP_JDK_OPTIONS +JDKOPT_SETUP_JLINK_OPTIONS HOTSPOT_SETUP_HOTSPOT_OPTIONS JDKVER_SETUP_JDK_VERSION_NUMBERS @@ -144,6 +145,7 @@ JDKVER_SETUP_JDK_VERSION_NUMBERS ############################################################################### BOOTJDK_SETUP_BOOT_JDK +BOOTJDK_SETUP_BUILD_JDK ############################################################################### # @@ -155,6 +157,8 @@ SRCDIRS_SETUP_TOPDIRS SRCDIRS_SETUP_ALTERNATIVE_TOPDIRS SRCDIRS_SETUP_OUTPUT_DIRS +SRCDIRS_SETUP_IMPORT_MODULES + ############################################################################### # # Setup the toolchain (compilers etc), i.e. tools used to compile and process diff --git a/common/autoconf/flags.m4 b/common/autoconf/flags.m4 index 6b08cbb0a4f..9ea5eaa2a9f 100644 --- a/common/autoconf/flags.m4 +++ b/common/autoconf/flags.m4 @@ -689,9 +689,6 @@ AC_DEFUN_ONCE([FLAGS_SETUP_COMPILER_FLAGS_FOR_JDK], ;; esac - # Setup LP64 - COMMON_CCXXFLAGS_JDK="$COMMON_CCXXFLAGS_JDK $ADD_LP64" - # Set some common defines. These works for all compilers, but assume # -D is universally accepted. @@ -722,7 +719,12 @@ AC_DEFUN_ONCE([FLAGS_SETUP_COMPILER_FLAGS_FOR_JDK], COMMON_CCXXFLAGS_JDK="$COMMON_CCXXFLAGS_JDK -D$OPENJDK_TARGET_OS_UPPERCASE" # Setup target CPU - COMMON_CCXXFLAGS_JDK="$COMMON_CCXXFLAGS_JDK -DARCH='\"$OPENJDK_TARGET_CPU_LEGACY\"' -D$OPENJDK_TARGET_CPU_LEGACY" + OPENJDK_TARGET_CCXXFLAGS_JDK="$OPENJDK_TARGET_CCXXFLAGS_JDK \ + $ADD_LP64 \ + -DARCH='\"$OPENJDK_TARGET_CPU_LEGACY\"' -D$OPENJDK_TARGET_CPU_LEGACY" + OPENJDK_BUILD_CCXXFLAGS_JDK="$OPENJDK_BUILD_CCXXFLAGS_JDK \ + $OPENJDK_BUILD_ADD_LP64 \ + -DARCH='\"$OPENJDK_BUILD_CPU_LEGACY\"' -D$OPENJDK_BUILD_CPU_LEGACY" # Setup debug/release defines if test "x$DEBUG_LEVEL" = xrelease; then @@ -766,17 +768,35 @@ AC_DEFUN_ONCE([FLAGS_SETUP_COMPILER_FLAGS_FOR_JDK], -I${JDK_TOPDIR}/src/java.base/$OPENJDK_TARGET_OS_TYPE/native/libjava" # The shared libraries are compiled using the picflag. - CFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $CFLAGS_JDK $PICFLAG $CFLAGS_JDKLIB_EXTRA" - CXXFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $CXXFLAGS_JDK $PICFLAG $CXXFLAGS_JDKLIB_EXTRA" + CFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $OPENJDK_TARGET_CCXXFLAGS_JDK \ + $CFLAGS_JDK $EXTRA_CFLAGS_JDK $PICFLAG $CFLAGS_JDKLIB_EXTRA" + CXXFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $OPENJDK_TARGET_CCXXFLAGS_JDK \ + $CXXFLAGS_JDK $EXTRA_CXXFLAGS_JDK $PICFLAG $CXXFLAGS_JDKLIB_EXTRA" # Executable flags - CFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $CFLAGS_JDK" - CXXFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $CXXFLAGS_JDK" + CFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $OPENJDK_TARGET_CCXXFLAGS_JDK \ + $CFLAGS_JDK $EXTRA_CFLAGS_JDK" + CXXFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $OPENJDK_TARGET_CCXXFLAGS_JDK \ + $CXXFLAGS_JDK $EXTRA_CXXFLAGS_JDK" + + # The corresponding flags for building for the build platform. This is still an + # approximation, we only need something that runs on this machine when cross + # compiling the product. + OPENJDK_BUILD_CFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $OPENJDK_BUILD_CCXXFLAGS_JDK \ + $PICFLAG $CFLAGS_JDKLIB_EXTRA" + OPENJDK_BUILD_CXXFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $OPENJDK_BUILD_CCXXFLAGS_JDK \ + $PICFLAG $CXXFLAGS_JDKLIB_EXTRA" + OPENJDK_BUILD_CFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $OPENJDK_BUILD_CCXXFLAGS_JDK" + OPENJDK_BUILD_CXXFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $OPENJDK_BUILD_CCXXFLAGS_JDK" AC_SUBST(CFLAGS_JDKLIB) AC_SUBST(CFLAGS_JDKEXE) AC_SUBST(CXXFLAGS_JDKLIB) AC_SUBST(CXXFLAGS_JDKEXE) + AC_SUBST(OPENJDK_BUILD_CFLAGS_JDKLIB) + AC_SUBST(OPENJDK_BUILD_CFLAGS_JDKEXE) + AC_SUBST(OPENJDK_BUILD_CXXFLAGS_JDKLIB) + AC_SUBST(OPENJDK_BUILD_CXXFLAGS_JDKEXE) # Flags for compiling test libraries CFLAGS_TESTLIB="$COMMON_CCXXFLAGS_JDK $CFLAGS_JDK $PICFLAG $CFLAGS_JDKLIB_EXTRA" @@ -872,6 +892,9 @@ AC_DEFUN_ONCE([FLAGS_SETUP_COMPILER_FLAGS_FOR_JDK], LDFLAGS_JDKEXE="$LDFLAGS_JDKEXE -Wl,--allow-shlib-undefined" fi + OPENJDK_BUILD_LDFLAGS_JDKEXE="${LDFLAGS_JDKEXE}" + LDFLAGS_JDKEXE="${LDFLAGS_JDKEXE} ${EXTRA_LDFLAGS_JDK}" + # Customize LDFLAGS for libs LDFLAGS_JDKLIB="${LDFLAGS_JDK}" @@ -882,30 +905,39 @@ AC_DEFUN_ONCE([FLAGS_SETUP_COMPILER_FLAGS_FOR_JDK], JDKLIB_LIBS="" else LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} \ - -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}" + -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)" # On some platforms (mac) the linker warns about non existing -L dirs. # Add server first if available. Linking aginst client does not always produce the same results. # Only add client dir if client is being built. Add minimal (note not minimal1) if only building minimal1. # Default to server for other variants. if test "x$JVM_VARIANT_SERVER" = xtrue; then - LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}/server" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/server" elif test "x$JVM_VARIANT_CLIENT" = xtrue; then - LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}/client" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/client" elif test "x$JVM_VARIANT_MINIMAL1" = xtrue; then - LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}/minimal" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/minimal" else - LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}/server" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/server" fi JDKLIB_LIBS="-ljava -ljvm" if test "x$TOOLCHAIN_TYPE" = xsolstudio; then JDKLIB_LIBS="$JDKLIB_LIBS -lc" fi + + # When building a buildjdk, it's always only the server variant + OPENJDK_BUILD_LDFLAGS_JDKLIB="${OPENJDK_BUILD_LDFLAGS_JDKLIB} \ + -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/server" fi + OPENJDK_BUILD_LDFLAGS_JDKLIB="${OPENJDK_BUILD_LDFLAGS_JDKLIB} ${LDFLAGS_JDKLIB}" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} ${EXTRA_LDFLAGS_JDK}" + AC_SUBST(LDFLAGS_JDKLIB) AC_SUBST(LDFLAGS_JDKEXE) + AC_SUBST(OPENJDK_BUILD_LDFLAGS_JDKLIB) + AC_SUBST(OPENJDK_BUILD_LDFLAGS_JDKEXE) AC_SUBST(JDKLIB_LIBS) AC_SUBST(JDKEXE_LIBS) AC_SUBST(LDFLAGS_CXX_JDK) @@ -1075,5 +1107,6 @@ AC_DEFUN_ONCE([FLAGS_SETUP_COMPILER_FLAGS_MISC], ;; esac AC_SUBST(DISABLE_WARNING_PREFIX) + AC_SUBST(BUILD_CC_DISABLE_WARNING_PREFIX) AC_SUBST(CFLAGS_WARNINGS_ARE_ERRORS) ]) diff --git a/common/autoconf/generated-configure.sh b/common/autoconf/generated-configure.sh index 58beee04e52..e4ea273f57f 100644 --- a/common/autoconf/generated-configure.sh +++ b/common/autoconf/generated-configure.sh @@ -694,6 +694,7 @@ ZIP_EXTERNAL_DEBUG_SYMBOLS COPY_DEBUG_SYMBOLS COMPILE_WITH_DEBUG_SYMBOLS CFLAGS_WARNINGS_ARE_ERRORS +BUILD_CC_DISABLE_WARNING_PREFIX DISABLE_WARNING_PREFIX HOTSPOT_SET_WARNINGS_AS_ERRORS WARNINGS_AS_ERRORS @@ -705,12 +706,18 @@ LDFLAGS_HASH_STYLE LDFLAGS_CXX_JDK JDKEXE_LIBS JDKLIB_LIBS +OPENJDK_BUILD_LDFLAGS_JDKEXE +OPENJDK_BUILD_LDFLAGS_JDKLIB LDFLAGS_JDKEXE LDFLAGS_JDKLIB CXXFLAGS_TESTEXE CXXFLAGS_TESTLIB CFLAGS_TESTEXE CFLAGS_TESTLIB +OPENJDK_BUILD_CXXFLAGS_JDKEXE +OPENJDK_BUILD_CXXFLAGS_JDKLIB +OPENJDK_BUILD_CFLAGS_JDKEXE +OPENJDK_BUILD_CFLAGS_JDKLIB CXXFLAGS_JDKEXE CXXFLAGS_JDKLIB CFLAGS_JDKEXE @@ -759,6 +766,8 @@ HOTSPOT_MT BUILD_AS BUILD_LDCXX BUILD_LD +BUILD_STRIP +BUILD_OBJCOPY BUILD_AR BUILD_NM BUILD_CXX @@ -821,6 +830,12 @@ SHARED_LIBRARY_SUFFIX LIBRARY_PREFIX TOOLCHAIN_TYPE STATIC_BUILD +IMPORT_MODULES_MAKE +IMPORT_MODULES_SRC +IMPORT_MODULES_CONF +IMPORT_MODULES_LIBS +IMPORT_MODULES_CMDS +IMPORT_MODULES_CLASSES BUILD_HOTSPOT HOTSPOT_DIST BUILD_OUTPUT @@ -831,8 +846,11 @@ JAXWS_TOPDIR JAXP_TOPDIR CORBA_TOPDIR LANGTOOLS_TOPDIR +BUILD_JDK +CREATE_BUILDJDK BOOT_JDK_BITS JAVAC_FLAGS +BOOT_JDK_MODULAR BOOT_JDK_SOURCETARGET JARSIGNER JAR @@ -862,6 +880,7 @@ PRODUCT_SUFFIX PRODUCT_NAME LAUNCHER_NAME TEST_IN_BUILD +JLINK_KEEP_PACKAGED_MODULES COPYRIGHT_YEAR COMPRESS_JARS INCLUDE_SA @@ -930,9 +949,13 @@ ZERO_ARCHDEF DEFINE_CROSS_COMPILE_ARCH LP64 OPENJDK_TARGET_OS_EXPORT_DIR +OPENJDK_BUILD_CPU_JLI_CFLAGS OPENJDK_TARGET_CPU_JLI_CFLAGS OPENJDK_TARGET_CPU_OSARCH OPENJDK_TARGET_CPU_ISADIR +OPENJDK_BUILD_CPU_LIBDIR +OPENJDK_BUILD_CPU_LEGACY_LIB +OPENJDK_BUILD_CPU_LEGACY OPENJDK_TARGET_CPU_LIBDIR OPENJDK_TARGET_CPU_LEGACY_LIB OPENJDK_TARGET_CPU_LEGACY @@ -1080,6 +1103,7 @@ enable_headful with_cacerts_file enable_unlimited_crypto with_copyright_year +enable_keep_packaged_modules enable_hotspot_test_in_build with_milestone with_update_version @@ -1094,6 +1118,7 @@ with_version_minor with_version_security with_version_patch with_boot_jdk +with_build_jdk with_add_source_root with_override_source_root with_adds_and_overrides @@ -1105,6 +1130,7 @@ with_override_hotspot with_override_nashorn with_override_jdk with_import_hotspot +with_import_modules enable_static_build with_toolchain_type with_extra_cflags @@ -1248,6 +1274,8 @@ BUILD_CC BUILD_CXX BUILD_NM BUILD_AR +BUILD_OBJCOPY +BUILD_STRIP JTREGEXE XMKMF FREETYPE_CFLAGS @@ -1890,6 +1918,8 @@ Optional Features: support) [enabled] --enable-unlimited-crypto Enable unlimited crypto policy [disabled] + --disable-keep-packaged-modules + Do not keep packaged modules in jdk image [enable] --enable-hotspot-test-in-build run the Queens test after Hotspot build [disabled] --enable-static-build enable static library build [disabled] @@ -1973,6 +2003,8 @@ Optional Packages: --with-version-patch Set version 'PATCH' field (fourth number) [not specified] --with-boot-jdk path to Boot JDK (used to bootstrap build) [probed] + --with-build-jdk path to JDK of same version as is being built[the + newly built JDK] --with-add-source-root Deprecated. Option is kept for backwards compatibility and is ignored --with-override-source-root @@ -1999,6 +2031,8 @@ Optional Packages: --with-import-hotspot import hotspot binaries from this jdk image or hotspot build dist dir instead of building from source + --with-import-modules import a set of prebuilt modules either as a zip + file or an exploded directory --with-toolchain-type the toolchain type (or family) to use, use '--help' to show possible values [platform dependent] --with-extra-cflags extra flags to be used when compiling jdk c-files @@ -2167,6 +2201,9 @@ Some influential environment variables: BUILD_CXX Override default value for BUILD_CXX BUILD_NM Override default value for BUILD_NM BUILD_AR Override default value for BUILD_AR + BUILD_OBJCOPY + Override default value for BUILD_OBJCOPY + BUILD_STRIP Override default value for BUILD_STRIP JTREGEXE Override default value for JTREGEXE XMKMF Path to xmkmf, Makefile generator for X Window System FREETYPE_CFLAGS @@ -3777,6 +3814,23 @@ ac_configure="$SHELL $ac_aux_dir/configure" # Please don't use this var. +# BUILD_JDK: the location of the latest JDK that can run +# on the host system and supports the target class file version +# generated in this JDK build. This variable should only be +# used after the launchers are built. +# + +# Execute the check given as argument, and verify the result. +# If the JDK was previously found, do nothing. +# $1 A command line (typically autoconf macro) to execute + + +# By default the BUILD_JDK is the JDK_OUTPUTDIR. If the target architecture +# is different than the host system doing the build (e.g. cross-compilation), +# a special BUILD_JDK is built as part of the build process. An external +# prebuilt BUILD_JDK can also be supplied. + + # # Copyright (c) 2011, 2016, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. @@ -4192,6 +4246,13 @@ pkgadd_help() { # +################################################################################ +# +# jlink options. +# We always keep packaged modules in JDK image. +# + + # # Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. @@ -4628,6 +4689,12 @@ pkgadd_help() { +################################################################################ +# Define a mechanism for importing extra prebuilt modules +# + + + # # Copyright (c) 2011, 2016, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. @@ -4862,7 +4929,7 @@ VS_SDK_PLATFORM_NAME_2013= #CUSTOM_AUTOCONF_INCLUDE # Do not change or remove the following line, it is needed for consistency checks: -DATE_WHEN_GENERATED=1457684806 +DATE_WHEN_GENERATED=1458008154 ############################################################################### # @@ -15157,6 +15224,37 @@ $as_echo "$COMPILE_TYPE" >&6; } fi + # Now do the same for OPENJDK_BUILD_CPU... + # Also store the legacy naming of the cpu. + # Ie i586 and amd64 instead of x86 and x86_64 + OPENJDK_BUILD_CPU_LEGACY="$OPENJDK_BUILD_CPU" + if test "x$OPENJDK_BUILD_CPU" = xx86; then + OPENJDK_BUILD_CPU_LEGACY="i586" + elif test "x$OPENJDK_BUILD_OS" != xmacosx && test "x$OPENJDK_BUILD_CPU" = xx86_64; then + # On all platforms except MacOSX replace x86_64 with amd64. + OPENJDK_BUILD_CPU_LEGACY="amd64" + fi + + + # And the second legacy naming of the cpu. + # Ie i386 and amd64 instead of x86 and x86_64. + OPENJDK_BUILD_CPU_LEGACY_LIB="$OPENJDK_BUILD_CPU" + if test "x$OPENJDK_BUILD_CPU" = xx86; then + OPENJDK_BUILD_CPU_LEGACY_LIB="i386" + elif test "x$OPENJDK_BUILD_CPU" = xx86_64; then + OPENJDK_BUILD_CPU_LEGACY_LIB="amd64" + fi + + + # This is the name of the cpu (but using i386 and amd64 instead of + # x86 and x86_64, respectively), preceeded by a /, to be used when + # locating libraries. On macosx, it's empty, though. + OPENJDK_BUILD_CPU_LIBDIR="/$OPENJDK_BUILD_CPU_LEGACY_LIB" + if test "x$OPENJDK_BUILD_OS" = xmacosx; then + OPENJDK_BUILD_CPU_LIBDIR="" + fi + + # OPENJDK_TARGET_CPU_ISADIR is normally empty. On 64-bit Solaris systems, it is set to # /amd64 or /sparcv9. This string is appended to some library paths, like this: # /usr/lib${OPENJDK_TARGET_CPU_ISADIR}/libexample.so @@ -15199,6 +15297,24 @@ $as_echo "$COMPILE_TYPE" >&6; } fi + OPENJDK_BUILD_CPU_JLI="$OPENJDK_BUILD_CPU" + if test "x$OPENJDK_BUILD_CPU" = xx86; then + OPENJDK_BUILD_CPU_JLI="i386" + elif test "x$OPENJDK_BUILD_OS" != xmacosx && test "x$OPENJDK_BUILD_CPU" = xx86_64; then + # On all platforms except macosx, we replace x86_64 with amd64. + OPENJDK_BUILD_CPU_JLI="amd64" + fi + # Now setup the -D flags for building libjli. + OPENJDK_BUILD_CPU_JLI_CFLAGS="-DLIBARCHNAME='\"$OPENJDK_BUILD_CPU_JLI\"'" + if test "x$OPENJDK_BUILD_OS" = xsolaris; then + if test "x$OPENJDK_BUILD_CPU_ARCH" = xsparc; then + OPENJDK_BUILD_CPU_JLI_CFLAGS="$OPENJDK_BUILD_CPU_JLI_CFLAGS -DLIBARCH32NAME='\"sparc\"' -DLIBARCH64NAME='\"sparcv9\"'" + elif test "x$OPENJDK_BUILD_CPU_ARCH" = xx86; then + OPENJDK_BUILD_CPU_JLI_CFLAGS="$OPENJDK_BUILD_CPU_JLI_CFLAGS -DLIBARCH32NAME='\"i386\"' -DLIBARCH64NAME='\"amd64\"'" + fi + fi + + if test "x$OPENJDK_TARGET_OS" = xmacosx; then OPENJDK_TARGET_OS_EXPORT_DIR=macosx else @@ -15216,6 +15332,11 @@ $as_echo "$COMPILE_TYPE" >&6; } fi LP64=$A_LP64 + if test "x$OPENJDK_BUILD_CPU_BITS" = x64; then + if test "x$OPENJDK_BUILD_OS" = xlinux || test "x$OPENJDK_BUILD_OS" = xmacosx; then + OPENJDK_BUILD_ADD_LP64="-D_LP64=1" + fi + fi if test "x$COMPILE_TYPE" = "xcross"; then # FIXME: ... or should this include reduced builds..? @@ -16701,6 +16822,9 @@ $as_echo "$as_me: The path of OUTPUT_ROOT, which resolves as \"$path\", is inval # The bootcycle-spec.gmk file contains support for boot cycle builds. ac_config_files="$ac_config_files $OUTPUT_ROOT/bootcycle-spec.gmk:$AUTOCONF_DIR/bootcycle-spec.gmk.in" + # The buildjdk-spec.gmk file contains support for building a buildjdk when cross compiling. + ac_config_files="$ac_config_files $OUTPUT_ROOT/buildjdk-spec.gmk:$AUTOCONF_DIR/buildjdk-spec.gmk.in" + # The compare.sh is used to compare the build output to other builds. ac_config_files="$ac_config_files $OUTPUT_ROOT/compare.sh:$AUTOCONF_DIR/compare.sh.in" @@ -23299,6 +23423,35 @@ fi + # Check whether --enable-keep-packaged-modules was given. +if test "${enable_keep_packaged_modules+set}" = set; then : + enableval=$enable_keep_packaged_modules; +fi + + + if test "x$enable_keep_packaged_modules" = "xyes"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: checking if packaged modules are kept" >&5 +$as_echo_n "checking if packaged modules are kept... " >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: result: yes" >&5 +$as_echo "yes" >&6; } + JLINK_KEEP_PACKAGED_MODULES=true + elif test "x$enable_keep_packaged_modules" = "xno"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: checking if packaged modules are kept" >&5 +$as_echo_n "checking if packaged modules are kept... " >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } + JLINK_KEEP_PACKAGED_MODULES=false + elif test "x$enable_keep_packaged_modules" = "x"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: yes (default)" >&5 +$as_echo "yes (default)" >&6; } + JLINK_KEEP_PACKAGED_MODULES=true + else + as_fn_error $? "--enable-keep-packaged-modules accepts no argument" "$LINENO" 5 + fi + + + + # Control wether Hotspot runs Queens test after build. # Check whether --enable-hotspot-test-in-build was given. if test "${enable_hotspot_test_in_build+set}" = set; then : @@ -29596,6 +29749,35 @@ $as_echo "$tool_specified" >&6; } + $ECHO "Check if jvm arg is ok: -Xpatch:" >&5 + $ECHO "Command: $JAVA -Xpatch: -version" >&5 + OUTPUT=`$JAVA -Xpatch: -version 2>&1` + FOUND_WARN=`$ECHO "$OUTPUT" | grep -i warn` + FOUND_VERSION=`$ECHO $OUTPUT | grep " version \""` + if test "x$FOUND_VERSION" != x && test "x$FOUND_WARN" = x; then + dummy="$dummy -Xpatch:" + JVM_ARG_OK=true + else + $ECHO "Arg failed:" >&5 + $ECHO "$OUTPUT" >&5 + JVM_ARG_OK=false + fi + + { $as_echo "$as_me:${as_lineno-$LINENO}: checking if Boot JDK supports modules" >&5 +$as_echo_n "checking if Boot JDK supports modules... " >&6; } + if test "x$JVM_ARG_OK" = "xtrue"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: yes" >&5 +$as_echo "yes" >&6; } + BOOT_JDK_MODULAR="true" + else + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } + BOOT_JDK_MODULAR="false" + fi + + + + # Check if the boot jdk is 32 or 64 bit if "$JAVA" -d64 -version > /dev/null 2>&1; then BOOT_JDK_BITS="64" @@ -29609,6 +29791,237 @@ $as_echo "$BOOT_JDK_BITS" >&6; } + +# Check whether --with-build-jdk was given. +if test "${with_build_jdk+set}" = set; then : + withval=$with_build_jdk; +fi + + + CREATE_BUILDJDK_FOR_HOST=false + BUILD_JDK_FOUND="no" + if test "x$with_build_jdk" != "x"; then + + if test "x$BUILD_JDK_FOUND" = xno; then + # Execute the test + + if test "x$with_build_jdk" != x; then + BUILD_JDK=$with_build_jdk + BUILD_JDK_FOUND=maybe + { $as_echo "$as_me:${as_lineno-$LINENO}: Found potential Build JDK using configure arguments" >&5 +$as_echo "$as_me: Found potential Build JDK using configure arguments" >&6;} + fi + + # If previous step claimed to have found a JDK, check it to see if it seems to be valid. + if test "x$BUILD_JDK_FOUND" = xmaybe; then + # Do we have a bin/java? + if test ! -x "$BUILD_JDK/bin/java"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: Potential Build JDK found at $BUILD_JDK did not contain bin/java; ignoring" >&5 +$as_echo "$as_me: Potential Build JDK found at $BUILD_JDK did not contain bin/java; ignoring" >&6;} + BUILD_JDK_FOUND=no + elif test ! -x "$BUILD_JDK/bin/jlink"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: Potential Build JDK found at $BUILD_JDK did not contain bin/jlink; ignoring" >&5 +$as_echo "$as_me: Potential Build JDK found at $BUILD_JDK did not contain bin/jlink; ignoring" >&6;} + BUILD_JDK_FOUND=no + elif test ! -x "$BUILD_JDK/bin/javac"; then + # Do we have a bin/javac? + { $as_echo "$as_me:${as_lineno-$LINENO}: Potential Build JDK found at $BUILD_JDK did not contain bin/javac; ignoring" >&5 +$as_echo "$as_me: Potential Build JDK found at $BUILD_JDK did not contain bin/javac; ignoring" >&6;} + { $as_echo "$as_me:${as_lineno-$LINENO}: (This might be a JRE instead of an JDK)" >&5 +$as_echo "$as_me: (This might be a JRE instead of an JDK)" >&6;} + BUILD_JDK_FOUND=no + else + # Oh, this is looking good! We probably have found a proper JDK. Is it the correct version? + BUILD_JDK_VERSION=`"$BUILD_JDK/bin/java" -version 2>&1 | head -n 1` + + # Extra M4 quote needed to protect [] in grep expression. + FOUND_CORRECT_VERSION=`echo $BUILD_JDK_VERSION | grep '\"1\.[9]\.'` + if test "x$FOUND_CORRECT_VERSION" = x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: Potential Boot JDK found at $BUILD_JDK is incorrect JDK version ($BUILD_JDK_VERSION); ignoring" >&5 +$as_echo "$as_me: Potential Boot JDK found at $BUILD_JDK is incorrect JDK version ($BUILD_JDK_VERSION); ignoring" >&6;} + { $as_echo "$as_me:${as_lineno-$LINENO}: (Your Build JDK must be version 9)" >&5 +$as_echo "$as_me: (Your Build JDK must be version 9)" >&6;} + BUILD_JDK_FOUND=no + else + # We're done! + BUILD_JDK_FOUND=yes + + # Only process if variable expands to non-empty + + if test "x$BUILD_JDK" != x; then + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + + # Input might be given as Windows format, start by converting to + # unix format. + path="$BUILD_JDK" + new_path=`$CYGPATH -u "$path"` + + # Cygwin tries to hide some aspects of the Windows file system, such that binaries are + # named .exe but called without that suffix. Therefore, "foo" and "foo.exe" are considered + # the same file, most of the time (as in "test -f"). But not when running cygpath -s, then + # "foo.exe" is OK but "foo" is an error. + # + # This test is therefore slightly more accurate than "test -f" to check for file precense. + # It is also a way to make sure we got the proper file name for the real test later on. + test_shortpath=`$CYGPATH -s -m "$new_path" 2> /dev/null` + if test "x$test_shortpath" = x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_JDK, which resolves as \"$path\", is invalid." >&5 +$as_echo "$as_me: The path of BUILD_JDK, which resolves as \"$path\", is invalid." >&6;} + as_fn_error $? "Cannot locate the the path of BUILD_JDK" "$LINENO" 5 + fi + + # Call helper function which possibly converts this using DOS-style short mode. + # If so, the updated path is stored in $new_path. + + input_path="$new_path" + # Check if we need to convert this using DOS-style short mode. If the path + # contains just simple characters, use it. Otherwise (spaces, weird characters), + # take no chances and rewrite it. + # Note: m4 eats our [], so we need to use [ and ] instead. + has_forbidden_chars=`$ECHO "$input_path" | $GREP [^-._/a-zA-Z0-9]` + if test "x$has_forbidden_chars" != x; then + # Now convert it to mixed DOS-style, short mode (no spaces, and / instead of \) + shortmode_path=`$CYGPATH -s -m -a "$input_path"` + path_after_shortmode=`$CYGPATH -u "$shortmode_path"` + if test "x$path_after_shortmode" != "x$input_to_shortpath"; then + # Going to short mode and back again did indeed matter. Since short mode is + # case insensitive, let's make it lowercase to improve readability. + shortmode_path=`$ECHO "$shortmode_path" | $TR 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + # Now convert it back to Unix-style (cygpath) + input_path=`$CYGPATH -u "$shortmode_path"` + new_path="$input_path" + fi + fi + + test_cygdrive_prefix=`$ECHO $input_path | $GREP ^/cygdrive/` + if test "x$test_cygdrive_prefix" = x; then + # As a simple fix, exclude /usr/bin since it's not a real path. + if test "x`$ECHO $new_path | $GREP ^/usr/bin/`" = x; then + # The path is in a Cygwin special directory (e.g. /home). We need this converted to + # a path prefixed by /cygdrive for fixpath to work. + new_path="$CYGWIN_ROOT_PATH$input_path" + fi + fi + + + if test "x$path" != "x$new_path"; then + BUILD_JDK="$new_path" + { $as_echo "$as_me:${as_lineno-$LINENO}: Rewriting BUILD_JDK to \"$new_path\"" >&5 +$as_echo "$as_me: Rewriting BUILD_JDK to \"$new_path\"" >&6;} + fi + + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + + path="$BUILD_JDK" + has_colon=`$ECHO $path | $GREP ^.:` + new_path="$path" + if test "x$has_colon" = x; then + # Not in mixed or Windows style, start by that. + new_path=`cmd //c echo $path` + fi + + + input_path="$new_path" + # Check if we need to convert this using DOS-style short mode. If the path + # contains just simple characters, use it. Otherwise (spaces, weird characters), + # take no chances and rewrite it. + # Note: m4 eats our [], so we need to use [ and ] instead. + has_forbidden_chars=`$ECHO "$input_path" | $GREP [^-_/:a-zA-Z0-9]` + if test "x$has_forbidden_chars" != x; then + # Now convert it to mixed DOS-style, short mode (no spaces, and / instead of \) + new_path=`cmd /c "for %A in (\"$input_path\") do @echo %~sA"|$TR \\\\\\\\ / | $TR 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + fi + + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + if test "x$path" != "x$new_path"; then + BUILD_JDK="$new_path" + { $as_echo "$as_me:${as_lineno-$LINENO}: Rewriting BUILD_JDK to \"$new_path\"" >&5 +$as_echo "$as_me: Rewriting BUILD_JDK to \"$new_path\"" >&6;} + fi + + # Save the first 10 bytes of this path to the storage, so fixpath can work. + all_fixpath_prefixes=("${all_fixpath_prefixes[@]}" "${new_path:0:10}") + + else + # We're on a unix platform. Hooray! :) + path="$BUILD_JDK" + has_space=`$ECHO "$path" | $GREP " "` + if test "x$has_space" != x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_JDK, which resolves as \"$path\", is invalid." >&5 +$as_echo "$as_me: The path of BUILD_JDK, which resolves as \"$path\", is invalid." >&6;} + as_fn_error $? "Spaces are not allowed in this path." "$LINENO" 5 + fi + + # Use eval to expand a potential ~ + eval path="$path" + if test ! -f "$path" && test ! -d "$path"; then + as_fn_error $? "The path of BUILD_JDK, which resolves as \"$path\", is not found." "$LINENO" 5 + fi + + if test -d "$path"; then + BUILD_JDK="`cd "$path"; $THEPWDCMD -L`" + else + dir="`$DIRNAME "$path"`" + base="`$BASENAME "$path"`" + BUILD_JDK="`cd "$dir"; $THEPWDCMD -L`/$base" + fi + fi + fi + + { $as_echo "$as_me:${as_lineno-$LINENO}: checking for Build JDK" >&5 +$as_echo_n "checking for Build JDK... " >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $BUILD_JDK" >&5 +$as_echo "$BUILD_JDK" >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: checking Build JDK version" >&5 +$as_echo_n "checking Build JDK version... " >&6; } + BUILD_JDK_VERSION=`"$BUILD_JDK/bin/java" -version 2>&1 | $TR '\n\r' ' '` + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $BUILD_JDK_VERSION" >&5 +$as_echo "$BUILD_JDK_VERSION" >&6; } + fi # end check jdk version + fi # end check java + fi # end check build jdk found + fi + + else + if test "x$COMPILE_TYPE" = "xcross"; then + BUILD_JDK="\$(BUILDJDK_OUTPUTDIR)/jdk" + BUILD_JDK_FOUND=yes + CREATE_BUILDJDK=true + { $as_echo "$as_me:${as_lineno-$LINENO}: checking for Build JDK" >&5 +$as_echo_n "checking for Build JDK... " >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: result: yes, will build it for the host platform" >&5 +$as_echo "yes, will build it for the host platform" >&6; } + else + BUILD_JDK="\$(JDK_OUTPUTDIR)" + BUILD_JDK_FOUND=yes + { $as_echo "$as_me:${as_lineno-$LINENO}: checking for Build JDK" >&5 +$as_echo_n "checking for Build JDK... " >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: result: yes, will use output dir" >&5 +$as_echo "yes, will use output dir" >&6; } + fi + fi + + if test "x$BUILD_JDK_FOUND" != "xyes"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: checking for Build JDK" >&5 +$as_echo_n "checking for Build JDK... " >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } + as_fn_error $? "Could not find a suitable Build JDK" "$LINENO" 5 + fi + + + + + ############################################################################### # # Configure the sources to use. We can add or override individual directories. @@ -29758,6 +30171,189 @@ $as_echo "yes from $HOTSPOT_DIST" >&6; } JDK_OUTPUTDIR="$OUTPUT_ROOT/jdk" + + +# Check whether --with-import-modules was given. +if test "${with_import_modules+set}" = set; then : + withval=$with_import_modules; +fi + + + if test "x$with_import_modules" != x \ + && test "x$with_import_modules" != "xno"; then + if test -d "$with_import_modules"; then + IMPORT_MODULES_TOPDIR="$with_import_modules" + + # Only process if variable expands to non-empty + + if test "x$IMPORT_MODULES_TOPDIR" != x; then + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + + # Input might be given as Windows format, start by converting to + # unix format. + path="$IMPORT_MODULES_TOPDIR" + new_path=`$CYGPATH -u "$path"` + + # Cygwin tries to hide some aspects of the Windows file system, such that binaries are + # named .exe but called without that suffix. Therefore, "foo" and "foo.exe" are considered + # the same file, most of the time (as in "test -f"). But not when running cygpath -s, then + # "foo.exe" is OK but "foo" is an error. + # + # This test is therefore slightly more accurate than "test -f" to check for file precense. + # It is also a way to make sure we got the proper file name for the real test later on. + test_shortpath=`$CYGPATH -s -m "$new_path" 2> /dev/null` + if test "x$test_shortpath" = x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of IMPORT_MODULES_TOPDIR, which resolves as \"$path\", is invalid." >&5 +$as_echo "$as_me: The path of IMPORT_MODULES_TOPDIR, which resolves as \"$path\", is invalid." >&6;} + as_fn_error $? "Cannot locate the the path of IMPORT_MODULES_TOPDIR" "$LINENO" 5 + fi + + # Call helper function which possibly converts this using DOS-style short mode. + # If so, the updated path is stored in $new_path. + + input_path="$new_path" + # Check if we need to convert this using DOS-style short mode. If the path + # contains just simple characters, use it. Otherwise (spaces, weird characters), + # take no chances and rewrite it. + # Note: m4 eats our [], so we need to use [ and ] instead. + has_forbidden_chars=`$ECHO "$input_path" | $GREP [^-._/a-zA-Z0-9]` + if test "x$has_forbidden_chars" != x; then + # Now convert it to mixed DOS-style, short mode (no spaces, and / instead of \) + shortmode_path=`$CYGPATH -s -m -a "$input_path"` + path_after_shortmode=`$CYGPATH -u "$shortmode_path"` + if test "x$path_after_shortmode" != "x$input_to_shortpath"; then + # Going to short mode and back again did indeed matter. Since short mode is + # case insensitive, let's make it lowercase to improve readability. + shortmode_path=`$ECHO "$shortmode_path" | $TR 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + # Now convert it back to Unix-style (cygpath) + input_path=`$CYGPATH -u "$shortmode_path"` + new_path="$input_path" + fi + fi + + test_cygdrive_prefix=`$ECHO $input_path | $GREP ^/cygdrive/` + if test "x$test_cygdrive_prefix" = x; then + # As a simple fix, exclude /usr/bin since it's not a real path. + if test "x`$ECHO $new_path | $GREP ^/usr/bin/`" = x; then + # The path is in a Cygwin special directory (e.g. /home). We need this converted to + # a path prefixed by /cygdrive for fixpath to work. + new_path="$CYGWIN_ROOT_PATH$input_path" + fi + fi + + + if test "x$path" != "x$new_path"; then + IMPORT_MODULES_TOPDIR="$new_path" + { $as_echo "$as_me:${as_lineno-$LINENO}: Rewriting IMPORT_MODULES_TOPDIR to \"$new_path\"" >&5 +$as_echo "$as_me: Rewriting IMPORT_MODULES_TOPDIR to \"$new_path\"" >&6;} + fi + + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + + path="$IMPORT_MODULES_TOPDIR" + has_colon=`$ECHO $path | $GREP ^.:` + new_path="$path" + if test "x$has_colon" = x; then + # Not in mixed or Windows style, start by that. + new_path=`cmd //c echo $path` + fi + + + input_path="$new_path" + # Check if we need to convert this using DOS-style short mode. If the path + # contains just simple characters, use it. Otherwise (spaces, weird characters), + # take no chances and rewrite it. + # Note: m4 eats our [], so we need to use [ and ] instead. + has_forbidden_chars=`$ECHO "$input_path" | $GREP [^-_/:a-zA-Z0-9]` + if test "x$has_forbidden_chars" != x; then + # Now convert it to mixed DOS-style, short mode (no spaces, and / instead of \) + new_path=`cmd /c "for %A in (\"$input_path\") do @echo %~sA"|$TR \\\\\\\\ / | $TR 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + fi + + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + if test "x$path" != "x$new_path"; then + IMPORT_MODULES_TOPDIR="$new_path" + { $as_echo "$as_me:${as_lineno-$LINENO}: Rewriting IMPORT_MODULES_TOPDIR to \"$new_path\"" >&5 +$as_echo "$as_me: Rewriting IMPORT_MODULES_TOPDIR to \"$new_path\"" >&6;} + fi + + # Save the first 10 bytes of this path to the storage, so fixpath can work. + all_fixpath_prefixes=("${all_fixpath_prefixes[@]}" "${new_path:0:10}") + + else + # We're on a unix platform. Hooray! :) + path="$IMPORT_MODULES_TOPDIR" + has_space=`$ECHO "$path" | $GREP " "` + if test "x$has_space" != x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of IMPORT_MODULES_TOPDIR, which resolves as \"$path\", is invalid." >&5 +$as_echo "$as_me: The path of IMPORT_MODULES_TOPDIR, which resolves as \"$path\", is invalid." >&6;} + as_fn_error $? "Spaces are not allowed in this path." "$LINENO" 5 + fi + + # Use eval to expand a potential ~ + eval path="$path" + if test ! -f "$path" && test ! -d "$path"; then + as_fn_error $? "The path of IMPORT_MODULES_TOPDIR, which resolves as \"$path\", is not found." "$LINENO" 5 + fi + + if test -d "$path"; then + IMPORT_MODULES_TOPDIR="`cd "$path"; $THEPWDCMD -L`" + else + dir="`$DIRNAME "$path"`" + base="`$BASENAME "$path"`" + IMPORT_MODULES_TOPDIR="`cd "$dir"; $THEPWDCMD -L`/$base" + fi + fi + fi + + elif test -e "$with_import_modules"; then + IMPORT_MODULES_TOPDIR="$CONFIGURESUPPORT_OUTPUTDIR/import-modules" + $RM -rf "$IMPORT_MODULES_TOPDIR" + $MKDIR -p "$IMPORT_MODULES_TOPDIR" + if ! $UNZIP -q "$with_import_modules" -d "$IMPORT_MODULES_TOPDIR"; then + as_fn_error $? "--with-import-modules=\"$with_import_modules\" must point to a dir or a zip file" "$LINENO" 5 + fi + else + as_fn_error $? "--with-import-modules=\"$with_import_modules\" must point to a dir or a zip file" "$LINENO" 5 + fi + fi + + if test -d "$IMPORT_MODULES_TOPDIR/modules"; then + IMPORT_MODULES_CLASSES="$IMPORT_MODULES_TOPDIR/modules" + fi + if test -d "$IMPORT_MODULES_TOPDIR/modules_cmds"; then + IMPORT_MODULES_CMDS="$IMPORT_MODULES_TOPDIR/modules_cmds" + fi + if test -d "$IMPORT_MODULES_TOPDIR/modules_libs"; then + IMPORT_MODULES_LIBS="$IMPORT_MODULES_TOPDIR/modules_libs" + fi + if test -d "$IMPORT_MODULES_TOPDIR/modules_conf"; then + IMPORT_MODULES_CONF="$IMPORT_MODULES_TOPDIR/modules_conf" + fi + if test -d "$IMPORT_MODULES_TOPDIR/modules_src"; then + IMPORT_MODULES_SRC="$IMPORT_MODULES_TOPDIR/modules_src" + fi + if test -d "$IMPORT_MODULES_TOPDIR/make"; then + IMPORT_MODULES_MAKE="$IMPORT_MODULES_TOPDIR/make" + fi + + + + + + + + + ############################################################################### # # Setup the toolchain (compilers etc), i.e. tools used to compile and process @@ -44514,6 +45110,972 @@ $as_echo "$as_me: Rewriting BUILD_AR to \"$new_complete\"" >&6;} fi fi + + + # Publish this variable in the help. + + + if [ -z "${BUILD_OBJCOPY+x}" ]; then + # The variable is not set by user, try to locate tool using the code snippet + for ac_prog in objcopy +do + # Extract the first word of "$ac_prog", so it can be a program name with args. +set dummy $ac_prog; ac_word=$2 +{ $as_echo "$as_me:${as_lineno-$LINENO}: checking for $ac_word" >&5 +$as_echo_n "checking for $ac_word... " >&6; } +if ${ac_cv_path_BUILD_OBJCOPY+:} false; then : + $as_echo_n "(cached) " >&6 +else + case $BUILD_OBJCOPY in + [\\/]* | ?:[\\/]*) + ac_cv_path_BUILD_OBJCOPY="$BUILD_OBJCOPY" # Let the user override the test with a path. + ;; + *) + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if as_fn_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_path_BUILD_OBJCOPY="$as_dir/$ac_word$ac_exec_ext" + $as_echo "$as_me:${as_lineno-$LINENO}: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done + done +IFS=$as_save_IFS + + ;; +esac +fi +BUILD_OBJCOPY=$ac_cv_path_BUILD_OBJCOPY +if test -n "$BUILD_OBJCOPY"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $BUILD_OBJCOPY" >&5 +$as_echo "$BUILD_OBJCOPY" >&6; } +else + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } +fi + + + test -n "$BUILD_OBJCOPY" && break +done + + else + # The variable is set, but is it from the command line or the environment? + + # Try to remove the string !BUILD_OBJCOPY! from our list. + try_remove_var=${CONFIGURE_OVERRIDDEN_VARIABLES//!BUILD_OBJCOPY!/} + if test "x$try_remove_var" = "x$CONFIGURE_OVERRIDDEN_VARIABLES"; then + # If it failed, the variable was not from the command line. Ignore it, + # but warn the user (except for BASH, which is always set by the calling BASH). + if test "xBUILD_OBJCOPY" != xBASH; then + { $as_echo "$as_me:${as_lineno-$LINENO}: WARNING: Ignoring value of BUILD_OBJCOPY from the environment. Use command line variables instead." >&5 +$as_echo "$as_me: WARNING: Ignoring value of BUILD_OBJCOPY from the environment. Use command line variables instead." >&2;} + fi + # Try to locate tool using the code snippet + for ac_prog in objcopy +do + # Extract the first word of "$ac_prog", so it can be a program name with args. +set dummy $ac_prog; ac_word=$2 +{ $as_echo "$as_me:${as_lineno-$LINENO}: checking for $ac_word" >&5 +$as_echo_n "checking for $ac_word... " >&6; } +if ${ac_cv_path_BUILD_OBJCOPY+:} false; then : + $as_echo_n "(cached) " >&6 +else + case $BUILD_OBJCOPY in + [\\/]* | ?:[\\/]*) + ac_cv_path_BUILD_OBJCOPY="$BUILD_OBJCOPY" # Let the user override the test with a path. + ;; + *) + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if as_fn_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_path_BUILD_OBJCOPY="$as_dir/$ac_word$ac_exec_ext" + $as_echo "$as_me:${as_lineno-$LINENO}: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done + done +IFS=$as_save_IFS + + ;; +esac +fi +BUILD_OBJCOPY=$ac_cv_path_BUILD_OBJCOPY +if test -n "$BUILD_OBJCOPY"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $BUILD_OBJCOPY" >&5 +$as_echo "$BUILD_OBJCOPY" >&6; } +else + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } +fi + + + test -n "$BUILD_OBJCOPY" && break +done + + else + # If it succeeded, then it was overridden by the user. We will use it + # for the tool. + + # First remove it from the list of overridden variables, so we can test + # for unknown variables in the end. + CONFIGURE_OVERRIDDEN_VARIABLES="$try_remove_var" + + # Check if we try to supply an empty value + if test "x$BUILD_OBJCOPY" = x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: Setting user supplied tool BUILD_OBJCOPY= (no value)" >&5 +$as_echo "$as_me: Setting user supplied tool BUILD_OBJCOPY= (no value)" >&6;} + { $as_echo "$as_me:${as_lineno-$LINENO}: checking for BUILD_OBJCOPY" >&5 +$as_echo_n "checking for BUILD_OBJCOPY... " >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: result: disabled" >&5 +$as_echo "disabled" >&6; } + else + # Check if the provided tool contains a complete path. + tool_specified="$BUILD_OBJCOPY" + tool_basename="${tool_specified##*/}" + if test "x$tool_basename" = "x$tool_specified"; then + # A command without a complete path is provided, search $PATH. + { $as_echo "$as_me:${as_lineno-$LINENO}: Will search for user supplied tool BUILD_OBJCOPY=$tool_basename" >&5 +$as_echo "$as_me: Will search for user supplied tool BUILD_OBJCOPY=$tool_basename" >&6;} + # Extract the first word of "$tool_basename", so it can be a program name with args. +set dummy $tool_basename; ac_word=$2 +{ $as_echo "$as_me:${as_lineno-$LINENO}: checking for $ac_word" >&5 +$as_echo_n "checking for $ac_word... " >&6; } +if ${ac_cv_path_BUILD_OBJCOPY+:} false; then : + $as_echo_n "(cached) " >&6 +else + case $BUILD_OBJCOPY in + [\\/]* | ?:[\\/]*) + ac_cv_path_BUILD_OBJCOPY="$BUILD_OBJCOPY" # Let the user override the test with a path. + ;; + *) + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if as_fn_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_path_BUILD_OBJCOPY="$as_dir/$ac_word$ac_exec_ext" + $as_echo "$as_me:${as_lineno-$LINENO}: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done + done +IFS=$as_save_IFS + + ;; +esac +fi +BUILD_OBJCOPY=$ac_cv_path_BUILD_OBJCOPY +if test -n "$BUILD_OBJCOPY"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $BUILD_OBJCOPY" >&5 +$as_echo "$BUILD_OBJCOPY" >&6; } +else + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } +fi + + + if test "x$BUILD_OBJCOPY" = x; then + as_fn_error $? "User supplied tool $tool_basename could not be found" "$LINENO" 5 + fi + else + # Otherwise we believe it is a complete path. Use it as it is. + { $as_echo "$as_me:${as_lineno-$LINENO}: Will use user supplied tool BUILD_OBJCOPY=$tool_specified" >&5 +$as_echo "$as_me: Will use user supplied tool BUILD_OBJCOPY=$tool_specified" >&6;} + { $as_echo "$as_me:${as_lineno-$LINENO}: checking for BUILD_OBJCOPY" >&5 +$as_echo_n "checking for BUILD_OBJCOPY... " >&6; } + if test ! -x "$tool_specified"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: not found" >&5 +$as_echo "not found" >&6; } + as_fn_error $? "User supplied tool BUILD_OBJCOPY=$tool_specified does not exist or is not executable" "$LINENO" 5 + fi + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $tool_specified" >&5 +$as_echo "$tool_specified" >&6; } + fi + fi + fi + + fi + + + + # Only process if variable expands to non-empty + + if test "x$BUILD_OBJCOPY" != x; then + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + + # First separate the path from the arguments. This will split at the first + # space. + complete="$BUILD_OBJCOPY" + path="${complete%% *}" + tmp="$complete EOL" + arguments="${tmp#* }" + + # Input might be given as Windows format, start by converting to + # unix format. + new_path=`$CYGPATH -u "$path"` + + # Now try to locate executable using which + new_path=`$WHICH "$new_path" 2> /dev/null` + # bat and cmd files are not always considered executable in cygwin causing which + # to not find them + if test "x$new_path" = x \ + && test "x`$ECHO \"$path\" | $GREP -i -e \"\\.bat$\" -e \"\\.cmd$\"`" != x \ + && test "x`$LS \"$path\" 2>/dev/null`" != x; then + new_path=`$CYGPATH -u "$path"` + fi + if test "x$new_path" = x; then + # Oops. Which didn't find the executable. + # The splitting of arguments from the executable at a space might have been incorrect, + # since paths with space are more likely in Windows. Give it another try with the whole + # argument. + path="$complete" + arguments="EOL" + new_path=`$CYGPATH -u "$path"` + new_path=`$WHICH "$new_path" 2> /dev/null` + # bat and cmd files are not always considered executable in cygwin causing which + # to not find them + if test "x$new_path" = x \ + && test "x`$ECHO \"$path\" | $GREP -i -e \"\\.bat$\" -e \"\\.cmd$\"`" != x \ + && test "x`$LS \"$path\" 2>/dev/null`" != x; then + new_path=`$CYGPATH -u "$path"` + fi + if test "x$new_path" = x; then + # It's still not found. Now this is an unrecoverable error. + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_OBJCOPY, which resolves as \"$complete\", is not found." >&5 +$as_echo "$as_me: The path of BUILD_OBJCOPY, which resolves as \"$complete\", is not found." >&6;} + has_space=`$ECHO "$complete" | $GREP " "` + if test "x$has_space" != x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: You might be mixing spaces in the path and extra arguments, which is not allowed." >&5 +$as_echo "$as_me: You might be mixing spaces in the path and extra arguments, which is not allowed." >&6;} + fi + as_fn_error $? "Cannot locate the the path of BUILD_OBJCOPY" "$LINENO" 5 + fi + fi + + # Cygwin tries to hide some aspects of the Windows file system, such that binaries are + # named .exe but called without that suffix. Therefore, "foo" and "foo.exe" are considered + # the same file, most of the time (as in "test -f"). But not when running cygpath -s, then + # "foo.exe" is OK but "foo" is an error. + # + # This test is therefore slightly more accurate than "test -f" to check for file presence. + # It is also a way to make sure we got the proper file name for the real test later on. + test_shortpath=`$CYGPATH -s -m "$new_path" 2> /dev/null` + if test "x$test_shortpath" = x; then + # Short path failed, file does not exist as specified. + # Try adding .exe or .cmd + if test -f "${new_path}.exe"; then + input_to_shortpath="${new_path}.exe" + elif test -f "${new_path}.cmd"; then + input_to_shortpath="${new_path}.cmd" + else + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_OBJCOPY, which resolves as \"$new_path\", is invalid." >&5 +$as_echo "$as_me: The path of BUILD_OBJCOPY, which resolves as \"$new_path\", is invalid." >&6;} + { $as_echo "$as_me:${as_lineno-$LINENO}: Neither \"$new_path\" nor \"$new_path.exe/cmd\" can be found" >&5 +$as_echo "$as_me: Neither \"$new_path\" nor \"$new_path.exe/cmd\" can be found" >&6;} + as_fn_error $? "Cannot locate the the path of BUILD_OBJCOPY" "$LINENO" 5 + fi + else + input_to_shortpath="$new_path" + fi + + # Call helper function which possibly converts this using DOS-style short mode. + # If so, the updated path is stored in $new_path. + new_path="$input_to_shortpath" + + input_path="$input_to_shortpath" + # Check if we need to convert this using DOS-style short mode. If the path + # contains just simple characters, use it. Otherwise (spaces, weird characters), + # take no chances and rewrite it. + # Note: m4 eats our [], so we need to use [ and ] instead. + has_forbidden_chars=`$ECHO "$input_path" | $GREP [^-._/a-zA-Z0-9]` + if test "x$has_forbidden_chars" != x; then + # Now convert it to mixed DOS-style, short mode (no spaces, and / instead of \) + shortmode_path=`$CYGPATH -s -m -a "$input_path"` + path_after_shortmode=`$CYGPATH -u "$shortmode_path"` + if test "x$path_after_shortmode" != "x$input_to_shortpath"; then + # Going to short mode and back again did indeed matter. Since short mode is + # case insensitive, let's make it lowercase to improve readability. + shortmode_path=`$ECHO "$shortmode_path" | $TR 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + # Now convert it back to Unix-style (cygpath) + input_path=`$CYGPATH -u "$shortmode_path"` + new_path="$input_path" + fi + fi + + test_cygdrive_prefix=`$ECHO $input_path | $GREP ^/cygdrive/` + if test "x$test_cygdrive_prefix" = x; then + # As a simple fix, exclude /usr/bin since it's not a real path. + if test "x`$ECHO $input_to_shortpath | $GREP ^/usr/bin/`" = x; then + # The path is in a Cygwin special directory (e.g. /home). We need this converted to + # a path prefixed by /cygdrive for fixpath to work. + new_path="$CYGWIN_ROOT_PATH$input_path" + fi + fi + + # remove trailing .exe if any + new_path="${new_path/%.exe/}" + + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + + # First separate the path from the arguments. This will split at the first + # space. + complete="$BUILD_OBJCOPY" + path="${complete%% *}" + tmp="$complete EOL" + arguments="${tmp#* }" + + # Input might be given as Windows format, start by converting to + # unix format. + new_path="$path" + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + + # Now try to locate executable using which + new_path=`$WHICH "$new_path" 2> /dev/null` + + if test "x$new_path" = x; then + # Oops. Which didn't find the executable. + # The splitting of arguments from the executable at a space might have been incorrect, + # since paths with space are more likely in Windows. Give it another try with the whole + # argument. + path="$complete" + arguments="EOL" + new_path="$path" + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + + new_path=`$WHICH "$new_path" 2> /dev/null` + # bat and cmd files are not always considered executable in MSYS causing which + # to not find them + if test "x$new_path" = x \ + && test "x`$ECHO \"$path\" | $GREP -i -e \"\\.bat$\" -e \"\\.cmd$\"`" != x \ + && test "x`$LS \"$path\" 2>/dev/null`" != x; then + new_path="$path" + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + fi + + if test "x$new_path" = x; then + # It's still not found. Now this is an unrecoverable error. + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_OBJCOPY, which resolves as \"$complete\", is not found." >&5 +$as_echo "$as_me: The path of BUILD_OBJCOPY, which resolves as \"$complete\", is not found." >&6;} + has_space=`$ECHO "$complete" | $GREP " "` + if test "x$has_space" != x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: You might be mixing spaces in the path and extra arguments, which is not allowed." >&5 +$as_echo "$as_me: You might be mixing spaces in the path and extra arguments, which is not allowed." >&6;} + fi + as_fn_error $? "Cannot locate the the path of BUILD_OBJCOPY" "$LINENO" 5 + fi + fi + + # Now new_path has a complete unix path to the binary + if test "x`$ECHO $new_path | $GREP ^/bin/`" != x; then + # Keep paths in /bin as-is, but remove trailing .exe if any + new_path="${new_path/%.exe/}" + # Do not save /bin paths to all_fixpath_prefixes! + else + # Not in mixed or Windows style, start by that. + new_path=`cmd //c echo $new_path` + + input_path="$new_path" + # Check if we need to convert this using DOS-style short mode. If the path + # contains just simple characters, use it. Otherwise (spaces, weird characters), + # take no chances and rewrite it. + # Note: m4 eats our [], so we need to use [ and ] instead. + has_forbidden_chars=`$ECHO "$input_path" | $GREP [^-_/:a-zA-Z0-9]` + if test "x$has_forbidden_chars" != x; then + # Now convert it to mixed DOS-style, short mode (no spaces, and / instead of \) + new_path=`cmd /c "for %A in (\"$input_path\") do @echo %~sA"|$TR \\\\\\\\ / | $TR 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + fi + + # Output is in $new_path + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + # remove trailing .exe if any + new_path="${new_path/%.exe/}" + + # Save the first 10 bytes of this path to the storage, so fixpath can work. + all_fixpath_prefixes=("${all_fixpath_prefixes[@]}" "${new_path:0:10}") + fi + + else + # We're on a unix platform. Hooray! :) + # First separate the path from the arguments. This will split at the first + # space. + complete="$BUILD_OBJCOPY" + path="${complete%% *}" + tmp="$complete EOL" + arguments="${tmp#* }" + + # Cannot rely on the command "which" here since it doesn't always work. + is_absolute_path=`$ECHO "$path" | $GREP ^/` + if test -z "$is_absolute_path"; then + # Path to executable is not absolute. Find it. + IFS_save="$IFS" + IFS=: + for p in $PATH; do + if test -f "$p/$path" && test -x "$p/$path"; then + new_path="$p/$path" + break + fi + done + IFS="$IFS_save" + else + # This is an absolute path, we can use it without further modifications. + new_path="$path" + fi + + if test "x$new_path" = x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_OBJCOPY, which resolves as \"$complete\", is not found." >&5 +$as_echo "$as_me: The path of BUILD_OBJCOPY, which resolves as \"$complete\", is not found." >&6;} + has_space=`$ECHO "$complete" | $GREP " "` + if test "x$has_space" != x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: This might be caused by spaces in the path, which is not allowed." >&5 +$as_echo "$as_me: This might be caused by spaces in the path, which is not allowed." >&6;} + fi + as_fn_error $? "Cannot locate the the path of BUILD_OBJCOPY" "$LINENO" 5 + fi + fi + + # Now join together the path and the arguments once again + if test "x$arguments" != xEOL; then + new_complete="$new_path ${arguments% *}" + else + new_complete="$new_path" + fi + + if test "x$complete" != "x$new_complete"; then + BUILD_OBJCOPY="$new_complete" + { $as_echo "$as_me:${as_lineno-$LINENO}: Rewriting BUILD_OBJCOPY to \"$new_complete\"" >&5 +$as_echo "$as_me: Rewriting BUILD_OBJCOPY to \"$new_complete\"" >&6;} + fi + fi + + + + # Publish this variable in the help. + + + if [ -z "${BUILD_STRIP+x}" ]; then + # The variable is not set by user, try to locate tool using the code snippet + for ac_prog in strip +do + # Extract the first word of "$ac_prog", so it can be a program name with args. +set dummy $ac_prog; ac_word=$2 +{ $as_echo "$as_me:${as_lineno-$LINENO}: checking for $ac_word" >&5 +$as_echo_n "checking for $ac_word... " >&6; } +if ${ac_cv_path_BUILD_STRIP+:} false; then : + $as_echo_n "(cached) " >&6 +else + case $BUILD_STRIP in + [\\/]* | ?:[\\/]*) + ac_cv_path_BUILD_STRIP="$BUILD_STRIP" # Let the user override the test with a path. + ;; + *) + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if as_fn_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_path_BUILD_STRIP="$as_dir/$ac_word$ac_exec_ext" + $as_echo "$as_me:${as_lineno-$LINENO}: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done + done +IFS=$as_save_IFS + + ;; +esac +fi +BUILD_STRIP=$ac_cv_path_BUILD_STRIP +if test -n "$BUILD_STRIP"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $BUILD_STRIP" >&5 +$as_echo "$BUILD_STRIP" >&6; } +else + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } +fi + + + test -n "$BUILD_STRIP" && break +done + + else + # The variable is set, but is it from the command line or the environment? + + # Try to remove the string !BUILD_STRIP! from our list. + try_remove_var=${CONFIGURE_OVERRIDDEN_VARIABLES//!BUILD_STRIP!/} + if test "x$try_remove_var" = "x$CONFIGURE_OVERRIDDEN_VARIABLES"; then + # If it failed, the variable was not from the command line. Ignore it, + # but warn the user (except for BASH, which is always set by the calling BASH). + if test "xBUILD_STRIP" != xBASH; then + { $as_echo "$as_me:${as_lineno-$LINENO}: WARNING: Ignoring value of BUILD_STRIP from the environment. Use command line variables instead." >&5 +$as_echo "$as_me: WARNING: Ignoring value of BUILD_STRIP from the environment. Use command line variables instead." >&2;} + fi + # Try to locate tool using the code snippet + for ac_prog in strip +do + # Extract the first word of "$ac_prog", so it can be a program name with args. +set dummy $ac_prog; ac_word=$2 +{ $as_echo "$as_me:${as_lineno-$LINENO}: checking for $ac_word" >&5 +$as_echo_n "checking for $ac_word... " >&6; } +if ${ac_cv_path_BUILD_STRIP+:} false; then : + $as_echo_n "(cached) " >&6 +else + case $BUILD_STRIP in + [\\/]* | ?:[\\/]*) + ac_cv_path_BUILD_STRIP="$BUILD_STRIP" # Let the user override the test with a path. + ;; + *) + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if as_fn_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_path_BUILD_STRIP="$as_dir/$ac_word$ac_exec_ext" + $as_echo "$as_me:${as_lineno-$LINENO}: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done + done +IFS=$as_save_IFS + + ;; +esac +fi +BUILD_STRIP=$ac_cv_path_BUILD_STRIP +if test -n "$BUILD_STRIP"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $BUILD_STRIP" >&5 +$as_echo "$BUILD_STRIP" >&6; } +else + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } +fi + + + test -n "$BUILD_STRIP" && break +done + + else + # If it succeeded, then it was overridden by the user. We will use it + # for the tool. + + # First remove it from the list of overridden variables, so we can test + # for unknown variables in the end. + CONFIGURE_OVERRIDDEN_VARIABLES="$try_remove_var" + + # Check if we try to supply an empty value + if test "x$BUILD_STRIP" = x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: Setting user supplied tool BUILD_STRIP= (no value)" >&5 +$as_echo "$as_me: Setting user supplied tool BUILD_STRIP= (no value)" >&6;} + { $as_echo "$as_me:${as_lineno-$LINENO}: checking for BUILD_STRIP" >&5 +$as_echo_n "checking for BUILD_STRIP... " >&6; } + { $as_echo "$as_me:${as_lineno-$LINENO}: result: disabled" >&5 +$as_echo "disabled" >&6; } + else + # Check if the provided tool contains a complete path. + tool_specified="$BUILD_STRIP" + tool_basename="${tool_specified##*/}" + if test "x$tool_basename" = "x$tool_specified"; then + # A command without a complete path is provided, search $PATH. + { $as_echo "$as_me:${as_lineno-$LINENO}: Will search for user supplied tool BUILD_STRIP=$tool_basename" >&5 +$as_echo "$as_me: Will search for user supplied tool BUILD_STRIP=$tool_basename" >&6;} + # Extract the first word of "$tool_basename", so it can be a program name with args. +set dummy $tool_basename; ac_word=$2 +{ $as_echo "$as_me:${as_lineno-$LINENO}: checking for $ac_word" >&5 +$as_echo_n "checking for $ac_word... " >&6; } +if ${ac_cv_path_BUILD_STRIP+:} false; then : + $as_echo_n "(cached) " >&6 +else + case $BUILD_STRIP in + [\\/]* | ?:[\\/]*) + ac_cv_path_BUILD_STRIP="$BUILD_STRIP" # Let the user override the test with a path. + ;; + *) + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if as_fn_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_path_BUILD_STRIP="$as_dir/$ac_word$ac_exec_ext" + $as_echo "$as_me:${as_lineno-$LINENO}: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done + done +IFS=$as_save_IFS + + ;; +esac +fi +BUILD_STRIP=$ac_cv_path_BUILD_STRIP +if test -n "$BUILD_STRIP"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $BUILD_STRIP" >&5 +$as_echo "$BUILD_STRIP" >&6; } +else + { $as_echo "$as_me:${as_lineno-$LINENO}: result: no" >&5 +$as_echo "no" >&6; } +fi + + + if test "x$BUILD_STRIP" = x; then + as_fn_error $? "User supplied tool $tool_basename could not be found" "$LINENO" 5 + fi + else + # Otherwise we believe it is a complete path. Use it as it is. + { $as_echo "$as_me:${as_lineno-$LINENO}: Will use user supplied tool BUILD_STRIP=$tool_specified" >&5 +$as_echo "$as_me: Will use user supplied tool BUILD_STRIP=$tool_specified" >&6;} + { $as_echo "$as_me:${as_lineno-$LINENO}: checking for BUILD_STRIP" >&5 +$as_echo_n "checking for BUILD_STRIP... " >&6; } + if test ! -x "$tool_specified"; then + { $as_echo "$as_me:${as_lineno-$LINENO}: result: not found" >&5 +$as_echo "not found" >&6; } + as_fn_error $? "User supplied tool BUILD_STRIP=$tool_specified does not exist or is not executable" "$LINENO" 5 + fi + { $as_echo "$as_me:${as_lineno-$LINENO}: result: $tool_specified" >&5 +$as_echo "$tool_specified" >&6; } + fi + fi + fi + + fi + + + + # Only process if variable expands to non-empty + + if test "x$BUILD_STRIP" != x; then + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + + # First separate the path from the arguments. This will split at the first + # space. + complete="$BUILD_STRIP" + path="${complete%% *}" + tmp="$complete EOL" + arguments="${tmp#* }" + + # Input might be given as Windows format, start by converting to + # unix format. + new_path=`$CYGPATH -u "$path"` + + # Now try to locate executable using which + new_path=`$WHICH "$new_path" 2> /dev/null` + # bat and cmd files are not always considered executable in cygwin causing which + # to not find them + if test "x$new_path" = x \ + && test "x`$ECHO \"$path\" | $GREP -i -e \"\\.bat$\" -e \"\\.cmd$\"`" != x \ + && test "x`$LS \"$path\" 2>/dev/null`" != x; then + new_path=`$CYGPATH -u "$path"` + fi + if test "x$new_path" = x; then + # Oops. Which didn't find the executable. + # The splitting of arguments from the executable at a space might have been incorrect, + # since paths with space are more likely in Windows. Give it another try with the whole + # argument. + path="$complete" + arguments="EOL" + new_path=`$CYGPATH -u "$path"` + new_path=`$WHICH "$new_path" 2> /dev/null` + # bat and cmd files are not always considered executable in cygwin causing which + # to not find them + if test "x$new_path" = x \ + && test "x`$ECHO \"$path\" | $GREP -i -e \"\\.bat$\" -e \"\\.cmd$\"`" != x \ + && test "x`$LS \"$path\" 2>/dev/null`" != x; then + new_path=`$CYGPATH -u "$path"` + fi + if test "x$new_path" = x; then + # It's still not found. Now this is an unrecoverable error. + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_STRIP, which resolves as \"$complete\", is not found." >&5 +$as_echo "$as_me: The path of BUILD_STRIP, which resolves as \"$complete\", is not found." >&6;} + has_space=`$ECHO "$complete" | $GREP " "` + if test "x$has_space" != x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: You might be mixing spaces in the path and extra arguments, which is not allowed." >&5 +$as_echo "$as_me: You might be mixing spaces in the path and extra arguments, which is not allowed." >&6;} + fi + as_fn_error $? "Cannot locate the the path of BUILD_STRIP" "$LINENO" 5 + fi + fi + + # Cygwin tries to hide some aspects of the Windows file system, such that binaries are + # named .exe but called without that suffix. Therefore, "foo" and "foo.exe" are considered + # the same file, most of the time (as in "test -f"). But not when running cygpath -s, then + # "foo.exe" is OK but "foo" is an error. + # + # This test is therefore slightly more accurate than "test -f" to check for file presence. + # It is also a way to make sure we got the proper file name for the real test later on. + test_shortpath=`$CYGPATH -s -m "$new_path" 2> /dev/null` + if test "x$test_shortpath" = x; then + # Short path failed, file does not exist as specified. + # Try adding .exe or .cmd + if test -f "${new_path}.exe"; then + input_to_shortpath="${new_path}.exe" + elif test -f "${new_path}.cmd"; then + input_to_shortpath="${new_path}.cmd" + else + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_STRIP, which resolves as \"$new_path\", is invalid." >&5 +$as_echo "$as_me: The path of BUILD_STRIP, which resolves as \"$new_path\", is invalid." >&6;} + { $as_echo "$as_me:${as_lineno-$LINENO}: Neither \"$new_path\" nor \"$new_path.exe/cmd\" can be found" >&5 +$as_echo "$as_me: Neither \"$new_path\" nor \"$new_path.exe/cmd\" can be found" >&6;} + as_fn_error $? "Cannot locate the the path of BUILD_STRIP" "$LINENO" 5 + fi + else + input_to_shortpath="$new_path" + fi + + # Call helper function which possibly converts this using DOS-style short mode. + # If so, the updated path is stored in $new_path. + new_path="$input_to_shortpath" + + input_path="$input_to_shortpath" + # Check if we need to convert this using DOS-style short mode. If the path + # contains just simple characters, use it. Otherwise (spaces, weird characters), + # take no chances and rewrite it. + # Note: m4 eats our [], so we need to use [ and ] instead. + has_forbidden_chars=`$ECHO "$input_path" | $GREP [^-._/a-zA-Z0-9]` + if test "x$has_forbidden_chars" != x; then + # Now convert it to mixed DOS-style, short mode (no spaces, and / instead of \) + shortmode_path=`$CYGPATH -s -m -a "$input_path"` + path_after_shortmode=`$CYGPATH -u "$shortmode_path"` + if test "x$path_after_shortmode" != "x$input_to_shortpath"; then + # Going to short mode and back again did indeed matter. Since short mode is + # case insensitive, let's make it lowercase to improve readability. + shortmode_path=`$ECHO "$shortmode_path" | $TR 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + # Now convert it back to Unix-style (cygpath) + input_path=`$CYGPATH -u "$shortmode_path"` + new_path="$input_path" + fi + fi + + test_cygdrive_prefix=`$ECHO $input_path | $GREP ^/cygdrive/` + if test "x$test_cygdrive_prefix" = x; then + # As a simple fix, exclude /usr/bin since it's not a real path. + if test "x`$ECHO $input_to_shortpath | $GREP ^/usr/bin/`" = x; then + # The path is in a Cygwin special directory (e.g. /home). We need this converted to + # a path prefixed by /cygdrive for fixpath to work. + new_path="$CYGWIN_ROOT_PATH$input_path" + fi + fi + + # remove trailing .exe if any + new_path="${new_path/%.exe/}" + + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + + # First separate the path from the arguments. This will split at the first + # space. + complete="$BUILD_STRIP" + path="${complete%% *}" + tmp="$complete EOL" + arguments="${tmp#* }" + + # Input might be given as Windows format, start by converting to + # unix format. + new_path="$path" + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + + # Now try to locate executable using which + new_path=`$WHICH "$new_path" 2> /dev/null` + + if test "x$new_path" = x; then + # Oops. Which didn't find the executable. + # The splitting of arguments from the executable at a space might have been incorrect, + # since paths with space are more likely in Windows. Give it another try with the whole + # argument. + path="$complete" + arguments="EOL" + new_path="$path" + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + + new_path=`$WHICH "$new_path" 2> /dev/null` + # bat and cmd files are not always considered executable in MSYS causing which + # to not find them + if test "x$new_path" = x \ + && test "x`$ECHO \"$path\" | $GREP -i -e \"\\.bat$\" -e \"\\.cmd$\"`" != x \ + && test "x`$LS \"$path\" 2>/dev/null`" != x; then + new_path="$path" + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + fi + + if test "x$new_path" = x; then + # It's still not found. Now this is an unrecoverable error. + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_STRIP, which resolves as \"$complete\", is not found." >&5 +$as_echo "$as_me: The path of BUILD_STRIP, which resolves as \"$complete\", is not found." >&6;} + has_space=`$ECHO "$complete" | $GREP " "` + if test "x$has_space" != x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: You might be mixing spaces in the path and extra arguments, which is not allowed." >&5 +$as_echo "$as_me: You might be mixing spaces in the path and extra arguments, which is not allowed." >&6;} + fi + as_fn_error $? "Cannot locate the the path of BUILD_STRIP" "$LINENO" 5 + fi + fi + + # Now new_path has a complete unix path to the binary + if test "x`$ECHO $new_path | $GREP ^/bin/`" != x; then + # Keep paths in /bin as-is, but remove trailing .exe if any + new_path="${new_path/%.exe/}" + # Do not save /bin paths to all_fixpath_prefixes! + else + # Not in mixed or Windows style, start by that. + new_path=`cmd //c echo $new_path` + + input_path="$new_path" + # Check if we need to convert this using DOS-style short mode. If the path + # contains just simple characters, use it. Otherwise (spaces, weird characters), + # take no chances and rewrite it. + # Note: m4 eats our [], so we need to use [ and ] instead. + has_forbidden_chars=`$ECHO "$input_path" | $GREP [^-_/:a-zA-Z0-9]` + if test "x$has_forbidden_chars" != x; then + # Now convert it to mixed DOS-style, short mode (no spaces, and / instead of \) + new_path=`cmd /c "for %A in (\"$input_path\") do @echo %~sA"|$TR \\\\\\\\ / | $TR 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + fi + + # Output is in $new_path + + windows_path="$new_path" + if test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.cygwin"; then + unix_path=`$CYGPATH -u "$windows_path"` + new_path="$unix_path" + elif test "x$OPENJDK_BUILD_OS_ENV" = "xwindows.msys"; then + unix_path=`$ECHO "$windows_path" | $SED -e 's,^\\(.\\):,/\\1,g' -e 's,\\\\,/,g'` + new_path="$unix_path" + fi + + # remove trailing .exe if any + new_path="${new_path/%.exe/}" + + # Save the first 10 bytes of this path to the storage, so fixpath can work. + all_fixpath_prefixes=("${all_fixpath_prefixes[@]}" "${new_path:0:10}") + fi + + else + # We're on a unix platform. Hooray! :) + # First separate the path from the arguments. This will split at the first + # space. + complete="$BUILD_STRIP" + path="${complete%% *}" + tmp="$complete EOL" + arguments="${tmp#* }" + + # Cannot rely on the command "which" here since it doesn't always work. + is_absolute_path=`$ECHO "$path" | $GREP ^/` + if test -z "$is_absolute_path"; then + # Path to executable is not absolute. Find it. + IFS_save="$IFS" + IFS=: + for p in $PATH; do + if test -f "$p/$path" && test -x "$p/$path"; then + new_path="$p/$path" + break + fi + done + IFS="$IFS_save" + else + # This is an absolute path, we can use it without further modifications. + new_path="$path" + fi + + if test "x$new_path" = x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: The path of BUILD_STRIP, which resolves as \"$complete\", is not found." >&5 +$as_echo "$as_me: The path of BUILD_STRIP, which resolves as \"$complete\", is not found." >&6;} + has_space=`$ECHO "$complete" | $GREP " "` + if test "x$has_space" != x; then + { $as_echo "$as_me:${as_lineno-$LINENO}: This might be caused by spaces in the path, which is not allowed." >&5 +$as_echo "$as_me: This might be caused by spaces in the path, which is not allowed." >&6;} + fi + as_fn_error $? "Cannot locate the the path of BUILD_STRIP" "$LINENO" 5 + fi + fi + + # Now join together the path and the arguments once again + if test "x$arguments" != xEOL; then + new_complete="$new_path ${arguments% *}" + else + new_complete="$new_path" + fi + + if test "x$complete" != "x$new_complete"; then + BUILD_STRIP="$new_complete" + { $as_echo "$as_me:${as_lineno-$LINENO}: Rewriting BUILD_STRIP to \"$new_complete\"" >&5 +$as_echo "$as_me: Rewriting BUILD_STRIP to \"$new_complete\"" >&6;} + fi + fi + # Assume the C compiler is the assembler BUILD_AS="$BUILD_CC -c" # Just like for the target compiler, use the compiler as linker @@ -44530,6 +46092,8 @@ $as_echo "$as_me: Rewriting BUILD_AR to \"$new_complete\"" >&6;} BUILD_LDCXX="$LDCXX" BUILD_NM="$NM" BUILD_AS="$AS" + BUILD_OBJCOPY="$OBJCOPY" + BUILD_STRIP="$STRIP" BUILD_SYSROOT_CFLAGS="$SYSROOT_CFLAGS" BUILD_SYSROOT_LDFLAGS="$SYSROOT_LDFLAGS" BUILD_AR="$AR" @@ -46684,9 +48248,6 @@ $as_echo "$supports" >&6; } ;; esac - # Setup LP64 - COMMON_CCXXFLAGS_JDK="$COMMON_CCXXFLAGS_JDK $ADD_LP64" - # Set some common defines. These works for all compilers, but assume # -D is universally accepted. @@ -46717,7 +48278,12 @@ $as_echo "$supports" >&6; } COMMON_CCXXFLAGS_JDK="$COMMON_CCXXFLAGS_JDK -D$OPENJDK_TARGET_OS_UPPERCASE" # Setup target CPU - COMMON_CCXXFLAGS_JDK="$COMMON_CCXXFLAGS_JDK -DARCH='\"$OPENJDK_TARGET_CPU_LEGACY\"' -D$OPENJDK_TARGET_CPU_LEGACY" + OPENJDK_TARGET_CCXXFLAGS_JDK="$OPENJDK_TARGET_CCXXFLAGS_JDK \ + $ADD_LP64 \ + -DARCH='\"$OPENJDK_TARGET_CPU_LEGACY\"' -D$OPENJDK_TARGET_CPU_LEGACY" + OPENJDK_BUILD_CCXXFLAGS_JDK="$OPENJDK_BUILD_CCXXFLAGS_JDK \ + $OPENJDK_BUILD_ADD_LP64 \ + -DARCH='\"$OPENJDK_BUILD_CPU_LEGACY\"' -D$OPENJDK_BUILD_CPU_LEGACY" # Setup debug/release defines if test "x$DEBUG_LEVEL" = xrelease; then @@ -46761,12 +48327,30 @@ $as_echo "$supports" >&6; } -I${JDK_TOPDIR}/src/java.base/$OPENJDK_TARGET_OS_TYPE/native/libjava" # The shared libraries are compiled using the picflag. - CFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $CFLAGS_JDK $PICFLAG $CFLAGS_JDKLIB_EXTRA" - CXXFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $CXXFLAGS_JDK $PICFLAG $CXXFLAGS_JDKLIB_EXTRA" + CFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $OPENJDK_TARGET_CCXXFLAGS_JDK \ + $CFLAGS_JDK $EXTRA_CFLAGS_JDK $PICFLAG $CFLAGS_JDKLIB_EXTRA" + CXXFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $OPENJDK_TARGET_CCXXFLAGS_JDK \ + $CXXFLAGS_JDK $EXTRA_CXXFLAGS_JDK $PICFLAG $CXXFLAGS_JDKLIB_EXTRA" # Executable flags - CFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $CFLAGS_JDK" - CXXFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $CXXFLAGS_JDK" + CFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $OPENJDK_TARGET_CCXXFLAGS_JDK \ + $CFLAGS_JDK $EXTRA_CFLAGS_JDK" + CXXFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $OPENJDK_TARGET_CCXXFLAGS_JDK \ + $CXXFLAGS_JDK $EXTRA_CXXFLAGS_JDK" + + # The corresponding flags for building for the build platform. This is still an + # approximation, we only need something that runs on this machine when cross + # compiling the product. + OPENJDK_BUILD_CFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $OPENJDK_BUILD_CCXXFLAGS_JDK \ + $PICFLAG $CFLAGS_JDKLIB_EXTRA" + OPENJDK_BUILD_CXXFLAGS_JDKLIB="$COMMON_CCXXFLAGS_JDK $OPENJDK_BUILD_CCXXFLAGS_JDK \ + $PICFLAG $CXXFLAGS_JDKLIB_EXTRA" + OPENJDK_BUILD_CFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $OPENJDK_BUILD_CCXXFLAGS_JDK" + OPENJDK_BUILD_CXXFLAGS_JDKEXE="$COMMON_CCXXFLAGS_JDK $OPENJDK_BUILD_CCXXFLAGS_JDK" + + + + @@ -46867,6 +48451,9 @@ $as_echo "$supports" >&6; } LDFLAGS_JDKEXE="$LDFLAGS_JDKEXE -Wl,--allow-shlib-undefined" fi + OPENJDK_BUILD_LDFLAGS_JDKEXE="${LDFLAGS_JDKEXE}" + LDFLAGS_JDKEXE="${LDFLAGS_JDKEXE} ${EXTRA_LDFLAGS_JDK}" + # Customize LDFLAGS for libs LDFLAGS_JDKLIB="${LDFLAGS_JDK}" @@ -46877,28 +48464,37 @@ $as_echo "$supports" >&6; } JDKLIB_LIBS="" else LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} \ - -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}" + -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)" # On some platforms (mac) the linker warns about non existing -L dirs. # Add server first if available. Linking aginst client does not always produce the same results. # Only add client dir if client is being built. Add minimal (note not minimal1) if only building minimal1. # Default to server for other variants. if test "x$JVM_VARIANT_SERVER" = xtrue; then - LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}/server" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/server" elif test "x$JVM_VARIANT_CLIENT" = xtrue; then - LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}/client" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/client" elif test "x$JVM_VARIANT_MINIMAL1" = xtrue; then - LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}/minimal" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/minimal" else - LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L${OUTPUT_ROOT}/support/modules_libs/java.base${OPENJDK_TARGET_CPU_LIBDIR}/server" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/server" fi JDKLIB_LIBS="-ljava -ljvm" if test "x$TOOLCHAIN_TYPE" = xsolstudio; then JDKLIB_LIBS="$JDKLIB_LIBS -lc" fi + + # When building a buildjdk, it's always only the server variant + OPENJDK_BUILD_LDFLAGS_JDKLIB="${OPENJDK_BUILD_LDFLAGS_JDKLIB} \ + -L\$(SUPPORT_OUTPUTDIR)/modules_libs/java.base\$(OPENJDK_TARGET_CPU_LIBDIR)/server" fi + OPENJDK_BUILD_LDFLAGS_JDKLIB="${OPENJDK_BUILD_LDFLAGS_JDKLIB} ${LDFLAGS_JDKLIB}" + LDFLAGS_JDKLIB="${LDFLAGS_JDKLIB} ${EXTRA_LDFLAGS_JDK}" + + + @@ -47503,6 +49099,7 @@ $as_echo "$supports" >&6; } + # Setup debug symbols (need objcopy from the toolchain for that) # @@ -61205,6 +62802,7 @@ do "$OUTPUT_ROOT/spec.gmk") CONFIG_FILES="$CONFIG_FILES $OUTPUT_ROOT/spec.gmk:$AUTOCONF_DIR/spec.gmk.in" ;; "$OUTPUT_ROOT/hotspot-spec.gmk") CONFIG_FILES="$CONFIG_FILES $OUTPUT_ROOT/hotspot-spec.gmk:$AUTOCONF_DIR/hotspot-spec.gmk.in" ;; "$OUTPUT_ROOT/bootcycle-spec.gmk") CONFIG_FILES="$CONFIG_FILES $OUTPUT_ROOT/bootcycle-spec.gmk:$AUTOCONF_DIR/bootcycle-spec.gmk.in" ;; + "$OUTPUT_ROOT/buildjdk-spec.gmk") CONFIG_FILES="$CONFIG_FILES $OUTPUT_ROOT/buildjdk-spec.gmk:$AUTOCONF_DIR/buildjdk-spec.gmk.in" ;; "$OUTPUT_ROOT/compare.sh") CONFIG_FILES="$CONFIG_FILES $OUTPUT_ROOT/compare.sh:$AUTOCONF_DIR/compare.sh.in" ;; "$OUTPUT_ROOT/Makefile") CONFIG_FILES="$CONFIG_FILES $OUTPUT_ROOT/Makefile:$AUTOCONF_DIR/Makefile.in" ;; diff --git a/common/autoconf/jdk-options.m4 b/common/autoconf/jdk-options.m4 index a5b403b0d30..3c677d8adde 100644 --- a/common/autoconf/jdk-options.m4 +++ b/common/autoconf/jdk-options.m4 @@ -405,3 +405,31 @@ AC_DEFUN_ONCE([JDKOPT_SETUP_STATIC_BUILD], AC_SUBST(STATIC_BUILD) ]) + +################################################################################ +# +# jlink options. +# We always keep packaged modules in JDK image. +# +AC_DEFUN_ONCE([JDKOPT_SETUP_JLINK_OPTIONS], +[ + AC_ARG_ENABLE([keep-packaged-modules], [AS_HELP_STRING([--disable-keep-packaged-modules], + [Do not keep packaged modules in jdk image @<:@enable@:>@])]) + + if test "x$enable_keep_packaged_modules" = "xyes"; then + AC_MSG_CHECKING([if packaged modules are kept]) + AC_MSG_RESULT([yes]) + JLINK_KEEP_PACKAGED_MODULES=true + elif test "x$enable_keep_packaged_modules" = "xno"; then + AC_MSG_CHECKING([if packaged modules are kept]) + AC_MSG_RESULT([no]) + JLINK_KEEP_PACKAGED_MODULES=false + elif test "x$enable_keep_packaged_modules" = "x"; then + AC_MSG_RESULT([yes (default)]) + JLINK_KEEP_PACKAGED_MODULES=true + else + AC_MSG_ERROR([--enable-keep-packaged-modules accepts no argument]) + fi + + AC_SUBST(JLINK_KEEP_PACKAGED_MODULES) +]) diff --git a/common/autoconf/platform.m4 b/common/autoconf/platform.m4 index 7553dfe0323..fe5a201ecd9 100644 --- a/common/autoconf/platform.m4 +++ b/common/autoconf/platform.m4 @@ -304,6 +304,37 @@ AC_DEFUN([PLATFORM_SETUP_LEGACY_VARS], fi AC_SUBST(OPENJDK_TARGET_CPU_LIBDIR) + # Now do the same for OPENJDK_BUILD_CPU... + # Also store the legacy naming of the cpu. + # Ie i586 and amd64 instead of x86 and x86_64 + OPENJDK_BUILD_CPU_LEGACY="$OPENJDK_BUILD_CPU" + if test "x$OPENJDK_BUILD_CPU" = xx86; then + OPENJDK_BUILD_CPU_LEGACY="i586" + elif test "x$OPENJDK_BUILD_OS" != xmacosx && test "x$OPENJDK_BUILD_CPU" = xx86_64; then + # On all platforms except MacOSX replace x86_64 with amd64. + OPENJDK_BUILD_CPU_LEGACY="amd64" + fi + AC_SUBST(OPENJDK_BUILD_CPU_LEGACY) + + # And the second legacy naming of the cpu. + # Ie i386 and amd64 instead of x86 and x86_64. + OPENJDK_BUILD_CPU_LEGACY_LIB="$OPENJDK_BUILD_CPU" + if test "x$OPENJDK_BUILD_CPU" = xx86; then + OPENJDK_BUILD_CPU_LEGACY_LIB="i386" + elif test "x$OPENJDK_BUILD_CPU" = xx86_64; then + OPENJDK_BUILD_CPU_LEGACY_LIB="amd64" + fi + AC_SUBST(OPENJDK_BUILD_CPU_LEGACY_LIB) + + # This is the name of the cpu (but using i386 and amd64 instead of + # x86 and x86_64, respectively), preceeded by a /, to be used when + # locating libraries. On macosx, it's empty, though. + OPENJDK_BUILD_CPU_LIBDIR="/$OPENJDK_BUILD_CPU_LEGACY_LIB" + if test "x$OPENJDK_BUILD_OS" = xmacosx; then + OPENJDK_BUILD_CPU_LIBDIR="" + fi + AC_SUBST(OPENJDK_BUILD_CPU_LIBDIR) + # OPENJDK_TARGET_CPU_ISADIR is normally empty. On 64-bit Solaris systems, it is set to # /amd64 or /sparcv9. This string is appended to some library paths, like this: # /usr/lib${OPENJDK_TARGET_CPU_ISADIR}/libexample.so @@ -346,6 +377,24 @@ AC_DEFUN([PLATFORM_SETUP_LEGACY_VARS], fi AC_SUBST(OPENJDK_TARGET_CPU_JLI_CFLAGS) + OPENJDK_BUILD_CPU_JLI="$OPENJDK_BUILD_CPU" + if test "x$OPENJDK_BUILD_CPU" = xx86; then + OPENJDK_BUILD_CPU_JLI="i386" + elif test "x$OPENJDK_BUILD_OS" != xmacosx && test "x$OPENJDK_BUILD_CPU" = xx86_64; then + # On all platforms except macosx, we replace x86_64 with amd64. + OPENJDK_BUILD_CPU_JLI="amd64" + fi + # Now setup the -D flags for building libjli. + OPENJDK_BUILD_CPU_JLI_CFLAGS="-DLIBARCHNAME='\"$OPENJDK_BUILD_CPU_JLI\"'" + if test "x$OPENJDK_BUILD_OS" = xsolaris; then + if test "x$OPENJDK_BUILD_CPU_ARCH" = xsparc; then + OPENJDK_BUILD_CPU_JLI_CFLAGS="$OPENJDK_BUILD_CPU_JLI_CFLAGS -DLIBARCH32NAME='\"sparc\"' -DLIBARCH64NAME='\"sparcv9\"'" + elif test "x$OPENJDK_BUILD_CPU_ARCH" = xx86; then + OPENJDK_BUILD_CPU_JLI_CFLAGS="$OPENJDK_BUILD_CPU_JLI_CFLAGS -DLIBARCH32NAME='\"i386\"' -DLIBARCH64NAME='\"amd64\"'" + fi + fi + AC_SUBST(OPENJDK_BUILD_CPU_JLI_CFLAGS) + if test "x$OPENJDK_TARGET_OS" = xmacosx; then OPENJDK_TARGET_OS_EXPORT_DIR=macosx else @@ -362,6 +411,11 @@ AC_DEFUN([PLATFORM_SETUP_LEGACY_VARS], fi fi AC_SUBST(LP64,$A_LP64) + if test "x$OPENJDK_BUILD_CPU_BITS" = x64; then + if test "x$OPENJDK_BUILD_OS" = xlinux || test "x$OPENJDK_BUILD_OS" = xmacosx; then + OPENJDK_BUILD_ADD_LP64="-D_LP64=1" + fi + fi if test "x$COMPILE_TYPE" = "xcross"; then # FIXME: ... or should this include reduced builds..? diff --git a/common/autoconf/source-dirs.m4 b/common/autoconf/source-dirs.m4 index 56f95dcba60..fa2f74d2237 100644 --- a/common/autoconf/source-dirs.m4 +++ b/common/autoconf/source-dirs.m4 @@ -84,3 +84,56 @@ AC_DEFUN_ONCE([SRCDIRS_SETUP_OUTPUT_DIRS], JDK_OUTPUTDIR="$OUTPUT_ROOT/jdk" ]) + +################################################################################ +# Define a mechanism for importing extra prebuilt modules +# + +AC_DEFUN_ONCE([SRCDIRS_SETUP_IMPORT_MODULES], +[ + AC_ARG_WITH(import-modules, [AS_HELP_STRING([--with-import-modules], + [import a set of prebuilt modules either as a zip file or an exploded directory])]) + + if test "x$with_import_modules" != x \ + && test "x$with_import_modules" != "xno"; then + if test -d "$with_import_modules"; then + IMPORT_MODULES_TOPDIR="$with_import_modules" + BASIC_FIXUP_PATH([IMPORT_MODULES_TOPDIR]) + elif test -e "$with_import_modules"; then + IMPORT_MODULES_TOPDIR="$CONFIGURESUPPORT_OUTPUTDIR/import-modules" + $RM -rf "$IMPORT_MODULES_TOPDIR" + $MKDIR -p "$IMPORT_MODULES_TOPDIR" + if ! $UNZIP -q "$with_import_modules" -d "$IMPORT_MODULES_TOPDIR"; then + AC_MSG_ERROR([--with-import-modules="$with_import_modules" must point to a dir or a zip file]) + fi + else + AC_MSG_ERROR([--with-import-modules="$with_import_modules" must point to a dir or a zip file]) + fi + fi + + if test -d "$IMPORT_MODULES_TOPDIR/modules"; then + IMPORT_MODULES_CLASSES="$IMPORT_MODULES_TOPDIR/modules" + fi + if test -d "$IMPORT_MODULES_TOPDIR/modules_cmds"; then + IMPORT_MODULES_CMDS="$IMPORT_MODULES_TOPDIR/modules_cmds" + fi + if test -d "$IMPORT_MODULES_TOPDIR/modules_libs"; then + IMPORT_MODULES_LIBS="$IMPORT_MODULES_TOPDIR/modules_libs" + fi + if test -d "$IMPORT_MODULES_TOPDIR/modules_conf"; then + IMPORT_MODULES_CONF="$IMPORT_MODULES_TOPDIR/modules_conf" + fi + if test -d "$IMPORT_MODULES_TOPDIR/modules_src"; then + IMPORT_MODULES_SRC="$IMPORT_MODULES_TOPDIR/modules_src" + fi + if test -d "$IMPORT_MODULES_TOPDIR/make"; then + IMPORT_MODULES_MAKE="$IMPORT_MODULES_TOPDIR/make" + fi + + AC_SUBST(IMPORT_MODULES_CLASSES) + AC_SUBST(IMPORT_MODULES_CMDS) + AC_SUBST(IMPORT_MODULES_LIBS) + AC_SUBST(IMPORT_MODULES_CONF) + AC_SUBST(IMPORT_MODULES_SRC) + AC_SUBST(IMPORT_MODULES_MAKE) +]) diff --git a/common/autoconf/spec.gmk.in b/common/autoconf/spec.gmk.in index d8322ebce9c..66374c8f649 100644 --- a/common/autoconf/spec.gmk.in +++ b/common/autoconf/spec.gmk.in @@ -130,6 +130,14 @@ JAXP_TOPDIR:=@JAXP_TOPDIR@ JAXWS_TOPDIR:=@JAXWS_TOPDIR@ HOTSPOT_TOPDIR:=@HOTSPOT_TOPDIR@ NASHORN_TOPDIR:=@NASHORN_TOPDIR@ + +IMPORT_MODULES_CLASSES:=@IMPORT_MODULES_CLASSES@ +IMPORT_MODULES_CMDS:=@IMPORT_MODULES_CMDS@ +IMPORT_MODULES_LIBS:=@IMPORT_MODULES_LIBS@ +IMPORT_MODULES_CONF:=@IMPORT_MODULES_CONF@ +IMPORT_MODULES_SRC:=@IMPORT_MODULES_SRC@ +IMPORT_MODULES_MAKE:=@IMPORT_MODULES_MAKE@ + COPYRIGHT_YEAR:=@COPYRIGHT_YEAR@ # New (JEP-223) version information @@ -246,6 +254,7 @@ TESTMAKE_OUTPUTDIR=$(BUILD_OUTPUT)/test-make MAKESUPPORT_OUTPUTDIR=$(BUILD_OUTPUT)/make-support # This does not get overridden in a bootcycle build CONFIGURESUPPORT_OUTPUTDIR:=@CONFIGURESUPPORT_OUTPUTDIR@ +BUILDJDK_OUTPUTDIR=$(BUILD_OUTPUT)/buildjdk HOTSPOT_DIST=@HOTSPOT_DIST@ @@ -255,6 +264,9 @@ BUILD_HOTSPOT=@BUILD_HOTSPOT@ # it in sync. BOOT_JDK:=@BOOT_JDK@ +BUILD_JDK:=@BUILD_JDK@ +CREATE_BUILDJDK:=@CREATE_BUILDJDK@ + # When compiling Java source to be run by the boot jdk # use these extra flags, eg -source 6 -target 6 BOOT_JDK_SOURCETARGET:=@BOOT_JDK_SOURCETARGET@ @@ -405,6 +417,8 @@ BUILD_LDCXX:=@FIXPATH@ @BUILD_LDCXX@ BUILD_AS:=@FIXPATH@ @BUILD_AS@ BUILD_AR:=@FIXPATH@ @BUILD_AR@ BUILD_NM:=@FIXPATH@ @BUILD_NM@ +BUILD_OBJCOPY:=@BUILD_OBJCOPY@ +BUILD_STRIP:=@BUILD_STRIP@ BUILD_SYSROOT_CFLAGS:=@BUILD_SYSROOT_CFLAGS@ BUILD_SYSROOT_LDFLAGS:=@BUILD_SYSROOT_LDFLAGS@ @@ -502,12 +516,40 @@ SJAVAC_SERVER_JAVA=@FIXPATH@ @FIXPATH_DETACH_FLAG@ $(SJAVAC_SERVER_JAVA_CMD) \ # overriding that value by using ?=. JAVAC_FLAGS?=@JAVAC_FLAGS@ + +BUILD_JAVA_FLAGS:=-Xms64M -Xmx1100M +BUILD_JAVA=@FIXPATH@ $(BUILD_JDK)/bin/java $(BUILD_JAVA_FLAGS) + +# Use ?= as this can be overridden from bootcycle-spec.gmk +BOOT_JDK_MODULAR ?= @BOOT_JDK_MODULAR@ + +ifeq ($(BOOT_JDK_MODULAR), true) + INTERIM_OVERRIDE_MODULES_ARGS = -Xpatch:$(BUILDTOOLS_OUTPUTDIR)/override_modules + INTERIM_LANGTOOLS_ARGS = $(INTERIM_OVERRIDE_MODULES_ARGS) + JAVAC_MAIN_CLASS = -m jdk.compiler/com.sun.tools.javac.Main + JAVADOC_MAIN_CLASS = -m jdk.javadoc/jdk.javadoc.internal.tool.Main +else + INTERIM_OVERRIDE_MODULES := java.compiler jdk.compiler \ + jdk.jdeps jdk.javadoc jdk.rmic + INTERIM_OVERRIDE_MODULES_ARGS = \ + -Xbootclasspath/p:$(call PathList, \ + $(addprefix $(BUILDTOOLS_OUTPUTDIR)/override_modules/, \ + $(INTERIM_OVERRIDE_MODULES))) + INTERIM_LANGTOOLS_ARGS = $(INTERIM_OVERRIDE_MODULES_ARGS) \ + -cp $(BUILDTOOLS_OUTPUTDIR)/override_modules/jdk.compiler + JAVAC_MAIN_CLASS = com.sun.tools.javac.Main + JAVADOC_MAIN_CLASS = jdk.javadoc.internal.tool.Main +endif # You run the new javac using the boot jdk with $(BOOT_JDK)/bin/java $(NEW_JAVAC) ... # Use = assignment to be able to override in bootcycle-spec.gmk -INTERIM_LANGTOOLS_JAR = $(BUILDTOOLS_OUTPUTDIR)/interim_langtools.jar -INTERIM_LANGTOOLS_ARGS = "-Xbootclasspath/p:$(INTERIM_LANGTOOLS_JAR)" -cp $(INTERIM_LANGTOOLS_JAR) -NEW_JAVAC = $(INTERIM_LANGTOOLS_ARGS) com.sun.tools.javac.Main -NEW_JAVADOC = $(INTERIM_LANGTOOLS_ARGS) jdk.javadoc.internal.tool.Main +NEW_JAVAC = $(INTERIM_LANGTOOLS_ARGS) $(JAVAC_MAIN_CLASS) +NEW_JAVADOC = $(INTERIM_LANGTOOLS_ARGS) $(JAVADOC_MAIN_CLASS) + +# JLink/Jmod are run using the BUILD_JDK, which is normally the jdk output dir. +JLINK_KEEP_PACKAGED_MODULES:=@JLINK_KEEP_PACKAGED_MODULES@ + +JLINK = @FIXPATH@ $(BUILD_JDK)/bin/jlink $(JAVA_TOOL_FLAGS_SMALL) +JMOD = @FIXPATH@ $(BUILD_JDK)/bin/jmod $(JAVA_TOOL_FLAGS_SMALL) # Base flags for RC # Guarding this against resetting value. Legacy make files include spec multiple diff --git a/common/autoconf/toolchain.m4 b/common/autoconf/toolchain.m4 index 5e26fd1e3e6..07a6f834bfc 100644 --- a/common/autoconf/toolchain.m4 +++ b/common/autoconf/toolchain.m4 @@ -797,6 +797,10 @@ AC_DEFUN_ONCE([TOOLCHAIN_SETUP_BUILD_COMPILERS], BASIC_FIXUP_EXECUTABLE(BUILD_NM) BASIC_PATH_PROGS(BUILD_AR, ar gcc-ar) BASIC_FIXUP_EXECUTABLE(BUILD_AR) + BASIC_PATH_PROGS(BUILD_OBJCOPY, objcopy) + BASIC_FIXUP_EXECUTABLE(BUILD_OBJCOPY) + BASIC_PATH_PROGS(BUILD_STRIP, strip) + BASIC_FIXUP_EXECUTABLE(BUILD_STRIP) # Assume the C compiler is the assembler BUILD_AS="$BUILD_CC -c" # Just like for the target compiler, use the compiler as linker @@ -813,6 +817,8 @@ AC_DEFUN_ONCE([TOOLCHAIN_SETUP_BUILD_COMPILERS], BUILD_LDCXX="$LDCXX" BUILD_NM="$NM" BUILD_AS="$AS" + BUILD_OBJCOPY="$OBJCOPY" + BUILD_STRIP="$STRIP" BUILD_SYSROOT_CFLAGS="$SYSROOT_CFLAGS" BUILD_SYSROOT_LDFLAGS="$SYSROOT_LDFLAGS" BUILD_AR="$AR" diff --git a/common/bin/compare.sh b/common/bin/compare.sh index a629ae084eb..d073fbbda32 100644 --- a/common/bin/compare.sh +++ b/common/bin/compare.sh @@ -290,9 +290,9 @@ compare_general_files() { GENERAL_FILES=$(cd $THIS_DIR && $FIND . -type f ! -name "*.so" ! -name "*.jar" \ ! -name "*.zip" ! -name "*.debuginfo" ! -name "*.dylib" ! -name "jexec" \ - ! -name "*.jimage" ! -name "ct.sym" ! -name "*.diz" ! -name "*.dll" \ + ! -name "modules" ! -name "ct.sym" ! -name "*.diz" ! -name "*.dll" \ ! -name "*.cpl" ! -name "*.pdb" ! -name "*.exp" ! -name "*.ilk" \ - ! -name "*.lib" ! -name "*.war" ! -name "JavaControlPanel" \ + ! -name "*.lib" ! -name "*.war" ! -name "JavaControlPanel" ! -name "*.jmod" \ ! -name "*.obj" ! -name "*.o" ! -name "JavaControlPanelHelper" \ ! -name "JavaUpdater" ! -name "JavaWSApplicationStub" \ ! -name "jspawnhelper" ! -name "JavawsLauncher" ! -name "*.a" \ @@ -389,13 +389,13 @@ compare_zip_file() { $RM -rf $THIS_UNZIPDIR $OTHER_UNZIPDIR $MKDIR -p $THIS_UNZIPDIR $MKDIR -p $OTHER_UNZIPDIR - if [ "$TYPE" = "jimage" ] + if [ "$TYPE" = "jar" || "$TYPE" = "war" || "$TYPE" = "zip" || "$TYPE" = "jmod"] then - (cd $THIS_UNZIPDIR && $JIMAGE extract $THIS_ZIP) - (cd $OTHER_UNZIPDIR && $JIMAGE extract $OTHER_ZIP) - else (cd $THIS_UNZIPDIR && $UNARCHIVE $THIS_ZIP) (cd $OTHER_UNZIPDIR && $UNARCHIVE $OTHER_ZIP) + else + (cd $THIS_UNZIPDIR && $JIMAGE extract $THIS_ZIP) + (cd $OTHER_UNZIPDIR && $JIMAGE extract $OTHER_ZIP) fi # Find all archives inside and unzip them as well to compare the contents rather than @@ -526,7 +526,7 @@ compare_all_jar_files() { # TODO filter? ZIPS=$(cd $THIS_DIR && $FIND . -type f -name "*.jar" -o -name "*.war" \ - -o -name "*.jimage" | $SORT | $FILTER) + -o -name "modules" -o -name "*.jmod" | $SORT | $FILTER) if [ -n "$ZIPS" ]; then echo Jar files... diff --git a/common/conf/jib-profiles.js b/common/conf/jib-profiles.js index 48519cfbe74..27fbe1f02cf 100644 --- a/common/conf/jib-profiles.js +++ b/common/conf/jib-profiles.js @@ -421,10 +421,10 @@ var getJibProfilesDependencies = function (input, common) { jtreg: { server: "javare", - revision: "4.1", - build_number: "b12", + revision: "4.2", + build_number: "b01", checksum_file: "MD5_VALUES", - file: "jtreg_bin-4.1.zip", + file: "jtreg_bin-4.2.zip", environment_name: "JT_HOME" }, diff --git a/make/CheckModules.gmk b/make/CheckModules.gmk deleted file mode 100644 index 4872c912e49..00000000000 --- a/make/CheckModules.gmk +++ /dev/null @@ -1,46 +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. Oracle designates this -# particular file as subject to the "Classpath" exception as provided -# by Oracle in the LICENSE file that accompanied this code. -# -# This code is distributed in the hope that it will be useful, but WITHOUT -# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or -# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License -# version 2 for more details (a copy is included in the LICENSE file that -# accompanied this code). -# -# You should have received a copy of the GNU General Public License version -# 2 along with this work; if not, write to the Free Software Foundation, -# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. -# -# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA -# or visit www.oracle.com if you need additional information or have any -# questions. -# - -# Default target declared first -default: all - -include $(SPEC) -include MakeBase.gmk - -JDEPS_MODULES_XML := $(JDK_OUTPUTDIR)/modules/jdk.jdeps/com/sun/tools/jdeps/resources/jdeps-modules.xml - -# -# Verify access across module boundaries -# -checkdeps: - $(ECHO) "Checking dependencies across JDK modules" - $(JAVA) -Xbootclasspath/p:$(INTERIM_LANGTOOLS_JAR) \ - -Djdeps.modules.xml=$(JDEPS_MODULES_XML) \ - com.sun.tools.jdeps.Main \ - -verify:access -mp $(JDK_OUTPUTDIR)/modules - -all: checkdeps - -.PHONY: all diff --git a/make/CompileJavaModules.gmk b/make/CompileJavaModules.gmk index e8928e2130d..a2650d730bb 100644 --- a/make/CompileJavaModules.gmk +++ b/make/CompileJavaModules.gmk @@ -46,6 +46,9 @@ java.base_ADD_JAVAC_FLAGS := -Xdoclint:all/protected,-reference '-Xdoclint/packa java.base_COPY := .icu .dat .spp content-types.properties hijrah-config-islamic-umalqura.properties java.base_CLEAN := intrinsic.properties +java.base_EXCLUDE_FILES += \ + $(JDK_TOPDIR)/src/java.base/share/classes/jdk/internal/module/ModuleLoaderMap.java + java.base_EXCLUDES += java/lang/doc-files # Exclude BreakIterator classes that are just used in compile process to generate @@ -359,8 +362,8 @@ jdk.charsets_COPY := .dat ################################################################################ -jdk.compiler_ADD_JAVAC_FLAGS := -Xdoclint:all/protected '-Xdoclint/package:-com.sun.tools.*' -XDstringConcat=inline -jdk.compiler_COPY := javax.tools.JavaCompilerTool +jdk.compiler_ADD_JAVAC_FLAGS := -Xdoclint:all/protected '-Xdoclint/package:-com.sun.tools.*' \ + -XDstringConcat=inline jdk.compiler_CLEAN_FILES := $(wildcard \ $(patsubst %, $(JDK_TOPDIR)/src/jdk.compiler/share/classes/%/*.properties, \ sun/tools/serialver/resources)) @@ -480,31 +483,28 @@ jdk.localedata_EXCLUDE_FILES += sun/text/resources/th/BreakIteratorRules_th.java # source before static source and platform specific source before shared. # GENERATED_SRC_DIRS += \ - $(SUPPORT_OUTPUTDIR)/gensrc/$(MODULE) \ - $(SUPPORT_OUTPUTDIR)/gensrc_no_docs/$(MODULE) \ + $(SUPPORT_OUTPUTDIR)/gensrc \ # -OS_SRC_DIRS += $(JDK_TOPDIR)/src/$(MODULE)/$(OPENJDK_TARGET_OS)/classes +TOP_SRC_DIRS += \ + $(HOTSPOT_TOPDIR)/src \ + $(CORBA_TOPDIR)/src \ + $(JDK_TOPDIR)/src \ + $(LANGTOOLS_TOPDIR)/src \ + $(JAXP_TOPDIR)/src \ + $(JAXWS_TOPDIR)/src \ + $(NASHORN_TOPDIR)/src \ + # + +SRC_SUBDIRS += $(OPENJDK_TARGET_OS)/classes ifneq ($(OPENJDK_TARGET_OS), $(OPENJDK_TARGET_OS_TYPE)) - OS_TYPE_SRC_DIRS += $(JDK_TOPDIR)/src/$(MODULE)/$(OPENJDK_TARGET_OS_TYPE)/classes + SRC_SUBDIRS += $(OPENJDK_TARGET_OS_TYPE)/classes endif +SRC_SUBDIRS += share/classes -SHARE_SRC_DIRS += \ - $(HOTSPOT_TOPDIR)/src/$(MODULE)/share/classes \ - $(JDK_TOPDIR)/src/$(MODULE)/share/classes \ - $(LANGTOOLS_TOPDIR)/src/$(MODULE)/share/classes \ - $(CORBA_TOPDIR)/src/$(MODULE)/share/classes \ - $(JAXP_TOPDIR)/src/$(MODULE)/share/classes \ - $(JAXWS_TOPDIR)/src/$(MODULE)/share/classes \ - $(NASHORN_TOPDIR)/src/$(MODULE)/share/classes \ - # - -ALL_SRC_DIRS = \ - $(GENERATED_SRC_DIRS) \ - $(OS_SRC_DIRS) \ - $(OS_TYPE_SRC_DIRS) \ - $(SHARE_SRC_DIRS) \ - # +MODULE_SRC_DIRS := $(strip \ + $(addsuffix /$(MODULE), $(GENERATED_SRC_DIRS) $(IMPORT_MODULES_SRC)) \ + $(foreach sub, $(SRC_SUBDIRS), $(addsuffix /$(MODULE)/$(sub), $(TOP_SRC_DIRS)))) # The JDK_USER_DEFINED_FILTER is a poor man's incremental build: by specifying # JDK_FILTER at the make command line, only a subset of the JDK java files will @@ -512,32 +512,41 @@ ALL_SRC_DIRS = \ # space separated list. JDK_USER_DEFINED_FILTER := $(strip $(subst $(COMMA),$(SPACE), $(JDK_FILTER))) -# Create an empty directory to set the bootclasspath to. -EMPTY_DIR := $(SUPPORT_OUTPUTDIR)/empty-dir -$(call MakeDir, $(EMPTY_DIR)) +# Rewrite the MODULE_SRC_DIRS with a wildcard for the module so that all module +# source dirs are available on the path. +MODULESOURCEPATH := $(subst $(SPACE),$(PATH_SEP),$(subst $(MODULE),*,$(MODULE_SRC_DIRS))) -# Find the module dependencies by parsing modules.list file -DEPS := $(call FindDepsForModule, $(MODULE)) +# Add imported modules to the moduleclasspath +MODULECLASSPATH := $(subst $(SPACE),$(PATH_SEP), $(IMPORT_MODULES_CLASSES)) -CLASSPATH := $(foreach d, $(DEPS), $(if $($d_BIN), $($d_BIN), \ - $(JDK_OUTPUTDIR)/modules/$d)) - # When crypto classes are prebuilt, need to look for classes already in - # output dir. - ifneq ($(BUILD_CRYPTO), true) - CLASSPATH += $(JDK_OUTPUTDIR)/modules/$(MODULE) - endif -JAVAC_FLAGS_BOOTCLASSPATH := -bootclasspath $(EMPTY_DIR) -extdirs $(EMPTY_DIR) \ - -endorseddirs $(EMPTY_DIR) $($(MODULE)_ADD_JAVAC_FLAGS) +ifeq ($(MODULE), jdk.vm.ci) + ## WORKAROUND jdk.vm.ci source structure issue + JVMCI_MODULESOURCEPATH := $(MODULESOURCEPATH) \ + $(subst /$(MODULE)/,/*/, $(filter-out %processor/src, \ + $(wildcard $(HOTSPOT_TOPDIR)/src/jdk.vm.ci/share/classes/*/src))) + MODULESOURCEPATH := $(subst $(SPACE),$(PATH_SEP), $(JVMCI_MODULESOURCEPATH)) +endif + +# Make sure the generated source base dirs exist. Not all modules have generated +# source in all of these directories and because of timing, all of them might not +# exist at the time this makefile gets called. Javac will complain if there are +# missing directories in the moduleclasspath. +$(call MakeDir, $(GENERATED_SRC_DIRS)) $(eval $(call SetupJavaCompilation, $(MODULE), \ SETUP := $(if $($(MODULE)_SETUP), $($(MODULE)_SETUP), GENERATE_JDKBYTECODE), \ - SRC := $(if $($(MODULE)_SRC), $($(MODULE)_SRC), $(wildcard $(ALL_SRC_DIRS))), \ - INCLUDES := $(JDK_USER_DEFINED_FILTER),\ - BIN := $(if $($(MODULE)_BIN), $($(MODULE)_BIN), $(JDK_OUTPUTDIR)/modules/$(MODULE)), \ - HEADERS := $(SUPPORT_OUTPUTDIR)/headers/$(MODULE), \ - CLASSPATH := $(CLASSPATH), \ - ADD_JAVAC_FLAGS := $($(MODULE)_ADD_JAVAC_FLAGS) $(JAVAC_FLAGS_BOOTCLASSPATH) \ + MODULE := $(MODULE), \ + SRC := $(wildcard $(MODULE_SRC_DIRS)), \ + INCLUDES := $(JDK_USER_DEFINED_FILTER),\ + BIN := $(if $($(MODULE)_BIN), $($(MODULE)_BIN), $(JDK_OUTPUTDIR)/modules), \ + HEADERS := $(SUPPORT_OUTPUTDIR)/headers, \ + ADD_JAVAC_FLAGS := \ + $($(MODULE)_ADD_JAVAC_FLAGS) \ + -modulesourcepath "$(MODULESOURCEPATH)" \ + $(if $(MODULECLASSPATH), -modulepath "$(MODULECLASSPATH)") \ + -system none, \ )) + TARGETS += $($(MODULE)) $($(MODULE)_COPY_EXTRA) # Declare dependencies between java compilations of different modules. @@ -545,7 +554,7 @@ TARGETS += $($(MODULE)) $($(MODULE)_COPY_EXTRA) # use the macro to find the correct target file to depend on. # Only the javac compilation actually depends on other modules so limit # dependency declaration to that by using the *_COMPILE_TARGET variable. -$($(MODULE)_COMPILE_TARGET): $(foreach d, $($(MODULE)_DEPS), \ +$($(MODULE)_COMPILE_TARGET): $(foreach d, $(call FindDepsForModule, $(MODULE)), \ $(call SetupJavaCompilationCompileTarget, $d, \ $(if $($d_BIN), $($d_BIN), $(JDK_OUTPUTDIR)/modules/$d))) @@ -568,6 +577,29 @@ ifeq ($(MODULE), java.rmi) TARGETS += $(call CreateHkTargets, $(java.rmi_CLEAN_FILES)) endif +################################################################################ +# If this is an imported module, copy the pre built classes and resources into +# the modules output dir + +ifneq ($(wildcard $(IMPORT_MODULES_CLASSES)/$(MODULE)), ) + $(JDK_OUTPUTDIR)/modules/$(MODULE)/_imported.marker: \ + $(call CacheFind, $(IMPORT_MODULES_CLASSES)/$(MODULE)) + $(RM) -r $(@D) + $(MKDIR) -p $(@D) + $(CP) -R $(IMPORT_MODULES_CLASSES)/$(MODULE)/* $(@D)/ + $(TOUCH) $@ + + TARGETS += $(JDK_OUTPUTDIR)/modules/$(MODULE)/_imported.marker + + # Add this dependency to avoid a race between compiling module-info.java and + # importing the classes. + $($(MODULE)_COMPILE_TARGET): $(JDK_OUTPUTDIR)/modules/$(MODULE)/_imported.marker +endif + +################################################################################ + +$(eval $(call IncludeCustomExtension, , CompileJavaModules-post.gmk)) + ################################################################################ all: $(TARGETS) diff --git a/make/CopyImportModules.gmk b/make/CopyImportModules.gmk new file mode 100644 index 00000000000..e79288557a1 --- /dev/null +++ b/make/CopyImportModules.gmk @@ -0,0 +1,86 @@ +# +# 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. Oracle designates this +# particular file as subject to the "Classpath" exception as provided +# by Oracle in the LICENSE file that accompanied this code. +# +# This code is distributed in the hope that it will be useful, but WITHOUT +# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +# version 2 for more details (a copy is included in the LICENSE file that +# accompanied this code). +# +# You should have received a copy of the GNU General Public License version +# 2 along with this work; if not, write to the Free Software Foundation, +# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. +# +# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA +# or visit www.oracle.com if you need additional information or have any +# questions. +# + +# This makefile is called for every imported module to copy the non class +# contents into the exploded jdk image. + +default: all + +include $(SPEC) +include MakeBase.gmk + +LIBS_DIR := $(wildcard $(addsuffix /$(MODULE), $(IMPORT_MODULES_LIBS))) +CMDS_DIR := $(wildcard $(addsuffix /$(MODULE), $(IMPORT_MODULES_CMDS))) +CONF_DIR := $(wildcard $(addsuffix /$(MODULE), $(IMPORT_MODULES_CONF))) + +$(eval $(call FillCacheFind, $(LIBS_DIR) $(CMDS_DIR) $(CONF_DIR))) + +ifneq ($(LIBS_DIR), ) + ifeq ($(OPENJDK_TARGET_OS), windows) + TO_BIN_FILTER := %$(SHARED_LIBRARY_SUFFIX) %.diz %.pdb %.map + + $(eval $(call SetupCopyFiles,COPY_LIBS_TO_BIN, \ + SRC := $(LIBS_DIR), \ + DEST := $(JDK_OUTPUTDIR)/bin, \ + FILES := $(filter $(TO_BIN_FILTER), \ + $(call CacheFind, $(LIBS_DIR))) \ + )) + + $(eval $(call SetupCopyFiles,COPY_LIBS_TO_LIB, \ + SRC := $(LIBS_DIR), \ + DEST := $(JDK_OUTPUTDIR)/lib, \ + FILES := $(filter-out $(TO_BIN_FILTER), \ + $(call CacheFind, $(LIBS_DIR))) \ + )) + TARGETS += $(COPY_LIBS_TO_BIN) $(COPY_LIBS_TO_LIB) + else + $(eval $(call SetupCopyFiles, COPY_LIBS, \ + SRC := $(LIBS_DIR), \ + DEST := $(JDK_OUTPUTDIR)/lib, \ + FILES := $(call CacheFind, $(LIBS_DIR)), \ + )) + TARGETS += $(COPY_LIBS) + endif +endif + +ifneq ($(CMDS_DIR), ) + $(eval $(call SetupCopyFiles, COPY_CMDS, \ + SRC := $(CMDS_DIR), \ + DEST := $(JDK_OUTPUTDIR)/bin, \ + FILES := $(call CacheFind, $(CMDS_DIR)), \ + )) + TARGETS += $(COPY_CMDS) +endif + +ifneq ($(CONF_DIR), ) + $(eval $(call SetupCopyFiles, COPY_CONF, \ + SRC := $(CONF_DIR), \ + DEST := $(JDK_OUTPUTDIR)/lib, \ + FILES := $(call CacheFind, $(CONF_DIR)), \ + )) + TARGETS += $(COPY_CONF) +endif + +all: $(TARGETS) diff --git a/make/CreateBuildJdkCopy.gmk b/make/CreateBuildJdkCopy.gmk new file mode 100644 index 00000000000..ca8b3e10491 --- /dev/null +++ b/make/CreateBuildJdkCopy.gmk @@ -0,0 +1,86 @@ +# +# 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. Oracle designates this +# particular file as subject to the "Classpath" exception as provided +# by Oracle in the LICENSE file that accompanied this code. +# +# This code is distributed in the hope that it will be useful, but WITHOUT +# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +# version 2 for more details (a copy is included in the LICENSE file that +# accompanied this code). +# +# You should have received a copy of the GNU General Public License version +# 2 along with this work; if not, write to the Free Software Foundation, +# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. +# +# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA +# or visit www.oracle.com if you need additional information or have any +# questions. +# + +default: all + +include $(SPEC) +include MakeBase.gmk +include Modules.gmk + +MODULES_TO_COPY := $(call FindTransitiveDepsForModule, jdk.jlink) jdk.jlink + +################################################################################ +# Copy the modules needed to run jlink and jmod. Use bulk copy instead of +# SetupCopyFiles since there are so many files. + +COPY_CLASSES_TARGET := $(BUILDJDK_OUTPUTDIR)/jdk/modules/_buildjdk-copy-maker + +$(COPY_CLASSES_TARGET): $(call CacheFind, $(wildcard \ + $(addprefix $(JDK_OUTPUTDIR)/modules/, $(MODULES_TO_COPY)))) + $(ECHO) $(LOG_INFO) "Copying java modules to buildjdk: $(MODULES_TO_COPY)" + $(RM) -r $(BUILDJDK_OUTPUTDIR)/jdk/modules + $(MKDIR) -p $(BUILDJDK_OUTPUTDIR)/jdk/modules + $(foreach m, $(MODULES_TO_COPY), \ + $(CP) -R $(JDK_OUTPUTDIR)/modules/$m \ + $(BUILDJDK_OUTPUTDIR)/jdk/modules/ $(NEWLINE)) + $(TOUCH) $@ + +TARGETS += $(COPY_CLASSES_TARGET) + +#$(eval $(call SetupCopyFiles, COPY_JDK_MODULES, \ + SRC := $(BUILD_OUTPUT), \ + DEST := $(BUILDJDK_OUTPUTDIR), \ + FILES := $(call DoubleDollar, $(call DoubleDollar, $(call CacheFind, $(wildcard \ + $(addprefix $(JDK_OUTPUTDIR)/modules/, $(MODULES_TO_COPY)))))), \ +)) + +#TARGETS += $(COPY_JDK_MODULES) + +################################################################################ + +$(eval $(call SetupCopyFiles, COPY_SUPPORT_HEADERS, \ + SRC := $(BUILD_OUTPUT), \ + DEST := $(BUILDJDK_OUTPUTDIR), \ + FILES := $(call CacheFind, $(wildcard \ + $(addprefix $(SUPPORT_OUTPUTDIR)/headers/, $(MODULES_TO_COPY)))), \ +)) + +TARGETS += $(COPY_SUPPORT_HEADERS) + +################################################################################ + +$(eval $(call SetupCopyFiles, COPY_JDK_LIB_FILES, \ + SRC := $(BUILD_OUTPUT), \ + DEST := $(BUILDJDK_OUTPUTDIR), \ + FILES := $(JDK_OUTPUTDIR)/lib/tzdb.dat, \ +)) + +TARGETS += $(COPY_JDK_LIB_FILES) + +################################################################################ + +all: $(TARGETS) + +.PHONY: default all diff --git a/make/CreateJmods.gmk b/make/CreateJmods.gmk new file mode 100644 index 00000000000..fdfb7dc6884 --- /dev/null +++ b/make/CreateJmods.gmk @@ -0,0 +1,95 @@ +# +# 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. Oracle designates this +# particular file as subject to the "Classpath" exception as provided +# by Oracle in the LICENSE file that accompanied this code. +# +# This code is distributed in the hope that it will be useful, but WITHOUT +# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +# version 2 for more details (a copy is included in the LICENSE file that +# accompanied this code). +# +# You should have received a copy of the GNU General Public License version +# 2 along with this work; if not, write to the Free Software Foundation, +# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. +# +# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA +# or visit www.oracle.com if you need additional information or have any +# questions. +# + +default: all + +include $(SPEC) +include MakeBase.gmk +include Modules.gmk + +ifeq ($(MODULE), ) + $(error MODULE must be set when calling CreateJmods.gmk) +endif + +################################################################################ + +LIBS_DIR := $(firstword $(wildcard $(addsuffix /$(MODULE), \ + $(SUPPORT_OUTPUTDIR)/modules_libs-stripped $(IMPORT_MODULES_LIBS)))) +CMDS_DIR := $(firstword $(wildcard $(addsuffix /$(MODULE), \ + $(SUPPORT_OUTPUTDIR)/modules_cmds-stripped $(IMPORT_MODULES_CMDS)))) +CONF_DIR := $(firstword $(wildcard $(addsuffix /$(MODULE), \ + $(SUPPORT_OUTPUTDIR)/modules_conf $(IMPORT_MODULES_CONF)))) +CLASSES_DIR := $(wildcard $(JDK_OUTPUTDIR)/modules/$(MODULE)) + +$(eval $(call FillCacheFind, \ + $(LIBS_DIR) $(CMDS_DIR) $(CONF_DIR) $(CLASSES_DIR) \ +)) + +ifneq ($(LIBS_DIR), ) + JMOD_FLAGS += --libs $(LIBS_DIR) + DEPS += $(call CacheFind, $(LIBS_DIR)) +endif +ifneq ($(CMDS_DIR), ) + JMOD_FLAGS += --cmds $(CMDS_DIR) + DEPS += $(call CacheFind, $(CMDS_DIR)) +endif +ifneq ($(CONF_DIR), ) + JMOD_FLAGS += --config $(CONF_DIR) + DEPS += $(call CacheFind, $(CONF_DIR)) +endif +ifneq ($(CLASSES_DIR), ) + JMOD_FLAGS += --class-path $(CLASSES_DIR) + DEPS += $(call CacheFind, $(CLASSES_DIR)) +endif + +# Add dependencies on other jmod files +DEPS += $(patsubst %, $(IMAGES_OUTPUTDIR)/jmods/%.jmod, \ + $(call FindDepsForModule, $(MODULE))) + +# TODO: What about headers? +# Create jmods in a temp dir and then move them into place to keep the +# module path in $(IMAGES_OUTPUTDIR)/jmods valid at all times. +$(IMAGES_OUTPUTDIR)/jmods/$(MODULE).jmod: $(DEPS) + $(call LogWarn, Creating $(notdir $@)) + $(call MakeDir, $(@D) $(SUPPORT_OUTPUTDIR)/jmods) + $(RM) $@ $(SUPPORT_OUTPUTDIR)/jmods/$(notdir $@) + $(JMOD) create \ + --module-version $(VERSION_SHORT) \ + --os-name $(REQUIRED_OS_NAME) \ + --os-arch $(OPENJDK_TARGET_CPU_LEGACY) \ + --os-version $(REQUIRED_OS_VERSION) \ + --modulepath $(IMAGES_OUTPUTDIR)/jmods\ + --hash-dependencies '.*' \ + --exclude '**_the.*' \ + $(JMOD_FLAGS) $(SUPPORT_OUTPUTDIR)/jmods/$(notdir $@) + $(MV) $(SUPPORT_OUTPUTDIR)/jmods/$(notdir $@) $@ + +TARGETS += $(IMAGES_OUTPUTDIR)/jmods/$(MODULE).jmod + +################################################################################ + +all: $(TARGETS) + +################################################################################ diff --git a/make/GenerateModuleDeps.gmk b/make/GenerateModuleDeps.gmk deleted file mode 100644 index b5427dd89a5..00000000000 --- a/make/GenerateModuleDeps.gmk +++ /dev/null @@ -1,64 +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. Oracle designates this -# particular file as subject to the "Classpath" exception as provided -# by Oracle in the LICENSE file that accompanied this code. -# -# This code is distributed in the hope that it will be useful, but WITHOUT -# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or -# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License -# version 2 for more details (a copy is included in the LICENSE file that -# accompanied this code). -# -# You should have received a copy of the GNU General Public License version -# 2 along with this work; if not, write to the Free Software Foundation, -# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. -# -# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA -# or visit www.oracle.com if you need additional information or have any -# questions. -# - -# This file is included from Main.gmk only. - -include $(SRC_ROOT)/make/common/JavaCompilation.gmk -include $(SRC_ROOT)/make/common/SetupJavaCompilers.gmk - -################################################################################ - -$(eval $(call SetupJavaCompilation, BUILD_GENMODULESLIST, \ - SETUP := BOOT_JAVAC, \ - SRC := $(JDK_TOPDIR)/make/src/classes, \ - INCLUDES := build/tools/module, \ - EXCLUDE_FILES := ImageBuilder.java ModuleArchive.java, \ - BIN := $(MAKESUPPORT_OUTPUTDIR)/bt_classes_moduleslist, \ - DISABLE_SJAVAC := true, \ -)) - -TOOL_GENMODULESLIST = $(JAVA_SMALL) \ - -cp "$(MAKESUPPORT_OUTPUTDIR)/bt_classes_moduleslist" \ - build.tools.module.GenModulesList - -MODULES_LIST_FILE := $(MAKESUPPORT_OUTPUTDIR)/modules.list -# The module deps makefile is used from make/common/Modules.gmk -MODULE_DEPS_MAKEFILE := $(MAKESUPPORT_OUTPUTDIR)/module-deps.gmk - -$(MODULES_LIST_FILE): $(SRC_ROOT)/modules.xml $(BUILD_GENMODULESLIST) - $(TOOL_GENMODULESLIST) -o $@ $(filter %.xml, $^) - -$(MODULE_DEPS_MAKEFILE): $(MODULES_LIST_FILE) - $(CAT) $^ | $(SED) -e 's/^\([^:]*\):/DEPS_\1 :=/g' > $@ - -TARGETS += $(MODULE_DEPS_MAKEFILE) - -################################################################################ - -# Hook to include the corresponding custom file, if present. -$(eval $(call IncludeCustomExtension, , GenerateModuleDeps.gmk)) - -# Trigger generation of this file and restart make if it changed. --include $(MODULE_DEPS_MAKEFILE) diff --git a/make/GensrcModuleInfo.gmk b/make/GensrcModuleInfo.gmk new file mode 100644 index 00000000000..b53fc5a1256 --- /dev/null +++ b/make/GensrcModuleInfo.gmk @@ -0,0 +1,154 @@ +# +# 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. Oracle designates this +# particular file as subject to the "Classpath" exception as provided +# by Oracle in the LICENSE file that accompanied this code. +# +# This code is distributed in the hope that it will be useful, but WITHOUT +# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or +# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +# version 2 for more details (a copy is included in the LICENSE file that +# accompanied this code). +# +# You should have received a copy of the GNU General Public License version +# 2 along with this work; if not, write to the Free Software Foundation, +# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. +# +# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA +# or visit www.oracle.com if you need additional information or have any +# questions. +# + +################################################################################ +# This file makes modifications to module-info.java files based on the build +# configuration. +# +# Depending on build platform, imported modules and optional parts of the build +# being active, some modules need to have extra exports, provides or uses +# declarations added to them. These optional extras are defined in .extra files: +# +# src///classes/module-info.java.extra +# +# The contents of the .extra files are simply extra lines that could fit into +# the module-info file. +# +# This makefile is called once for each from-module with the variable +# MODULE naming the from-module. +# +# The modified module-info.java files are put in the gensrc directory where +# they will automatically override the static versions in the src tree. +# +################################################################################ + +default: all + +include $(SPEC) +include MakeBase.gmk +include Modules.gmk +#include TextFileProcessing.gmk + +################################################################################ +# Define this here since jdk/make/Tools.gmk cannot be included from the top +# make directory. Should probably move some tools away from the jdk repo. +TOOL_GENMODULEINFOSOURCE = $(JAVA_SMALL) \ + $(INTERIM_LANGTOOLS_ARGS) \ + -cp "$(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes" \ + build.tools.module.GenModuleInfoSource + +################################################################################ + +# Name of data file. Keep module-info.java.ext until javafx has changed. +MOD_FILENAME := module-info.java.extra module-info.java.ext + +# List all the possible sub directories inside a module source directory where +# data might be stored. +CLASSES_SUBDIRS += $(OPENJDK_TARGET_OS)/classes +ifneq ($(OPENJDK_TARGET_OS), $(OPENJDK_TARGET_OS_TYPE)) + CLASSES_SUBDIRS += $(OPENJDK_TARGET_OS_TYPE)/classes +endif +CLASSES_SUBDIRS += share/classes + +# TODO: When the deploy build is better integrated, this will get added globally +# but for now need to add it here. +ifeq ($(BUILD_DEPLOY), true) + ALL_TOP_SRC_DIRS += $(DEPLOY_TOPDIR)/src +endif + +# Construct all possible src directories for the module. +MODULE_CLASSES_DIRS := $(strip \ + $(foreach sub, $(CLASSES_SUBDIRS), \ + $(addsuffix /$(MODULE)/$(sub), $(ALL_TOP_SRC_DIRS))) \ + $(addsuffix /$(MODULE), $(IMPORT_MODULES_SRC))) + +# Find all the .extra files in the src dirs. +MOD_FILES := $(wildcard $(foreach f, $(MOD_FILENAME), $(addsuffix /$(f), \ + $(MODULE_CLASSES_DIRS)))) + +ifneq ($(MOD_FILES), ) + # Only make this call if modification files are found for this module + ALL_MODULES := $(call FindAllModules) + + # Read the contents of all the files into a variable. Replace space with / to + # let space represent new lines in the variable as $(shell) normalizes all + # whitespace. + $(foreach f, $(MOD_FILES), \ + $(eval MOD_FILE_CONTENTS += $(shell $(GREP) -v ".\*" $f | $(TR) ' ' '/'))) + + # Filter the contents for modules that are actually being built + MODULES_FILTER := $(addprefix %/, $(addsuffix ;, $(ALL_MODULES))) + MODULES_FILTER += provides% + MODIFICATIONS := $(filter $(MODULES_FILTER), $(MOD_FILE_CONTENTS)) + + # Convert the modification lines into arguments for the modification tool. + # Filter out modifications for non existing to-modules. + $(foreach line, $(MODIFICATIONS), \ + $(eval split_line := $(subst /,$(SPACE),$(line))) \ + $(eval command := $(word 1, $(split_line))) \ + $(eval package := $(word 2, $(split_line))) \ + $(eval to_module := $(patsubst %;,%,$(word 4, $(split_line)))) \ + $(eval ARGS += -$(command) $(package)/$(to_module))) + + ifneq ($(ARGS), ) + $(SUPPORT_OUTPUTDIR)/gensrc/$(MODULE)/module-info.java: \ + $(firstword $(call FindAllModuleInfos, $(MODULE))) \ + $(BUILD_TOOLS_JDK) \ + $(call DependOnVariable, ARGS) + $(MKDIR) -p $(@D) + $(RM) $@ $@.tmp + $(TOOL_GENMODULEINFOSOURCE) $(ARGS) -o $@.tmp $< + $(MV) $@.tmp $@ + + TARGETS += $(SUPPORT_OUTPUTDIR)/gensrc/$(MODULE)/module-info.java + endif + +# This doesn't work because javac only accepts one single exports line per +# exported package. + # Restore the modifications to separate lines with spaces +# MODIFICATIONS := $(subst /,$(SPACE),$(MODIFICATIONS)) + +# ifneq ($(MODIFICATIONS), ) +# $(eval $(call SetupTextFileProcessing, PROCESS_MODULE_INFO, \ +# SOURCE_FILES := $(firstword $(call FindAllModuleInfos, $(MODULE))), \ +# OUTPUT_FILE := $(SUPPORT_OUTPUTDIR)/gensrc/$(MODULE)/module-info.java, \ +# REPLACEMENTS := } => $(MODIFICATIONS) }, \ +# )) + +# TARGETS += $(PROCESS_MODULE_INFO) +# endif +endif + +# If no modifications are found for this module, remove any module-info.java +# created by a previous build since that is no longer valid. +ifeq ($(MODIFICATIONS), ) + ifneq ($(wildcard $(SUPPORT_OUTPUTDIR)/gensrc/$(MODULE)/module-info.java), ) + $(shell $(RM) $(SUPPORT_OUTPUTDIR)/gensrc/$(MODULE)/module-info.java) + endif +endif + +################################################################################ + +all: $(TARGETS) diff --git a/make/HotspotWrapper.gmk b/make/HotspotWrapper.gmk index eec08d6e129..663215525a3 100644 --- a/make/HotspotWrapper.gmk +++ b/make/HotspotWrapper.gmk @@ -25,7 +25,10 @@ # Include the legacy hotspot-spec.gmk (which in turns includes spec.gmk) BASE_SPEC:=$(SPEC) -include $(dir $(SPEC))hotspot-spec.gmk +# Assign to HOTSPOT_SPEC so that the variable HOTSPOT_SPEC can be +# overridden when building the buildjdk. +HOTSPOT_SPEC := $(dir $(SPEC))hotspot-spec.gmk +include $(HOTSPOT_SPEC) include MakeBase.gmk # Inclusion of this pseudo-target will cause make to execute this file @@ -45,7 +48,7 @@ HOTSPOT_FILES := $(shell $(FIND) -L \ # not doing it breaks builds on msys. $(HOTSPOT_OUTPUTDIR)/_hotspot.timestamp: $(HOTSPOT_FILES) @$(MKDIR) -p $(HOTSPOT_OUTPUTDIR) - @($(CD) $(HOTSPOT_TOPDIR)/make && $(MAKE) $(HOTSPOT_MAKE_ARGS) \ + ($(CD) $(HOTSPOT_TOPDIR)/make && $(MAKE) $(HOTSPOT_MAKE_ARGS) \ LOG_LEVEL=$(LOG_LEVEL) SPEC=$(HOTSPOT_SPEC) BASE_SPEC=$(BASE_SPEC)) $(TOUCH) $@ diff --git a/make/Images.gmk b/make/Images.gmk index 7fc063773d6..3aa4173b4cf 100644 --- a/make/Images.gmk +++ b/make/Images.gmk @@ -38,7 +38,7 @@ $(eval $(call IncludeCustomExtension, , Images-pre.gmk)) ############################################################################ -MAIN_MODULES += java.se java.smartcardio jdk.httpserver jdk.sctp \ +MAIN_MODULES += java.se.ee java.smartcardio jdk.httpserver jdk.sctp \ jdk.security.auth jdk.security.jgss jdk.pack200 jdk.xml.dom \ jdk.accessibility jdk.internal.le jdk.dynalink \ jdk.scripting.nashorn jdk.scripting.nashorn.shell \ @@ -49,10 +49,10 @@ PROVIDER_MODULES += jdk.charsets jdk.crypto.ec jdk.crypto.pkcs11 jdk.jvmstat jdk jdk.localedata jdk.naming.dns jdk.naming.rmi jdk.zipfs # tools -TOOLS_MODULES += jdk.attach jdk.compiler jdk.dev \ +TOOLS_MODULES += jdk.attach jdk.compiler \ jdk.javadoc jdk.jcmd jdk.jconsole jdk.hotspot.agent jdk.jartool \ - jdk.jdeps jdk.jdi jdk.jdwp.agent jdk.jshell jdk.policytool jdk.rmic \ - jdk.xml.bind jdk.xml.ws + jdk.jdeps jdk.jdi jdk.jdwp.agent jdk.jlink jdk.jshell \ + jdk.policytool jdk.rmic jdk.xml.bind jdk.xml.ws jdk.internal.opt ifeq ($(OPENJDK_TARGET_OS), windows) PROVIDER_MODULES += jdk.crypto.mscapi @@ -65,104 +65,151 @@ endif JRE_MODULES := $(filter-out $(MODULES_FILTER), $(MAIN_MODULES) $(PROVIDER_MODULES)) JDK_MODULES := $(filter-out $(MODULES_FILTER), $(JRE_MODULES) $(TOOLS_MODULES)) -# compact3 builds have additional modules -JDK_COMPACT3_MODULES := java.compact3 java.smartcardio jdk.httpserver jdk.naming.dns \ - jdk.naming.rmi jdk.sctp jdk.security.auth jdk.management +# Param 1 - Name of module +define ReadImportMetaData + ifneq ($$(wildcard $(IMPORT_MODULES_MAKE)/$$(strip $1)/build.properties), ) + include_in_jre := + include_in_jdk := + include $(IMPORT_MODULES_MAKE)/$$(strip $1)/build.properties + ifeq ($$(include_in_jre), true) + JRE_MODULES += $1 + endif + ifeq ($$(include_in_jdk), true) + JDK_MODULES += $1 + endif + else + # Default to include in all + JRE_MODULES += $1 + JDK_MODULES += $1 + endif +endef + +IMPORTED_MODULES := $(call FindImportedModules) +$(foreach m, $(IMPORTED_MODULES), $(eval $(call ReadImportMetaData, $m))) + +# Compact builds have additional modules +COMPACT_EXTRA_MODULES := jdk.localedata jdk.crypto.pkcs11 jdk.crypto.ec +JRE_COMPACT1_MODULES := $(COMPACT_EXTRA_MODULES) java.compact1 +JRE_COMPACT2_MODULES := $(JRE_COMPACT1_MODULES) java.compact2 jdk.xml.dom jdk.httpserver +JRE_COMPACT3_MODULES := $(JRE_COMPACT2_MODULES) java.compact3 java.smartcardio jdk.management \ + jdk.naming.dns jdk.naming.rmi jdk.sctp jdk.security.auth # Replacing double-comma with a single comma is to workaround the issue # with some version of make on windows that doesn't substitute spaces # with one comma properly as with make 4.0 -define SubstComma - $(subst $(COMMA)$(COMMA),$(COMMA),$(subst $(SPACE),$(COMMA),$(strip $1))) -endef +SubstComma = \ + $(strip \ + $(subst $(COMMA)$(COMMA),$(COMMA),$(subst $(SPACE),$(COMMA),$(strip $1))) \ + ) + JRE_MODULES_LIST := $(call SubstComma, $(JRE_MODULES)) JDK_MODULES_LIST := $(call SubstComma, $(JDK_MODULES)) +JRE_COMPACT1_MODULES_LIST := $(call SubstComma, $(JRE_COMPACT1_MODULES)) +JRE_COMPACT2_MODULES_LIST := $(call SubstComma, $(JRE_COMPACT2_MODULES)) +JRE_COMPACT3_MODULES_LIST := $(call SubstComma, $(JRE_COMPACT3_MODULES)) + +################################################################################ +# Release file + +BASE_RELEASE_FILE := $(JDK_OUTPUTDIR)/release + +# Common way to emit a line into the release or info file +define info-file-item # name value + $(PRINTF) '%s="%s"\n' $1 $2 >> $@ +endef + +# Param 1 - The file containing the MODULES list +define create-info-file + $(call info-file-item, "JAVA_VERSION", "$(VERSION_NUMBER)") + $(call info-file-item, "JAVA_FULL_VERSION", "$(VERSION_STRING)") + $(call info-file-item, "OS_NAME", "$(REQUIRED_OS_NAME)") + $(call info-file-item, "OS_VERSION", "$(REQUIRED_OS_VERSION)") + $(call info-file-item, "OS_ARCH", "$(OPENJDK_TARGET_CPU_LEGACY)") + $(if $(JDK_ARCH_ABI_PROP_NAME), \ + $(call info-file-item, "SUN_ARCH_ABI", "$(JDK_ARCH_ABI_PROP_NAME)")) + $(call info-file-item, "SOURCE", "$(strip $(ALL_SOURCE_TIPS))") +endef + +# Param 1 - The file containing the MODULES list +define prepare-info-file + $(ECHO) $(LOG_INFO) Generating $(patsubst $(OUTPUT_ROOT)/%,%,$@) + $(MKDIR) -p $(@D) + $(RM) $@ +endef + +define info-file + $(call prepare-info-file) + $(call create-info-file) +endef + +# Create a variable dependency file common for all release info files. +INFO_FILE_VARDEPS := $(call DependOnVariable, create-info-file) + +ALL_SOURCE_TIPS = $(shell \ + if [ -f $(SUPPORT_OUTPUTDIR)/source_tips ] ; then \ + $(CAT) $(SUPPORT_OUTPUTDIR)/source_tips ; \ + fi) + +$(BASE_RELEASE_FILE): $(INFO_FILE_VARDEPS) $(SUPPORT_OUTPUTDIR)/source_tips + $(info-file) ################################################################################ -JRE_SORTED_MODULES := $(SUPPORT_OUTPUTDIR)/jre-sorted-modules -JDK_SORTED_MODULES := $(SUPPORT_OUTPUTDIR)/jdk-sorted-modules -JRE_COMPACT1_SORTED_MODULES := $(SUPPORT_OUTPUTDIR)/compact1-sorted-modules -JRE_COMPACT2_SORTED_MODULES := $(SUPPORT_OUTPUTDIR)/compact2-sorted-modules -JRE_COMPACT3_SORTED_MODULES := $(SUPPORT_OUTPUTDIR)/compact3-sorted-modules - - -MODULES_CMDS := $(SUPPORT_OUTPUTDIR)/modules_cmds-stripped -MODULES_LIBS := $(SUPPORT_OUTPUTDIR)/modules_libs-stripped -MODULES_CONF := $(SUPPORT_OUTPUTDIR)/modules_conf - -JIMAGE_TOOL := $(JAVA_SMALL) \ - -Xbootclasspath/p:$(BUILDTOOLS_OUTPUTDIR)/interim_jimage_classes \ - -cp $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes build.tools.module.ImageBuilder \ - --cmds $(MODULES_CMDS) \ - --libs $(MODULES_LIBS) \ - --configs $(MODULES_CONF) \ - --classes $(JDK_OUTPUTDIR)/modules \ - --endian $(OPENJDK_TARGET_CPU_ENDIAN) \ - # - -MODULES_XML += $(SRC_ROOT)/modules.xml -DEPENDENCIES := $(call CacheFind, \ - $(SUPPORT_OUTPUTDIR)/modules_cmds \ - $(SUPPORT_OUTPUTDIR)/modules_conf \ - $(SUPPORT_OUTPUTDIR)/modules_libs) \ - $(wildcard $(JDK_OUTPUTDIR)/modules/*/_*) \ - $(MODULES_XML) - # +JMODS := $(wildcard $(IMAGES_OUTPUTDIR)/jmods/*.jmod) # Use this file inside the image as target for make rule JIMAGE_TARGET_FILE := bin/java$(EXE_SUFFIX) -$(JDK_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(DEPENDENCIES) \ - $(call DependOnVariable, JDK_MODULES_LIST) +JLINK_TOOL := $(JLINK) --modulepath $(IMAGES_OUTPUTDIR)/jmods \ + --endian $(OPENJDK_BUILD_CPU_ENDIAN) \ + --release-info $(BASE_RELEASE_FILE) + +ifeq ($(JLINK_KEEP_PACKAGED_MODULES), true) + JLINK_EXTRA_OPTS := --keep-packaged-modules $(JDK_IMAGE_DIR)/jmods +endif + +$(JDK_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(JMODS) \ + $(call DependOnVariable, JDK_MODULES_LIST) $(BASE_RELEASE_FILE) $(ECHO) Creating jdk jimage - $(RM) -r $(JDK_IMAGE_DIR) $(JDK_SORTED_MODULES) - $(JIMAGE_TOOL) --mods $(JDK_MODULES_LIST) --output $(JDK_IMAGE_DIR) \ - $(MODULES_XML) > $(JDK_SORTED_MODULES) + $(RM) -r $(JDK_IMAGE_DIR) + $(JLINK_TOOL) --output $(JDK_IMAGE_DIR) \ + --addmods $(JDK_MODULES_LIST) $(JLINK_EXTRA_OPTS) $(TOUCH) $@ -$(JRE_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(DEPENDENCIES) \ - $(call DependOnVariable, JRE_MODULES_LIST) +$(JRE_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(JMODS) \ + $(call DependOnVariable, JRE_MODULES_LIST) $(BASE_RELEASE_FILE) $(ECHO) Creating jre jimage - $(RM) -r $(JRE_IMAGE_DIR) $(JRE_SORTED_MODULES) - $(JIMAGE_TOOL) --mods $(JRE_MODULES_LIST) --output $(JRE_IMAGE_DIR) \ - $(MODULES_XML) > $(JRE_SORTED_MODULES) + $(RM) -r $(JRE_IMAGE_DIR) + $(JLINK_TOOL) --output $(JRE_IMAGE_DIR) \ + --addmods $(JRE_MODULES_LIST) $(TOUCH) $@ JRE_COMPACT1_IMAGE_DIR := $(JRE_IMAGE_DIR)-compact1 JRE_COMPACT2_IMAGE_DIR := $(JRE_IMAGE_DIR)-compact2 JRE_COMPACT3_IMAGE_DIR := $(JRE_IMAGE_DIR)-compact3 -COMPACT_EXTRA_MODULES := jdk.localedata jdk.crypto.pkcs11 jdk.crypto.ec - -$(JRE_COMPACT1_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(DEPENDENCIES) \ - $(call DependOnVariable, JRE_COMPACT1_MODULES_LIST) +$(JRE_COMPACT1_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(JMODS) \ + $(call DependOnVariable, JRE_COMPACT1_MODULES_LIST) $(BASE_RELEASE_FILE) $(ECHO) Creating jre compact1 jimage - $(RM) -r $(JRE_COMPACT1_IMAGE_DIR) $(JRE_COMPACT1_SORTED_MODULES) - $(JIMAGE_TOOL) \ - --mods $(call SubstComma, java.compact1 $(COMPACT_EXTRA_MODULES)) \ - --output $(JRE_COMPACT1_IMAGE_DIR) \ - $(MODULES_XML) > $(JRE_COMPACT1_SORTED_MODULES) + $(RM) -r $(JRE_COMPACT1_IMAGE_DIR) + $(JLINK_TOOL) --addmods $(JRE_COMPACT1_MODULES_LIST) \ + --output $(JRE_COMPACT1_IMAGE_DIR) $(TOUCH) $@ -$(JRE_COMPACT2_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(DEPENDENCIES) \ - $(call DependOnVariable, JRE_COMPACT2_MODULES_LIST) +$(JRE_COMPACT2_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(JMODS) \ + $(call DependOnVariable, JRE_COMPACT2_MODULES_LIST) $(BASE_RELEASE_FILE) $(ECHO) Creating jre compact2 jimage - $(RM) -r $(JRE_COMPACT2_IMAGE_DIR) $(JRE_COMPACT2_SORTED_MODULES) - $(JIMAGE_TOOL) \ - --mods $(call SubstComma, java.compact2 $(COMPACT_EXTRA_MODULES)) \ - --output $(JRE_COMPACT2_IMAGE_DIR) \ - $(MODULES_XML) > $(JRE_COMPACT2_SORTED_MODULES) + $(RM) -r $(JRE_COMPACT2_IMAGE_DIR) + $(JLINK_TOOL) --addmods $(JRE_COMPACT2_MODULES_LIST) \ + --output $(JRE_COMPACT2_IMAGE_DIR) $(TOUCH) $@ -$(JRE_COMPACT3_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(DEPENDENCIES) \ - $(call DependOnVariable, JRE_COMPACT3_MODULES_LIST) +$(JRE_COMPACT3_IMAGE_DIR)/$(JIMAGE_TARGET_FILE): $(JMODS) \ + $(call DependOnVariable, JRE_COMPACT3_MODULES_LIST) $(BASE_RELEASE_FILE) $(ECHO) Creating jre compact3 jimage - $(RM) -r $(JRE_COMPACT3_IMAGE_DIR) $(JRE_COMPACT3_SORTED_MODULES) - $(JIMAGE_TOOL) \ - --mods $(call SubstComma, $(JDK_COMPACT3_MODULES) $(COMPACT_EXTRA_MODULES)) \ - --output $(JRE_COMPACT3_IMAGE_DIR) \ - $(MODULES_XML) > $(JRE_COMPACT3_SORTED_MODULES) + $(RM) -r $(JRE_COMPACT3_IMAGE_DIR) + $(JLINK_TOOL) --addmods $(JRE_COMPACT3_MODULES_LIST) \ + --output $(JRE_COMPACT3_IMAGE_DIR) $(TOUCH) $@ TOOL_JRE_TARGETS := $(JRE_IMAGE_DIR)/$(JIMAGE_TARGET_FILE) @@ -345,79 +392,6 @@ $(JDK_IMAGE_DIR)/%: $(JDK_DOC_LOCATION)/% JRE_TARGETS += $(JRE_DOC_TARGETS) JDK_TARGETS += $(JDK_DOC_TARGETS) -################################################################################ -# Release file - -JRE_INFO_FILE := $(JRE_IMAGE_DIR)/release -JDK_INFO_FILE := $(JDK_IMAGE_DIR)/release -JRE_COMPACT1_INFO_FILE := $(JRE_COMPACT1_IMAGE_DIR)/release -JRE_COMPACT2_INFO_FILE := $(JRE_COMPACT2_IMAGE_DIR)/release -JRE_COMPACT3_INFO_FILE := $(JRE_COMPACT3_IMAGE_DIR)/release - -# Common way to emit a line into the release or info file -define info-file-item # name value - $(PRINTF) '%s="%s"\n' $1 $2 >> $@ -endef - -# Param 1 - The file containing the MODULES list -define create-info-file - $(call info-file-item, "JAVA_VERSION", "$(VERSION_NUMBER)") - $(call info-file-item, "JAVA_FULL_VERSION", "$(VERSION_STRING)") - $(call info-file-item, "OS_NAME", "$(REQUIRED_OS_NAME)") - $(call info-file-item, "OS_VERSION", "$(REQUIRED_OS_VERSION)") - $(call info-file-item, "OS_ARCH", "$(OPENJDK_TARGET_CPU_LEGACY)") - $(if $(JDK_ARCH_ABI_PROP_NAME), \ - $(call info-file-item, "SUN_ARCH_ABI", "$(JDK_ARCH_ABI_PROP_NAME)")) - $(call info-file-item, "SOURCE", "$(strip $(ALL_SOURCE_TIPS))") - $(call info-file-item, "MODULES", "`$(CAT) $1`") -endef - -# Param 1 - The file containing the MODULES list -define prepare-info-file - $(call LogInfo, Generating $(patsubst $(OUTPUT_ROOT)/%,%,$@)) - $(MKDIR) -p $(@D) - $(RM) $@ -endef - -define info-file - $(call prepare-info-file, $1) - $(call create-info-file, $1) -endef - -# Create a variable dependency file common for all release info files. The -# sorted module list will only change if the image is regenerated, which will -# trigger a rebuild of these files anyway. -INFO_FILE_VARDEPS := $(call DependOnVariable, create-info-file) - -ALL_SOURCE_TIPS = $(shell \ - if [ -f $(SUPPORT_OUTPUTDIR)/source_tips ] ; then \ - $(CAT) $(SUPPORT_OUTPUTDIR)/source_tips ; \ - fi) - -$(JRE_INFO_FILE): $(INFO_FILE_VARDEPS) $(SUPPORT_OUTPUTDIR)/source_tips - $(call info-file, $(JRE_SORTED_MODULES)) - -$(JDK_INFO_FILE): $(INFO_FILE_VARDEPS) $(SUPPORT_OUTPUTDIR)/source_tips - $(call info-file, $(JDK_SORTED_MODULES)) - -$(JRE_COMPACT1_INFO_FILE): $(INFO_FILE_VARDEPS) $(SUPPORT_OUTPUTDIR)/source_tips - $(call info-file, $(JRE_COMPACT1_SORTED_MODULES)) - $(call info-file-item, "JAVA_PROFILE", "compact1") - -$(JRE_COMPACT2_INFO_FILE): $(INFO_FILE_VARDEPS) $(SUPPORT_OUTPUTDIR)/source_tips - $(call info-file, $(JRE_COMPACT2_SORTED_MODULES)) - $(call info-file-item, "JAVA_PROFILE", "compact2") - -$(JRE_COMPACT3_INFO_FILE): $(INFO_FILE_VARDEPS) $(SUPPORT_OUTPUTDIR)/source_tips - $(call info-file, $(JRE_COMPACT3_SORTED_MODULES)) - $(call info-file-item, "JAVA_PROFILE", "compact3") - -JRE_TARGETS += $(JRE_INFO_FILE) -JDK_TARGETS += $(JDK_INFO_FILE) -JRE_COMPACT1_TARGETS += $(JRE_COMPACT1_INFO_FILE) -JRE_COMPACT2_TARGETS += $(JRE_COMPACT2_INFO_FILE) -JRE_COMPACT3_TARGETS += $(JRE_COMPACT3_INFO_FILE) - ################################################################################ # src.zip @@ -435,7 +409,7 @@ ifneq ($(findstring images, $(MAKECMDGOALS)), ) DEST := $(JDK_IMAGE_DIR)/demo, \ FILES := $(if $(wildcard $(SUPPORT_OUTPUTDIR)/demos/image), \ $(call DoubleDollar, $(call DoubleDollar, \ - $(shell $(FIND) $(SUPPORT_OUTPUTDIR)/demos/image \ + $(shell $(FIND) $(SUPPORT_OUTPUTDIR)/demos/image \ -type f -a ! \( -name "_the*" -o -name "javac_state" \) )))), \ )) @@ -445,7 +419,7 @@ endif ################################################################################ # /sample dir -$(eval $(call SetupCopyFiles, COPY_SAMPLES, \ +$(eval $(call SetupCopyFiles,COPY_SAMPLES, \ SRC := $(SUPPORT_OUTPUTDIR)/sample/image, \ DEST := $(JDK_IMAGE_DIR)/sample, \ FILES := $(if $(wildcard $(SUPPORT_OUTPUTDIR)/sample/image), \ diff --git a/make/Javadoc.gmk b/make/Javadoc.gmk index 0e0be3fc47e..f0b170186cf 100644 --- a/make/Javadoc.gmk +++ b/make/Javadoc.gmk @@ -1,4 +1,4 @@ -# Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 1997, 2016, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # # This code is free software; you can redistribute it and/or modify it @@ -138,26 +138,48 @@ $(FULL_COMPANY_NAME) in the US and other countries. # to $(PATH_SEP) being interpreted as an end of # command (newline or shell ; character) ALL_SOURCE_DIRS := $(wildcard \ - $(JDK_TOPDIR)/src/*/share/classes \ + $(SUPPORT_OUTPUTDIR)/gensrc/j* \ + $(if $(IMPORT_MODULES_SRC), $(IMPORT_MODULES_SRC)/*) \ $(JDK_TOPDIR)/src/*/$(OPENJDK_TARGET_OS)/classes \ $(JDK_TOPDIR)/src/*/$(OPENJDK_TARGET_OS_TYPE)/classes \ + $(JDK_TOPDIR)/src/*/share/classes \ + $(HOTSPOT_TOPDIR)/src/*/share/classes \ $(LANGTOOLS_TOPDIR)/src/*/share/classes \ $(NASHORN_TOPDIR)/src/*/share/classes \ $(CORBA_TOPDIR)/src/*/share/classes \ $(JAXP_TOPDIR)/src/*/share/classes \ $(JAXWS_TOPDIR)/src/*/share/classes \ - $(SUPPORT_OUTPUTDIR)/gensrc/j* \ $(SUPPORT_OUTPUTDIR)/rmic/j* \ $(JDK_TOPDIR)/src/*/share/doc/stub \ ) \ # +ALL_MODULE_SOURCE_DIRS := \ + $(SUPPORT_OUTPUTDIR)/gensrc/* \ + $(if $(IMPORT_MODULES_SRC), $(IMPORT_MODULES_SRC)/*) \ + $(JDK_TOPDIR)/src/*/$(OPENJDK_TARGET_OS)/classes \ + $(JDK_TOPDIR)/src/*/$(OPENJDK_TARGET_OS_TYPE)/classes \ + $(JDK_TOPDIR)/src/*/share/classes \ + $(HOTSPOT_TOPDIR)/src/*/share/classes \ + $(LANGTOOLS_TOPDIR)/src/*/share/classes \ + $(NASHORN_TOPDIR)/src/*/share/classes \ + $(CORBA_TOPDIR)/src/*/share/classes \ + $(JAXP_TOPDIR)/src/*/share/classes \ + $(JAXWS_TOPDIR)/src/*/share/classes \ + $(SUPPORT_OUTPUTDIR)/rmic/* \ + $(JDK_TOPDIR)/src/*/share/doc/stub \ + # + + # List with classpath separator between them EMPTY:= SPACE:= $(EMPTY) $(EMPTY) RELEASEDOCS_SOURCEPATH = \ $(subst $(SPACE),$(PATH_SEP),$(strip $(ALL_SOURCE_DIRS))) +RELEASEDOCS_MODULESOURCEPATH = \ + $(subst $(SPACE),$(PATH_SEP),$(strip $(ALL_MODULE_SOURCE_DIRS))) + define prep-target $(MKDIR) -p $(@D) $(RM) $@ @@ -372,6 +394,9 @@ COREAPI_OVERVIEW = $(JDK_TOPDIR)/src/java.base/share/classes/overview-core.html COREAPI_OPTIONS_FILE = $(DOCSTMPDIR)/coredocs.options COREAPI_PACKAGES_FILE = $(DOCSTMPDIR)/coredocs.packages +# The modules required to be documented +COREAPI_MODULES = java.se.ee + coredocs: $(COREAPI_INDEX_FILE) # Set relative location to core api document root @@ -391,7 +416,9 @@ $(COREAPI_OPTIONS_FILE): $(COREAPI_OVERVIEW) $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:reference) ; \ $(call OptionOnly,-Xdoclint/package:-org.omg.*) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(COREAPI_MODULES)) ; \ $(call OptionPair,-encoding,ISO-8859-1) ; \ $(call OptionOnly,-splitIndex) ; \ $(call OptionPair,-overview,$(COREAPI_OVERVIEW)) ; \ @@ -431,6 +458,9 @@ DOCLETAPI_INDEX_FILE = $(DOCLETAPI_DOCDIR)/index.html DOCLETAPI_OPTIONS_FILE = $(DOCSTMPDIR)/docletapi.options DOCLETAPI_PACKAGES_FILE = $(DOCSTMPDIR)/docletapi.packages +# The modules required to be documented +DOCLETAPI_MODULES = jdk.javadoc + docletapidocs: $(DOCLETAPI_INDEX_FILE) # Set relative location to core api document root @@ -449,7 +479,9 @@ $(DOCLETAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:all) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(DOCLETAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-breakiterator) ; \ $(call OptionPair,-doctitle,$(DOCLETAPI_DOCTITLE)) ; \ @@ -488,6 +520,9 @@ OLD_DOCLETAPI_INDEX_FILE = $(OLD_DOCLETAPI_DOCDIR)/index.html OLD_DOCLETAPI_OPTIONS_FILE = $(DOCSTMPDIR)/old-docletapi.options OLD_DOCLETAPI_PACKAGES_FILE = $(DOCSTMPDIR)/old-docletapi.packages +# The modules required to be documented +OLD_DOCLETAPI_MODULES = jdk.javadoc + olddocletapidocs: $(OLD_DOCLETAPI_INDEX_FILE) # Set relative location to core api document root @@ -506,7 +541,9 @@ $(OLD_DOCLETAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:all) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(OLD_DOCLETAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-breakiterator) ; \ $(call OptionPair,-doctitle,$(OLD_DOCLETAPI_DOCTITLE)) ; \ @@ -541,6 +578,9 @@ TAGLETAPI_INDEX_FILE = $(TAGLETAPI_DOCDIR)/com/sun/tools/doclets/Taglet.html TAGLETAPI_OPTIONS_FILE = $(DOCSTMPDIR)/tagletapi.options TAGLETAPI_PACKAGES_FILE = $(DOCSTMPDIR)/tagletapi.packages +# The modules required to be documented +TAGLETAPI_MODULES = jdk.javadoc + tagletapidocs: $(TAGLETAPI_INDEX_FILE) # Set relative location to core api document root @@ -564,7 +604,9 @@ $(TAGLETAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:all) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(TAGLETAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nonavbar) ; \ $(call OptionOnly,-noindex) ; \ @@ -599,6 +641,9 @@ DOMAPI_INDEX_FILE = $(DOMAPI_DOCDIR)/index.html DOMAPI_OPTIONS_FILE = $(DOCSTMPDIR)/domapi.options DOMAPI_PACKAGES_FILE = $(DOCSTMPDIR)/domapi.packages +# The modules required to be documented +DOMAPI_MODULES = java.xml,jdk.xml.dom + domapidocs: $(DOMAPI_INDEX_FILE) # Set relative location to core api document root @@ -617,7 +662,9 @@ $(DOMAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(DOMAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-splitIndex) ; \ $(call OptionPair,-doctitle,$(DOMAPI_DOCTITLE)) ; \ @@ -663,6 +710,9 @@ JDI_INDEX_FILE = $(JDI_DOCDIR)/index.html JDI_OPTIONS_FILE = $(DOCSTMPDIR)/jdi.options JDI_PACKAGES_FILE = $(DOCSTMPDIR)/jdi.packages +# The modules required to be documented +JDI_MODULES = jdk.jdi + jdidocs: $(JDI_INDEX_FILE) # Set relative location to core api document root @@ -681,7 +731,9 @@ $(JDI_OPTIONS_FILE): $(JDI_OVERVIEW) @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(JDI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionPair,-overview,$(JDI_OVERVIEW)) ; \ $(call OptionPair,-doctitle,$(JDI_DOCTITLE)) ; \ @@ -751,6 +803,9 @@ JAAS_INDEX_FILE = $(JAAS_DOCDIR)/index.html JAAS_OPTIONS_FILE = $(DOCSTMPDIR)/jaas.options JAAS_PACKAGES_FILE = $(DOCSTMPDIR)/jaas.packages +# The modules required to be documented +JAAS_MODULES = jdk.security.auth + jaasdocs: $(JAAS_INDEX_FILE) # Set relative location to core api document root @@ -769,7 +824,9 @@ $(JAAS_OPTIONS_FILE): $(JAAS_OVERVIEW) @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(JAAS_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionPair,-overview,$(JAAS_OVERVIEW)) ; \ $(call OptionPair,-doctitle,$(JAAS_DOCTITLE)) ; \ @@ -805,6 +862,9 @@ JGSS_INDEX_FILE = $(JGSS_DOCDIR)/index.html JGSS_OPTIONS_FILE = $(DOCSTMPDIR)/jgss.options JGSS_PACKAGES_FILE = $(DOCSTMPDIR)/jgss.packages +# The modules required to be documented +JGSS_MODULES = jdk.security.jgss + jgssdocs: $(JGSS_INDEX_FILE) # Set relative location to core api document root @@ -823,7 +883,9 @@ $(JGSS_OPTIONS_FILE): $(JGSS_OVERVIEW) @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(JGSS_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-overview,$(JGSS_OVERVIEW)) ; \ @@ -859,6 +921,9 @@ SMARTCARDIO_INDEX_FILE = $(SMARTCARDIO_DOCDIR)/index.html SMARTCARDIO_OPTIONS_FILE = $(DOCSTMPDIR)/smartcardio.options SMARTCARDIO_PACKAGES_FILE = $(DOCSTMPDIR)/smartcardio.packages +# The modules required to be documented +SMARTCARDIO_MODULES = java.smartcardio + smartcardiodocs: $(SMARTCARDIO_INDEX_FILE) # Set relative location to core api document root @@ -877,7 +942,9 @@ $(SMARTCARDIO_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(SMARTCARDIO_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-doctitle,$(SMARTCARDIO_DOCTITLE)) ; \ @@ -911,6 +978,9 @@ HTTPSERVER_INDEX_HTML = $(HTTPSERVER_DOCDIR)/index.html HTTPSERVER_OPTIONS_FILE = $(DOCSTMPDIR)/httpserver.options HTTPSERVER_PACKAGES_FILE = $(DOCSTMPDIR)/httpserver.packages +# The modules required to be documented +HTTPSERVER_MODULES = jdk.httpserver + httpserverdocs: $(HTTPSERVER_INDEX_HTML) # Set relative location to core api document root @@ -929,7 +999,9 @@ $(HTTPSERVER_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(HTTPSERVER_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-doctitle,$(HTTPSERVER_DOCTITLE)) ; \ @@ -967,6 +1039,9 @@ MGMT_INDEX_FILE = $(MGMT_DOCDIR)/index.html MGMT_OPTIONS_FILE = $(DOCSTMPDIR)/mgmt.options MGMT_PACKAGES_FILE = $(DOCSTMPDIR)/mgmt.packages +# The modules required to be documented +MGMT_MODULES = jdk.management + mgmtdocs: $(MGMT_INDEX_FILE) # Set relative location to core api document root @@ -990,7 +1065,9 @@ $(MGMT_OPTIONS_FILE): $(MGMT_OVERVIEW) @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(MGMT_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-overview,$(MGMT_OVERVIEW)) ; \ @@ -1025,6 +1102,9 @@ ATTACH_INDEX_HTML = $(ATTACH_DOCDIR)/index.html ATTACH_OPTIONS_FILE = $(DOCSTMPDIR)/attach.options ATTACH_PACKAGES_FILE = $(DOCSTMPDIR)/attach.packages +# The modules required to be documented +ATTACH_MODULES = jdk.attach + attachdocs: $(ATTACH_INDEX_HTML) # Set relative location to core api document root @@ -1043,7 +1123,9 @@ $(ATTACH_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(ATTACH_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-doctitle,$(ATTACH_DOCTITLE)) ; \ @@ -1077,6 +1159,9 @@ JCONSOLE_INDEX_HTML = $(JCONSOLE_DOCDIR)/index.html JCONSOLE_OPTIONS_FILE = $(DOCSTMPDIR)/jconsole.options JCONSOLE_PACKAGES_FILE = $(DOCSTMPDIR)/jconsole.packages +# The modules required to be documented +JCONSOLE_MODULES = jdk.jconsole + jconsoledocs: $(JCONSOLE_INDEX_HTML) # Set relative location to core api document root @@ -1095,7 +1180,9 @@ $(JCONSOLE_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(JCONSOLE_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-doctitle,$(JCONSOLE_DOCTITLE)) ; \ @@ -1131,6 +1218,9 @@ TREEAPI_INDEX_HTML = $(TREEAPI_DOCDIR)/index.html TREEAPI_OPTIONS_FILE = $(DOCSTMPDIR)/treeapi.options TREEAPI_PACKAGES_FILE = $(DOCSTMPDIR)/treeapi.packages +# The modules required to be documented +TREEAPI_MODULES = jdk.compiler + treeapidocs: $(TREEAPI_INDEX_HTML) # Set relative location to core api document root @@ -1149,7 +1239,9 @@ $(TREEAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:all) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(TREEAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionPair,-doctitle,$(TREEAPI_DOCTITLE)) ; \ $(call OptionPair,-windowtitle,$(TREEAPI_WINDOWTITLE) $(DRAFT_WINTITLE)); \ @@ -1185,6 +1277,9 @@ NASHORNAPI_INDEX_HTML = $(NASHORNAPI_DOCDIR)/index.html NASHORNAPI_OPTIONS_FILE = $(DOCSTMPDIR)/nashornapi.options NASHORNAPI_PACKAGES_FILE = $(DOCSTMPDIR)/nashornapi.packages +# The modules required to be documented +NASHORNAPI_MODULES = jdk.scripting.nashorn + nashornapidocs: $(NASHORNAPI_INDEX_HTML) # Set relative location to core api document root @@ -1203,7 +1298,9 @@ $(NASHORNAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:all) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(NASHORNAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionPair,-doctitle,$(NASHORNAPI_DOCTITLE)) ; \ $(call OptionPair,-windowtitle,$(NASHORNAPI_WINDOWTITLE) $(DRAFT_WINTITLE)); \ @@ -1239,6 +1336,9 @@ DYNALINKAPI_INDEX_HTML = $(DYNALINKAPI_DOCDIR)/index.html DYNALINKAPI_OPTIONS_FILE = $(DOCSTMPDIR)/dynalinkapi.options DYNALINKAPI_PACKAGES_FILE = $(DOCSTMPDIR)/dynalinkapi.packages +# The modules required to be documented +DYNALINKAPI_MODULES = jdk.dynalink + dynalinkapidocs: $(DYNALINKAPI_INDEX_HTML) # Set relative location to core api document root @@ -1257,7 +1357,9 @@ $(DYNALINKAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:all) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(DYNALINKAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionPair,-doctitle,$(DYNALINKAPI_DOCTITLE)) ; \ $(call OptionPair,-windowtitle,$(DYNALINKAPI_WINDOWTITLE) $(DRAFT_WINTITLE)); \ @@ -1291,6 +1393,9 @@ SCTPAPI_INDEX_HTML = $(SCTPAPI_DOCDIR)/index.html SCTPAPI_OPTIONS_FILE = $(DOCSTMPDIR)/sctp.options SCTPAPI_PACKAGES_FILE = $(DOCSTMPDIR)/sctp.packages +# The modules required to be documented +SCTPAPI_MODULES = jdk.sctp + sctpdocs: $(SCTPAPI_INDEX_HTML) # Set relative location to core api document root @@ -1309,7 +1414,9 @@ $(SCTPAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(SCTPAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-doctitle,$(SCTPAPI_DOCTITLE)) ; \ @@ -1343,6 +1450,9 @@ JACCESSAPI_INDEX_HTML = $(JACCESSAPI_DOCDIR)/index.html JACCESSAPI_OPTIONS_FILE = $(DOCSTMPDIR)/jaccess.options JACCESSAPI_PACKAGES_FILE = $(DOCSTMPDIR)/jaccess.packages +# The modules required to be documented +JACCESSAPI_MODULES = jdk.accessibility + jaccessdocs: $(JACCESSAPI_INDEX_HTML) # Set relative location to core api document root @@ -1361,7 +1471,9 @@ $(JACCESSAPI_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:all) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(JACCESSAPI_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-doctitle,$(JACCESSAPI_DOCTITLE)) ; \ @@ -1395,6 +1507,9 @@ JDKNET_INDEX_HTML = $(JDKNET_DOCDIR)/index.html JDKNET_OPTIONS_FILE = $(DOCSTMPDIR)/jdknet.options JDKNET_PACKAGES_FILE = $(DOCSTMPDIR)/jdknet.packages +# The modules required to be documented +JDKNET_MODULES = java.base + jdknetdocs: $(JDKNET_INDEX_HTML) # Set relative location to core api document root @@ -1413,7 +1528,9 @@ $(JDKNET_OPTIONS_FILE): @($(call COMMON_JAVADOCFLAGS) ; \ $(call COMMON_JAVADOCTAGS) ; \ $(call OptionOnly,-Xdoclint:none) ; \ - $(call OptionPair,-sourcepath,$(RELEASEDOCS_SOURCEPATH)) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(JDKNET_MODULES)) ; \ $(call OptionPair,-encoding,ascii) ; \ $(call OptionOnly,-nodeprecatedlist) ; \ $(call OptionPair,-doctitle,$(JDKNET_DOCTITLE)) ; \ @@ -1428,6 +1545,69 @@ $(JDKNET_PACKAGES_FILE): $(call PackageDependencies,$(JDKNET_PKGS)) $(prep-target) $(call PackageFilter,$(JDKNET_PKGS)) +############################################################# +# +# jlink plugin API docs +# +# TODO: Need to decide when the plugin API is ready to publish as experimental API. +# This target is temporarily added for internal use for now. +# + +ALL_OTHER_TARGETS += jlinkdocs + +JLINK_PLUGIN_FIRST_COPYRIGHT_YEAR = 2015 +JLINK_PLUGIN_DOCDIR := $(JDK_API_DOCSDIR)/jlink +JLINK_PLUGIN2COREAPI := ../$(JDKJRE2COREAPI) +JLINK_PLUGIN_DOCTITLE := JLink Plugin API - EXPERIMENTAL +JLINK_PLUGIN_WINDOWTITLE := JLink Plugin API - EXPERIMENTAL +JLINK_PLUGIN_HEADER := JLink Plugin API - EXPERIMENTAL +JLINK_PLUGIN_BOTTOM := $(call CommonBottom,$(JLINK_PLUGIN_FIRST_COPYRIGHT_YEAR)) +JLINK_PLUGIN_PKGS = jdk.tools.jlink \ + jdk.tools.jlink.builder \ + jdk.tools.jlink.plugin + +JLINK_PLUGIN_INDEX_HTML = $(JLINK_PLUGIN_DOCDIR)/index.html +JLINK_PLUGIN_OPTIONS_FILE = $(DOCSTMPDIR)/jlinkplugins.options +JLINK_PLUGIN_PACKAGES_FILE = $(DOCSTMPDIR)/jlinkplugins.packages + +# The modules required to be documented +JLINK_PLUGIN_MODULES = jdk.jlink + +jlinkdocs: $(JLINK_PLUGIN_INDEX_HTML) + +# Set relative location to core api document root +$(JLINK_PLUGIN_INDEX_HTML): GET2DOCSDIR=$(JLINK_PLUGIN2COREAPI)/.. + +# Run javadoc if the index file is out of date or missing +$(JLINK_PLUGIN_INDEX_HTML): $(JLINK_PLUGIN_OPTIONS_FILE) $(JLINK_PLUGIN_PACKAGES_FILE) $(COREAPI_INDEX_FILE) + $(prep-javadoc) + $(call JavadocSummary,$(JLINK_PLUGIN_OPTIONS_FILE),$(JLINK_PLUGIN_PACKAGES_FILE)) + $(JAVADOC_CMD_SMALL) -d $(@D) \ + @$(JLINK_PLUGIN_OPTIONS_FILE) @$(JLINK_PLUGIN_PACKAGES_FILE) + +# Create file with javadoc options in it +$(JLINK_PLUGIN_OPTIONS_FILE): + $(prep-target) + @($(call COMMON_JAVADOCFLAGS) ; \ + $(call COMMON_JAVADOCTAGS) ; \ + $(call OptionOnly,-Xdoclint:none) ; \ + $(call OptionPair,-system,none) ; \ + $(call OptionPair,-modulesourcepath,$(RELEASEDOCS_MODULESOURCEPATH)) ; \ + $(call OptionPair,-addmods,$(JLINK_PLUGIN_MODULES)) ; \ + $(call OptionPair,-encoding,ascii) ; \ + $(call OptionOnly,-nodeprecatedlist) ; \ + $(call OptionPair,-doctitle,$(JLINK_PLUGIN_DOCTITLE)) ; \ + $(call OptionPair,-windowtitle,$(JLINK_PLUGIN_WINDOWTITLE) $(DRAFT_WINTITLE)); \ + $(call OptionPair,-header,$(JLINK_PLUGIN_HEADER)$(DRAFT_HEADER)); \ + $(call OptionPair,-bottom,$(JLINK_PLUGIN_BOTTOM)$(DRAFT_BOTTOM)); \ + $(call OptionTrip,-linkoffline,$(JLINK_PLUGIN2COREAPI),$(COREAPI_DOCSDIR)/); \ + ) >> $@ + +# Create a file with the package names in it +$(JLINK_PLUGIN_PACKAGES_FILE): $(call PackageDependencies,$(JLINK_PLUGIN_PKGS)) + $(prep-target) + $(call PackageFilter,$(JLINK_PLUGIN_PKGS)) + otherdocs: $(ALL_OTHER_TARGETS) diff --git a/make/Jprt.gmk b/make/Jprt.gmk index 892e46cc243..6b778293d2a 100644 --- a/make/Jprt.gmk +++ b/make/Jprt.gmk @@ -36,6 +36,10 @@ ifneq ($(CYGPATH), ) override JPRT_ARCHIVE_INSTALL_BUNDLE := \ $(shell $(CYGPATH) -u $(JPRT_ARCHIVE_INSTALL_BUNDLE)) endif + ifdef JPRT_ARCHIVE_MODULES_BUNDLE + override JPRT_ARCHIVE_MODULES_BUNDLE := \ + $(shell $(CYGPATH) -u $(JPRT_ARCHIVE_MODULES_BUNDLE)) + endif ifdef JPRT_ARCHIVE_TEST_BUNDLE override JPRT_ARCHIVE_TEST_BUNDLE := \ $(shell $(CYGPATH) -u $(JPRT_ARCHIVE_TEST_BUNDLE)) @@ -54,6 +58,9 @@ endif ifndef JPRT_ARCHIVE_INSTALL_BUNDLE JPRT_ARCHIVE_INSTALL_BUNDLE=/tmp/jprt_bundles/product-install.zip endif +ifndef JPRT_ARCHIVE_MODULES_BUNDLE + JPRT_ARCHIVE_MODULES_BUNDLE=/tmp/jprt_bundles/modules.zip +endif ifndef JPRT_ARCHIVE_TEST_BUNDLE JPRT_ARCHIVE_TEST_BUNDLE=/tmp/jprt_bundles/test-image.zip endif @@ -66,7 +73,8 @@ ifeq ($(SKIP_BOOT_CYCLE), false) endif # This target must be called in the context of a SPEC file -jprt_bundle: $(JPRT_ARCHIVE_BUNDLE) $(JPRT_ARCHIVE_TEST_BUNDLE) +jprt_bundle: $(JPRT_ARCHIVE_BUNDLE) $(JPRT_ARCHIVE_TEST_BUNDLE) \ + $(JPRT_ARCHIVE_MODULES_BUNDLE) ifeq ($(GCOV_ENABLED), true) jprt_bundle: $(JPRT_ARCHIVE_SYMBOLS_BUNDLE) @@ -78,6 +86,11 @@ $(JPRT_ARCHIVE_BUNDLE): bundles $(RM) $@ $(CP) $(BUILD_OUTPUT)/bundles/$(JDK_IMAGE_SUBDIR).zip $@ +$(JPRT_ARCHIVE_MODULES_BUNDLE): bundles + $(MKDIR) -p $(@D) + $(RM) $@ + $(CP) $(BUILD_OUTPUT)/bundles/modules.zip $@ + $(JPRT_ARCHIVE_TEST_BUNDLE): bundles $(MKDIR) -p $(@D) $(RM) $@ @@ -104,6 +117,8 @@ bundles: all $(BUILD_OUTPUT)/bundles/$(JRE_IMAGE_SUBDIR).zip . $(CD) $(SRC_TEST_IMAGE_DIR) && $(ZIP) -y -q -r \ $(BUILD_OUTPUT)/bundles/$(TEST_IMAGE_SUBDIR).zip . + $(CD) $(IMAGES_OUTPUTDIR)/jmods && $(ZIP) -y -q -r \ + $(BUILD_OUTPUT)/bundles/modules.zip . if [ -d $(BUILD_OUTPUT)/install/bundles ] ; then \ $(CD) $(BUILD_OUTPUT)/install/bundles && $(ZIP) -y -q -r \ $(JPRT_ARCHIVE_INSTALL_BUNDLE) . ; \ diff --git a/make/JrtfsJar.gmk b/make/JrtfsJar.gmk index 2f8931aee7b..ce5d5bea6c3 100644 --- a/make/JrtfsJar.gmk +++ b/make/JrtfsJar.gmk @@ -28,6 +28,7 @@ default: all include $(SPEC) include MakeBase.gmk include JarArchive.gmk +include SetupJavaCompilers.gmk include TextFileProcessing.gmk # This rule will be depended on due to the MANIFEST line @@ -40,10 +41,32 @@ $(eval $(call SetupTextFileProcessing, BUILD_JAVA_MANIFEST, \ @@COMPANY_NAME@@ => $(COMPANY_NAME) , \ )) -$(eval $(call SetupJarArchive, JRTFS_JAR, \ - SRCS := $(BUILDTOOLS_OUTPUTDIR)/interim_jimage_classes, \ +JIMAGE_PKGS := \ + jdk/internal/jimage \ + jdk/internal/jrtfs \ + # + +$(eval $(call SetupJavaCompilation,BUILD_JRTFS, \ + SETUP := GENERATE_OLDBYTECODE, \ + SRC := $(JDK_TOPDIR)/src/java.base/share/classes, \ + EXCLUDE_FILES := module-info.java, \ + INCLUDES := $(JIMAGE_PKGS), \ + BIN := $(SUPPORT_OUTPUTDIR)/jrtfs_classes)) + +# Because of the explicit INCLUDES in the compilation setup above, the service provider +# file will not be copied unless META-INF/services would also be added to the INCLUDES. +# Adding META-INF/services would include all files in that directory when only the one +# is needed, which is why this explicit copy is defined instead. +$(eval $(call SetupCopyFiles,COPY_JIMAGE_SERVICE_PROVIDER, \ + SRC := $(JDK_TOPDIR)/src/java.base/share/classes, \ + DEST := $(SUPPORT_OUTPUTDIR)/jrtfs_classes, \ + FILES := META-INF/services/java.nio.file.spi.FileSystemProvider)) + +$(eval $(call SetupJarArchive,BUILD_JRTFS_JAR, \ + DEPENDENCIES := $(BUILD_JRTFS) $(COPY_JIMAGE_SERVICE_PROVIDER), \ + SRCS := $(SUPPORT_OUTPUTDIR)/jrtfs_classes, \ JAR := $(SUPPORT_OUTPUTDIR)/jrt-fs.jar, \ MANIFEST := $(SUPPORT_OUTPUTDIR)/java-main-manifest.mf, \ )) -all: $(JRTFS_JAR) +all: $(BUILD_JRTFS_JAR) diff --git a/make/Main.gmk b/make/Main.gmk index 66509ddfdcb..5d8137f2b47 100644 --- a/make/Main.gmk +++ b/make/Main.gmk @@ -42,8 +42,6 @@ include $(SRC_ROOT)/make/MainSupport.gmk # Load the vital tools for all the makefiles. include $(SRC_ROOT)/make/common/MakeBase.gmk -# Explicitly generate module deps makefile data -include $(SRC_ROOT)/make/GenerateModuleDeps.gmk include $(SRC_ROOT)/make/common/Modules.gmk # Declare ALL_TARGETS as an immediate variable. This variable is a list of all @@ -113,6 +111,24 @@ LANGTOOLS_GENSRC_TARGETS := $(filter %-gensrc-langtools, $(GENSRC_TARGETS)) CORBA_GENSRC_TARGETS := $(filter %-gensrc-corba, $(GENSRC_TARGETS)) HOTSPOT_GENSRC_TARGETS := $(filter %-gensrc-hotspot, $(GENSRC_TARGETS)) +GENSRC_MODULEINFO_MODULES := $(ALL_MODULES) +GENSRC_MODULEINFO_TARGETS := $(addsuffix -gensrc-moduleinfo, \ + $(GENSRC_MODULEINFO_MODULES)) + +GENSRC_MODULES := $(GENSRC_MODULEINFO_MODULES) +GENSRC_TARGETS += $(sort $(GENSRC_MODULEINFO_TARGETS) \ + $(addsuffix -gensrc, $(GENSRC_MODULES))) + +define DeclareModuleInfoRecipe + $1-gensrc-moduleinfo: + +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) \ + -f GensrcModuleInfo.gmk MODULE=$1) + + $1-gensrc: $1-gensrc-moduleinfo +endef + +$(foreach m, $(GENSRC_MODULEINFO_MODULES), $(eval $(call DeclareModuleInfoRecipe,$m))) + ALL_TARGETS += $(GENSRC_TARGETS) ################################################################################ @@ -136,11 +152,27 @@ $(eval $(call DeclareRecipesForPhase, COPY, \ USE_WRAPPER := true, \ MULTIPLE_MAKEFILES := true)) -ALL_TARGETS += $(COPY_TARGETS) +ALL_COPY_MODULES += $(COPY_MODULES) +ALL_COPY_TARGETS += $(COPY_TARGETS) + +IMPORT_COPY_MODULES := $(call FindImportedModules) +IMPORT_COPY_TARGETS := $(addsuffix -copy, $(IMPORT_COPY_MODULES)) +ALL_COPY_MODULES += $(IMPORT_COPY_MODULES) +ALL_COPY_TARGETS += $(IMPORT_COPY_TARGETS) + +define DeclareImportCopyRecipe + $1-copy: + +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) \ + -f CopyImportModules.gmk MODULE=$1) +endef + +$(foreach m, $(IMPORT_COPY_MODULES), $(eval $(call DeclareImportCopyRecipe,$m))) + +ALL_TARGETS += $(ALL_COPY_TARGETS) ################################################################################ # Targets for compiling all java modules. Nashorn is treated separately. -JAVA_MODULES := $(call FindJavaModules) +JAVA_MODULES := $(ALL_MODULES) JAVA_TARGETS := $(addsuffix -java, $(JAVA_MODULES)) define DeclareCompileJavaRecipe @@ -214,7 +246,48 @@ samples-jdk: ALL_TARGETS += demos-jdk samples-jdk ################################################################################ -# Image targets +# Jigsaw specific data and analysis targets. + +generate-summary: + +($(CD) $(JDK_TOPDIR)/make && $(MAKE) -f GenerateModuleSummary.gmk) + +ALL_TARGETS += generate-summary + +################################################################################ +# Strip binaries targets + +STRIP_MODULES := $(sort $(LIBS_MODULES) $(LAUNCHER_MODULES) $(COPY_MODULES) \ + $(GENDATA_MODULES)) +STRIP_TARGETS := $(addsuffix -strip, $(STRIP_MODULES)) + +define DeclareStripRecipe + $1-strip: + +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f StripBinaries.gmk \ + MODULE=$1) +endef + +$(foreach m, $(STRIP_MODULES), $(eval $(call DeclareStripRecipe,$m))) + +ALL_TARGETS += $(STRIP_TARGETS) + +################################################################################ +# Jmod targets + +JMOD_MODULES := $(ALL_MODULES) +JMOD_TARGETS := $(addsuffix -jmod, $(JMOD_MODULES)) + +define DeclareJmodRecipe + $1-jmod: + +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f CreateJmods.gmk \ + MODULE=$1) +endef + +$(foreach m, $(JMOD_MODULES), $(eval $(call DeclareJmodRecipe,$m))) + +ALL_TARGETS += $(JMOD_TARGETS) + +################################################################################ +# Images targets # Stores the tips for each repository. This file is be used when constructing the jdk image and can be # used to track the exact sources used to build that image. @@ -236,9 +309,6 @@ zip-security: zip-source: +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f ZipSource.gmk) -strip-binaries: - +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f StripBinaries.gmk) - jrtfs-jar: +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f JrtfsJar.gmk) @@ -251,7 +321,7 @@ profiles: mac-bundles-jdk: +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f MacBundles.gmk) -ALL_TARGETS += source-tips bootcycle-images zip-security zip-source strip-binaries \ +ALL_TARGETS += source-tips bootcycle-images zip-security zip-source \ jrtfs-jar jimages profiles mac-bundles-jdk ################################################################################ @@ -265,6 +335,28 @@ docs-jvmtidoc: ALL_TARGETS += docs-javadoc docs-jvmtidoc +################################################################################ +# Cross compilation support + +ifeq ($(CREATE_BUILDJDK), true) + # This target is only called by the recursive call below. + create-buildjdk-compile-hotspot-helper: hotspot + create-buildjdk-compile-modules-helper: jdk.jlink-launchers java.base-copy +endif + +create-buildjdk-copy: + +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f CreateBuildJdkCopy.gmk) + +create-buildjdk-compile-hotspot create-buildjdk-compile-modules: + +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f Main.gmk \ + $@-helper \ + SPEC=$(dir $(SPEC))buildjdk-spec.gmk \ + HOTSPOT_SPEC=$(dir $(SPEC))buildjdk-spec.gmk \ + CREATING_BUILDJDK=true) + +ALL_TARGETS += create-buildjdk-copy create-buildjdk-compile-hotspot \ + create-buildjdk-compile-modules + ################################################################################ # Build tests # @@ -321,16 +413,6 @@ test-make: ALL_TARGETS += test test-hotspot-jtreg test-hotspot-jtreg-native \ test-hotspot-internal test-jdk-jtreg-native test-make -################################################################################ -# Verification targets - -verify-modules: - @$(call TargetEnter) - +($(CD) $(SRC_ROOT)/make && $(MAKE) $(MAKE_ARGS) -f CheckModules.gmk) - @$(call TargetExit) - -ALL_TARGETS += verify-modules - ################################################################################ # Install targets @@ -370,6 +452,8 @@ else $(JDK_GENSRC_TARGETS): interim-langtools buildtools-jdk + $(GENSRC_MODULEINFO_TARGETS): buildtools-jdk + $(GENDATA_TARGETS): interim-langtools buildtools-jdk interim-rmic: interim-langtools @@ -407,8 +491,11 @@ else $(foreach m, $(RMIC_MODULES), $(eval $m-rmic: $m-java)) # Declare dependencies from -lib to -java - # Skip modules that do not have java source. + # Skip modules that do not have java source. When creating a BUILD_JDK, the + # java compilation has already been done by the normal build and copied in. + ifneq ($(CREATING_BUILDJDK), true) $(foreach m, $(filter $(JAVA_MODULES), $(LIBS_MODULES)), $(eval $m-libs: $m-java)) + endif # Declare dependencies from all other -lib to java.base-lib $(foreach t, $(filter-out java.base-libs, $(LIBS_TARGETS)), \ @@ -438,31 +525,57 @@ else jdk.jdeps-gendata: java rmic + # Declare dependencies from -strip to libs, launchers, gendata and copy + $(foreach m, $(LIBS_MODULES), $(eval $m-strip: $m-libs)) + $(foreach m, $(LAUNCHER_MODULES), $(eval $m-strip: $m-launchers)) + $(foreach m, $(GENDATA_MODULES), $(eval $m-strip: $m-gendata)) + $(foreach m, $(COPY_MODULES), $(eval $m-strip: $m-copy)) + + # Declare dependencies between jmod targets + $(foreach m, $(JMOD_MODULES), \ + $(eval $m-jmod: $(addsuffix -jmod, $(call FindDepsForModule,$m)))) + + # Declare dependencies from -jmod to all other module targets + $(foreach m, $(STRIP_MODULES), $(eval $m-jmod: $m-strip)) + $(foreach m, $(JAVA_MODULES), $(eval $m-jmod: $m-java)) + $(foreach m, $(GENDATA_MODULES), $(eval $m-jmod: $m-gendata)) + $(foreach m, $(RMIC_MODULES), $(eval $m-jmod: $m-rmic)) + $(foreach m, $(LIBS_MODULES), $(eval $m-jmod: $m-libs)) + $(foreach m, $(LAUNCHER_MODULES), $(eval $m-jmod: $m-launchers)) + $(foreach m, $(COPY_MODULES), $(eval $m-jmod: $m-copy)) + + # Jmods cannot be created until we have the jlink tool ready to run, which requires + # all java modules to be compiled and jdk.jlink-launchers. + $(JMOD_TARGETS): java java.base-libs jdk.jlink-launchers + + ifeq ($(CREATE_BUILDJDK), true) + $(JMOD_TARGETS): create-buildjdk + endif + zip-security: java.base-java java.security.jgss-java java.security.jgss-libs \ $(filter jdk.crypto%, $(JAVA_TARGETS)) zip-source: gensrc rmic - strip-binaries: libs launchers gendata copy + jrtfs-jar: interim-langtools - jrtfs-jar: buildtools-jdk + jimages: jmods zip-source source-tips demos samples jrtfs-jar - jimages: exploded-image zip-source strip-binaries source-tips demos samples \ - jrtfs-jar - - profiles: exploded-image strip-binaries source-tips + profiles: jmods zip-source source-tips jrtfs-jar mac-bundles-jdk: jimages bootcycle-images: jimages - docs-javadoc: gensrc rmic + docs-javadoc: $(GENSRC_TARGETS) rmic docs-jvmtidoc: hotspot test: jimages test-image - verify-modules: exploded-image + create-buildjdk-copy: jdk.jlink-java java.base-gendata + + create-buildjdk-compile-modules: create-buildjdk-copy create-buildjdk-compile-hotspot test-make: clean-test-make @@ -482,6 +595,8 @@ else install: product-images + generate-summary: jmods + endif ################################################################################ @@ -494,7 +609,7 @@ gensrc: $(GENSRC_TARGETS) gendata: $(GENDATA_TARGETS) -copy: $(COPY_TARGETS) +copy: $(ALL_COPY_TARGETS) java: $(JAVA_TARGETS) @@ -504,6 +619,10 @@ libs: $(LIBS_TARGETS) launchers: $(LAUNCHER_TARGETS) +jmods: $(JMOD_TARGETS) + +strip-binaries: $(STRIP_TARGETS) + # Explicitly declare dependency for virtual target jdk.jdwp.agent-gensrc which # is actually handled by jdk.jdi-gensrc jdk.jdwp.agent-gensrc: jdk.jdi-gensrc @@ -516,8 +635,7 @@ $(foreach m, $(GENDATA_MODULES), $(eval $m: $m-gendata)) $(foreach m, $(RMIC_MODULES), $(eval $m: $m-rmic)) $(foreach m, $(LIBS_MODULES), $(eval $m: $m-libs)) $(foreach m, $(LAUNCHER_MODULES), $(eval $m: $m-launchers)) -$(foreach m, $(COPY_MODULES), $(eval $m: $m-copy)) - +$(foreach m, $(ALL_COPY_MODULES), $(eval $m: $m-copy)) demos: demos-jdk samples: samples-jdk @@ -525,6 +643,9 @@ samples: samples-jdk # The "exploded image" is a locally runnable JDK in $(BUILD_OUTPUT)/jdk. exploded-image: $(ALL_MODULES) +create-buildjdk: create-buildjdk-compile-modules create-buildjdk-copy \ + create-buildjdk-compile-hotspot + mac-bundles: mac-bundles-jdk # The $(BUILD_OUTPUT)/images directory contain the resulting deliverables, @@ -532,7 +653,13 @@ mac-bundles: mac-bundles-jdk # This target builds the product images, e.g. the JRE and JDK image # (and possibly other, more specific versions) -product-images: jimages demos samples zip-security verify-modules +product-images: jimages demos samples zip-security exploded-image + +# When cross compiling and building a partial BUILDJDK for the build host, +# the summary generation cannot be run. +ifneq ($(CREATE_BUILDJDK), true) + product-images: generate-summary +endif ifeq ($(OPENJDK_TARGET_OS), macosx) product-images: mac-bundles @@ -548,9 +675,9 @@ test-image: prepare-test-image test-image-hotspot-jtreg-native \ # all-images is the top-most target, it builds all our deliverables ("images"). all-images: product-images test-image docs-image -ALL_TARGETS += buildtools gensrc gendata copy java rmic libs launchers \ +ALL_TARGETS += buildtools gensrc gendata copy java rmic libs launchers jmods \ jdk.jdwp.agent-gensrc $(ALL_MODULES) demos samples exploded-image \ - mac-bundles product-images docs-image test-image all-images + create-buildjdk mac-bundles product-images docs-image test-image all-images ################################################################################ @@ -575,7 +702,7 @@ ALL_TARGETS += default jdk images docs all # file. CLEAN_DIRS += hotspot jdk bootcycle-build test buildtools support \ - images make-support test-make bundles + images make-support test-make bundles buildjdk CLEAN_DIR_TARGETS := $(addprefix clean-, $(CLEAN_DIRS)) CLEAN_SUPPORT_DIRS += demos CLEAN_SUPPORT_DIR_TARGETS := $(addprefix clean-, $(CLEAN_SUPPORT_DIRS)) diff --git a/make/MainSupport.gmk b/make/MainSupport.gmk index be697f663cf..9d5865d2b2c 100644 --- a/make/MainSupport.gmk +++ b/make/MainSupport.gmk @@ -67,7 +67,6 @@ define Clean-gensrc @$(PRINTF) "Cleaning gensrc $(if $1,for $(strip $1) )..." @$(PRINTF) "\n" $(LOG_DEBUG) $(RM) -r $(SUPPORT_OUTPUTDIR)/gensrc/$(strip $1) - $(RM) -r $(SUPPORT_OUTPUTDIR)/gensrc_no_docs/$(strip $1) @$(PRINTF) " done\n" endef diff --git a/make/StripBinaries.gmk b/make/StripBinaries.gmk index 07669bf93aa..26b3ee92338 100644 --- a/make/StripBinaries.gmk +++ b/make/StripBinaries.gmk @@ -57,25 +57,29 @@ endif NO_STRIP_CMDS_FILTER += %.cgi # Don't include debug info for executables. +ifneq ($(wildcard $(SUPPORT_OUTPUTDIR)/modules_cmds/$(MODULE)), ) + # OS X stores symbol information in a .dylib file inside a .dSYM directory - + # that file should not be stripped, so we prune the tree at the .dSYM directory. + ALL_CMDS_SRC := $(filter-out %.bc %.debuginfo %.diz %.map %.pdb, \ + $(shell $(FIND) $(SUPPORT_OUTPUTDIR)/modules_cmds/$(MODULE) \( -type f -o -type l \) \ + -print -o -name "*.dSYM" -prune)) + COPY_CMDS_SRC := $(filter $(NO_STRIP_CMDS_FILTER), $(ALL_CMDS_SRC)) + STRIP_CMDS_SRC := $(filter-out $(NO_STRIP_CMDS_FILTER), $(ALL_CMDS_SRC)) +endif -# OS X stores symbol information in a .dylib file inside a .dSYM directory - -# that file should not be stripped, so we prune the tree at the .dSYM directory. -ALL_CMDS_SRC := $(filter-out %.bc %.debuginfo %.diz %.map %.pdb, \ - $(shell $(FIND) $(SUPPORT_OUTPUTDIR)/modules_cmds \( -type f -o -type l \) \ - -print -o -name "*.dSYM" -prune)) -COPY_CMDS_SRC := $(filter $(NO_STRIP_CMDS_FILTER), $(ALL_CMDS_SRC)) -STRIP_CMDS_SRC := $(filter-out $(NO_STRIP_CMDS_FILTER), $(ALL_CMDS_SRC)) - -# OS X stores symbol information in a .dylib file inside a .dSYM directory - -# that file should not be stripped, so we prune the tree at the .dSYM directory. -# Example: support/modules_libs/java.base/libjsig.dylib.dSYM/Contents/Resources/DWARF/libjsig.dylib -STRIP_LIBS_SRC := \ - $(shell $(FIND) $(SUPPORT_OUTPUTDIR)/modules_libs \ +ifneq ($(wildcard $(SUPPORT_OUTPUTDIR)/modules_libs/$(MODULE)), ) + # OS X stores symbol information in a .dylib file inside a .dSYM directory - + # that file should not be stripped, so we prune the tree at the .dSYM directory. + # Example: support/modules_libs/java.base/libjsig.dylib.dSYM/Contents/Resources/DWARF/libjsig.dylib + STRIP_LIBS_SRC := \ + $(shell $(FIND) $(SUPPORT_OUTPUTDIR)/modules_libs/$(MODULE) \ -name '*$(SHARED_LIBRARY_SUFFIX)' -type f -print -o -name "*.dSYM" -prune) -# Make sure symbolic links are copied and not stripped. -COPY_LIBS_SRC := \ - $(filter-out $(STRIP_LIBS_SRC), \ - $(shell $(FIND) $(SUPPORT_OUTPUTDIR)/modules_libs -type f -o -type l)) + + # Make sure symbolic links are copied and not stripped. + COPY_LIBS_SRC := \ + $(filter-out $(STRIP_LIBS_SRC), \ + $(shell $(FIND) $(SUPPORT_OUTPUTDIR)/modules_libs/$(MODULE) -type f -o -type l)) +endif $(eval $(call SetupCopyFiles,STRIP_MODULES_CMDS, \ SRC := $(SUPPORT_OUTPUTDIR)/modules_cmds, \ diff --git a/make/common/CORE_PKGS.gmk b/make/common/CORE_PKGS.gmk index eb6ac3d769e..850073385a4 100644 --- a/make/common/CORE_PKGS.gmk +++ b/make/common/CORE_PKGS.gmk @@ -1,5 +1,5 @@ # -# Copyright (c) 2001, 2011, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2001, 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 @@ -99,6 +99,7 @@ CORE_PKGS = \ java.lang.instrument \ java.lang.invoke \ java.lang.management \ + java.lang.module \ java.lang.ref \ java.lang.reflect \ java.math \ diff --git a/make/common/JavaCompilation.gmk b/make/common/JavaCompilation.gmk index 29442c23151..22e9d908bfc 100644 --- a/make/common/JavaCompilation.gmk +++ b/make/common/JavaCompilation.gmk @@ -75,13 +75,13 @@ define add_file_to_copy ifneq ($$($1_COPY_$$($2_TARGET)), 1) $1_COPY_$$($2_TARGET) := 1 # Now we can setup the dependency that will trigger the copying. - $$($1_BIN)$$($2_TARGET) : $2 + $$($1_BIN)$$($1_MODULE_SUBDIR)$$($2_TARGET) : $2 $(MKDIR) -p $$(@D) $(CP) $$< $$@ $(CHMOD) -f ug+w $$@ # And do not forget this target - $1_ALL_COPY_TARGETS += $$($1_BIN)$$($2_TARGET) + $1_ALL_COPY_TARGETS += $$($1_BIN)$$($1_MODULE_SUBDIR)$$($2_TARGET) endif endef @@ -133,7 +133,7 @@ define add_file_to_clean $(CHMOD) -f ug+w $$@ # And do not forget this target - $1_ALL_COPY_CLEAN_TARGETS += $$($1_BIN)$$($2_TARGET) + $1_ALL_COPY_CLEAN_TARGETS += $$($1_BIN)$$($1_MODULE_SUBDIR)$$($2_TARGET) endif endef @@ -154,6 +154,7 @@ endef # SRC:=one or more directories to search for sources. The order of the source roots # is significant. The first found file of a certain name has priority. # BIN:=store classes here +# MODULE:=Name of module being compiled. If set, classes are put in BIN/MODULE. # CLASSPATH:=a list of additional entries to set as classpath to javac # INCLUDES:=myapp.foo means will only compile java files in myapp.foo or any of its sub-packages. # EXCLUDES:=myapp.foo means will do not compile java files in myapp.foo or any of its sub-packages. @@ -190,6 +191,10 @@ define SetupJavaCompilationBody $1_SJAVAC_PORTFILE := $$($$($1_SETUP)_SJAVAC_PORTFILE) $1_SERVER_JVM := $$($$($1_SETUP)_SERVER_JVM) $1_DISABLE_SJAVAC := $$($$($1_SETUP)_DISABLE_SJAVAC) + + ifneq ($$($1_MODULE), ) + $1_MODULE_SUBDIR := /$$($1_MODULE) + endif # Make sure the dirs exist. $$(foreach d,$$($1_SRC), $$(if $$(wildcard $$d),,$$(error SRC specified to SetupJavaCompilation $1 contains missing directory >$$d<))) @@ -303,7 +308,7 @@ define SetupJavaCompilationBody $1_REMOTE:=--server:portfile=$$($1_SJAVAC_PORTFILE),id=$1,sjavac=$$(subst \ $$(SPACE),%20,$$(subst $$(COMMA),%2C,$$(strip $$($1_SERVER_JVM) $$($1_SJAVAC)))) - $1_COMPILE_TARGET := $$($1_BIN)/_the.$1_batch + $1_COMPILE_TARGET := $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1_batch ifeq ($$($1_DISABLE_SJAVAC)x$$(ENABLE_SJAVAC),xyes) # Using sjavac to compile. @@ -336,7 +341,8 @@ define SetupJavaCompilationBody $1_VARDEPS := $$($1_JVM) $$($1_SJAVAC) $$($1_SJAVAC_ARGS_STRING) $$($1_FLAGS) \ $$($1_HEADERS_ARG) $$($1_BIN) $$($1_EXCLUDES) $$($1_INCLUDES) \ $$($1_EXCLUDE_FILES) $$($1_INCLUDE_FILES) - $1_VARDEPS_FILE := $$(call DependOnVariable, $1_VARDEPS, $$($1_BIN)/_the.$1.vardeps) + $1_VARDEPS_FILE := $$(call DependOnVariable, $1_VARDEPS, \ + $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1.vardeps) $$($1_COMPILE_TARGET): $$($1_SRCS) $$($1_DEPENDS) $$($1_VARDEPS_FILE) $$(call MakeDir, $$(@D) $$(dir $$($1_SJAVAC_PORTFILE))) @@ -345,7 +351,7 @@ define SetupJavaCompilationBody $$(eval $$(call ListPathsSafely,$1_SJAVAC_ARGS_STRING, $$($1_SJAVAC_ARGS_FILE))) endif $$(call LogWarn, Compiling $1) - $$(call ExecuteWithLog, $$($1_BIN)/_the.$$($1_SAFE_NAME)_batch, \ + $$(call ExecuteWithLog, $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$$($1_SAFE_NAME)_batch, \ $$($1_JVM) $$($1_SJAVAC) \ $$($1_REMOTE) \ -j 1 \ @@ -353,7 +359,7 @@ define SetupJavaCompilationBody --permit-sources-without-package \ --compare-found-sources $$@.tmp \ --log=$(LOG_LEVEL) \ - --state-dir=$$($1_BIN) \ + --state-dir=$$($1_BIN)$$($1_MODULE_SUBDIR) \ $$($1_SJAVAC_ARGS) \ $$($1_FLAGS) \ $$($1_HEADERS_ARG) \ @@ -362,11 +368,13 @@ define SetupJavaCompilationBody # Create a pubapi file that only changes when the pubapi changes. Dependent # compilations can use this file to only get recompiled when pubapi has changed. # Grep returns 1 if no matching lines are found. Do not fail for this. - $(GREP) -e "^I" $$($1_BIN)/javac_state > $$($1_BIN)/_the.$1_pubapi.tmp \ + $(GREP) -e "^I" $$($1_BIN)$$($1_MODULE_SUBDIR)/javac_state > $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1_pubapi.tmp \ || test "$$$$?" = "1" - if [ ! -f $$($1_BIN)/_the.$1_pubapi ] \ - || [ "`$(DIFF) $$($1_BIN)/_the.$1_pubapi $$($1_BIN)/_the.$1_pubapi.tmp`" != "" ]; then \ - $(MV) $$($1_BIN)/_the.$1_pubapi.tmp $$($1_BIN)/_the.$1_pubapi; \ + if [ ! -f $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1_pubapi ] \ + || [ "`$(DIFF) $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1_pubapi \ + $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1_pubapi.tmp`" != "" ]; then \ + $(MV) $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1_pubapi.tmp \ + $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1_pubapi; \ fi else @@ -380,11 +388,12 @@ define SetupJavaCompilationBody $$($1_HEADERS)/_the.$1_headers: $$($1_COMPILE_TARGET) $(MKDIR) -p $$(@D) if [ -d "$$($1_HEADERS).$1.tmp" ]; then \ - for f in `ls $$($1_HEADERS).$1.tmp`; do \ + for f in `$(CD) $$($1_HEADERS).$1.tmp && $(FIND) . -type f`; do \ if [ ! -f "$$($1_HEADERS)/$$$$f" ] \ || [ "`$(DIFF) $$($1_HEADERS)/$$$$f $$($1_HEADERS).$1.tmp/$$$$f`" != "" ]; then \ - $(CP) -f $$($1_HEADERS).$1.tmp/$$$$f $$($1_HEADERS)/$$$$f; \ - fi; \ + $(MKDIR) -p `$(DIRNAME) $$($1_HEADERS)/$$$$f`; \ + $(CP) -f $$($1_HEADERS).$1.tmp/$$$$f $$($1_HEADERS)/$$$$f; \ + fi; \ done; \ fi $(RM) -r $$($1_HEADERS).$1.tmp @@ -396,7 +405,8 @@ define SetupJavaCompilationBody $1_VARDEPS := $$($1_JVM) $$($1_JAVAC) $$($1_FLAGS) $$($1_BIN) \ $$($1_HEADERS_ARG) $$($1_EXCLUDES) $$($1_INCLUDES) \ $$($1_EXCLUDE_FILES) $$($1_INCLUDE_FILES) - $1_VARDEPS_FILE := $$(call DependOnVariable, $1_VARDEPS, $$($1_BIN)/_the.$1.vardeps) + $1_VARDEPS_FILE := $$(call DependOnVariable, $1_VARDEPS, \ + $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$1.vardeps) ifeq ($$($1_DISABLE_SJAVAC)x$(ENABLE_JAVAC_SERVER), xyes) $1_JAVAC_CMD := $$($1_SJAVAC) $$($1_REMOTE) @@ -409,7 +419,7 @@ define SetupJavaCompilationBody $$(call MakeDir, $$(@D)) $$(eval $$(call ListPathsSafely,$1_SRCS, $$@.tmp)) $$(call LogWarn, Compiling $$(words $$($1_SRCS)) files for $1) - $$(call ExecuteWithLog, $$($1_BIN)/_the.$$($1_SAFE_NAME)_batch, \ + $$(call ExecuteWithLog, $$($1_BIN)$$($1_MODULE_SUBDIR)/_the.$$($1_SAFE_NAME)_batch, \ $$($1_JVM) $$($1_JAVAC_CMD) $$($1_FLAGS) \ -implicit:none \ -d $$($1_BIN) $$($1_HEADERS_ARG) @$$@.tmp) && \ @@ -431,7 +441,7 @@ define SetupJavaCompilationBody $$(eval $$(call SetupJarArchive, ARCHIVE_$1, \ DEPENDENCIES:=$$($1), \ - SRCS:=$$($1_BIN), \ + SRCS:=$$($1_BIN)$$($1_MODULE_SUBDIR), \ SUFFIXES:=$$($1_SUFFIXES), \ EXCLUDE:=$$($1_EXCLUDES), \ INCLUDES:=$$($1_INCLUDES), \ diff --git a/make/common/MakeBase.gmk b/make/common/MakeBase.gmk index 6200851793c..7ed52135f28 100644 --- a/make/common/MakeBase.gmk +++ b/make/common/MakeBase.gmk @@ -247,7 +247,7 @@ REPO_LIST = $(patsubst ./%,%,$(patsubst %/,%,$(sort $(dir \ $(HG_SEARCH:%/REPO=%/$(HGTIP_FILENAME)) \ 2> /dev/null))))) -# Emit the repo:tip pairs to $@ +# Emit the repo:tip pairs to $@, but only if they changed since last time define GetSourceTips $(CD) $(SRC_ROOT) ; \ for i in $(REPO_LIST) IGNORE ; do \ @@ -260,8 +260,13 @@ define GetSourceTips $(PRINTF) " %s:%s" \ "$${i}" `$(CAT) $${i}/$(HGTIP_FILENAME)` ; \ fi; \ - done >> $@ - $(PRINTF) "\n" >> $@ + done > $@.tmp + $(PRINTF) "\n" >> $@.tmp + if [ ! -f $@ ] || [ "`$(CAT) $@`" != "`$(CAT) $@.tmp`" ]; then \ + $(MV) $@.tmp $@ ; \ + else \ + $(RM) $@.tmp ; \ + fi endef # Create the HGTIP_FILENAME file. Called from closed/make/SourceBundles.gmk @@ -644,10 +649,10 @@ endif # Defines the sub directory structure to store variable value file in DependOnVariableDirName = \ - $(strip $(subst $(SRC_ROOT)/,,\ - $(if $(filter /%, $(firstword $(MAKEFILE_LIST))), \ + $(strip $(addsuffix $(if $(MODULE),/$(MODULE)), \ + $(subst $(SRC_ROOT)/,, $(if $(filter /%, $(firstword $(MAKEFILE_LIST))), \ $(firstword $(MAKEFILE_LIST)), \ - $(CURDIR)/$(firstword $(MAKEFILE_LIST))))) + $(CURDIR)/$(firstword $(MAKEFILE_LIST)))))) # Defines the name of the file to store variable value in. Generates a name # unless parameter 2 is given. diff --git a/make/common/Modules.gmk b/make/common/Modules.gmk index 54f7a2b44a9..b3478f29aeb 100644 --- a/make/common/Modules.gmk +++ b/make/common/Modules.gmk @@ -26,6 +26,9 @@ ifndef _MODULES_GMK _MODULES_GMK := 1 +# Hook to include the corresponding custom file, if present. +$(eval $(call IncludeCustomExtension, , common/Modules.gmk)) + ################################################################################ # Some platforms don't have the serviceability agent ifeq ($(INCLUDE_SA), false) @@ -35,7 +38,9 @@ endif ################################################################################ # Module list macros -ALL_TOP_SRC_DIRS := \ +# Use append so that the custom extension may add to this variable + +ALL_TOP_SRC_DIRS += \ $(HOTSPOT_TOPDIR)/src \ $(JDK_TOPDIR)/src \ $(LANGTOOLS_TOPDIR)/src \ @@ -45,37 +50,74 @@ ALL_TOP_SRC_DIRS := \ $(NASHORN_TOPDIR)/src \ # -# Find all modules with java sources by looking in the source dirs -define FindJavaModules - $(filter-out $(MODULES_FILTER), $(sort $(notdir \ - $(patsubst %/,%, $(dir $(patsubst %/,%, $(dir $(patsubst %/,%, $(dir \ - $(wildcard $(patsubst %,%/*/share/classes/*, $(ALL_TOP_SRC_DIRS)) \ - $(patsubst %,%/*/$(OPENJDK_TARGET_OS)/classes/*, $(ALL_TOP_SRC_DIRS)) \ - $(patsubst %,%/*/$(OPENJDK_TARGET_OS_TYPE)/classes/*, $(ALL_TOP_SRC_DIRS)))))))))))) -endef +# Find all module-info.java files for the current build target platform and +# configuration. +# Param 1 - Module to find for, set to * for finding all +FindAllModuleInfos = \ + $(wildcard \ + $(patsubst %,%/$(strip $1)/$(OPENJDK_TARGET_OS)/classes/module-info.java, $(ALL_TOP_SRC_DIRS)) \ + $(patsubst %,%/$(strip $1)/$(OPENJDK_TARGET_OS_TYPE)/classes/module-info.java, $(ALL_TOP_SRC_DIRS)) \ + $(patsubst %,%/$(strip $1)/share/classes/module-info.java, $(ALL_TOP_SRC_DIRS)) \ + $(patsubst %,%/$(strip $1)/module-info.java, $(IMPORT_MODULES_SRC))) -# Find all modules with source for the target platform. -define FindAllModules - $(sort $(filter-out $(MODULES_FILTER) closed demo sample, \ - $(notdir $(patsubst %/,%, $(dir \ - $(wildcard $(patsubst %, %/*/share, $(ALL_TOP_SRC_DIRS)) \ - $(patsubst %, %/*/$(OPENJDK_TARGET_OS), $(ALL_TOP_SRC_DIRS)) \ - $(patsubst %, %/*/$(OPENJDK_TARGET_OS_TYPE), $(ALL_TOP_SRC_DIRS)))))))) -endef +# Extract the module names from the paths of module-info.java files. The +# position of the module directory differs depending on if this is an imported +# src dir or not. +GetModuleNameFromModuleInfo = \ + $(strip $(foreach mi, $1, \ + $(if $(filter $(addsuffix %, $(IMPORT_MODULES_SRC)), $(mi)), \ + $(notdir $(patsubst %/,%, $(dir $(mi)))), \ + $(notdir $(patsubst %/,%, $(dir $(patsubst %/,%, $(dir $(patsubst %/,%, $(dir $(mi))))))))))) + +# Find all modules by looking for module-info.java files and looking at parent +# directories. +FindAllModules = \ + $(sort $(filter-out $(MODULES_FILTER), \ + $(call GetModuleNameFromModuleInfo, $(MODULE_INFOS)))) + +FindImportedModules = \ + $(if $(IMPORT_MODULES_CLASSES), $(notdir $(wildcard $(IMPORT_MODULES_CLASSES)/*))) ################################################################################ -# The module deps makefile is generated in make/GenerateModuleDeps.gmk +# Extract module dependencies from module-info.java files. + MODULE_DEPS_MAKEFILE := $(MAKESUPPORT_OUTPUTDIR)/module-deps.gmk + +MODULE_INFOS := $(call FindAllModuleInfos, *) + +$(MODULE_DEPS_MAKEFILE): $(MODULE_INFOS) \ + $(call DependOnVariable, MODULE_INFOS, $(MAKESUPPORT_OUTPUTDIR)/MODULE_INFOS.vardeps) + $(MKDIR) -p $(@D) + $(RM) $@ + $(foreach m, $(MODULE_INFOS), \ + ( $(PRINTF) "DEPS_$(call GetModuleNameFromModuleInfo, $m) :=" && \ + $(NAWK) -v MODULE=$(call GetModuleNameFromModuleInfo, $m) '\ + BEGIN { if (MODULE != "java.base") printf(" java.base"); } \ + /requires/ { sub(/;/, ""); \ + sub(/requires/, ""); \ + sub(/public/, ""); \ + sub(/\/\/.*/, ""); \ + sub(/\/\*.*\*\//, ""); \ + gsub(/ /, ""); \ + printf(" %s", $$0) } \ + END { printf("\n") }' $m \ + ) >> $@ $(NEWLINE)) + -include $(MODULE_DEPS_MAKEFILE) # Param 1: Module to find deps for -define FindDepsForModule +FindDepsForModule = \ $(DEPS_$(strip $1)) -endef + +# Finds transitive dependencies in 3 levels. +# Param 1: Module to find transitive deps for +FindTransitiveDepsForModule = \ + $(sort $(call FindDepsForModule, $1) \ + $(foreach m, $(call FindDepsForModule, $1), \ + $(call FindDepsForModule, $m) \ + $(foreach n, $(call FindDepsForModule, $m), \ + $(call FindDepsForModule, $n)))) ################################################################################ -# Hook to include the corresponding custom file, if present. -$(eval $(call IncludeCustomExtension, , common/Modules.gmk)) - endif # _MODULES_GMK diff --git a/make/common/NativeCompilation.gmk b/make/common/NativeCompilation.gmk index f675790efc6..2ddd3760559 100644 --- a/make/common/NativeCompilation.gmk +++ b/make/common/NativeCompilation.gmk @@ -74,6 +74,7 @@ endef # AS - Assembler # MT - Windows MT tool # RC - Windows RC tool +# OBJCOPY - The objcopy tool for debug symbol handling # STRIP - The tool to use for stripping debug symbols # SYSROOT_CFLAGS - Compiler flags for using the specific sysroot # SYSROOT_LDFLAGS - Linker flags for using the specific sysroot @@ -90,6 +91,7 @@ define DefineNativeToolchainBody $$(call SetIfEmpty, $1_AS, $$($$($1_EXTENDS)_AS)) $$(call SetIfEmpty, $1_MT, $$($$($1_EXTENDS)_MT)) $$(call SetIfEmpty, $1_RC, $$($$($1_EXTENDS)_RC)) + $$(call SetIfEmpty, $1_OBJCOPY, $$($$($1_EXTENDS)_OBJCOPY)) $$(call SetIfEmpty, $1_STRIP, $$($$($1_EXTENDS)_STRIP)) $$(call SetIfEmpty, $1_SYSROOT_CFLAGS, $$($$($1_EXTENDS)_SYSROOT_CFLAGS)) $$(call SetIfEmpty, $1_SYSROOT_LDFLAGS, $$($$($1_EXTENDS)_SYSROOT_LDFLAGS)) @@ -105,6 +107,7 @@ $(eval $(call DefineNativeToolchain, TOOLCHAIN_DEFAULT, \ AS := $(AS), \ MT := $(MT), \ RC := $(RC), \ + OBJCOPY := $(OBJCOPY), \ STRIP := $(STRIP), \ SYSROOT_CFLAGS := $(SYSROOT_CFLAGS), \ SYSROOT_LDFLAGS := $(SYSROOT_LDFLAGS), \ @@ -118,13 +121,14 @@ $(eval $(call DefineNativeToolchain, TOOLCHAIN_LINK_CXX, \ # Create a toolchain with the BUILD compiler, used for build tools that # are to be run during the build. -# The BUILD_SYSROOT_*FLAGS variables are empty for now. $(eval $(call DefineNativeToolchain, TOOLCHAIN_BUILD, \ CC := $(BUILD_CC), \ CXX := $(BUILD_CXX), \ LD := $(BUILD_LD), \ AR := $(BUILD_AR), \ AS := $(BUILD_AS), \ + OBJCOPY := $(BUILD_OBJCOPY), \ + STRIP := $(BUILD_STRIP), \ SYSROOT_CFLAGS := $(BUILD_SYSROOT_CFLAGS), \ SYSROOT_LDFLAGS := $(BUILD_SYSROOT_LDFLAGS), \ )) @@ -437,6 +441,7 @@ define SetupNativeCompilationBody $$(call SetIfEmpty, $1_AS, $$($$($1_TOOLCHAIN)_AS)) $$(call SetIfEmpty, $1_MT, $$($$($1_TOOLCHAIN)_MT)) $$(call SetIfEmpty, $1_RC, $$($$($1_TOOLCHAIN)_RC)) + $$(call SetIfEmpty, $1_OBJCOPY, $$($$($1_TOOLCHAIN)_OBJCOPY)) $$(call SetIfEmpty, $1_STRIP, $$($$($1_TOOLCHAIN)_STRIP)) $$(call SetIfEmpty, $1_SYSROOT_CFLAGS, $$($$($1_TOOLCHAIN)_SYSROOT_CFLAGS)) $$(call SetIfEmpty, $1_SYSROOT_LDFLAGS, $$($$($1_TOOLCHAIN)_SYSROOT_LDFLAGS)) @@ -720,9 +725,9 @@ define SetupNativeCompilationBody # Setup the command line creating debuginfo files, to be run after linking. # It cannot be run separately since it updates the original target file $1_CREATE_DEBUGINFO_CMDS := \ - $(OBJCOPY) --only-keep-debug $$($1_TARGET) $$($1_DEBUGINFO_FILES) $$(NEWLINE) \ - $(CD) $$($1_OUTPUT_DIR) && \ - $(OBJCOPY) --add-gnu-debuglink=$$($1_DEBUGINFO_FILES) $$($1_TARGET) + $$($1_OBJCOPY) --only-keep-debug $$($1_TARGET) $$($1_DEBUGINFO_FILES) $$(NEWLINE) \ + $(CD) $$($1_OUTPUT_DIR) && \ + $$($1_OBJCOPY) --add-gnu-debuglink=$$($1_DEBUGINFO_FILES) $$($1_TARGET) else ifeq ($(OPENJDK_TARGET_OS), macosx) $1_DEBUGINFO_FILES := $$($1_OBJECT_DIR)/$$($1_BASENAME).dSYM diff --git a/make/common/SetupJavaCompilers.gmk b/make/common/SetupJavaCompilers.gmk index a381d1c65a4..8100cdbeccb 100644 --- a/make/common/SetupJavaCompilers.gmk +++ b/make/common/SetupJavaCompilers.gmk @@ -38,7 +38,8 @@ JAVAC_WARNINGS := -Xlint:all -Werror # and the interim javac, to be run by the boot jdk. $(eval $(call SetupJavaCompiler,BOOT_JAVAC, \ JAVAC := $(JAVAC), \ - FLAGS := -XDignore.symbol.file=true -g -Xlint:all$(COMMA)-deprecation -Werror, \ + FLAGS := $(BOOT_JDK_SOURCETARGET) -XDignore.symbol.file=true -g \ + -Xlint:all$(COMMA)-deprecation$(COMMA)-options -Werror, \ DISABLE_SJAVAC := true, \ )) @@ -80,9 +81,6 @@ $(eval $(call SetupJavaCompiler,GENERATE_JDKBYTECODE_NOWARNINGS, \ SERVER_DIR := $(SJAVAC_SERVER_DIR), \ SERVER_JVM := $(SJAVAC_SERVER_JAVA))) -JDK_BOOTCLASSPATH := $(call PathList, \ - $(filter-out $(JDK_OUTPUTDIR)/modules/_%, $(wildcard $(JDK_OUTPUTDIR)/modules/*))) - # After the jdk is built, we want to build demos using only the recently # generated jdk classes and nothing else, no jdk source, etc etc. # I.e. the rt.jar, but since rt.jar has not yet been generated @@ -90,7 +88,7 @@ JDK_BOOTCLASSPATH := $(call PathList, \ $(eval $(call SetupJavaCompiler,GENERATE_USINGJDKBYTECODE, \ JVM := $(JAVA_SMALL), \ JAVAC := $(NEW_JAVAC), \ - FLAGS := -bootclasspath $(JDK_BOOTCLASSPATH) $(DISABLE_WARNINGS), \ + FLAGS := -upgrademodulepath $(JDK_OUTPUTDIR)/modules -system none $(DISABLE_WARNINGS), \ SERVER_DIR := $(SJAVAC_SERVER_DIR), \ SERVER_JVM := $(SJAVAC_SERVER_JAVA))) diff --git a/modules.xml b/modules.xml deleted file mode 100644 index 7061f52301b..00000000000 --- a/modules.xml +++ /dev/null @@ -1,2139 +0,0 @@ - - - - - - - - - java.activation - java.base - java.datatransfer - java.desktop - java.logging - - javax.activation - - - - java.annotations.common - java.base - - javax.annotation - - - - java.base - - java.io - - - java.lang - - - java.lang.annotation - - - java.lang.invoke - - - java.lang.ref - - - java.lang.reflect - - - java.math - - - java.net - - - java.nio - - - java.nio.channels - - - java.nio.channels.spi - - - java.nio.charset - - - java.nio.charset.spi - - - java.nio.file - - - java.nio.file.attribute - - - java.nio.file.spi - - - java.security - - - java.security.acl - - - java.security.cert - - - java.security.interfaces - - - java.security.spec - - - java.text - - - java.text.spi - - - java.time - - - java.time.chrono - - - java.time.format - - - java.time.temporal - - - java.time.zone - - - java.util - - - java.util.concurrent - - - java.util.concurrent.atomic - - - java.util.concurrent.locks - - - java.util.function - - - java.util.jar - - - java.util.regex - - - java.util.spi - - - java.util.stream - - - java.util.zip - - - javax.crypto - - - javax.crypto.interfaces - - - javax.crypto.spec - - - javax.net - - - javax.net.ssl - - - javax.security.auth - - - javax.security.auth.callback - - - javax.security.auth.login - - - javax.security.auth.spi - - - javax.security.auth.x500 - - - javax.security.cert - - - jdk - - - jdk.net - - - com.sun.security.ntlm - java.security.sasl - - - jdk.internal.jimage - jdk.dev - - - jdk.internal.jimage.decompressor - jdk.dev - - - jdk.internal.math - java.desktop - - - jdk.internal.misc - java.corba - java.desktop - java.logging - java.management - java.naming - java.rmi - java.security.jgss - java.sql - java.xml - jdk.charsets - jdk.management.resource - jdk.jfr - jdk.net - jdk.scripting.nashorn - jdk.vm.ci - - - jdk.internal.perf - java.desktop - java.management - jdk.jvmstat - - - jdk.internal.org.xml.sax - jdk.jfr - - - jdk.internal.org.xml.sax.helpers - jdk.jfr - - - jdk.internal.util.xml - jdk.jfr - - - jdk.internal.util.xml.impl - jdk.jfr - - - jdk.internal.org.objectweb.asm - java.instrument - jdk.jfr - jdk.scripting.nashorn - jdk.vm.ci - - - jdk.internal.org.objectweb.asm.commons - java.instrument - jdk.jfr - jdk.scripting.nashorn - - - jdk.internal.org.objectweb.asm.signature - jdk.scripting.nashorn - - - jdk.internal.org.objectweb.asm.tree - java.instrument - jdk.jfr - - - jdk.internal.org.objectweb.asm.util - java.instrument - jdk.jfr - jdk.scripting.nashorn - - - jdk.internal.ref - java.desktop - - - jdk.internal - jdk.jfr - - - sun.misc - java.corba - java.desktop - java.instrument - java.logging - java.management - java.naming - java.prefs - java.rmi - java.security.jgss - java.security.sasl - java.xml - jdk.charsets - jdk.crypto.pkcs11 - jdk.httpserver - jdk.jartool - jdk.jconsole - jdk.jvmstat - jdk.management.resource - jdk.pack200 - jdk.scripting.nashorn - jdk.security.auth - jdk.security.jgss - jdk.snmp - jdk.vm.ci - jdk.zipfs - java.instrument - jdk.jfr - - - sun.net - java.httpclient - - - sun.net.dns - java.security.jgss - jdk.naming.dns - - - sun.net.sdp - jdk.net - - - sun.net.spi.nameservice - jdk.naming.dns - - - sun.net.util - java.desktop - jdk.jconsole - jdk.naming.dns - - - sun.net.www - java.desktop - jdk.compiler - jdk.jartool - - - sun.net.www.protocol.http - java.security.jgss - - - sun.nio.ch - java.management - jdk.crypto.pkcs11 - jdk.crypto.ucrypto - jdk.management.resource - jdk.net - jdk.sctp - - - sun.nio.cs - java.desktop - jdk.charsets - - - sun.reflect - java.corba - java.instrument - java.logging - java.sql - java.sql.rowset - jdk.dynalink - jdk.scripting.nashorn - jdk.vm.cds - - - sun.reflect.annotation - jdk.compiler - - - sun.reflect.generics.reflectiveObjects - java.desktop - - - sun.reflect.misc - java.corba - java.datatransfer - java.desktop - java.management - java.rmi - java.sql.rowset - java.xml - - - sun.security.action - java.desktop - java.security.jgss - jdk.crypto.ec - jdk.crypto.pkcs11 - jdk.crypto.ucrypto - - - sun.security.internal.interfaces - jdk.crypto.pkcs11 - - - sun.security.internal.spec - jdk.crypto.mscapi - jdk.crypto.pkcs11 - jdk.crypto.ucrypto - - - sun.security.jca - java.smartcardio - java.xml.crypto - jdk.crypto.ec - jdk.crypto.pkcs11 - jdk.crypto.ucrypto - jdk.naming.dns - - - sun.security.pkcs - jdk.crypto.ec - jdk.jartool - - - sun.security.provider - java.rmi - java.security.jgss - jdk.crypto.pkcs11 - jdk.jartool - jdk.policytool - jdk.security.auth - - - sun.security.provider.certpath - java.naming - - - sun.security.rsa - jdk.crypto.mscapi - jdk.crypto.pkcs11 - - - sun.security.ssl - java.security.jgss - - - sun.security.tools - jdk.jartool - - - sun.security.util - java.desktop - java.naming - java.rmi - java.security.jgss - java.security.sasl - java.smartcardio - jdk.crypto.ec - jdk.crypto.mscapi - jdk.crypto.pkcs11 - jdk.crypto.ucrypto - jdk.jartool - jdk.policytool - jdk.security.auth - jdk.security.jgss - - - sun.security.x509 - java.naming - jdk.crypto.ec - jdk.crypto.pkcs11 - jdk.jartool - jdk.security.auth - - - sun.text - java.desktop - - - sun.util - java.desktop - - - sun.util.locale.provider - java.desktop - jdk.localedata - - - jdk.internal.logger - java.logging - - - sun.util.logging - java.desktop - java.logging - java.management - java.prefs - java.httpclient - - - sun.util.resources - jdk.localedata - - - - java.compact1 - java.base - java.logging - java.scripting - - - java.compact2 - java.compact1 - java.rmi - java.sql - java.xml - - - java.compact3 - java.compact2 - java.compiler - java.instrument - java.httpclient - java.management - java.naming - java.prefs - java.security.jgss - java.security.sasl - java.sql.rowset - java.xml.crypto - - - java.compiler - java.base - java.logging - - javax.annotation.processing - - - javax.lang.model - - - javax.lang.model.element - - - javax.lang.model.type - - - javax.lang.model.util - - - javax.tools - - - - java.corba - java.base - java.desktop - java.logging - java.naming - java.rmi - java.transaction - - javax.activity - - - javax.rmi - - - javax.rmi.CORBA - - - org.omg.CORBA - - - org.omg.CORBA.DynAnyPackage - - - org.omg.CORBA.ORBPackage - - - org.omg.CORBA.TypeCodePackage - - - org.omg.CORBA.portable - - - org.omg.CORBA_2_3 - - - org.omg.CORBA_2_3.portable - - - org.omg.CosNaming - - - org.omg.CosNaming.NamingContextExtPackage - - - org.omg.CosNaming.NamingContextPackage - - - org.omg.Dynamic - - - org.omg.DynamicAny - - - org.omg.DynamicAny.DynAnyFactoryPackage - - - org.omg.DynamicAny.DynAnyPackage - - - org.omg.IOP - - - org.omg.IOP.CodecFactoryPackage - - - org.omg.IOP.CodecPackage - - - org.omg.Messaging - - - org.omg.PortableInterceptor - - - org.omg.PortableInterceptor.ORBInitInfoPackage - - - org.omg.PortableServer - - - org.omg.PortableServer.CurrentPackage - - - org.omg.PortableServer.POAManagerPackage - - - org.omg.PortableServer.POAPackage - - - org.omg.PortableServer.ServantLocatorPackage - - - org.omg.PortableServer.portable - - - org.omg.SendingContext - - - org.omg.stub.java.rmi - - - com.sun.corba.se.impl.util - jdk.rmic - - - - java.datatransfer - java.base - - java.awt.datatransfer - - - sun.datatransfer - java.desktop - - - - java.desktop - java.base - java.logging - java.prefs - java.xml - java.datatransfer - - java.applet - - - java.awt - - - java.awt.color - - - java.awt.dnd - - - java.awt.event - - - java.awt.font - - - java.awt.geom - - - java.awt.im - - - java.awt.im.spi - - - java.awt.image - - - java.awt.image.renderable - - - java.awt.print - - - java.beans - - - java.beans.beancontext - - - javax.accessibility - - - javax.imageio - - - javax.imageio.event - - - javax.imageio.metadata - - - javax.imageio.plugins.bmp - - - javax.imageio.plugins.jpeg - - - javax.imageio.spi - - - javax.imageio.stream - - - javax.print - - - javax.print.attribute - - - javax.print.attribute.standard - - - javax.print.event - - - javax.sound.midi - - - javax.sound.midi.spi - - - javax.sound.sampled - - - javax.sound.sampled.spi - - - javax.swing - - - javax.swing.border - - - javax.swing.colorchooser - - - javax.swing.event - - - javax.swing.filechooser - - - javax.swing.plaf - - - javax.swing.plaf.basic - - - javax.swing.plaf.metal - - - javax.swing.plaf.multi - - - javax.swing.plaf.nimbus - - - javax.swing.plaf.synth - - - javax.swing.table - - - javax.swing.text - - - javax.swing.text.html - - - javax.swing.text.html.parser - - - javax.swing.text.rtf - - - javax.swing.tree - - - javax.swing.undo - - - sun.awt - jdk.accessibility - - - - jdk.jshell - java.base - java.compiler - jdk.compiler - java.desktop - java.prefs - jdk.jdi - jdk.internal.le - - jdk.jshell - - - - jdk.jsobject - java.base - java.desktop - - netscape.javascript - - - jdk.internal.netscape.javascript.spi - jdk.plugin - - - - java.httpclient - java.base - - java.net.http - - - - java.instrument - java.base - - java.lang.instrument - - - jdk.internal.instrumentation - jdk.jfr - jdk.management.resource - - - - java.logging - java.base - - java.util.logging - - - - java.management - java.base - java.logging - java.naming - java.rmi - - java.lang.management - - - javax.management - - - javax.management.loading - - - javax.management.modelmbean - - - javax.management.monitor - - - javax.management.openmbean - - - javax.management.relation - - - javax.management.remote - - - javax.management.remote.rmi - - - javax.management.timer - - - sun.management - jdk.jconsole - jdk.management - - - sun.management.spi - jdk.management - jdk.management.cmm - jdk.management.jfr - - - - java.naming - java.base - java.security.sasl - - javax.naming - - - javax.naming.directory - - - javax.naming.event - - - javax.naming.ldap - - - javax.naming.spi - - - com.sun.jndi.toolkit.ctx - jdk.naming.dns - - - com.sun.jndi.toolkit.url - java.corba - jdk.naming.dns - jdk.naming.rmi - - - - java.prefs - java.base - java.xml - - java.util.prefs - - - - java.rmi - java.base - java.logging - - java.rmi - - - java.rmi.activation - - - java.rmi.dgc - - - java.rmi.registry - - - java.rmi.server - - - javax.rmi.ssl - - - sun.rmi.registry - java.management - - - sun.rmi.server - java.management - jdk.jconsole - - - sun.rmi.transport - java.management - jdk.jconsole - - - - java.scripting - java.base - - javax.script - - - - java.se - java.activation - java.compact3 - java.corba - java.datatransfer - java.desktop - java.transaction - java.xml.bind - java.xml.ws - - - java.security.jgss - java.base - java.naming - - javax.security.auth.kerberos - - - org.ietf.jgss - - - sun.security.jgss.krb5 - jdk.security.auth - - - sun.security.krb5 - jdk.security.auth - - - sun.security.krb5.internal.ktab - jdk.security.auth - - - sun.security.jgss - jdk.security.jgss - - - sun.security.krb5.internal - jdk.security.jgss - - - - java.security.sasl - java.base - java.logging - - javax.security.sasl - - - com.sun.security.sasl.util - jdk.security.jgss - - - - java.smartcardio - java.base - - javax.smartcardio - - - - java.sql - java.base - java.logging - java.xml - - java.sql - - - javax.sql - - - javax.transaction.xa - - - - java.sql.rowset - java.base - java.logging - java.naming - java.sql - java.xml - - javax.sql.rowset - - - javax.sql.rowset.serial - - - javax.sql.rowset.spi - - - - java.transaction - java.base - java.rmi - - javax.transaction - - - - java.xml - java.base - - javax.xml - - - javax.xml.catalog - - - javax.xml.datatype - - - javax.xml.namespace - - - javax.xml.parsers - - - javax.xml.stream - - - javax.xml.stream.events - - - javax.xml.stream.util - - - javax.xml.transform - - - javax.xml.transform.dom - - - javax.xml.transform.sax - - - javax.xml.transform.stax - - - javax.xml.transform.stream - - - javax.xml.validation - - - javax.xml.xpath - - - org.w3c.dom - - - org.w3c.dom.bootstrap - - - org.w3c.dom.events - - - org.w3c.dom.ls - - - org.w3c.dom.ranges - - - org.w3c.dom.traversal - - - org.w3c.dom.views - - - org.xml.sax - - - org.xml.sax.ext - - - org.xml.sax.helpers - - - com.sun.org.apache.xerces.internal.dom - java.xml.ws - - - com.sun.org.apache.xerces.internal.jaxp - java.xml.ws - - - com.sun.org.apache.xerces.internal.util - java.xml.ws - - - com.sun.org.apache.xml.internal.dtm - java.xml.crypto - - - com.sun.org.apache.xml.internal.resolver - java.xml.ws - jdk.xml.bind - - - com.sun.org.apache.xml.internal.resolver.tools - java.xml.ws - jdk.xml.bind - - - com.sun.org.apache.xml.internal.utils - java.xml.crypto - - - com.sun.org.apache.xpath.internal - java.xml.crypto - - - com.sun.org.apache.xpath.internal.compiler - java.xml.crypto - - - com.sun.org.apache.xpath.internal.functions - java.xml.crypto - - - com.sun.org.apache.xpath.internal.objects - java.xml.crypto - - - com.sun.org.apache.xpath.internal.res - java.xml.crypto - - - - java.xml.bind - java.activation - java.base - java.compiler - java.datatransfer - java.desktop - java.logging - java.xml - - javax.xml.bind - - - javax.xml.bind.annotation - - - javax.xml.bind.annotation.adapters - - - javax.xml.bind.attachment - - - javax.xml.bind.helpers - - - javax.xml.bind.util - - - com.sun.istack.internal - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.istack.internal.localization - java.xml.ws - jdk.xml.ws - - - com.sun.istack.internal.logging - java.xml.ws - jdk.xml.ws - - - com.sun.xml.internal.bind - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.bind.annotation - jdk.xml.bind - - - com.sun.xml.internal.bind.api - java.xml.ws - jdk.xml.bind - - - com.sun.xml.internal.bind.api.impl - java.xml.ws - jdk.xml.bind - - - com.sun.xml.internal.bind.marshaller - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.bind.unmarshaller - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.bind.util - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.bind.v2 - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.bind.v2.model.annotation - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.bind.v2.model.core - jdk.xml.bind - - - com.sun.xml.internal.bind.v2.model.impl - jdk.xml.bind - - - com.sun.xml.internal.bind.v2.model.nav - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.bind.v2.model.runtime - java.xml.ws - - - com.sun.xml.internal.bind.v2.model.util - jdk.xml.bind - - - com.sun.xml.internal.bind.v2.runtime - java.xml.ws - jdk.xml.bind - - - com.sun.xml.internal.bind.v2.runtime.unmarshaller - java.xml.ws - - - com.sun.xml.internal.bind.v2.schemagen - java.xml.ws - jdk.xml.bind - - - com.sun.xml.internal.bind.v2.schemagen.episode - jdk.xml.bind - - - com.sun.xml.internal.bind.v2.schemagen.xmlschema - java.xml.ws - - - com.sun.xml.internal.bind.v2.util - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.fastinfoset.stax - java.xml.ws - - - com.sun.xml.internal.fastinfoset.vocab - java.xml.ws - - - com.sun.xml.internal.org.jvnet.fastinfoset - java.xml.ws - - - com.sun.xml.internal.org.jvnet.mimepull - java.xml.ws - - - com.sun.xml.internal.org.jvnet.staxex - java.xml.ws - - - com.sun.xml.internal.org.jvnet.staxex.util - java.xml.ws - - - com.sun.xml.internal.txw2 - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.txw2.annotation - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - com.sun.xml.internal.txw2.output - java.xml.ws - jdk.xml.bind - jdk.xml.ws - - - - java.xml.crypto - java.base - java.logging - java.xml - - javax.xml.crypto - - - javax.xml.crypto.dom - - - javax.xml.crypto.dsig - - - javax.xml.crypto.dsig.dom - - - javax.xml.crypto.dsig.keyinfo - - - javax.xml.crypto.dsig.spec - - - - java.xml.ws - java.annotations.common - java.base - java.datatransfer - java.desktop - java.logging - java.management - java.rmi - java.activation - java.xml - java.xml.bind - jdk.httpserver - - javax.jws - - - javax.jws.soap - - - javax.xml.soap - - - javax.xml.ws - - - javax.xml.ws.handler - - - javax.xml.ws.handler.soap - - - javax.xml.ws.http - - - javax.xml.ws.soap - - - javax.xml.ws.spi - - - javax.xml.ws.spi.http - - - javax.xml.ws.wsaddressing - - - com.oracle.webservices.internal.api.databinding - jdk.xml.ws - - - com.sun.xml.internal.ws.addressing - jdk.xml.ws - - - com.sun.xml.internal.ws.addressing.v200408 - jdk.xml.ws - - - com.sun.xml.internal.ws.api - jdk.xml.ws - - - com.sun.xml.internal.ws.api.addressing - jdk.xml.ws - - - com.sun.xml.internal.ws.api.databinding - jdk.xml.ws - - - com.sun.xml.internal.ws.api.model - jdk.xml.ws - - - com.sun.xml.internal.ws.api.server - jdk.xml.ws - - - com.sun.xml.internal.ws.api.streaming - jdk.xml.ws - - - com.sun.xml.internal.ws.api.wsdl.parser - jdk.xml.ws - - - com.sun.xml.internal.ws.api.wsdl.writer - jdk.xml.ws - - - com.sun.xml.internal.ws.binding - jdk.xml.ws - - - com.sun.xml.internal.ws.db - jdk.xml.ws - - - com.sun.xml.internal.ws.model - jdk.xml.ws - - - com.sun.xml.internal.ws.policy.sourcemodel.wspolicy - jdk.xml.ws - - - com.sun.xml.internal.ws.spi.db - jdk.xml.ws - - - com.sun.xml.internal.ws.streaming - jdk.xml.ws - - - com.sun.xml.internal.ws.util - jdk.xml.ws - - - com.sun.xml.internal.ws.util.exception - jdk.xml.ws - - - com.sun.xml.internal.ws.util.xml - jdk.xml.ws - - - com.sun.xml.internal.ws.wsdl.parser - jdk.xml.ws - - - com.sun.xml.internal.ws.wsdl.writer - jdk.xml.ws - - - - jdk.attach - java.base - jdk.jvmstat - - com.sun.tools.attach - - - com.sun.tools.attach.spi - - - sun.tools.attach - jdk.jcmd - - - - jdk.charsets - java.base - - - jdk.compiler - java.base - java.compiler - - com.sun.source.doctree - - - com.sun.source.tree - - - com.sun.source.util - - - com.sun.tools.javac - - - com.sun.tools.doclint - jdk.javadoc - - - com.sun.tools.javac.api - jdk.jshell - jdk.javadoc - - - com.sun.tools.javac.code - jdk.jshell - jdk.javadoc - - - com.sun.tools.javac.comp - jdk.jshell - jdk.javadoc - - - com.sun.tools.javac.file - jdk.javadoc - jdk.jdeps - - - com.sun.tools.javac.jvm - jdk.jshell - jdk.javadoc - - - com.sun.tools.javac.main - jdk.jshell - jdk.javadoc - - - com.sun.tools.javac.nio - jdk.javadoc - - - com.sun.tools.javac.parser - jdk.jshell - - - com.sun.tools.javac.platform - jdk.javadoc - - - com.sun.tools.javac.sym - jdk.javadoc - - - com.sun.tools.javac.tree - jdk.jshell - jdk.javadoc - - - com.sun.tools.javac.util - jdk.jshell - jdk.javadoc - jdk.jdeps - - - com.sun.tools.javac.model - jdk.javadoc - - - - jdk.crypto.ec - java.base - - sun.security.ec - jdk.crypto.pkcs11 - - - - jdk.crypto.mscapi - java.base - - - jdk.crypto.pkcs11 - java.base - jdk.crypto.ec - - - jdk.crypto.ucrypto - java.base - - - jdk.dev - java.base - - - jdk.dynalink - java.base - java.logging - - jdk.dynalink - - - jdk.dynalink.beans - - - jdk.dynalink.linker - - - jdk.dynalink.linker.support - - - jdk.dynalink.support - - - - jdk.hotspot.agent - java.base - java.datatransfer - java.desktop - java.rmi - java.scripting - jdk.jdi - - - jdk.httpserver - java.base - java.logging - - com.sun.net.httpserver - - - com.sun.net.httpserver.spi - - - - jdk.internal.le - java.base - - jdk.internal.jline - jdk.jshell - jdk.scripting.nashorn.shell - - - jdk.internal.jline.console - jdk.jshell - jdk.scripting.nashorn.shell - - - jdk.internal.jline.console.completer - jdk.jshell - jdk.scripting.nashorn.shell - - - jdk.internal.jline.console.history - jdk.jshell - jdk.scripting.nashorn.shell - - - jdk.internal.jline.internal - jdk.jshell - jdk.scripting.nashorn.shell - - - - jdk.internal.opt - java.base - - - jdk.jartool - java.base - - com.sun.jarsigner - - - jdk.security.jarsigner - - - - jdk.javadoc - java.base - java.compiler - java.xml - jdk.compiler - - com.sun.javadoc - - - com.sun.tools.doclets - - - com.sun.tools.javadoc - - - jdk.javadoc.doclet - - - jdk.javadoc.doclet.taglet - - - - jdk.jcmd - java.base - jdk.attach - jdk.jvmstat - - - jdk.jconsole - java.base - java.desktop - java.logging - java.management - java.rmi - jdk.attach - jdk.jvmstat - jdk.management - - com.sun.tools.jconsole - - - - jdk.jdeps - java.base - java.compiler - java.xml - jdk.compiler - - - jdk.jdi - java.base - jdk.attach - - com.sun.jdi - - - com.sun.jdi.connect - - - com.sun.jdi.connect.spi - - - com.sun.jdi.event - - - com.sun.jdi.request - - - com.sun.tools.jdi - jdk.hotspot.agent - - - - jdk.jdwp.agent - java.base - - - jdk.jvmstat - java.base - - sun.jvmstat.monitor - jdk.attach - jdk.jcmd - jdk.jconsole - jdk.jvmstat.rmi - - - sun.jvmstat.monitor.event - jdk.jcmd - jdk.jvmstat.rmi - - - sun.jvmstat.perfdata.monitor - jdk.jvmstat.rmi - - - - jdk.jvmstat.rmi - java.base - java.rmi - jdk.jvmstat - - sun.jvmstat.monitor.remote - java.rmi - - - - jdk.localedata - java.base - - - jdk.management - java.base - java.management - - com.sun.management - - - - jdk.naming.dns - java.base - java.naming - - - jdk.naming.rmi - java.base - java.naming - java.rmi - - - jdk.pack200 - java.base - - - jdk.policytool - java.base - java.desktop - java.logging - java.management - java.security.jgss - java.sql - jdk.security.jgss - - - jdk.rmic - java.base - java.corba - jdk.compiler - jdk.javadoc - - - jdk.scripting.nashorn - java.base - java.logging - jdk.dynalink - java.scripting - - jdk.nashorn.internal.runtime - jdk.scripting.nashorn.shell - - - jdk.nashorn.internal.objects - jdk.scripting.nashorn.shell - - - jdk.nashorn.tools - jdk.scripting.nashorn.shell - - - jdk.nashorn.api.scripting - - - jdk.nashorn.api.tree - - - - jdk.scripting.nashorn.shell - java.base - jdk.scripting.nashorn - jdk.internal.le - java.desktop - java.compiler - - - jdk.sctp - java.base - - com.sun.nio.sctp - - - - jdk.security.auth - java.base - java.naming - java.security.jgss - - com.sun.security.auth - - - com.sun.security.auth.callback - - - com.sun.security.auth.login - - - com.sun.security.auth.module - - - - jdk.security.jgss - java.base - java.logging - java.security.jgss - java.security.sasl - - com.sun.security.jgss - - - - jdk.vm.ci - java.base - - jdk.vm.ci.hotspot - jdk.jfr - - - jdk.vm.ci.hotspot.events - jdk.jfr - - - - jdk.xml.bind - java.activation - java.base - java.compiler - java.datatransfer - java.desktop - java.logging - java.xml - java.xml.bind - jdk.compiler - - com.sun.codemodel.internal - jdk.xml.ws - - - com.sun.codemodel.internal.writer - jdk.xml.ws - - - com.sun.istack.internal.tools - jdk.xml.ws - - - com.sun.tools.internal.jxc.ap - jdk.xml.ws - - - com.sun.tools.internal.jxc.model.nav - jdk.xml.ws - - - com.sun.tools.internal.xjc - jdk.xml.ws - - - com.sun.tools.internal.xjc.api - jdk.xml.ws - - - com.sun.tools.internal.xjc.api.util - jdk.xml.ws - - - com.sun.tools.internal.xjc.reader - jdk.xml.ws - - - com.sun.tools.internal.xjc.reader.internalizer - jdk.xml.ws - - - com.sun.tools.internal.xjc.util - jdk.xml.ws - - - com.sun.xml.internal.xsom.parser - jdk.xml.ws - - - - jdk.xml.dom - java.base - java.xml - - org.w3c.dom.css - - - org.w3c.dom.html - - - org.w3c.dom.stylesheets - - - org.w3c.dom.xpath - - - - jdk.xml.ws - java.base - java.compiler - java.logging - java.rmi - java.xml - java.xml.bind - java.xml.ws - jdk.xml.bind - - - jdk.zipfs - java.base - - - jdk.accessibility - java.base - java.desktop - - diff --git a/test/lib/sun/hotspot/WhiteBox.java b/test/lib/sun/hotspot/WhiteBox.java index 668d2f00c1a..4eb5dfa4af3 100644 --- a/test/lib/sun/hotspot/WhiteBox.java +++ b/test/lib/sun/hotspot/WhiteBox.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 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 @@ -32,7 +32,6 @@ import java.util.function.BiFunction; import java.util.function.Function; import java.security.BasicPermission; import java.util.Objects; -import jdk.internal.HotSpotIntrinsicCandidate; import sun.hotspot.parser.DiagnosticCommand; @@ -211,8 +210,6 @@ public class WhiteBox { public native int deoptimizeFrames(boolean makeNotEntrant); public native void deoptimizeAll(); - @HotSpotIntrinsicCandidate - public void deoptimize() {} public boolean isMethodCompiled(Executable method) { return isMethodCompiled(method, false /*not osr*/); } @@ -422,6 +419,19 @@ public class WhiteBox { .findAny() .orElse(null); } + + // Jigsaw + public native void DefineModule(Object module, String version, String location, + Object[] packages); + public native void AddModuleExports(Object from_module, String pkg, Object to_module); + public native void AddReadsModule(Object from_module, Object source_module); + public native boolean CanReadModule(Object asking_module, Object source_module); + public native boolean IsExportedToModule(Object from_module, String pkg, Object to_module); + public native void AddModulePackage(Object module, String pkg); + public native void AddModuleExportsToAllUnnamed(Object module, String pkg); + public native void AddModuleExportsToAll(Object module, String pkg); + public native Object GetModuleByPackageName(Object ldr, String pkg); + public native int getOffsetForName0(String name); public int getOffsetForName(String name) throws Exception { int offset = getOffsetForName0(name); @@ -452,7 +462,7 @@ public class WhiteBox { public native void assertMatchingSafepointCalls(boolean mutexSafepointValue, boolean attemptedNoSafepointValue); // Sharing - public native boolean isSharedClass(Class c); public native boolean isShared(Object o); + public native boolean isSharedClass(Class c); public native boolean areSharedStringsIgnored(); } From f30fc1c88bf2e8233f64e77886dc49cd94257aef Mon Sep 17 00:00:00 2001 From: Alan Bateman Date: Thu, 17 Mar 2016 19:04:01 +0000 Subject: [PATCH 4/4] 8142968: Module System implementation Initial integration of JEP 200, JEP 260, JEP 261, and JEP 282 Co-authored-by: Alex Buckley Co-authored-by: Jonathan Gibbons Co-authored-by: Karen Kinnear Co-authored-by: Mandy Chung Co-authored-by: Mark Reinhold Co-authored-by: Harold Seigel Co-authored-by: Lois Foltan Co-authored-by: Calvin Cheung Co-authored-by: Christian Tornqvist Co-authored-by: Erik Joelsson Co-authored-by: George Triantafillou Co-authored-by: Igor Ignatyev Co-authored-by: Ioi Lam Co-authored-by: James Laskey Co-authored-by: Jean-Francois Denise Co-authored-by: Jiangli Zhou Co-authored-by: Markus Gronlund Co-authored-by: Serguei Spitsyn Co-authored-by: Staffan Larsen Co-authored-by: Sundararajan Athijegannathan Reviewed-by: acorn, ccheung, coleenp, ctornqvi, dholmes, dsimms, gtriantafill, iklam, jiangli, mgronlun, mseledtsov, cjplummer, sspitsyn, stefank, twisti, hseigel, lfoltan, alanb, mchung, dfazunen --- hotspot/make/share/makefiles/mapfile-vers | 12 + hotspot/make/test/JtregNative.gmk | 1 + .../share/classes/module-info.java | 49 + .../jvm/hotspot/tools/ClassLoaderStats.java | 14 +- .../sun/jvm/hotspot/tools/FinalizerInfo.java | 19 +- .../sun/jvm/hotspot/tools/HeapDumper.java | 25 +- .../sun/jvm/hotspot/tools/HeapSummary.java | 13 +- .../classes/sun/jvm/hotspot/tools/JInfo.java | 21 +- .../classes/sun/jvm/hotspot/tools/JStack.java | 16 +- .../jvm/hotspot/tools/ObjectHistogram.java | 14 +- .../classes/sun/jvm/hotspot/tools/PMap.java | 14 +- ...k.vm.ci.hotspot.HotSpotJVMCIBackendFactory | 3 - .../jdk.vm.ci/share/classes/module-info.java | 37 + hotspot/src/os/posix/dtrace/hotspot_jni.d | 2 + hotspot/src/share/vm/c1/c1_Runtime1.cpp | 2 +- hotspot/src/share/vm/ci/ciEnv.cpp | 6 +- .../share/vm/classfile/classFileParser.cpp | 214 ++-- .../share/vm/classfile/classFileParser.hpp | 7 +- .../src/share/vm/classfile/classLoader.cpp | 655 +++++++----- .../src/share/vm/classfile/classLoader.hpp | 119 ++- .../share/vm/classfile/classLoaderData.cpp | 147 ++- .../share/vm/classfile/classLoaderData.hpp | 24 +- .../src/share/vm/classfile/classLoaderExt.hpp | 24 +- .../src/share/vm/classfile/javaClasses.cpp | 183 +++- .../src/share/vm/classfile/javaClasses.hpp | 72 +- .../share/vm/classfile/javaClasses.inline.hpp | 4 + hotspot/src/share/vm/classfile/jimage.hpp | 8 +- .../src/share/vm/classfile/klassFactory.cpp | 8 +- .../src/share/vm/classfile/moduleEntry.cpp | 405 ++++++++ .../src/share/vm/classfile/moduleEntry.hpp | 230 +++++ hotspot/src/share/vm/classfile/modules.cpp | 964 ++++++++++++++++++ hotspot/src/share/vm/classfile/modules.hpp | 151 +++ .../src/share/vm/classfile/packageEntry.cpp | 322 ++++++ .../src/share/vm/classfile/packageEntry.hpp | 202 ++++ .../vm/classfile/sharedPathsMiscInfo.cpp | 4 +- .../share/vm/classfile/systemDictionary.cpp | 174 +++- .../share/vm/classfile/systemDictionary.hpp | 11 +- .../vm/classfile/systemDictionaryShared.hpp | 10 + hotspot/src/share/vm/classfile/vmSymbols.hpp | 48 +- .../vm/interpreter/bytecodeInterpreter.cpp | 4 +- .../vm/interpreter/interpreterRuntime.cpp | 2 +- .../src/share/vm/interpreter/linkResolver.cpp | 29 +- hotspot/src/share/vm/jvmci/jvmciEnv.cpp | 4 +- hotspot/src/share/vm/logging/logTag.hpp | 1 + hotspot/src/share/vm/memory/filemap.cpp | 28 +- hotspot/src/share/vm/memory/filemap.hpp | 12 +- .../src/share/vm/memory/metaspaceShared.cpp | 47 +- .../src/share/vm/memory/metaspaceShared.hpp | 1 + hotspot/src/share/vm/memory/universe.cpp | 6 +- hotspot/src/share/vm/memory/universe.hpp | 3 + hotspot/src/share/vm/oops/arrayKlass.cpp | 11 +- hotspot/src/share/vm/oops/arrayKlass.hpp | 2 +- hotspot/src/share/vm/oops/instanceKlass.cpp | 186 +++- hotspot/src/share/vm/oops/instanceKlass.hpp | 71 +- hotspot/src/share/vm/oops/klass.cpp | 16 +- hotspot/src/share/vm/oops/method.cpp | 2 +- hotspot/src/share/vm/oops/objArrayKlass.cpp | 12 +- hotspot/src/share/vm/oops/typeArrayKlass.cpp | 2 +- hotspot/src/share/vm/opto/library_call.cpp | 14 - .../src/share/vm/precompiled/precompiled.hpp | 3 + hotspot/src/share/vm/prims/jni.cpp | 51 +- hotspot/src/share/vm/prims/jni.h | 25 + hotspot/src/share/vm/prims/jniCheck.cpp | 51 +- hotspot/src/share/vm/prims/jvm.cpp | 58 +- hotspot/src/share/vm/prims/jvm.h | 36 + hotspot/src/share/vm/prims/jvmti.xml | 127 ++- hotspot/src/share/vm/prims/jvmti.xsl | 4 +- hotspot/src/share/vm/prims/jvmtiEnter.xsl | 4 +- hotspot/src/share/vm/prims/jvmtiEnv.cpp | 18 +- hotspot/src/share/vm/prims/jvmtiEnvBase.cpp | 45 +- hotspot/src/share/vm/prims/jvmtiEnvBase.hpp | 25 +- .../share/vm/prims/jvmtiEventController.cpp | 7 +- hotspot/src/share/vm/prims/jvmtiExport.cpp | 148 ++- hotspot/src/share/vm/prims/jvmtiExport.hpp | 8 +- hotspot/src/share/vm/prims/jvmtiH.xsl | 20 +- hotspot/src/share/vm/prims/jvmtiLib.xsl | 10 +- .../vm/prims/jvmtiManageCapabilities.cpp | 8 +- hotspot/src/share/vm/prims/methodHandles.cpp | 6 +- hotspot/src/share/vm/prims/nativeLookup.cpp | 7 +- hotspot/src/share/vm/prims/whitebox.cpp | 69 +- hotspot/src/share/vm/runtime/arguments.cpp | 263 +++-- hotspot/src/share/vm/runtime/arguments.hpp | 158 ++- hotspot/src/share/vm/runtime/frame.cpp | 14 + hotspot/src/share/vm/runtime/jniHandles.cpp | 8 + hotspot/src/share/vm/runtime/jniHandles.hpp | 3 + hotspot/src/share/vm/runtime/mutexLocker.cpp | 4 +- hotspot/src/share/vm/runtime/mutexLocker.hpp | 2 +- hotspot/src/share/vm/runtime/os.cpp | 80 +- hotspot/src/share/vm/runtime/os.hpp | 3 +- hotspot/src/share/vm/runtime/reflection.cpp | 184 +++- hotspot/src/share/vm/runtime/reflection.hpp | 19 +- .../src/share/vm/runtime/sharedRuntime.cpp | 46 +- .../src/share/vm/runtime/sharedRuntime.hpp | 14 +- hotspot/src/share/vm/runtime/statSampler.cpp | 3 +- hotspot/src/share/vm/runtime/thread.cpp | 122 ++- hotspot/src/share/vm/services/jmm.h | 1 - hotspot/src/share/vm/services/management.cpp | 1 - hotspot/src/share/vm/trace/traceBackend.cpp | 28 + hotspot/src/share/vm/trace/traceDataTypes.hpp | 4 + hotspot/src/share/vm/trace/traceMacros.hpp | 4 + hotspot/src/share/vm/trace/tracetypes.xml | 22 + .../share/vm/utilities/dtrace_disabled.hpp | 6 + hotspot/src/share/vm/utilities/ostream.cpp | 2 + hotspot/src/share/vm/utilities/utf8.cpp | 62 ++ hotspot/src/share/vm/utilities/utf8.hpp | 3 + hotspot/test/TEST.ROOT | 4 +- .../TestMonomorphicObjectCall.java | 9 +- .../jsr292/CallSiteDepContextTest.java | 34 +- .../compiler/jsr292/NonInlinedCall/Agent.java | 27 +- .../jsr292/NonInlinedCall/GCTest.java | 49 +- .../jsr292/NonInlinedCall/InvokeTest.java | 41 +- .../jsr292/NonInlinedCall/RedefineTest.java | 43 +- .../java/lang/invoke/MethodHandleHelper.java | 92 ++ .../jvmci/SecurityRestrictionsTest.java | 15 +- .../compiler/jvmci/code/DataPatchTest.java | 9 +- .../code/SimpleCodeInstallationTest.java | 9 +- .../jvmci/code/SimpleDebugInfoTest.java | 9 +- .../code/VirtualObjectDebugInfoTest.java | 9 +- .../compiler/jvmci/common/CTVMUtilities.java | 27 +- .../vm/ci/hotspot}/CompilerToVMHelper.java | 0 .../jdk/vm/ci/hotspot}/MetaAccessWrapper.java | 0 .../PublicMetaspaceWrapperObject.java | 0 .../compilerToVM/AllocateCompileIdTest.java | 18 +- .../compilerToVM/CanInlineMethodTest.java | 16 +- .../compilerToVM/CollectCountersTest.java | 22 +- .../jvmci/compilerToVM/DebugOutputTest.java | 17 +- .../compilerToVM/DisassembleCodeBlobTest.java | 15 +- .../DoNotInlineOrCompileTest.java | 16 +- .../ExecuteInstalledCodeTest.java | 17 +- .../FindUniqueConcreteMethodTest.java | 13 +- .../jvmci/compilerToVM/GetBytecodeTest.java | 12 +- .../compilerToVM/GetClassInitializerTest.java | 9 +- .../compilerToVM/GetConstantPoolTest.java | 17 +- .../compilerToVM/GetExceptionTableTest.java | 12 +- .../compilerToVM/GetImplementorTest.java | 10 +- .../compilerToVM/GetLineNumberTableTest.java | 13 +- .../GetLocalVariableTableTest.java | 12 +- .../GetMaxCallTargetOffsetTest.java | 10 +- .../compilerToVM/GetNextStackFrameTest.java | 15 +- .../GetResolvedJavaMethodAtSlotTest.java | 13 +- .../GetResolvedJavaMethodTest.java | 15 +- .../compilerToVM/GetResolvedJavaTypeTest.java | 28 +- .../GetStackTraceElementTest.java | 11 +- .../jvmci/compilerToVM/GetSymbolTest.java | 17 +- .../GetVtableIndexForInterfaceTest.java | 13 +- .../HasCompiledCodeForOSRTest.java | 18 +- .../HasFinalizableSubclassTest.java | 9 +- .../InitializeConfigurationTest.java | 11 +- .../InvalidateInstalledCodeTest.java | 19 +- .../jvmci/compilerToVM/IsMatureTest.java | 15 +- .../compilerToVM/LookupKlassInPoolTest.java | 16 +- .../LookupKlassRefIndexInPoolTest.java | 9 +- .../compilerToVM/LookupMethodInPoolTest.java | 9 +- .../LookupNameAndTypeRefIndexInPoolTest.java | 9 +- .../compilerToVM/LookupNameInPoolTest.java | 9 +- .../LookupSignatureInPoolTest.java | 9 +- .../jvmci/compilerToVM/LookupTypeTest.java | 10 +- .../MaterializeVirtualObjectTest.java | 38 +- ...ethodIsIgnoredBySecurityStackWalkTest.java | 12 +- .../compilerToVM/ReadUncompressedOopTest.java | 22 +- .../jvmci/compilerToVM/ReprofileTest.java | 19 +- .../ResolveConstantInPoolTest.java | 13 +- .../compilerToVM/ResolveFieldInPoolTest.java | 9 +- .../jvmci/compilerToVM/ResolveMethodTest.java | 13 +- ...solvePossiblyCachedConstantInPoolTest.java | 9 +- .../compilerToVM/ResolveTypeInPoolTest.java | 15 +- .../ShouldDebugNonSafepointsTest.java | 28 +- .../compilerToVM/ShouldInlineMethodTest.java | 16 +- .../JvmciCreateMetaAccessContextTest.java | 8 +- .../events/JvmciNotifyInstallEventTest.java | 11 +- .../jdk/vm/ci/runtime/test/ConstantTest.java | 7 +- .../vm/ci/runtime/test/RedefineClassTest.java | 5 +- .../test/TestConstantReflectionProvider.java | 5 +- .../jdk/vm/ci/runtime/test/TestJavaField.java | 5 +- .../vm/ci/runtime/test/TestJavaMethod.java | 5 +- .../jdk/vm/ci/runtime/test/TestJavaType.java | 5 +- .../runtime/test/TestMetaAccessProvider.java | 5 +- .../runtime/test/TestResolvedJavaField.java | 5 +- .../runtime/test/TestResolvedJavaMethod.java | 5 +- .../ci/runtime/test/TestResolvedJavaType.java | 6 +- .../compiler/stable/StableConfiguration.java | 14 +- .../compiler/stable/TestStableBoolean.java | 131 +-- .../test/compiler/stable/TestStableByte.java | 149 ++- .../test/compiler/stable/TestStableChar.java | 149 ++- .../compiler/stable/TestStableDouble.java | 149 ++- .../test/compiler/stable/TestStableFloat.java | 149 ++- .../test/compiler/stable/TestStableInt.java | 149 ++- .../test/compiler/stable/TestStableLong.java | 149 ++- .../stable/TestStableMemoryBarrier.java | 60 +- .../compiler/stable/TestStableObject.java | 150 ++- .../test/compiler/stable/TestStableShort.java | 149 ++- .../unsafe/UnsafeGetConstantField.java | 161 ++- hotspot/test/gc/TestSmallHeap.java | 1 + .../lib => gc/metaspace}/PerfCounter.java | 2 - .../lib => gc/metaspace}/PerfCounters.java | 3 +- .../BadObjectClass/BootstrapRedefine.java | 16 +- .../BootClassPathAppend.java | 46 + .../BootClassPathAppendProp.java | 48 + .../BootClassAppendProp/SunBootClassPath.java | 39 + .../TestOptionsWithRanges.java | 5 +- .../TestOptionsWithRangesDynamic.java | 3 +- .../SharedArchiveFile/BasicJarBuilder.java | 14 +- .../SharedArchiveFile/BootAppendTests.java | 252 +++++ .../runtime/SharedArchiveFile/LoadClass.java | 40 + .../PrintSharedArchiveAndExit.java | 18 - .../SharedArchiveFile/SharedStrings.java | 5 +- .../javax/sound/sampled/MyClass.jasm | 46 + .../nonjdk/myPackage/MyClass.java | 30 + .../org/omg/CORBA/Context.jasm | 46 + .../runtime/getSysPackage/GetSysPkgTest.java | 160 +++ hotspot/test/runtime/logging/ModulesTest.java | 53 + .../modules/AccModuleTest.java} | 34 +- .../modules/AccessCheck/CheckRead.java | 138 +++ .../modules/AccessCheck/DiffCL_CheckRead.java | 138 +++ .../AccessCheck/DiffCL_ExpQualOther.java | 140 +++ .../AccessCheck/DiffCL_ExpQualToM1.java | 120 +++ .../modules/AccessCheck/DiffCL_ExpUnqual.java | 121 +++ .../modules/AccessCheck/DiffCL_PkgNotExp.java | 125 +++ .../modules/AccessCheck/DiffCL_Umod.java | 230 +++++ .../modules/AccessCheck/DiffCL_UmodUpkg.java | 173 ++++ .../modules/AccessCheck/ExpQualOther.java | 140 +++ .../modules/AccessCheck/ExpQualToM1.java | 110 ++ .../modules/AccessCheck/ExpUnqual.java | 109 ++ .../modules/AccessCheck/ExportAllUnnamed.java | 128 +++ .../modules/AccessCheck/ModuleLibrary.java | 82 ++ .../modules/AccessCheck/PkgNotExp.java | 123 +++ .../runtime/modules/AccessCheck/Umod.java | 221 ++++ .../AccessCheck/UmodDiffCL_ExpQualOther.java | 127 +++ .../AccessCheck/UmodDiffCL_ExpUnqual.java | 122 +++ .../AccessCheck/UmodDiffCL_PkgNotExp.java | 126 +++ .../modules/AccessCheck/UmodDiffCL_Umod.java | 56 + .../AccessCheck/UmodDiffCL_UmodUpkg.java | 58 ++ .../runtime/modules/AccessCheck/UmodUPkg.java | 167 +++ .../UmodUpkgDiffCL_ExpQualOther.java | 125 +++ .../AccessCheck/UmodUpkgDiffCL_NotExp.java | 127 +++ .../AccessCheck/UmodUpkgDiffCL_Umod.java | 57 ++ .../AccessCheck/UmodUpkg_ExpQualOther.java | 136 +++ .../modules/AccessCheck/UmodUpkg_NotExp.java | 122 +++ .../modules/AccessCheck/UmodUpkg_Umod.java | 57 ++ .../AccessCheck/Umod_ExpQualOther.java | 136 +++ .../modules/AccessCheck/Umod_ExpUnqual.java | 121 +++ .../modules/AccessCheck/Umod_PkgNotExp.java | 122 +++ .../modules/AccessCheck/Umod_UmodUpkg.java | 58 ++ .../test/runtime/modules/AccessCheck/c4.java | 28 + .../AccessCheck/c5.java} | 20 +- .../myloaders/MyDiffClassLoader.java | 72 ++ .../myloaders/MySameClassLoader.java | 68 ++ .../runtime/modules/AccessCheck/p1/c1.java | 36 + .../modules/AccessCheck/p1/c1Loose.java | 33 + .../modules/AccessCheck/p1/c1ReadEdge.java | 41 + .../AccessCheck/p1/c1ReadEdgeDiffLoader.java | 61 ++ .../runtime/modules/AccessCheck/p2/c2.java | 30 + .../runtime/modules/AccessCheck/p3/c3.jcod | 108 ++ .../modules/AccessCheck/p3/c3ReadEdge.jcod | 142 +++ .../AccessCheck/p3/c3ReadEdgeDiffLoader.jcod | 175 ++++ .../runtime/modules/AccessCheck/p6/c6.java | 30 + .../modules/AccessCheckAllUnnamed.java | 117 +++ .../test/runtime/modules/AccessCheckExp.java | 86 ++ .../runtime/modules/AccessCheckJavaBase.java | 58 ++ .../test/runtime/modules/AccessCheckRead.java | 86 ++ .../runtime/modules/AccessCheckSuper.java | 71 ++ .../runtime/modules/AccessCheckUnnamed.java | 75 ++ .../runtime/modules/AccessCheckWorks.java | 79 ++ .../test/runtime/modules/CCE_module_msg.java | 58 ++ hotspot/test/runtime/modules/ExportTwice.java | 92 ++ .../JVMAddModuleExportToAllUnnamed.java | 76 ++ .../runtime/modules/JVMAddModuleExports.java | 145 +++ .../modules/JVMAddModuleExportsToAll.java | 122 +++ .../runtime/modules/JVMAddModulePackage.java | 129 +++ .../runtime/modules/JVMAddReadsModule.java | 83 ++ .../runtime/modules/JVMCanReadModule.java | 96 ++ .../test/runtime/modules/JVMDefineModule.java | 262 +++++ .../modules/JVMGetModuleByPkgName.java | 90 ++ .../modules/JVMIsExportedToModule.java | 123 +++ .../modules/LoadUnloadModuleStress.java | 70 ++ .../test/runtime/modules/ModuleHelper.java | 104 ++ .../Visibility/XbootcpNoVisibility.java | 77 ++ .../modules/Visibility/XbootcpVisibility.java | 109 ++ .../modules/Visibility/XpatchVisibility.java | 91 ++ .../runtime/modules/Xpatch/Xpatch2Dirs.java | 68 ++ .../modules/Xpatch/Xpatch2DirsMain.java | 31 + .../runtime/modules/Xpatch/XpatchMain.java | 30 + .../runtime/modules/Xpatch/XpatchTest.java | 56 + .../runtime/modules/Xpatch/XpatchTraceCL.java | 83 ++ hotspot/test/runtime/modules/XpatchCDS.java | 62 ++ hotspot/test/runtime/modules/acc_module.jcod | 137 +++ .../modules/getModuleJNI/GetModule.java | 167 +++ .../modules/getModuleJNI/libGetModule.c | 42 + .../java/lang/reflect/ModuleHelper.java | 69 ++ hotspot/test/runtime/modules/p1/c1.java | 35 + hotspot/test/runtime/modules/p2/c2.java | 29 + hotspot/test/runtime/modules/p3/c3.java | 29 + .../sa/jmap-hprof/JMapHProfLargeHeapTest.java | 2 +- .../test/testlibrary/ClassFileInstaller.java | 58 +- .../jdk/test/lib/InMemoryJavaCompiler.java | 9 +- .../test/lib/cli/CommandLineOptionTest.java | 1 + .../whitebox/sun/hotspot/WhiteBox.java | 264 +++++ 297 files changed, 15848 insertions(+), 2302 deletions(-) create mode 100644 hotspot/src/jdk.hotspot.agent/share/classes/module-info.java delete mode 100644 hotspot/src/jdk.vm.ci/share/classes/META-INF/services/jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory create mode 100644 hotspot/src/jdk.vm.ci/share/classes/module-info.java create mode 100644 hotspot/src/share/vm/classfile/moduleEntry.cpp create mode 100644 hotspot/src/share/vm/classfile/moduleEntry.hpp create mode 100644 hotspot/src/share/vm/classfile/modules.cpp create mode 100644 hotspot/src/share/vm/classfile/modules.hpp create mode 100644 hotspot/src/share/vm/classfile/packageEntry.cpp create mode 100644 hotspot/src/share/vm/classfile/packageEntry.hpp create mode 100644 hotspot/src/share/vm/trace/traceBackend.cpp create mode 100644 hotspot/test/compiler/jsr292/patches/java.base/java/lang/invoke/MethodHandleHelper.java rename hotspot/test/compiler/jvmci/common/{ => patches/jdk.vm.ci/jdk/vm/ci/hotspot}/CompilerToVMHelper.java (100%) rename hotspot/test/compiler/jvmci/{events => common/patches/jdk.vm.ci/jdk/vm/ci/hotspot}/MetaAccessWrapper.java (100%) rename hotspot/test/compiler/jvmci/common/{ => patches/jdk.vm.ci/jdk/vm/ci/hotspot}/PublicMetaspaceWrapperObject.java (100%) rename hotspot/test/{testlibrary/jdk/test/lib => gc/metaspace}/PerfCounter.java (98%) rename hotspot/test/{testlibrary/jdk/test/lib => gc/metaspace}/PerfCounters.java (98%) create mode 100644 hotspot/test/runtime/BootClassAppendProp/BootClassPathAppend.java create mode 100644 hotspot/test/runtime/BootClassAppendProp/BootClassPathAppendProp.java create mode 100644 hotspot/test/runtime/BootClassAppendProp/SunBootClassPath.java create mode 100644 hotspot/test/runtime/SharedArchiveFile/BootAppendTests.java create mode 100644 hotspot/test/runtime/SharedArchiveFile/LoadClass.java create mode 100644 hotspot/test/runtime/SharedArchiveFile/javax/sound/sampled/MyClass.jasm create mode 100644 hotspot/test/runtime/SharedArchiveFile/nonjdk/myPackage/MyClass.java create mode 100644 hotspot/test/runtime/SharedArchiveFile/org/omg/CORBA/Context.jasm create mode 100644 hotspot/test/runtime/getSysPackage/GetSysPkgTest.java create mode 100644 hotspot/test/runtime/logging/ModulesTest.java rename hotspot/test/{compiler/jsr292/NonInlinedCall/NonInlinedReinvoker.java => runtime/modules/AccModuleTest.java} (56%) create mode 100644 hotspot/test/runtime/modules/AccessCheck/CheckRead.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/DiffCL_CheckRead.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpQualOther.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpQualToM1.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpUnqual.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/DiffCL_PkgNotExp.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/DiffCL_Umod.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/DiffCL_UmodUpkg.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/ExpQualOther.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/ExpQualToM1.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/ExpUnqual.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/ExportAllUnnamed.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/ModuleLibrary.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/PkgNotExp.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/Umod.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_ExpQualOther.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_ExpUnqual.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_PkgNotExp.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_Umod.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_UmodUpkg.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodUPkg.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_ExpQualOther.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_NotExp.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_Umod.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodUpkg_ExpQualOther.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodUpkg_NotExp.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/UmodUpkg_Umod.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/Umod_ExpQualOther.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/Umod_ExpUnqual.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/Umod_PkgNotExp.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/Umod_UmodUpkg.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/c4.java rename hotspot/test/runtime/{BadObjectClass/Object.java => modules/AccessCheck/c5.java} (71%) create mode 100644 hotspot/test/runtime/modules/AccessCheck/myloaders/MyDiffClassLoader.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/myloaders/MySameClassLoader.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/p1/c1.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/p1/c1Loose.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/p1/c1ReadEdge.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/p1/c1ReadEdgeDiffLoader.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/p2/c2.java create mode 100644 hotspot/test/runtime/modules/AccessCheck/p3/c3.jcod create mode 100644 hotspot/test/runtime/modules/AccessCheck/p3/c3ReadEdge.jcod create mode 100644 hotspot/test/runtime/modules/AccessCheck/p3/c3ReadEdgeDiffLoader.jcod create mode 100644 hotspot/test/runtime/modules/AccessCheck/p6/c6.java create mode 100644 hotspot/test/runtime/modules/AccessCheckAllUnnamed.java create mode 100644 hotspot/test/runtime/modules/AccessCheckExp.java create mode 100644 hotspot/test/runtime/modules/AccessCheckJavaBase.java create mode 100644 hotspot/test/runtime/modules/AccessCheckRead.java create mode 100644 hotspot/test/runtime/modules/AccessCheckSuper.java create mode 100644 hotspot/test/runtime/modules/AccessCheckUnnamed.java create mode 100644 hotspot/test/runtime/modules/AccessCheckWorks.java create mode 100644 hotspot/test/runtime/modules/CCE_module_msg.java create mode 100644 hotspot/test/runtime/modules/ExportTwice.java create mode 100644 hotspot/test/runtime/modules/JVMAddModuleExportToAllUnnamed.java create mode 100644 hotspot/test/runtime/modules/JVMAddModuleExports.java create mode 100644 hotspot/test/runtime/modules/JVMAddModuleExportsToAll.java create mode 100644 hotspot/test/runtime/modules/JVMAddModulePackage.java create mode 100644 hotspot/test/runtime/modules/JVMAddReadsModule.java create mode 100644 hotspot/test/runtime/modules/JVMCanReadModule.java create mode 100644 hotspot/test/runtime/modules/JVMDefineModule.java create mode 100644 hotspot/test/runtime/modules/JVMGetModuleByPkgName.java create mode 100644 hotspot/test/runtime/modules/JVMIsExportedToModule.java create mode 100644 hotspot/test/runtime/modules/LoadUnloadModuleStress.java create mode 100644 hotspot/test/runtime/modules/ModuleHelper.java create mode 100644 hotspot/test/runtime/modules/Visibility/XbootcpNoVisibility.java create mode 100644 hotspot/test/runtime/modules/Visibility/XbootcpVisibility.java create mode 100644 hotspot/test/runtime/modules/Visibility/XpatchVisibility.java create mode 100644 hotspot/test/runtime/modules/Xpatch/Xpatch2Dirs.java create mode 100644 hotspot/test/runtime/modules/Xpatch/Xpatch2DirsMain.java create mode 100644 hotspot/test/runtime/modules/Xpatch/XpatchMain.java create mode 100644 hotspot/test/runtime/modules/Xpatch/XpatchTest.java create mode 100644 hotspot/test/runtime/modules/Xpatch/XpatchTraceCL.java create mode 100644 hotspot/test/runtime/modules/XpatchCDS.java create mode 100644 hotspot/test/runtime/modules/acc_module.jcod create mode 100644 hotspot/test/runtime/modules/getModuleJNI/GetModule.java create mode 100644 hotspot/test/runtime/modules/getModuleJNI/libGetModule.c create mode 100644 hotspot/test/runtime/modules/java.base/java/lang/reflect/ModuleHelper.java create mode 100644 hotspot/test/runtime/modules/p1/c1.java create mode 100644 hotspot/test/runtime/modules/p2/c2.java create mode 100644 hotspot/test/runtime/modules/p3/c3.java create mode 100644 hotspot/test/testlibrary/whitebox/sun/hotspot/WhiteBox.java diff --git a/hotspot/make/share/makefiles/mapfile-vers b/hotspot/make/share/makefiles/mapfile-vers index 28338d046f6..120103a04ba 100644 --- a/hotspot/make/share/makefiles/mapfile-vers +++ b/hotspot/make/share/makefiles/mapfile-vers @@ -168,3 +168,15 @@ JVM_TotalMemory; JVM_UnloadLibrary; JVM_Yield; + + # Module related API's + JVM_AddModuleExports; + JVM_AddModuleExportsToAll; + JVM_AddModuleExportsToAllUnnamed; + JVM_AddModulePackage; + JVM_AddReadsModule; + JVM_CanReadModule; + JVM_DefineModule; + JVM_IsExportedToModule; + JVM_SetBootLoaderUnnamedModule; + JVM_GetModuleByPackageName; diff --git a/hotspot/make/test/JtregNative.gmk b/hotspot/make/test/JtregNative.gmk index bced741d6a7..95c6bf242b7 100644 --- a/hotspot/make/test/JtregNative.gmk +++ b/hotspot/make/test/JtregNative.gmk @@ -45,6 +45,7 @@ BUILD_HOTSPOT_JTREG_NATIVE_SRC := \ $(HOTSPOT_TOPDIR)/test/runtime/jni/8025979 \ $(HOTSPOT_TOPDIR)/test/runtime/jni/8033445 \ $(HOTSPOT_TOPDIR)/test/runtime/jni/ToStringInInterfaceTest \ + $(HOTSPOT_TOPDIR)/test/runtime/modules/getModuleJNI \ $(HOTSPOT_TOPDIR)/test/runtime/SameObject \ $(HOTSPOT_TOPDIR)/test/compiler/floatingpoint/ \ $(HOTSPOT_TOPDIR)/test/compiler/calls \ diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/module-info.java b/hotspot/src/jdk.hotspot.agent/share/classes/module-info.java new file mode 100644 index 00000000000..85b163bcd02 --- /dev/null +++ b/hotspot/src/jdk.hotspot.agent/share/classes/module-info.java @@ -0,0 +1,49 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +module jdk.hotspot.agent { + requires java.datatransfer; + requires java.desktop; + requires java.rmi; + requires java.scripting; + requires jdk.jcmd; + requires jdk.jdi; + + // RMI needs to serialize types in this package + exports sun.jvm.hotspot.debugger.remote to java.rmi; + provides com.sun.jdi.connect.Connector with sun.jvm.hotspot.jdi.SACoreAttachingConnector; + provides com.sun.jdi.connect.Connector with sun.jvm.hotspot.jdi.SADebugServerAttachingConnector; + provides com.sun.jdi.connect.Connector with sun.jvm.hotspot.jdi.SAPIDAttachingConnector; + + provides jdk.internal.vm.agent.spi.ToolProvider with sun.jvm.hotspot.tools.JStack; + provides jdk.internal.vm.agent.spi.ToolProvider with sun.jvm.hotspot.tools.JInfo; + provides jdk.internal.vm.agent.spi.ToolProvider with sun.jvm.hotspot.tools.ClassLoaderStats; + provides jdk.internal.vm.agent.spi.ToolProvider with sun.jvm.hotspot.tools.FinalizerInfo; + provides jdk.internal.vm.agent.spi.ToolProvider with sun.jvm.hotspot.tools.HeapDumper; + provides jdk.internal.vm.agent.spi.ToolProvider with sun.jvm.hotspot.tools.HeapSummary; + provides jdk.internal.vm.agent.spi.ToolProvider with sun.jvm.hotspot.tools.ObjectHistogram; + provides jdk.internal.vm.agent.spi.ToolProvider with sun.jvm.hotspot.tools.PMap; +} + diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/ClassLoaderStats.java b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/ClassLoaderStats.java index a5fac511d18..34f5bfbefef 100644 --- a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/ClassLoaderStats.java +++ b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/ClassLoaderStats.java @@ -31,14 +31,14 @@ import sun.jvm.hotspot.debugger.*; import sun.jvm.hotspot.memory.*; import sun.jvm.hotspot.oops.*; import sun.jvm.hotspot.runtime.*; -import sun.jvm.hotspot.tools.*; import sun.jvm.hotspot.utilities.*; +import jdk.internal.vm.agent.spi.ToolProvider; /** A command line tool to print class loader statistics. */ -public class ClassLoaderStats extends Tool { +public class ClassLoaderStats extends Tool implements ToolProvider { boolean verbose = true; public ClassLoaderStats() { @@ -49,6 +49,16 @@ public class ClassLoaderStats extends Tool { super(d); } + @Override + public String getName() { + return "classLoaderStats"; + } + + @Override + public void run(String... arguments) { + execute(arguments); + } + public static void main(String[] args) { ClassLoaderStats cls = new ClassLoaderStats(); cls.execute(args); diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/FinalizerInfo.java b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/FinalizerInfo.java index 58819a0a7dd..9da8e523afb 100644 --- a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/FinalizerInfo.java +++ b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/FinalizerInfo.java @@ -25,24 +25,21 @@ package sun.jvm.hotspot.tools; import sun.jvm.hotspot.debugger.JVMDebugger; -import sun.jvm.hotspot.tools.*; import sun.jvm.hotspot.oops.*; -import sun.jvm.hotspot.runtime.VM; import sun.jvm.hotspot.utilities.SystemDictionaryHelper; -import sun.jvm.hotspot.utilities.ObjectReader; -import sun.jvm.hotspot.utilities.MarkBits; +import jdk.internal.vm.agent.spi.ToolProvider; -import java.util.HashMap; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; +import java.util.HashMap; /* * Iterates over the queue of object pending finalization and prints a * summary of these objects in the form of a histogram. */ -public class FinalizerInfo extends Tool { +public class FinalizerInfo extends Tool implements ToolProvider { public FinalizerInfo() { super(); @@ -52,6 +49,16 @@ public class FinalizerInfo extends Tool { super(d); } + @Override + public String getName() { + return "finalizerInfo"; + } + + @Override + public void run(String... arguments) { + execute(arguments); + } + public static void main(String[] args) { FinalizerInfo finfo = new FinalizerInfo(); finfo.execute(args); diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/HeapDumper.java b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/HeapDumper.java index 1b93504313a..d0a11abd095 100644 --- a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/HeapDumper.java +++ b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/HeapDumper.java @@ -26,6 +26,8 @@ package sun.jvm.hotspot.tools; import sun.jvm.hotspot.utilities.HeapHprofBinWriter; import sun.jvm.hotspot.debugger.JVMDebugger; +import jdk.internal.vm.agent.spi.ToolProvider; + import java.io.IOException; /* @@ -33,12 +35,16 @@ import java.io.IOException; * process/core as a HPROF binary file. It can also be used as a standalone * tool if required. */ -public class HeapDumper extends Tool { +public class HeapDumper extends Tool implements ToolProvider { private static String DEFAULT_DUMP_FILE = "heap.bin"; private String dumpFile; + public HeapDumper() { + this.dumpFile = DEFAULT_DUMP_FILE; + } + public HeapDumper(String dumpFile) { this.dumpFile = dumpFile; } @@ -48,6 +54,11 @@ public class HeapDumper extends Tool { this.dumpFile = dumpFile; } + @Override + public String getName() { + return "heapDumper"; + } + protected void printFlagsUsage() { System.out.println(" \tto dump heap to " + DEFAULT_DUMP_FILE); @@ -69,18 +80,22 @@ public class HeapDumper extends Tool { // JDK jmap utility will always invoke this tool as: // HeapDumper -f public static void main(String args[]) { - String file = DEFAULT_DUMP_FILE; + HeapDumper dumper = new HeapDumper(); + dumper.run(args); + } + + @Override + public void run(String... args) { if (args.length > 2) { if (args[0].equals("-f")) { - file = args[1]; + this.dumpFile = args[1]; String[] newargs = new String[args.length-2]; System.arraycopy(args, 2, newargs, 0, args.length-2); args = newargs; } } - HeapDumper dumper = new HeapDumper(file); - dumper.execute(args); + execute(args); } } diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/HeapSummary.java b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/HeapSummary.java index 601283072ff..5c1012bf644 100644 --- a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/HeapSummary.java +++ b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/HeapSummary.java @@ -33,8 +33,9 @@ import sun.jvm.hotspot.debugger.JVMDebugger; import sun.jvm.hotspot.memory.*; import sun.jvm.hotspot.oops.*; import sun.jvm.hotspot.runtime.*; +import jdk.internal.vm.agent.spi.ToolProvider; -public class HeapSummary extends Tool { +public class HeapSummary extends Tool implements ToolProvider { public HeapSummary() { super(); @@ -49,6 +50,16 @@ public class HeapSummary extends Tool { hs.execute(args); } + @Override + public String getName() { + return "heapSummary"; + } + + @Override + public void run(String... arguments) { + execute(arguments); + } + public void run() { CollectedHeap heap = VM.getVM().getUniverse().heap(); VM.Flag[] flags = VM.getVM().getCommandLineFlags(); diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/JInfo.java b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/JInfo.java index 7469de28f40..50f9c30975b 100644 --- a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/JInfo.java +++ b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/JInfo.java @@ -27,8 +27,9 @@ package sun.jvm.hotspot.tools; import sun.jvm.hotspot.debugger.JVMDebugger; import sun.jvm.hotspot.runtime.Arguments; import sun.jvm.hotspot.runtime.VM; +import jdk.internal.vm.agent.spi.ToolProvider; -public class JInfo extends Tool { +public class JInfo extends Tool implements ToolProvider { public JInfo() { super(); } @@ -94,13 +95,14 @@ public class JInfo extends Tool { tool.run(); } - public static void main(String[] args) { + @Override + public void run(String... args) { int mode = -1; switch (args.length) { case 1: if (args[0].charAt(0) == '-') { // -h or -help or some invalid flag - new JInfo(mode).usage(); + usage(); } else { mode = MODE_BOTH; } @@ -114,7 +116,7 @@ public class JInfo extends Tool { mode = MODE_SYSPROPS; } else if (modeFlag.charAt(0) == '-') { // -h or -help or some invalid flag - new JInfo(mode).usage(); + usage(); } else { mode = MODE_BOTH; } @@ -131,11 +133,16 @@ public class JInfo extends Tool { } default: - new JInfo(mode).usage(); + usage(); } - JInfo jinfo = new JInfo(mode); - jinfo.execute(args); + this.mode = mode; + execute(args); + } + + public static void main(String[] args) { + JInfo jinfo = new JInfo(); + jinfo.run(args); } private void printVMFlags() { diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/JStack.java b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/JStack.java index b8609bcdbeb..821ed388853 100644 --- a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/JStack.java +++ b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/JStack.java @@ -25,8 +25,9 @@ package sun.jvm.hotspot.tools; import sun.jvm.hotspot.debugger.JVMDebugger; +import jdk.internal.vm.agent.spi.ToolProvider; -public class JStack extends Tool { +public class JStack extends Tool implements ToolProvider { public JStack(boolean mixedMode, boolean concurrentLocks) { this.mixedMode = mixedMode; this.concurrentLocks = concurrentLocks; @@ -66,9 +67,8 @@ public class JStack extends Tool { tool.run(); } - public static void main(String[] args) { - boolean mixedMode = false; - boolean concurrentLocks = false; + @Override + public void run(String... args) { int used = 0; for (int i = 0; i < args.length; i++) { if (args[i].equals("-m")) { @@ -88,8 +88,12 @@ public class JStack extends Tool { args = newArgs; } - JStack jstack = new JStack(mixedMode, concurrentLocks); - jstack.execute(args); + execute(args); + } + + public static void main(String[] args) { + JStack jstack = new JStack(); + jstack.run(args); } private boolean mixedMode; diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/ObjectHistogram.java b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/ObjectHistogram.java index ddedcae5394..0dc100eebe5 100644 --- a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/ObjectHistogram.java +++ b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/ObjectHistogram.java @@ -27,11 +27,13 @@ package sun.jvm.hotspot.tools; import sun.jvm.hotspot.debugger.*; import sun.jvm.hotspot.oops.*; import sun.jvm.hotspot.runtime.*; +import jdk.internal.vm.agent.spi.ToolProvider; + import java.io.PrintStream; /** A sample tool which uses the Serviceability Agent's APIs to obtain an object histogram from a remote or crashed VM. */ -public class ObjectHistogram extends Tool { +public class ObjectHistogram extends Tool implements ToolProvider { public ObjectHistogram() { super(); @@ -41,6 +43,16 @@ public class ObjectHistogram extends Tool { super(d); } + @Override + public String getName() { + return "objectHistogram"; + } + + @Override + public void run(String... arguments) { + execute(arguments); + } + public void run() { run(System.out, System.err); } diff --git a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/PMap.java b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/PMap.java index 43bcd593f30..0c0ccbf6c8c 100644 --- a/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/PMap.java +++ b/hotspot/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/tools/PMap.java @@ -28,9 +28,9 @@ import java.io.*; import java.util.*; import sun.jvm.hotspot.debugger.*; import sun.jvm.hotspot.debugger.cdbg.*; -import sun.jvm.hotspot.runtime.*; +import jdk.internal.vm.agent.spi.ToolProvider; -public class PMap extends Tool { +public class PMap extends Tool implements ToolProvider { public PMap() { super(); @@ -40,6 +40,16 @@ public class PMap extends Tool { super(d); } + @Override + public String getName() { + return "pmap"; + } + + @Override + public void run(String... arguments) { + execute(arguments); + } + public void run() { run(System.out); } diff --git a/hotspot/src/jdk.vm.ci/share/classes/META-INF/services/jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory b/hotspot/src/jdk.vm.ci/share/classes/META-INF/services/jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory deleted file mode 100644 index fe3a4573058..00000000000 --- a/hotspot/src/jdk.vm.ci/share/classes/META-INF/services/jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory +++ /dev/null @@ -1,3 +0,0 @@ -jdk.vm.ci.hotspot.aarch64.AArch64HotSpotJVMCIBackendFactory -jdk.vm.ci.hotspot.amd64.AMD64HotSpotJVMCIBackendFactory -jdk.vm.ci.hotspot.sparc.SPARCHotSpotJVMCIBackendFactory diff --git a/hotspot/src/jdk.vm.ci/share/classes/module-info.java b/hotspot/src/jdk.vm.ci/share/classes/module-info.java new file mode 100644 index 00000000000..a59a37d56e2 --- /dev/null +++ b/hotspot/src/jdk.vm.ci/share/classes/module-info.java @@ -0,0 +1,37 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +module jdk.vm.ci { + uses jdk.vm.ci.hotspot.HotSpotVMEventListener; + uses jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory; + uses jdk.vm.ci.runtime.JVMCICompilerFactory; + + provides jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory with + jdk.vm.ci.hotspot.aarch64.AArch64HotSpotJVMCIBackendFactory; + provides jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory with + jdk.vm.ci.hotspot.amd64.AMD64HotSpotJVMCIBackendFactory; + provides jdk.vm.ci.hotspot.HotSpotJVMCIBackendFactory with + jdk.vm.ci.hotspot.sparc.SPARCHotSpotJVMCIBackendFactory; +} diff --git a/hotspot/src/os/posix/dtrace/hotspot_jni.d b/hotspot/src/os/posix/dtrace/hotspot_jni.d index cca1c517650..eb95b7e4c3a 100644 --- a/hotspot/src/os/posix/dtrace/hotspot_jni.d +++ b/hotspot/src/os/posix/dtrace/hotspot_jni.d @@ -300,6 +300,8 @@ provider hotspot_jni { probe GetLongField__return(uintptr_t); probe GetMethodID__entry(void*, void*, const char*, const char*); probe GetMethodID__return(uintptr_t); + probe GetModule__entry(void*, void*); + probe GetModule__return(void*); probe GetObjectArrayElement__entry(void*, void*, uintptr_t); probe GetObjectArrayElement__return(void*); probe GetObjectClass__entry(void*, void*); diff --git a/hotspot/src/share/vm/c1/c1_Runtime1.cpp b/hotspot/src/share/vm/c1/c1_Runtime1.cpp index 92c721575d0..12b58adb872 100644 --- a/hotspot/src/share/vm/c1/c1_Runtime1.cpp +++ b/hotspot/src/share/vm/c1/c1_Runtime1.cpp @@ -661,7 +661,7 @@ JRT_ENTRY(void, Runtime1::throw_class_cast_exception(JavaThread* thread, oopDesc NOT_PRODUCT(_throw_class_cast_exception_count++;) ResourceMark rm(thread); char* message = SharedRuntime::generate_class_cast_message( - thread, object->klass()->external_name()); + thread, object->klass()); SharedRuntime::throw_and_post_jvmti_exception( thread, vmSymbols::java_lang_ClassCastException(), message); JRT_END diff --git a/hotspot/src/share/vm/ci/ciEnv.cpp b/hotspot/src/share/vm/ci/ciEnv.cpp index 2775c6776f5..21e952ee57f 100644 --- a/hotspot/src/share/vm/ci/ciEnv.cpp +++ b/hotspot/src/share/vm/ci/ciEnv.cpp @@ -370,9 +370,9 @@ bool ciEnv::check_klass_accessibility(ciKlass* accessing_klass, resolved_klass = ObjArrayKlass::cast(resolved_klass)->bottom_klass(); } if (resolved_klass->is_instance_klass()) { - return Reflection::verify_class_access(accessing_klass->get_Klass(), - resolved_klass, - true); + return (Reflection::verify_class_access(accessing_klass->get_Klass(), + resolved_klass, + true) == Reflection::ACCESS_OK); } return true; } diff --git a/hotspot/src/share/vm/classfile/classFileParser.cpp b/hotspot/src/share/vm/classfile/classFileParser.cpp index 97d6bdfcd56..3be9d2f9b47 100644 --- a/hotspot/src/share/vm/classfile/classFileParser.cpp +++ b/hotspot/src/share/vm/classfile/classFileParser.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2016, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -28,6 +28,7 @@ #include "classfile/classLoaderData.inline.hpp" #include "classfile/defaultMethods.hpp" #include "classfile/javaClasses.inline.hpp" +#include "classfile/moduleEntry.hpp" #include "classfile/symbolTable.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/verificationType.hpp" @@ -103,8 +104,6 @@ #define JAVA_9_VERSION 53 -enum { LegalClass, LegalField, LegalMethod }; // used to verify unqualified names - void ClassFileParser::parse_constant_pool_entries(const ClassFileStream* const stream, ConstantPool* cp, const int length, @@ -1965,7 +1964,7 @@ AnnotationCollector::annotation_index(const ClassLoaderData* loader_data, const vmSymbols::SID sid = vmSymbols::find_sid(name); // Privileged code can use all annotations. Other code silently drops some. const bool privileged = loader_data->is_the_null_class_loader_data() || - loader_data->is_ext_class_loader_data() || + loader_data->is_platform_class_loader_data() || loader_data->is_anonymous(); switch (sid) { case vmSymbols::VM_SYMBOL_ENUM_NAME(sun_reflect_CallerSensitive_signature): { @@ -4358,17 +4357,29 @@ static Array* compute_transitive_interfaces(const InstanceKlass* super, static void check_super_class_access(const InstanceKlass* this_klass, TRAPS) { assert(this_klass != NULL, "invariant"); const Klass* const super = this_klass->super(); - if ((super != NULL) && - (!Reflection::verify_class_access(this_klass, super, false))) { - ResourceMark rm(THREAD); - Exceptions::fthrow( - THREAD_AND_LOCATION, - vmSymbols::java_lang_IllegalAccessError(), - "class %s cannot access its superclass %s", - this_klass->external_name(), - super->external_name() - ); - return; + if (super != NULL) { + Reflection::VerifyClassAccessResults vca_result = + Reflection::verify_class_access(this_klass, super, false); + if (vca_result != Reflection::ACCESS_OK) { + ResourceMark rm(THREAD); + char* msg = Reflection::verify_class_access_msg(this_klass, super, vca_result); + if (msg == NULL) { + ResourceMark rm(THREAD); + Exceptions::fthrow( + THREAD_AND_LOCATION, + vmSymbols::java_lang_IllegalAccessError(), + "class %s cannot access its superclass %s", + this_klass->external_name(), + super->external_name()); + } else { + // Add additional message content. + Exceptions::fthrow( + THREAD_AND_LOCATION, + vmSymbols::java_lang_IllegalAccessError(), + "superclass access check failed: %s", + msg); + } + } } } @@ -4380,16 +4391,26 @@ static void check_super_interface_access(const InstanceKlass* this_klass, TRAPS) for (int i = lng - 1; i >= 0; i--) { Klass* const k = local_interfaces->at(i); assert (k != NULL && k->is_interface(), "invalid interface"); - if (!Reflection::verify_class_access(this_klass, k, false)) { + Reflection::VerifyClassAccessResults vca_result = + Reflection::verify_class_access(this_klass, k, false); + if (vca_result != Reflection::ACCESS_OK) { ResourceMark rm(THREAD); - Exceptions::fthrow( - THREAD_AND_LOCATION, - vmSymbols::java_lang_IllegalAccessError(), - "class %s cannot access its superinterface %s", - this_klass->external_name(), - k->external_name() - ); - return; + char* msg = Reflection::verify_class_access_msg(this_klass, k, vca_result); + if (msg == NULL) { + Exceptions::fthrow( + THREAD_AND_LOCATION, + vmSymbols::java_lang_IllegalAccessError(), + "class %s cannot access its superinterface %s", + this_klass->external_name(), + k->external_name()); + } else { + // Add additional message content. + Exceptions::fthrow( + THREAD_AND_LOCATION, + vmSymbols::java_lang_IllegalAccessError(), + "superinterface check failed: %s", + msg); + } } } } @@ -4489,12 +4510,14 @@ void ClassFileParser::verify_legal_class_modifiers(jint flags, TRAPS) const { const bool is_super = (flags & JVM_ACC_SUPER) != 0; const bool is_enum = (flags & JVM_ACC_ENUM) != 0; const bool is_annotation = (flags & JVM_ACC_ANNOTATION) != 0; + const bool is_module_info= (flags & JVM_ACC_MODULE) != 0; const bool major_gte_15 = _major_version >= JAVA_1_5_VERSION; if ((is_abstract && is_final) || (is_interface && !is_abstract) || (is_interface && major_gte_15 && (is_super || is_enum)) || - (!is_interface && major_gte_15 && is_annotation)) { + (!is_interface && major_gte_15 && is_annotation) || + is_module_info) { ResourceMark rm(THREAD); Exceptions::fthrow( THREAD_AND_LOCATION, @@ -4650,65 +4673,9 @@ void ClassFileParser::verify_legal_utf8(const unsigned char* buffer, int length, TRAPS) const { assert(_need_verify, "only called when _need_verify is true"); - int i = 0; - const int count = length >> 2; - for (int k=0; k= 128 (highest bit 1) for v == 0 or v >= 128. - const unsigned char res = b0 | b0 - 1 | - b1 | b1 - 1 | - b2 | b2 - 1 | - b3 | b3 - 1; - if (res >= 128) break; - i += 4; + if (!UTF8::is_legal_utf8(buffer, length, _major_version <= 47)) { + classfile_parse_error("Illegal UTF8 string in constant pool in class file %s", CHECK); } - for(; i < length; i++) { - unsigned short c; - // no embedded zeros - guarantee_property((buffer[i] != 0), "Illegal UTF8 string in constant pool in class file %s", CHECK); - if(buffer[i] < 128) { - continue; - } - if ((i + 5) < length) { // see if it's legal supplementary character - if (UTF8::is_supplementary_character(&buffer[i])) { - c = UTF8::get_supplementary_character(&buffer[i]); - i += 5; - continue; - } - } - switch (buffer[i] >> 4) { - default: break; - case 0x8: case 0x9: case 0xA: case 0xB: case 0xF: - classfile_parse_error("Illegal UTF8 string in constant pool in class file %s", CHECK); - case 0xC: case 0xD: // 110xxxxx 10xxxxxx - c = (buffer[i] & 0x1F) << 6; - i++; - if ((i < length) && ((buffer[i] & 0xC0) == 0x80)) { - c += buffer[i] & 0x3F; - if (_major_version <= 47 || c == 0 || c >= 0x80) { - // for classes with major > 47, c must a null or a character in its shortest form - break; - } - } - classfile_parse_error("Illegal UTF8 string in constant pool in class file %s", CHECK); - case 0xE: // 1110xxxx 10xxxxxx 10xxxxxx - c = (buffer[i] & 0xF) << 12; - i += 2; - if ((i < length) && ((buffer[i-1] & 0xC0) == 0x80) && ((buffer[i] & 0xC0) == 0x80)) { - c += ((buffer[i-1] & 0x3F) << 6) + (buffer[i] & 0x3F); - if (_major_version <= 47 || c >= 0x800) { - // for classes with major > 47, c must be in its shortest form - break; - } - } - classfile_parse_error("Illegal UTF8 string in constant pool in class file %s", CHECK); - } // end of switch - } // end of for } // Unqualified names may not contain the characters '.', ';', '[', or '/'. @@ -4716,24 +4683,35 @@ void ClassFileParser::verify_legal_utf8(const unsigned char* buffer, // or . Note that method names may not be or in this // method. Because these names have been checked as special cases before // calling this method in verify_legal_method_name. -static bool verify_unqualified_name(const char* name, - unsigned int length, - int type) { +// +// This method is also called from the modular system APIs in modules.cpp +// to verify the validity of module and package names. +bool ClassFileParser::verify_unqualified_name(const char* name, + unsigned int length, + int type) { for (const char* p = name; p != name + length;) { jchar ch = *p; if (ch < 128) { - p++; - if (ch == '.' || ch == ';' || ch == '[') { + if (ch == '.') { + // permit '.' in module names unless it's the first char, or + // preceding char is also a '.', or last char is a '.'. + if ((type != ClassFileParser::LegalModule) || + (p == name) || (*(p-1) == '.') || + (p == name + length - 1)) { + return false; + } + } + if (ch == ';' || ch == '[' ) { return false; // do not permit '.', ';', or '[' } - if (type != LegalClass && ch == '/') { + if (type != ClassFileParser::LegalClass && ch == '/') { return false; // do not permit '/' unless it's class name } - if (type == LegalMethod && (ch == '<' || ch == '>')) { + if (type == ClassFileParser::LegalMethod && (ch == '<' || ch == '>')) { return false; // do not permit '<' or '>' in method names } - } - else { + p++; + } else { char* tmp_p = UTF8::next(p, &ch); p = tmp_p; } @@ -5192,7 +5170,7 @@ static void check_methods_for_intrinsics(const InstanceKlass* ik, } } -InstanceKlass* ClassFileParser::create_instance_klass(TRAPS) { +InstanceKlass* ClassFileParser::create_instance_klass(bool changed_by_loadhook, TRAPS) { if (_klass != NULL) { return _klass; } @@ -5200,14 +5178,14 @@ InstanceKlass* ClassFileParser::create_instance_klass(TRAPS) { InstanceKlass* const ik = InstanceKlass::allocate_instance_klass(*this, CHECK_NULL); - fill_instance_klass(ik, CHECK_NULL); + fill_instance_klass(ik, changed_by_loadhook, CHECK_NULL); assert(_klass == ik, "invariant"); return ik; } -void ClassFileParser::fill_instance_klass(InstanceKlass* ik, TRAPS) { +void ClassFileParser::fill_instance_klass(InstanceKlass* ik, bool changed_by_loadhook, TRAPS) { assert(ik != NULL, "invariant"); set_klass_to_deallocate(ik); @@ -5272,6 +5250,12 @@ void ClassFileParser::fill_instance_klass(InstanceKlass* ik, TRAPS) { ik->set_host_klass(_host_klass); } + // Set PackageEntry for this_klass + oop cl = ik->class_loader(); + Handle clh = Handle(THREAD, java_lang_ClassLoader::non_reflection_class_loader(cl)); + ClassLoaderData* cld = ClassLoaderData::class_loader_data_or_null(clh()); + ik->set_package(cld, CHECK); + const Array* const methods = ik->methods(); assert(methods != NULL, "invariant"); const int methods_len = methods->length(); @@ -5327,10 +5311,18 @@ void ClassFileParser::fill_instance_klass(InstanceKlass* ik, TRAPS) { } } + // Obtain this_klass' module entry + ModuleEntry* module_entry = ik->module(); + assert(module_entry != NULL, "module_entry should always be set"); + + // Obtain java.lang.reflect.Module + Handle module_handle(THREAD, JNIHandles::resolve(module_entry->module())); + // Allocate mirror and initialize static fields // The create_mirror() call will also call compute_modifiers() java_lang_Class::create_mirror(ik, _loader_data->class_loader(), + module_handle, _protection_domain, CHECK); @@ -5344,6 +5336,15 @@ void ClassFileParser::fill_instance_klass(InstanceKlass* ik, TRAPS) { CHECK); } + // Add read edges to the unnamed modules of the bootstrap and app class loaders. + if (changed_by_loadhook && !module_handle.is_null() && module_entry->is_named() && + !module_entry->has_default_read_edges()) { + if (!module_entry->set_has_default_read_edges()) { + // We won a potential race + JvmtiExport::add_default_read_edges(module_handle, THREAD); + } + } + // Update the loader_data graph. record_defined_class_dependencies(ik, CHECK); @@ -5351,11 +5352,24 @@ void ClassFileParser::fill_instance_klass(InstanceKlass* ik, TRAPS) { if (!is_internal()) { if (log_is_enabled(Info, classload)) { - ik->print_loading_log(LogLevel::Info, _loader_data, _stream); - } - // No 'else' here as logging levels are not mutually exclusive - if (log_is_enabled(Debug, classload)) { - ik->print_loading_log(LogLevel::Debug, _loader_data, _stream); + ResourceMark rm; + const char* module_name = NULL; + static const size_t modules_image_name_len = strlen(MODULES_IMAGE_NAME); + size_t stream_len = strlen(_stream->source()); + // See if _stream->source() ends in "modules" + if (module_entry->is_named() && modules_image_name_len < stream_len && + (strncmp(_stream->source() + stream_len - modules_image_name_len, + MODULES_IMAGE_NAME, modules_image_name_len) == 0)) { + module_name = module_entry->name()->as_C_string(); + } + + if (log_is_enabled(Info, classload)) { + ik->print_loading_log(LogLevel::Info, _loader_data, module_name, _stream); + } + // No 'else' here as logging levels are not mutually exclusive + if (log_is_enabled(Debug, classload)) { + ik->print_loading_log(LogLevel::Debug, _loader_data, module_name, _stream); + } } if (log_is_enabled(Info, classresolve)) { diff --git a/hotspot/src/share/vm/classfile/classFileParser.hpp b/hotspot/src/share/vm/classfile/classFileParser.hpp index d6fd836d61c..3ee99df2e3b 100644 --- a/hotspot/src/share/vm/classfile/classFileParser.hpp +++ b/hotspot/src/share/vm/classfile/classFileParser.hpp @@ -73,6 +73,8 @@ class ClassFileParser VALUE_OBJ_CLASS_SPEC { NOF_PUBLICITY_LEVELS }; + enum { LegalClass, LegalField, LegalMethod, LegalModule }; // used to verify unqualified names + private: const ClassFileStream* _stream; // Actual input stream const Symbol* _requested_name; @@ -155,7 +157,7 @@ class ClassFileParser VALUE_OBJ_CLASS_SPEC { ConstantPool* cp, TRAPS); - void fill_instance_klass(InstanceKlass* ik, TRAPS); + void fill_instance_klass(InstanceKlass* ik, bool cf_changed_in_CFLH, TRAPS); void set_klass(InstanceKlass* instance); void set_class_synthetic_flag(bool x) { _synthetic_flag = x; } @@ -482,7 +484,7 @@ class ClassFileParser VALUE_OBJ_CLASS_SPEC { ~ClassFileParser(); - InstanceKlass* create_instance_klass(TRAPS); + InstanceKlass* create_instance_klass(bool cf_changed_in_CFLH, TRAPS); const ClassFileStream* clone_stream() const; @@ -512,6 +514,7 @@ class ClassFileParser VALUE_OBJ_CLASS_SPEC { bool is_internal() const { return INTERNAL == _pub_level; } + static bool verify_unqualified_name(const char* name, unsigned int length, int type); }; #endif // SHARE_VM_CLASSFILE_CLASSFILEPARSER_HPP diff --git a/hotspot/src/share/vm/classfile/classLoader.cpp b/hotspot/src/share/vm/classfile/classLoader.cpp index 0a1e20df3f0..d4196c38bd6 100644 --- a/hotspot/src/share/vm/classfile/classLoader.cpp +++ b/hotspot/src/share/vm/classfile/classLoader.cpp @@ -29,6 +29,9 @@ #include "classfile/classLoaderExt.hpp" #include "classfile/javaClasses.hpp" #include "classfile/jimage.hpp" +#include "classfile/moduleEntry.hpp" +#include "classfile/modules.hpp" +#include "classfile/packageEntry.hpp" #include "classfile/klassFactory.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -138,11 +141,14 @@ PerfCounter* ClassLoader::_load_instance_class_failCounter = NULL; ClassPathEntry* ClassLoader::_first_entry = NULL; ClassPathEntry* ClassLoader::_last_entry = NULL; int ClassLoader::_num_entries = 0; -PackageHashtable* ClassLoader::_package_hash_table = NULL; - +ClassPathEntry* ClassLoader::_first_append_entry = NULL; +bool ClassLoader::_has_jimage = false; #if INCLUDE_CDS +GrowableArray* ClassLoader::_boot_modules_array = NULL; +GrowableArray* ClassLoader::_platform_modules_array = NULL; SharedPathsMiscInfo* ClassLoader::_shared_paths_misc_info = NULL; #endif + // helper routines bool string_starts_with(const char* str, const char* str_to_find) { size_t str_len = strlen(str); @@ -162,7 +168,7 @@ static const char* get_jimage_version_string() { return (const char*)version_string; } -bool string_ends_with(const char* str, const char* str_to_find) { +bool ClassLoader::string_ends_with(const char* str, const char* str_to_find) { size_t str_len = strlen(str); size_t str_to_find_len = strlen(str_to_find); if (str_to_find_len > str_len) { @@ -356,15 +362,49 @@ ClassFileStream* ClassPathImageEntry::open_stream(const char* name, TRAPS) { if (location == 0) { char package[JIMAGE_MAX_PATH]; name_to_package(name, package, JIMAGE_MAX_PATH); + +#if INCLUDE_CDS + if (package[0] == '\0' && DumpSharedSpaces) { + return NULL; + } +#endif if (package[0] != '\0') { - const char* module = (*JImagePackageToModule)(_jimage, package); - if (module == NULL) { - module = "java.base"; + if (!Universe::is_module_initialized()) { + location = (*JImageFindResource)(_jimage, "java.base", get_jimage_version_string(), name, &size); +#if INCLUDE_CDS + // CDS uses the boot class loader to load classes whose packages are in + // modules defined for other class loaders. So, for now, get their module + // names from the "modules" jimage file. + if (DumpSharedSpaces && location == 0) { + const char* module_name = (*JImagePackageToModule)(_jimage, package); + if (module_name != NULL) { + location = (*JImageFindResource)(_jimage, module_name, get_jimage_version_string(), name, &size); + } } - location = (*JImageFindResource)(_jimage, module, get_jimage_version_string(), name, &size); +#endif + + } else { + // Get boot class loader's package entry table + PackageEntryTable* pkgEntryTable = + ClassLoaderData::the_null_class_loader_data()->packages(); + // Get package's package entry + TempNewSymbol pkg_symbol = SymbolTable::new_symbol(package, CHECK_NULL); + PackageEntry* package_entry = pkgEntryTable->lookup_only(pkg_symbol); + + if (package_entry != NULL) { + ResourceMark rm; + // Get the module name + ModuleEntry* module = package_entry->module(); + assert(module != NULL, "Boot classLoader package missing module"); + assert(module->is_named(), "Boot classLoader package is in unnamed module"); + const char* module_name = module->name()->as_C_string(); + if (module_name != NULL) { + location = (*JImageFindResource)(_jimage, module_name, get_jimage_version_string(), name, &size); + } + } + } } } - if (location != 0) { if (UsePerfData) { ClassLoader::perf_sys_classfile_bytes_read()->inc(size); @@ -409,11 +449,11 @@ void ClassPathImageEntry::compile_the_world(Handle loader, TRAPS) { } } } +#endif bool ClassPathImageEntry::is_jrt() { - return string_ends_with(name(), BOOT_IMAGE_NAME); + return ClassLoader::is_jrt(name()); } -#endif #if INCLUDE_CDS void ClassLoader::exit_with_path_failure(const char* error, const char* message) { @@ -480,7 +520,7 @@ void ClassLoader::setup_bootstrap_search_path() { _shared_paths_misc_info->add_boot_classpath(sys_class_path); } #endif - setup_search_path(sys_class_path); + setup_search_path(sys_class_path, true); } #if INCLUDE_CDS @@ -500,10 +540,11 @@ bool ClassLoader::check_shared_paths_misc_info(void *buf, int size) { } #endif -void ClassLoader::setup_search_path(const char *class_path) { +void ClassLoader::setup_search_path(const char *class_path, bool bootstrap_search) { int offset = 0; int len = (int)strlen(class_path); int end = 0; + bool mark_append_entry = false; // Iterate over class path entries for (int start = 0; start < len; start = end) { @@ -512,10 +553,23 @@ void ClassLoader::setup_search_path(const char *class_path) { } EXCEPTION_MARK; ResourceMark rm(THREAD); + mark_append_entry = (mark_append_entry || + (bootstrap_search && (start == Arguments::bootclassloader_append_index()))); char* path = NEW_RESOURCE_ARRAY(char, end - start + 1); strncpy(path, &class_path[start], end - start); path[end - start] = '\0'; - update_class_path_entry_list(path, false); + update_class_path_entry_list(path, false, mark_append_entry, false); + + // Check on the state of the boot loader's append path + if (mark_append_entry && (_first_append_entry == NULL)) { + // Failure to mark the first append entry, most likely + // due to a non-existent path. Record the next entry + // as the first boot loader append entry. + mark_append_entry = true; + } else { + mark_append_entry = false; + } + #if INCLUDE_CDS if (DumpSharedSpaces) { check_shared_classpath(path); @@ -616,6 +670,18 @@ ClassPathZipEntry* ClassLoader::create_class_path_zip_entry(const char *path) { return NULL; } +// The boot class loader must adhere to specfic visibility rules. +// Prior to loading a class in a named package, the package is checked +// to see if it is in a module defined to the boot loader. If the +// package is not in a module defined to the boot loader, the class +// must be loaded only in the boot loader's append path, which +// consists of [-Xbootclasspath/a]; [jvmti appended entries] +void ClassLoader::set_first_append_entry(ClassPathEntry *new_entry) { + if (_first_append_entry == NULL) { + _first_append_entry = new_entry; + } +} + // returns true if entry already on class path bool ClassLoader::contains_entry(ClassPathEntry *entry) { ClassPathEntry* e = _first_entry; @@ -641,9 +707,31 @@ void ClassLoader::add_to_list(ClassPathEntry *new_entry) { _num_entries ++; } +void ClassLoader::prepend_to_list(ClassPathEntry *new_entry) { + if (new_entry != NULL) { + if (_last_entry == NULL) { + _first_entry = _last_entry = new_entry; + } else { + new_entry->set_next(_first_entry); + _first_entry = new_entry; + } + } + _num_entries ++; +} + +void ClassLoader::add_to_list(const char *apath) { + update_class_path_entry_list((char*)apath, false, false, false); +} + +void ClassLoader::prepend_to_list(const char *apath) { + update_class_path_entry_list((char*)apath, false, false, true); +} + // Returns true IFF the file/dir exists and the entry was successfully created. bool ClassLoader::update_class_path_entry_list(const char *path, bool check_for_duplicates, + bool mark_append_entry, + bool prepend_entry, bool throw_exception) { struct stat st; if (os::stat(path, &st) == 0) { @@ -654,12 +742,20 @@ bool ClassLoader::update_class_path_entry_list(const char *path, if (new_entry == NULL) { return false; } - // The kernel VM adds dynamically to the end of the classloader path and - // doesn't reorder the bootclasspath which would break java.lang.Package - // (see PackageInfo). + + // Ensure that the first boot loader append entry will always be set correctly. + assert((!mark_append_entry || + (mark_append_entry && (!check_for_duplicates || !contains_entry(new_entry)))), + "failed to mark boot loader's first append boundary"); + + // Do not reorder the bootclasspath which would break get_system_package(). // Add new entry to linked list + if (!check_for_duplicates || !contains_entry(new_entry)) { - ClassLoaderExt::add_class_path_entry(path, check_for_duplicates, new_entry); + ClassLoaderExt::add_class_path_entry(path, check_for_duplicates, new_entry, prepend_entry); + if (mark_append_entry) { + set_first_append_entry(new_entry); + } } return true; } else { @@ -760,246 +856,205 @@ int ClassLoader::crc32(int crc, const char* buf, int len) { return (*Crc32)(crc, (const jbyte*)buf, len); } -// PackageInfo data exists in order to support the java.lang.Package -// class. A Package object provides information about a java package -// (version, vendor, etc.) which originates in the manifest of the jar -// file supplying the package. For application classes, the ClassLoader -// object takes care of this. - -// For system (boot) classes, the Java code in the Package class needs -// to be able to identify which source jar file contained the boot -// class, so that it can extract the manifest from it. This table -// identifies java packages with jar files in the boot classpath. - -// Because the boot classpath cannot change, the classpath index is -// sufficient to identify the source jar file or directory. (Since -// directories have no manifests, the directory name is not required, -// but is available.) - -// When using sharing -- the pathnames of entries in the boot classpath -// may not be the same at runtime as they were when the archive was -// created (NFS, Samba, etc.). The actual files and directories named -// in the classpath must be the same files, in the same order, even -// though the exact name is not the same. - -class PackageInfo: public BasicHashtableEntry { -public: - const char* _pkgname; // Package name - int _classpath_index; // Index of directory or JAR file loaded from - - PackageInfo* next() { - return (PackageInfo*)BasicHashtableEntry::next(); - } - - const char* pkgname() { return _pkgname; } - void set_pkgname(char* pkgname) { _pkgname = pkgname; } - - const char* filename() { - return ClassLoader::classpath_entry(_classpath_index)->name(); - } - - void set_index(int index) { - _classpath_index = index; - } -}; - - -class PackageHashtable : public BasicHashtable { -private: - inline unsigned int compute_hash(const char *s, int n) { - unsigned int val = 0; - while (--n >= 0) { - val = *s++ + 31 * val; - } - return val; - } - - PackageInfo* bucket(int index) { - return (PackageInfo*)BasicHashtable::bucket(index); - } - - PackageInfo* get_entry(int index, unsigned int hash, - const char* pkgname, size_t n) { - for (PackageInfo* pp = bucket(index); pp != NULL; pp = pp->next()) { - if (pp->hash() == hash && - strncmp(pkgname, pp->pkgname(), n) == 0 && - pp->pkgname()[n] == '\0') { - return pp; - } - } - return NULL; - } - -public: - PackageHashtable(int table_size) - : BasicHashtable(table_size, sizeof(PackageInfo)) {} - - PackageHashtable(int table_size, HashtableBucket* t, int number_of_entries) - : BasicHashtable(table_size, sizeof(PackageInfo), t, number_of_entries) {} - - PackageInfo* get_entry(const char* pkgname, int n) { - unsigned int hash = compute_hash(pkgname, n); - return get_entry(hash_to_index(hash), hash, pkgname, n); - } - - PackageInfo* new_entry(char* pkgname, int n) { - unsigned int hash = compute_hash(pkgname, n); - PackageInfo* pp; - pp = (PackageInfo*)BasicHashtable::new_entry(hash); - pp->set_pkgname(pkgname); - return pp; - } - - void add_entry(PackageInfo* pp) { - int index = hash_to_index(pp->hash()); - BasicHashtable::add_entry(index, pp); - } - - void copy_pkgnames(const char** packages) { - int n = 0; - for (int i = 0; i < table_size(); ++i) { - for (PackageInfo* pp = bucket(i); pp != NULL; pp = pp->next()) { - packages[n++] = pp->pkgname(); - } - } - assert(n == number_of_entries(), "just checking"); - } - - CDS_ONLY(void copy_table(char** top, char* end, PackageHashtable* table);) -}; - #if INCLUDE_CDS -void PackageHashtable::copy_table(char** top, char* end, - PackageHashtable* table) { - // Copy (relocate) the table to the shared space. - BasicHashtable::copy_table(top, end); - - // Calculate the space needed for the package name strings. - int i; - intptr_t* tableSize = (intptr_t*)(*top); - *top += sizeof(intptr_t); // For table size - char* tableStart = *top; - - for (i = 0; i < table_size(); ++i) { - for (PackageInfo* pp = table->bucket(i); - pp != NULL; - pp = pp->next()) { - int n1 = (int)(strlen(pp->pkgname()) + 1); - if (*top + n1 >= end) { - report_out_of_shared_space(SharedMiscData); +void ClassLoader::initialize_module_loader_map(JImageFile* jimage) { + jlong size; + JImageLocationRef location = (*JImageFindResource)(jimage, "java.base", get_jimage_version_string(), MODULE_LOADER_MAP, &size); + if (location == 0) { + vm_exit_during_initialization( + "Cannot find ModuleLoaderMap location from modules jimage.", NULL); + } + char* buffer = NEW_RESOURCE_ARRAY(char, size); + jlong read = (*JImageGetResource)(jimage, location, buffer, size); + if (read != size) { + vm_exit_during_initialization( + "Cannot find ModuleLoaderMap resource from modules jimage.", NULL); + } + char* char_buf = (char*)buffer; + int buflen = (int)strlen(char_buf); + char* begin_ptr = char_buf; + char* end_ptr = strchr(begin_ptr, '\n'); + bool process_boot_modules = false; + _boot_modules_array = new (ResourceObj::C_HEAP, mtInternal) + GrowableArray(INITIAL_BOOT_MODULES_ARRAY_SIZE, true); + _platform_modules_array = new (ResourceObj::C_HEAP, mtInternal) + GrowableArray(INITIAL_PLATFORM_MODULES_ARRAY_SIZE, true); + while (end_ptr != NULL && (end_ptr - char_buf) < buflen) { + // Allocate a buffer from the C heap to be appended to the _boot_modules_array + // or the _platform_modules_array. + char* temp_name = NEW_C_HEAP_ARRAY(char, (size_t)(end_ptr - begin_ptr + 1), mtInternal); + strncpy(temp_name, begin_ptr, end_ptr - begin_ptr); + temp_name[end_ptr - begin_ptr] = '\0'; + if (strncmp(temp_name, "BOOT", 4) == 0) { + process_boot_modules = true; + FREE_C_HEAP_ARRAY(char, temp_name); + } else if (strncmp(temp_name, "PLATFORM", 8) == 0) { + process_boot_modules = false; + FREE_C_HEAP_ARRAY(char, temp_name); + } else { + // module name + if (process_boot_modules) { + _boot_modules_array->append(temp_name); + } else { + _platform_modules_array->append(temp_name); } - pp->set_pkgname((char*)memcpy(*top, pp->pkgname(), n1)); - *top += n1; } + begin_ptr = ++end_ptr; + end_ptr = strchr(begin_ptr, '\n'); } - *top = (char*)align_size_up((intptr_t)*top, sizeof(HeapWord)); - if (*top >= end) { - report_out_of_shared_space(SharedMiscData); - } - - // Write table size - intptr_t len = *top - (char*)tableStart; - *tableSize = len; -} - - -void ClassLoader::copy_package_info_buckets(char** top, char* end) { - _package_hash_table->copy_buckets(top, end); -} - -void ClassLoader::copy_package_info_table(char** top, char* end) { - _package_hash_table->copy_table(top, end, _package_hash_table); + FREE_RESOURCE_ARRAY(u1, buffer, size); } #endif -PackageInfo* ClassLoader::lookup_package(const char *pkgname) { - const char *cp = strrchr(pkgname, '/'); +// Function add_package extracts the package from the fully qualified class name +// and checks if the package is in the boot loader's package entry table. If so, +// then it sets the classpath_index in the package entry record. +// +// The classpath_index field is used to find the entry on the boot loader class +// path for packages with classes loaded by the boot loader from -Xbootclasspath/a +// in an unnamed module. It is also used to indicate (for all packages whose +// classes are loaded by the boot loader) that at least one of the package's +// classes has been loaded. +bool ClassLoader::add_package(const char *fullq_class_name, s2 classpath_index, TRAPS) { + assert(fullq_class_name != NULL, "just checking"); + + // Get package name from fully qualified class name. + const char *cp = strrchr(fullq_class_name, '/'); if (cp != NULL) { - // Package prefix found - int n = cp - pkgname + 1; - return _package_hash_table->get_entry(pkgname, n); + int len = cp - fullq_class_name; + PackageEntryTable* pkg_entry_tbl = + ClassLoaderData::the_null_class_loader_data()->packages(); + TempNewSymbol pkg_symbol = + SymbolTable::new_symbol(fullq_class_name, len, CHECK_false); + PackageEntry* pkg_entry = pkg_entry_tbl->lookup_only(pkg_symbol); + if (pkg_entry != NULL) { + assert(classpath_index != -1, "Unexpected classpath_index"); + pkg_entry->set_classpath_index(classpath_index); + } else { + return false; + } + } + return true; +} + +oop ClassLoader::get_system_package(const char* name, TRAPS) { + // Look up the name in the boot loader's package entry table. + if (name != NULL) { + TempNewSymbol package_sym = SymbolTable::new_symbol(name, (int)strlen(name), CHECK_NULL); + // Look for the package entry in the boot loader's package entry table. + PackageEntry* package = + ClassLoaderData::the_null_class_loader_data()->packages()->lookup_only(package_sym); + + // Return NULL if package does not exist or if no classes in that package + // have been loaded. + if (package != NULL && package->has_loaded_class()) { + ModuleEntry* module = package->module(); + if (module->location() != NULL) { + ResourceMark rm(THREAD); + Handle ml = java_lang_String::create_from_str( + module->location()->as_C_string(), THREAD); + return ml(); + } + // Return entry on boot loader class path. + Handle cph = java_lang_String::create_from_str( + ClassLoader::classpath_entry(package->classpath_index())->name(), THREAD); + return cph(); + } } return NULL; } - -bool ClassLoader::add_package(const char *pkgname, int classpath_index, TRAPS) { - assert(pkgname != NULL, "just checking"); - // Bootstrap loader no longer holds system loader lock obj serializing - // load_instance_class and thereby add_package - { - MutexLocker ml(PackageTable_lock, THREAD); - // First check for previously loaded entry - PackageInfo* pp = lookup_package(pkgname); - if (pp != NULL) { - // Existing entry found, check source of package - pp->set_index(classpath_index); - return true; - } - - const char *cp = strrchr(pkgname, '/'); - if (cp != NULL) { - // Package prefix found - int n = cp - pkgname + 1; - - char* new_pkgname = NEW_C_HEAP_ARRAY(char, n + 1, mtClass); - if (new_pkgname == NULL) { - return false; - } - - memcpy(new_pkgname, pkgname, n); - new_pkgname[n] = '\0'; - pp = _package_hash_table->new_entry(new_pkgname, n); - pp->set_index(classpath_index); - - // Insert into hash table - _package_hash_table->add_entry(pp); - } - return true; - } -} - - -oop ClassLoader::get_system_package(const char* name, TRAPS) { - PackageInfo* pp; - { - MutexLocker ml(PackageTable_lock, THREAD); - pp = lookup_package(name); - } - if (pp == NULL) { - return NULL; - } else { - Handle p = java_lang_String::create_from_str(pp->filename(), THREAD); - return p(); - } -} - - objArrayOop ClassLoader::get_system_packages(TRAPS) { ResourceMark rm(THREAD); - int nof_entries; - const char** packages; + // List of pointers to PackageEntrys that have loaded classes. + GrowableArray* loaded_class_pkgs = new GrowableArray(50); { - MutexLocker ml(PackageTable_lock, THREAD); - // Allocate resource char* array containing package names - nof_entries = _package_hash_table->number_of_entries(); - if ((packages = NEW_RESOURCE_ARRAY(const char*, nof_entries)) == NULL) { - return NULL; + MutexLocker ml(Module_lock, THREAD); + + PackageEntryTable* pe_table = + ClassLoaderData::the_null_class_loader_data()->packages(); + + // Collect the packages that have at least one loaded class. + for (int x = 0; x < pe_table->table_size(); x++) { + for (PackageEntry* package_entry = pe_table->bucket(x); + package_entry != NULL; + package_entry = package_entry->next()) { + if (package_entry->has_loaded_class()) { + loaded_class_pkgs->append(package_entry); + } + } } - _package_hash_table->copy_pkgnames(packages); - } - // Allocate objArray and fill with java.lang.String - objArrayOop r = oopFactory::new_objArray(SystemDictionary::String_klass(), - nof_entries, CHECK_0); - objArrayHandle result(THREAD, r); - for (int i = 0; i < nof_entries; i++) { - Handle str = java_lang_String::create_from_str(packages[i], CHECK_0); - result->obj_at_put(i, str()); } + + // Allocate objArray and fill with java.lang.String + objArrayOop r = oopFactory::new_objArray(SystemDictionary::String_klass(), + loaded_class_pkgs->length(), CHECK_NULL); + objArrayHandle result(THREAD, r); + for (int x = 0; x < loaded_class_pkgs->length(); x++) { + PackageEntry* package_entry = loaded_class_pkgs->at(x); + Handle str = java_lang_String::create_from_symbol(package_entry->name(), CHECK_NULL); + result->obj_at_put(x, str()); + } return result(); } +#if INCLUDE_CDS +s2 ClassLoader::module_to_classloader(const char* module_name) { + + assert(_boot_modules_array != NULL, "_boot_modules_array is NULL"); + assert(_platform_modules_array != NULL, "_platform_modules_array is NULL"); + + int array_size = _boot_modules_array->length(); + for (int i = 0; i < array_size; i++) { + if (strcmp(module_name, _boot_modules_array->at(i)) == 0) { + return BOOT_LOADER; + } + } + + array_size = _platform_modules_array->length(); + for (int i = 0; i < array_size; i++) { + if (strcmp(module_name, _platform_modules_array->at(i)) == 0) { + return PLATFORM_LOADER; + } + } + + return APP_LOADER; +} +#endif + +s2 ClassLoader::classloader_type(Symbol* class_name, ClassPathEntry* e, + int classpath_index, TRAPS) { +#if INCLUDE_CDS + // obtain the classloader type based on the class name. + // First obtain the package name based on the class name. Then obtain + // the classloader type based on the package name from the jimage using + // a jimage API. If the classloader type cannot be found from the + // jimage, it is determined by the class path entry. + jshort loader_type = ClassLoader::APP_LOADER; + if (e->is_jrt()) { + int length = 0; + const jbyte* pkg_string = InstanceKlass::package_from_name(class_name, length); + if (pkg_string != NULL) { + ResourceMark rm; + TempNewSymbol pkg_name = SymbolTable::new_symbol((const char*)pkg_string, length, THREAD); + const char* pkg_name_C_string = (const char*)(pkg_name->as_C_string()); + ClassPathImageEntry* cpie = (ClassPathImageEntry*)e; + JImageFile* jimage = cpie->jimage(); + char* module_name = (char*)(*JImagePackageToModule)(jimage, pkg_name_C_string); + if (module_name != NULL) { + loader_type = ClassLoader::module_to_classloader(module_name); + } + } + } else if (ClassLoaderExt::is_boot_classpath(classpath_index)) { + loader_type = ClassLoader::BOOT_LOADER; + } + return loader_type; +#endif + return ClassLoader::BOOT_LOADER; // the classloader type is ignored in non-CDS cases +} + + // caller needs ResourceMark const char* ClassLoader::file_name_for_class_name(const char* class_name, int class_name_len) { @@ -1018,7 +1073,7 @@ const char* ClassLoader::file_name_for_class_name(const char* class_name, return file_name; } -instanceKlassHandle ClassLoader::load_class(Symbol* name, TRAPS) { +instanceKlassHandle ClassLoader::load_class(Symbol* name, bool search_append_only, TRAPS) { assert(name != NULL, "invariant"); assert(THREAD->is_Java_thread(), "must be a JavaThread"); @@ -1037,24 +1092,54 @@ instanceKlassHandle ClassLoader::load_class(Symbol* name, TRAPS) { ClassLoaderExt::Context context(class_name, file_name, THREAD); - // Lookup stream + // Lookup stream for parsing .class file ClassFileStream* stream = NULL; - int classpath_index = 0; - ClassPathEntry* e = _first_entry; - { - PerfClassTraceTime vmtimer(perf_sys_class_lookup_time(), - ((JavaThread*)THREAD)->get_thread_stat()->perf_timers_addr(), - PerfClassTraceTime::CLASS_LOAD); + s2 classpath_index = 0; - for (; e != NULL; e = e->next(), ++classpath_index) { - stream = e->open_stream(file_name, CHECK_NULL); - if (NULL == stream) { - continue; + // If DumpSharedSpaces is true, boot loader visibility boundaries are set + // to be _first_entry to the end (all path entries). + // + // If search_append_only is true, boot loader visibility boundaries are + // set to be _fist_append_entry to the end. This includes: + // [-Xbootclasspath/a]; [jvmti appended entries] + // + // If both DumpSharedSpaces and search_append_only are false, boot loader + // visibility boundaries are set to be _first_entry to the entry before + // the _first_append_entry. This would include: + // [-Xpatch:]; [exploded build | modules] + // + // DumpSharedSpaces and search_append_only are mutually exclusive and cannot + // be true at the same time. + ClassPathEntry* e = (search_append_only ? _first_append_entry : _first_entry); + ClassPathEntry* last_e = + (search_append_only || DumpSharedSpaces ? NULL : _first_append_entry); + + { + if (search_append_only) { + // For the boot loader append path search, must calculate + // the starting classpath_index prior to attempting to + // load the classfile. + ClassPathEntry *tmp_e = _first_entry; + while ((tmp_e != NULL) && (tmp_e != _first_append_entry)) { + tmp_e = tmp_e->next(); + ++classpath_index; } + } + + // Attempt to load the classfile from either: + // - [-Xpatch:dir]; exploded build | modules + // or + // - [-Xbootclasspath/a]; [jvmti appended entries] + while ((e != NULL) && (e != last_e)) { + stream = e->open_stream(file_name, CHECK_NULL); if (!context.check(stream, classpath_index)) { return NULL; } - break; + if (NULL != stream) { + break; + } + e = e->next(); + ++classpath_index; } } @@ -1085,32 +1170,16 @@ instanceKlassHandle ClassLoader::load_class(Symbol* name, TRAPS) { return NULL; } - return context.record_result(classpath_index, e, result, THREAD); + jshort loader_type = classloader_type(name, e, classpath_index, CHECK_NULL); + return context.record_result(classpath_index, loader_type, e, result, THREAD); } -void ClassLoader::create_package_info_table(HashtableBucket *t, int length, - int number_of_entries) { - assert(_package_hash_table == NULL, "One package info table allowed."); - assert(length == package_hash_table_size * sizeof(HashtableBucket), - "bad shared package info size."); - _package_hash_table = new PackageHashtable(package_hash_table_size, t, - number_of_entries); -} - - -void ClassLoader::create_package_info_table() { - assert(_package_hash_table == NULL, "shouldn't have one yet"); - _package_hash_table = new PackageHashtable(package_hash_table_size); -} - - // Initialize the class loader's access to methods in libzip. Parse and // process the boot classpath into a list ClassPathEntry objects. Once // this list has been created, it must not change order (see class PackageInfo) // it can be appended to and is by jvmti and the kernel vm. void ClassLoader::initialize() { - assert(_package_hash_table == NULL, "should have been initialized by now."); EXCEPTION_MARK; if (UsePerfData) { @@ -1258,12 +1327,48 @@ bool ClassLoader::get_canonical_path(const char* orig, char* out, int len) { return true; } -#ifndef PRODUCT +void ClassLoader::create_javabase() { + Thread* THREAD = Thread::current(); -void ClassLoader::verify() { - _package_hash_table->verify(); + // Create java.base's module entry for the boot + // class loader prior to loading j.l.Ojbect. + ClassLoaderData* null_cld = ClassLoaderData::the_null_class_loader_data(); + + // Get module entry table + ModuleEntryTable* null_cld_modules = null_cld->modules(); + if (null_cld_modules == NULL) { + vm_exit_during_initialization("No ModuleEntryTable for the boot class loader"); + } + + { + MutexLocker ml(Module_lock, THREAD); + ModuleEntry* jb_module = null_cld_modules->locked_create_entry_or_null(Handle(NULL), vmSymbols::java_base(), NULL, NULL, null_cld); + if (jb_module == NULL) { + vm_exit_during_initialization("Unable to create ModuleEntry for java.base"); + } + ModuleEntryTable::set_javabase_module(jb_module); + } + + // When looking for the jimage file, only + // search the boot loader's module path which + // can consist of [-Xpatch]; exploded build | modules + // Do not search the boot loader's append path. + ClassPathEntry* e = _first_entry; + ClassPathEntry* last_e = _first_append_entry; + while ((e != NULL) && (e != last_e)) { + JImageFile *jimage = e->jimage(); + if (jimage != NULL && e->is_jrt()) { + set_has_jimage(true); +#if INCLUDE_CDS + ClassLoader::initialize_module_loader_map(jimage); +#endif + return; + } + e = e->next(); + } } +#ifndef PRODUCT // CompileTheWorld // @@ -1325,10 +1430,6 @@ void ClassPathDirEntry::compile_the_world(Handle loader, TRAPS) { tty->cr(); } -bool ClassPathDirEntry::is_jrt() { - return false; -} - void ClassPathZipEntry::compile_the_world(Handle loader, TRAPS) { real_jzfile* zip = (real_jzfile*) _zip; tty->print_cr("CompileTheWorld : Compiling all classes in %s", zip->name); @@ -1350,10 +1451,6 @@ void ClassPathZipEntry::compile_the_world(Handle loader, TRAPS) { } } -bool ClassPathZipEntry::is_jrt() { - return false; -} - void ClassLoader::compile_the_world() { EXCEPTION_MARK; HandleMark hm(THREAD); @@ -1366,7 +1463,7 @@ void ClassLoader::compile_the_world() { ClassPathEntry* e = _first_entry; jlong start = os::javaTimeMillis(); while (e != NULL) { - // We stop at bootmodules.jimage, unless it is the first bootstrap path entry + // We stop at "modules" jimage, unless it is the first bootstrap path entry if (e->is_jrt() && e != _first_entry) break; e->compile_the_world(system_class_loader, CATCH); e = e->next(); diff --git a/hotspot/src/share/vm/classfile/classLoader.hpp b/hotspot/src/share/vm/classfile/classLoader.hpp index c2a68c04b09..4d5d2c9231c 100644 --- a/hotspot/src/share/vm/classfile/classLoader.hpp +++ b/hotspot/src/share/vm/classfile/classLoader.hpp @@ -33,8 +33,17 @@ // The VM class loader. #include -// Name of boot module image -#define BOOT_IMAGE_NAME "bootmodules.jimage" +// Name of boot "modules" image +#define MODULES_IMAGE_NAME "modules" + +// Name of the resource containing mapping from module names to defining class loader type +#define MODULE_LOADER_MAP "jdk/internal/vm/cds/resources/ModuleLoaderMap.dat" + +// Initial sizes of the following arrays are based on the generated ModuleLoaderMap.dat +#define INITIAL_BOOT_MODULES_ARRAY_SIZE 30 +#define INITIAL_PLATFORM_MODULES_ARRAY_SIZE 15 + +// Class path entry (directory or zip file) class JImageFile; class ClassFileStream; @@ -49,6 +58,7 @@ public: // may have unlocked readers, so write atomically. OrderAccess::release_store_ptr(&_next, next); } + virtual bool is_jrt() = 0; virtual bool is_jar_file() const = 0; virtual const char* name() const = 0; virtual JImageFile* jimage() const = 0; @@ -59,13 +69,13 @@ public: virtual ClassFileStream* open_stream(const char* name, TRAPS) = 0; // Debugging NOT_PRODUCT(virtual void compile_the_world(Handle loader, TRAPS) = 0;) - NOT_PRODUCT(virtual bool is_jrt() = 0;) }; class ClassPathDirEntry: public ClassPathEntry { private: const char* _dir; // Name of directory public: + bool is_jrt() { return false; } bool is_jar_file() const { return false; } const char* name() const { return _dir; } JImageFile* jimage() const { return NULL; } @@ -73,7 +83,6 @@ class ClassPathDirEntry: public ClassPathEntry { ClassFileStream* open_stream(const char* name, TRAPS); // Debugging NOT_PRODUCT(void compile_the_world(Handle loader, TRAPS);) - NOT_PRODUCT(bool is_jrt();) }; @@ -96,6 +105,7 @@ class ClassPathZipEntry: public ClassPathEntry { jzfile* _zip; // The zip archive const char* _zip_name; // Name of zip archive public: + bool is_jrt() { return false; } bool is_jar_file() const { return true; } const char* name() const { return _zip_name; } JImageFile* jimage() const { return NULL; } @@ -106,7 +116,6 @@ class ClassPathZipEntry: public ClassPathEntry { void contents_do(void f(const char* name, void* context), void* context); // Debugging NOT_PRODUCT(void compile_the_world(Handle loader, TRAPS);) - NOT_PRODUCT(bool is_jrt();) }; @@ -116,29 +125,28 @@ private: JImageFile* _jimage; const char* _name; public: + bool is_jrt(); bool is_jar_file() const { return false; } bool is_open() const { return _jimage != NULL; } const char* name() const { return _name == NULL ? "" : _name; } JImageFile* jimage() const { return _jimage; } ClassPathImageEntry(JImageFile* jimage, const char* name); ~ClassPathImageEntry(); - static void name_to_package(const char* name, char* buffer, int length); + void name_to_package(const char* name, char* package, int length); ClassFileStream* open_stream(const char* name, TRAPS); // Debugging NOT_PRODUCT(void compile_the_world(Handle loader, TRAPS);) - NOT_PRODUCT(bool is_jrt();) }; -class PackageHashtable; -class PackageInfo; class SharedPathsMiscInfo; -template class HashtableBucket; class ClassLoader: AllStatic { public: - enum SomeConstants { - package_hash_table_size = 31 // Number of buckets + enum ClassLoaderType { + BOOT_LOADER = 1, /* boot loader */ + PLATFORM_LOADER = 2, /* PlatformClassLoader */ + APP_LOADER = 3 /* AppClassLoader */ }; protected: @@ -177,41 +185,60 @@ class ClassLoader: AllStatic { static PerfCounter* _isUnsyncloadClass; static PerfCounter* _load_instance_class_failCounter; - // First entry in linked list of ClassPathEntry instances + // First entry in linked list of ClassPathEntry instances. + // This consists of entries made up by: + // - boot loader modules + // [-Xpatch]; exploded build | modules; + // - boot loader append path + // [-Xbootclasspath/a]; [jvmti appended entries] static ClassPathEntry* _first_entry; // Last entry in linked list of ClassPathEntry instances static ClassPathEntry* _last_entry; static int _num_entries; - // Hash table used to keep track of loaded packages - static PackageHashtable* _package_hash_table; + // Pointer into the linked list of ClassPathEntry instances. + // Marks the start of: + // - the boot loader's append path + // [-Xbootclasspath/a]; [jvmti appended entries] + static ClassPathEntry* _first_append_entry; + static const char* _shared_archive; + // True if the boot path has a "modules" jimage + static bool _has_jimage; + + // Array of module names associated with the boot class loader + CDS_ONLY(static GrowableArray* _boot_modules_array;) + + // Array of module names associated with the platform class loader + CDS_ONLY(static GrowableArray* _platform_modules_array;) + // Info used by CDS CDS_ONLY(static SharedPathsMiscInfo * _shared_paths_misc_info;) - // Hash function - static unsigned int hash(const char *s, int n); - // Returns the package file name corresponding to the specified package - // or class name, or null if not found. - static PackageInfo* lookup_package(const char *pkgname); - // Adds a new package entry for the specified class or package name and - // corresponding directory or jar file name. - static bool add_package(const char *pkgname, int classpath_index, TRAPS); - // Initialization static void setup_bootstrap_search_path(); - static void setup_search_path(const char *class_path); + static void setup_search_path(const char *class_path, bool setting_bootstrap); static void load_zip_library(); static void load_jimage_library(); static ClassPathEntry* create_class_path_entry(const char *path, const struct stat* st, bool throw_exception, TRAPS); + public: + + // If the package for the fully qualified class name is in the boot + // loader's package entry table then add_package() sets the classpath_index + // field so that get_system_package() will know to return a non-null value + // for the package's location. And, so that the package will be added to + // the list of packages returned by get_system_packages(). + // For packages whose classes are loaded from the boot loader class path, the + // classpath_index indicates which entry on the boot loader class path. + static bool add_package(const char *fullq_class_name, s2 classpath_index, TRAPS); + // Canonicalizes path names, so strcmp will work properly. This is mainly // to avoid confusing the zip library static bool get_canonical_path(const char* orig, char* out, int len); - static const char* file_name_for_class_name(const char* class_name, int class_name_len); @@ -220,6 +247,8 @@ class ClassLoader: AllStatic { static int crc32(int crc, const char* buf, int len); static bool update_class_path_entry_list(const char *path, bool check_for_duplicates, + bool mark_append_entry, + bool prepend_entry, bool throw_exception=true); static void print_bootclasspath(); @@ -284,8 +313,18 @@ class ClassLoader: AllStatic { return _load_instance_class_failCounter; } + // Sets _has_jimage to TRUE if "modules" jimage file exists + static void set_has_jimage(bool val) { + _has_jimage = val; + } + + static bool has_jimage() { return _has_jimage; } + + // Create the ModuleEntry for java.base + static void create_javabase(); + // Load individual .class file - static instanceKlassHandle load_class(Symbol* class_name, TRAPS); + static instanceKlassHandle load_class(Symbol* class_name, bool search_append_only, TRAPS); // If the specified package has been loaded by the system, then returns // the name of the directory or ZIP file that the package was loaded from. @@ -304,9 +343,7 @@ class ClassLoader: AllStatic { // Initialization static void initialize(); CDS_ONLY(static void initialize_shared_path();) - static void create_package_info_table(); - static void create_package_info_table(HashtableBucket *t, int length, - int number_of_entries); + static int compute_Object_vtable(); static ClassPathEntry* classpath_entry(int n) { @@ -320,8 +357,6 @@ class ClassLoader: AllStatic { #if INCLUDE_CDS // Sharing dump and restore - static void copy_package_info_buckets(char** top, char* end); - static void copy_package_info_table(char** top, char* end); static void check_shared_classpath(const char *path); static void finalize_shared_paths_misc_info(); @@ -329,7 +364,12 @@ class ClassLoader: AllStatic { static void* get_shared_paths_misc_info(); static bool check_shared_paths_misc_info(void* info, int size); static void exit_with_path_failure(const char* error, const char* message); + + static s2 module_to_classloader(const char* module_name); + static void initialize_module_loader_map(JImageFile* jimage); #endif + static s2 classloader_type(Symbol* class_name, ClassPathEntry* e, + int classpath_index, TRAPS); static void trace_class_path(const char* msg, const char* name = NULL); @@ -342,15 +382,30 @@ class ClassLoader: AllStatic { static jlong class_link_count(); static jlong class_link_time_ms(); + static void set_first_append_entry(ClassPathEntry* entry); + // indicates if class path already contains a entry (exact match by name) static bool contains_entry(ClassPathEntry* entry); // adds a class path list static void add_to_list(ClassPathEntry* new_entry); + // prepends a class path list + static void prepend_to_list(ClassPathEntry* new_entry); + // creates a class path zip entry (returns NULL if JAR file cannot be opened) static ClassPathZipEntry* create_class_path_zip_entry(const char *apath); + // add a path to class path list + static void add_to_list(const char* apath); + + // prepend a path to class path list + static void prepend_to_list(const char* apath); + + static bool string_ends_with(const char* str, const char* str_to_find); + + static bool is_jrt(const char* name) { return string_ends_with(name, MODULES_IMAGE_NAME); } + // Debugging static void verify() PRODUCT_RETURN; diff --git a/hotspot/src/share/vm/classfile/classLoaderData.cpp b/hotspot/src/share/vm/classfile/classLoaderData.cpp index 9543a1fac7b..f3278a7f8de 100644 --- a/hotspot/src/share/vm/classfile/classLoaderData.cpp +++ b/hotspot/src/share/vm/classfile/classLoaderData.cpp @@ -31,7 +31,7 @@ // // Class loaders that implement a deterministic name resolution strategy // (including with respect to their delegation behavior), such as the boot, the -// extension, and the system loaders of the JDK's built-in class loader +// platform, and the system loaders of the JDK's built-in class loader // hierarchy, always produce the same linkset for a given configuration. // // ClassLoaderData carries information related to a linkset (e.g., @@ -51,6 +51,8 @@ #include "classfile/classLoaderData.inline.hpp" #include "classfile/javaClasses.hpp" #include "classfile/metadataOnStackMark.hpp" +#include "classfile/moduleEntry.hpp" +#include "classfile/packageEntry.hpp" #include "classfile/systemDictionary.hpp" #include "code/codeCache.hpp" #include "gc/shared/gcLocker.hpp" @@ -83,6 +85,7 @@ ClassLoaderData::ClassLoaderData(Handle h_class_loader, bool is_anonymous, Depen // The null-class-loader should always be kept alive. _keep_alive(is_anonymous || h_class_loader.is_null()), _metaspace(NULL), _unloading(false), _klasses(NULL), + _modules(NULL), _packages(NULL), _claimed(0), _jmethod_ids(NULL), _handles(NULL), _deallocate_list(NULL), _next(NULL), _dependencies(dependencies), _shared_class_loader_id(-1), _metaspace_lock(new Mutex(Monitor::leaf+1, "Metaspace allocation lock", true, @@ -168,6 +171,30 @@ void ClassLoaderData::classes_do(void f(InstanceKlass*)) { } } +void ClassLoaderData::modules_do(void f(ModuleEntry*)) { + if (_modules != NULL) { + for (int i = 0; i < _modules->table_size(); i++) { + for (ModuleEntry* entry = _modules->bucket(i); + entry != NULL; + entry = entry->next()) { + f(entry); + } + } + } +} + +void ClassLoaderData::packages_do(void f(PackageEntry*)) { + if (_packages != NULL) { + for (int i = 0; i < _packages->table_size(); i++) { + for (PackageEntry* entry = _packages->bucket(i); + entry != NULL; + entry = entry->next()) { + f(entry); + } + } + } +} + void ClassLoaderData::record_dependency(const Klass* k, TRAPS) { assert(k != NULL, "invariant"); @@ -341,6 +368,46 @@ void ClassLoaderData::unload() { } } +PackageEntryTable* ClassLoaderData::packages() { + // Lazily create the package entry table at first request. + if (_packages == NULL) { + MutexLockerEx m1(metaspace_lock(), Mutex::_no_safepoint_check_flag); + // Check again if _packages has been allocated while we were getting this lock. + if (_packages != NULL) { + return _packages; + } + // Ensure _packages is stable, since it is examined without a lock + OrderAccess::storestore(); + _packages = new PackageEntryTable(PackageEntryTable::_packagetable_entry_size); + } + return _packages; +} + +ModuleEntryTable* ClassLoaderData::modules() { + // Lazily create the module entry table at first request. + if (_modules == NULL) { + MutexLocker m1(Module_lock); + // Check again if _modules has been allocated while we were getting this lock. + if (_modules != NULL) { + return _modules; + } + + ModuleEntryTable* temp_table = new ModuleEntryTable(ModuleEntryTable::_moduletable_entry_size); + // Each loader has one unnamed module entry. Create it before + // any classes, loaded by this loader, are defined in case + // they end up being defined in loader's unnamed module. + temp_table->create_unnamed_module(this); + + { + MutexLockerEx m1(metaspace_lock(), Mutex::_no_safepoint_check_flag); + // Ensure _modules is stable, since it is examined without a lock + OrderAccess::storestore(); + _modules = temp_table; + } + } + return _modules; +} + oop ClassLoaderData::keep_alive_object() const { assert(!keep_alive(), "Don't use with CLDs that are artificially kept alive"); return is_anonymous() ? _klasses->java_mirror() : class_loader(); @@ -358,16 +425,30 @@ ClassLoaderData::~ClassLoaderData() { // Release C heap structures for all the classes. classes_do(InstanceKlass::release_C_heap_structures); + // Release C heap allocated hashtable for all the packages. + if (_packages != NULL) { + // Destroy the table itself + delete _packages; + _packages = NULL; + } + + // Release C heap allocated hashtable for all the modules. + if (_modules != NULL) { + // Destroy the table itself + delete _modules; + _modules = NULL; + } + + // release the metaspace Metaspace *m = _metaspace; if (m != NULL) { _metaspace = NULL; - // release the metaspace delete m; - // release the handles - if (_handles != NULL) { - JNIHandleBlock::release_block(_handles); - _handles = NULL; - } + } + // release the handles + if (_handles != NULL) { + JNIHandleBlock::release_block(_handles); + _handles = NULL; } // Clear all the JNI handles for methods @@ -389,10 +470,10 @@ ClassLoaderData::~ClassLoaderData() { } /** - * Returns true if this class loader data is for the extension class loader. + * Returns true if this class loader data is for the platform class loader. */ -bool ClassLoaderData::is_ext_class_loader_data() const { - return SystemDictionary::is_ext_class_loader(class_loader()); +bool ClassLoaderData::is_platform_class_loader_data() const { + return SystemDictionary::is_platform_class_loader(class_loader()); } Metaspace* ClassLoaderData::metaspace_non_null() { @@ -438,6 +519,10 @@ jobject ClassLoaderData::add_handle(Handle h) { return handles()->allocate_handle(h()); } +void ClassLoaderData::remove_handle(jobject h) { + _handles->release_handle(h); +} + // Add this metadata pointer to be freed when it's safe. This is only during // class unloading because Handles might point to this metadata field. void ClassLoaderData::add_to_deallocate_list(Metadata* m) { @@ -712,6 +797,40 @@ void ClassLoaderDataGraph::methods_do(void f(Method*)) { } } +void ClassLoaderDataGraph::modules_do(void f(ModuleEntry*)) { + assert_locked_or_safepoint(Module_lock); + for (ClassLoaderData* cld = _head; cld != NULL; cld = cld->next()) { + cld->modules_do(f); + } +} + +void ClassLoaderDataGraph::modules_unloading_do(void f(ModuleEntry*)) { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint!"); + // Only walk the head until any clds not purged from prior unloading + // (CMS doesn't purge right away). + for (ClassLoaderData* cld = _unloading; cld != _saved_unloading; cld = cld->next()) { + assert(cld->is_unloading(), "invariant"); + cld->modules_do(f); + } +} + +void ClassLoaderDataGraph::packages_do(void f(PackageEntry*)) { + assert_locked_or_safepoint(Module_lock); + for (ClassLoaderData* cld = _head; cld != NULL; cld = cld->next()) { + cld->packages_do(f); + } +} + +void ClassLoaderDataGraph::packages_unloading_do(void f(PackageEntry*)) { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint!"); + // Only walk the head until any clds not purged from prior unloading + // (CMS doesn't purge right away). + for (ClassLoaderData* cld = _unloading; cld != _saved_unloading; cld = cld->next()) { + assert(cld->is_unloading(), "invariant"); + cld->packages_do(f); + } +} + void ClassLoaderDataGraph::loaded_classes_do(KlassClosure* klass_closure) { for (ClassLoaderData* cld = _head; cld != NULL; cld = cld->next()) { cld->loaded_classes_do(klass_closure); @@ -723,6 +842,7 @@ void ClassLoaderDataGraph::classes_unloading_do(void f(Klass* const)) { // Only walk the head until any clds not purged from prior unloading // (CMS doesn't purge right away). for (ClassLoaderData* cld = _unloading; cld != _saved_unloading; cld = cld->next()) { + assert(cld->is_unloading(), "invariant"); cld->classes_do(f); } } @@ -800,6 +920,12 @@ bool ClassLoaderDataGraph::do_unloading(BoolObjectClosure* is_alive_closure, data = _head; while (data != NULL) { if (data->is_alive(is_alive_closure)) { + if (data->packages_defined()) { + data->packages()->purge_all_package_exports(); + } + if (data->modules_defined()) { + data->modules()->purge_all_module_reads(); + } // clean metaspace if (walk_all_metadata) { data->classes_do(InstanceKlass::purge_previous_versions); @@ -992,6 +1118,7 @@ void ClassLoaderData::print_value_on(outputStream* out) const { Ticks ClassLoaderDataGraph::_class_unload_time; void ClassLoaderDataGraph::class_unload_event(Klass* const k) { + assert(k != NULL, "invariant"); // post class unload event EventClassUnload event(UNTIMED); diff --git a/hotspot/src/share/vm/classfile/classLoaderData.hpp b/hotspot/src/share/vm/classfile/classLoaderData.hpp index f6e82fa12bc..4b6d24722d3 100644 --- a/hotspot/src/share/vm/classfile/classLoaderData.hpp +++ b/hotspot/src/share/vm/classfile/classLoaderData.hpp @@ -53,6 +53,10 @@ class ClassLoaderData; class JNIMethodBlock; class JNIHandleBlock; class Metadebug; +class ModuleEntry; +class PackageEntry; +class ModuleEntryTable; +class PackageEntryTable; // GC root for walking class loader data created @@ -92,6 +96,10 @@ class ClassLoaderDataGraph : public AllStatic { static void classes_do(KlassClosure* klass_closure); static void classes_do(void f(Klass* const)); static void methods_do(void f(Method*)); + static void modules_do(void f(ModuleEntry*)); + static void modules_unloading_do(void f(ModuleEntry*)); + static void packages_do(void f(PackageEntry*)); + static void packages_unloading_do(void f(PackageEntry*)); static void loaded_classes_do(KlassClosure* klass_closure); static void classes_unloading_do(void f(Klass* const)); static bool do_unloading(BoolObjectClosure* is_alive, bool clean_previous_versions); @@ -172,9 +180,12 @@ class ClassLoaderData : public CHeapObj { volatile int _claimed; // true if claimed, for example during GC traces. // To avoid applying oop closure more than once. // Has to be an int because we cas it. - Klass* _klasses; // The classes defined by the class loader. + JNIHandleBlock* _handles; // Handles to constant pool arrays, Modules, etc, which + // have the same life cycle of the corresponding ClassLoader. - JNIHandleBlock* _handles; // Handles to constant pool arrays + Klass* _klasses; // The classes defined by the class loader. + PackageEntryTable* _packages; // The packages defined by the class loader. + ModuleEntryTable* _modules; // The modules defined by the class loader. // These method IDs are created for the class loader and set to NULL when the // class loader is unloaded. They are rarely freed, only for redefine classes @@ -218,6 +229,8 @@ class ClassLoaderData : public CHeapObj { void loaded_classes_do(KlassClosure* klass_closure); void classes_do(void f(InstanceKlass*)); void methods_do(void f(Method*)); + void modules_do(void f(ModuleEntry*)); + void packages_do(void f(PackageEntry*)); // Deallocate free list during class unloading. void free_deallocate_list(); @@ -256,7 +269,7 @@ class ClassLoaderData : public CHeapObj { bool is_the_null_class_loader_data() const { return this == _the_null_class_loader_data; } - bool is_ext_class_loader_data() const; + bool is_platform_class_loader_data() const; // The Metaspace is created lazily so may be NULL. This // method will allocate a Metaspace if needed. @@ -293,11 +306,16 @@ class ClassLoaderData : public CHeapObj { const char* loader_name(); jobject add_handle(Handle h); + void remove_handle(jobject h); void add_class(Klass* k, bool publicize = true); void remove_class(Klass* k); bool contains_klass(Klass* k); void record_dependency(const Klass* to, TRAPS); void init_dependencies(TRAPS); + PackageEntryTable* packages(); + bool packages_defined() { return (_packages != NULL); } + ModuleEntryTable* modules(); + bool modules_defined() { return (_modules != NULL); } void add_to_deallocate_list(Metadata* m); diff --git a/hotspot/src/share/vm/classfile/classLoaderExt.hpp b/hotspot/src/share/vm/classfile/classLoaderExt.hpp index 990a8b61f04..7fa11b2f614 100644 --- a/hotspot/src/share/vm/classfile/classLoaderExt.hpp +++ b/hotspot/src/share/vm/classfile/classLoaderExt.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2014, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -49,12 +49,14 @@ public: return false; } - instanceKlassHandle record_result(const int classpath_index, + instanceKlassHandle record_result(const s2 classpath_index, + const jshort classloader_type, const ClassPathEntry* e, instanceKlassHandle result, TRAPS) { if (ClassLoader::add_package(_file_name, classpath_index, THREAD)) { if (DumpSharedSpaces) { result->set_shared_classpath_index(classpath_index); + result->set_class_loader_type(classloader_type); } return result; } else { @@ -65,13 +67,27 @@ public: static void add_class_path_entry(const char* path, bool check_for_duplicates, - ClassPathEntry* new_entry) { - ClassLoader::add_to_list(new_entry); + ClassPathEntry* new_entry, bool prepend_entry) { + if (prepend_entry) { + ClassLoader::prepend_to_list(new_entry); + } else { + ClassLoader::add_to_list(new_entry); + } } static void append_boot_classpath(ClassPathEntry* new_entry) { ClassLoader::add_to_list(new_entry); + // During jvmti live phase an entry can be appended to the boot + // loader's ClassPathEntry instances. Need to mark the start + // of the boot loader's append path in case there was no reason + // to mark it initially in setup_bootstrap_search_path. + if (ClassLoader::_first_append_entry == NULL) { + ClassLoader::set_first_append_entry(new_entry); + } } static void setup_search_paths() {} + static bool is_boot_classpath(int classpath_index) { + return true; + } static Klass* load_one_class(ClassListParser* parser, TRAPS); }; diff --git a/hotspot/src/share/vm/classfile/javaClasses.cpp b/hotspot/src/share/vm/classfile/javaClasses.cpp index 2df6a47088c..a69b87e22d4 100644 --- a/hotspot/src/share/vm/classfile/javaClasses.cpp +++ b/hotspot/src/share/vm/classfile/javaClasses.cpp @@ -24,7 +24,9 @@ #include "precompiled.hpp" #include "classfile/altHashing.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/javaClasses.inline.hpp" +#include "classfile/moduleEntry.hpp" #include "classfile/stringTable.hpp" #include "classfile/vmSymbols.hpp" #include "code/debugInfo.hpp" @@ -768,7 +770,7 @@ void java_lang_Class::fixup_mirror(KlassHandle k, TRAPS) { } } } - create_mirror(k, Handle(NULL), Handle(NULL), CHECK); + create_mirror(k, Handle(NULL), Handle(NULL), Handle(NULL), CHECK); } void java_lang_Class::initialize_mirror_fields(KlassHandle k, @@ -789,7 +791,7 @@ void java_lang_Class::initialize_mirror_fields(KlassHandle k, } void java_lang_Class::create_mirror(KlassHandle k, Handle class_loader, - Handle protection_domain, TRAPS) { + Handle module, Handle protection_domain, TRAPS) { assert(k->java_mirror() == NULL, "should only assign mirror once"); // Use this moment of initialization to cache modifier_flags also, // to support Class.getModifiers(). Instance classes recalculate @@ -849,11 +851,25 @@ void java_lang_Class::create_mirror(KlassHandle k, Handle class_loader, assert(class_loader() == k->class_loader(), "should be same"); set_class_loader(mirror(), class_loader()); + // set the module field in the java_lang_Class instance + // This may be null during bootstrap but will get fixed up later on. + set_module(mirror(), module()); + // Setup indirection from klass->mirror last // after any exceptions can happen during allocations. if (!k.is_null()) { k->set_java_mirror(mirror()); } + + // Keep list of classes needing java.base module fixup. + if (!ModuleEntryTable::javabase_defined()) { + if (fixup_module_field_list() == NULL) { + GrowableArray* list = + new (ResourceObj::C_HEAP, mtClass) GrowableArray(500, true); + set_fixup_module_field_list(list); + } + fixup_module_field_list()->push(k()); + } } else { if (fixup_mirror_list() == NULL) { GrowableArray* list = @@ -864,6 +880,10 @@ void java_lang_Class::create_mirror(KlassHandle k, Handle class_loader, } } +void java_lang_Class::fixup_module_field(KlassHandle k, Handle module) { + assert(_module_offset != 0, "must have been computed already"); + java_lang_Class::set_module(k->java_mirror(), module()); +} int java_lang_Class::oop_size(oop java_class) { assert(_oop_size_offset != 0, "must be set"); @@ -931,6 +951,16 @@ oop java_lang_Class::class_loader(oop java_class) { return java_class->obj_field(_class_loader_offset); } +oop java_lang_Class::module(oop java_class) { + assert(_module_offset != 0, "must be set"); + return java_class->obj_field(_module_offset); +} + +void java_lang_Class::set_module(oop java_class, oop module) { + assert(_module_offset != 0, "must be set"); + java_class->obj_field_put(_module_offset, module); +} + oop java_lang_Class::create_basic_type_mirror(const char* basic_type_name, BasicType type, TRAPS) { // This should be improved by adding a field at the Java level or by // introducing a new VM klass (see comment in ClassFileParser) @@ -1116,6 +1146,10 @@ void java_lang_Class::compute_offsets() { k, vmSymbols::componentType_name(), vmSymbols::class_signature()); + compute_offset(_module_offset, + k, vmSymbols::module_name(), + vmSymbols::module_signature()); + // Init lock is a C union with component_mirror. Only instanceKlass mirrors have // init_lock and only ArrayKlass mirrors have component_mirror. Since both are oops // GC treats them the same. @@ -1668,28 +1702,48 @@ char* java_lang_Throwable::print_stack_element_to_buffer(Handle mirror, buf_len += (int)strlen(source_file_name); } + char *module_name = NULL, *module_version = NULL; + ModuleEntry* module = holder->module(); + if (module->is_named()) { + module_name = module->name()->as_C_string(); + buf_len += (int)strlen(module_name); + if (module->version() != NULL) { + module_version = module->version()->as_C_string(); + buf_len += (int)strlen(module_version); + } + } + // Allocate temporary buffer with extra space for formatting and line number char* buf = NEW_RESOURCE_ARRAY(char, buf_len + 64); // Print stack trace line in buffer - sprintf(buf, "\tat %s.%s", klass_name, method_name); + sprintf(buf, "\tat %s.%s(", klass_name, method_name); + + // Print module information + if (module_name != NULL) { + if (module_version != NULL) { + sprintf(buf + (int)strlen(buf), "%s@%s/", module_name, module_version); + } else { + sprintf(buf + (int)strlen(buf), "%s/", module_name); + } + } if (!version_matches(method, version)) { - strcat(buf, "(Redefined)"); + strcat(buf, "Redefined)"); } else { int line_number = Backtrace::get_line_number(method, bci); if (line_number == -2) { - strcat(buf, "(Native Method)"); + strcat(buf, "Native Method)"); } else { if (source_file_name != NULL && (line_number != -1)) { // Sourcename and linenumber - sprintf(buf + (int)strlen(buf), "(%s:%d)", source_file_name, line_number); + sprintf(buf + (int)strlen(buf), "%s:%d)", source_file_name, line_number); } else if (source_file_name != NULL) { // Just sourcename - sprintf(buf + (int)strlen(buf), "(%s)", source_file_name); + sprintf(buf + (int)strlen(buf), "%s)", source_file_name); } else { // Neither sourcename nor linenumber - sprintf(buf + (int)strlen(buf), "(Unknown Source)"); + sprintf(buf + (int)strlen(buf), "Unknown Source)"); } nmethod* nm = method->code(); if (WizardMode && nm != NULL) { @@ -2094,6 +2148,20 @@ oop java_lang_StackTraceElement::create(Handle mirror, int method_id, oop methodname = StringTable::intern(sym, CHECK_0); java_lang_StackTraceElement::set_methodName(element(), methodname); + // Fill in module name and version + ModuleEntry* module = holder->module(); + if (module->is_named()) { + oop module_name = StringTable::intern(module->name(), CHECK_0); + java_lang_StackTraceElement::set_moduleName(element(), module_name); + oop module_version; + if (module->version() != NULL) { + module_version = StringTable::intern(module->version(), CHECK_0); + } else { + module_version = NULL; + } + java_lang_StackTraceElement::set_moduleVersion(element(), module_version); + } + if (!version_matches(method, version)) { // The method was redefined, accurate line number information isn't available java_lang_StackTraceElement::set_fileName(element(), NULL); @@ -2753,6 +2821,80 @@ void java_lang_reflect_Parameter::set_executable(oop param, oop value) { } +int java_lang_reflect_Module::loader_offset; +int java_lang_reflect_Module::name_offset; +int java_lang_reflect_Module::_module_entry_offset = -1; + +Handle java_lang_reflect_Module::create(Handle loader, Handle module_name, TRAPS) { + assert(Universe::is_fully_initialized(), "Need to find another solution to the reflection problem"); + + Symbol* name = vmSymbols::java_lang_reflect_Module(); + Klass* k = SystemDictionary::resolve_or_fail(name, true, CHECK_NH); + instanceKlassHandle klass (THREAD, k); + + Handle jlrmh = klass->allocate_instance_handle(CHECK_NH); + JavaValue result(T_VOID); + JavaCalls::call_special(&result, jlrmh, KlassHandle(THREAD, klass()), + vmSymbols::object_initializer_name(), + vmSymbols::java_lang_reflect_module_init_signature(), + loader, module_name, CHECK_NH); + return jlrmh; +} + +void java_lang_reflect_Module::compute_offsets() { + Klass* k = SystemDictionary::reflect_Module_klass(); + if(NULL != k) { + compute_offset(loader_offset, k, vmSymbols::loader_name(), vmSymbols::classloader_signature()); + compute_offset(name_offset, k, vmSymbols::name_name(), vmSymbols::string_signature()); + MODULE_INJECTED_FIELDS(INJECTED_FIELD_COMPUTE_OFFSET); + } +} + + +oop java_lang_reflect_Module::loader(oop module) { + assert(Universe::is_fully_initialized(), "Need to find another solution to the reflection problem"); + return module->obj_field(loader_offset); +} + +void java_lang_reflect_Module::set_loader(oop module, oop value) { + assert(Universe::is_fully_initialized(), "Need to find another solution to the reflection problem"); + module->obj_field_put(loader_offset, value); +} + +oop java_lang_reflect_Module::name(oop module) { + assert(Universe::is_fully_initialized(), "Need to find another solution to the reflection problem"); + return module->obj_field(name_offset); +} + +void java_lang_reflect_Module::set_name(oop module, oop value) { + assert(Universe::is_fully_initialized(), "Need to find another solution to the reflection problem"); + module->obj_field_put(name_offset, value); +} + +ModuleEntry* java_lang_reflect_Module::module_entry(oop module, TRAPS) { + assert(_module_entry_offset != -1, "Uninitialized module_entry_offset"); + assert(module != NULL, "module can't be null"); + assert(module->is_oop(), "module must be oop"); + + ModuleEntry* module_entry = (ModuleEntry*)module->address_field(_module_entry_offset); + if (module_entry == NULL) { + // If the inject field containing the ModuleEntry* is null then return the + // class loader's unnamed module. + oop loader = java_lang_reflect_Module::loader(module); + Handle h_loader = Handle(THREAD, loader); + ClassLoaderData* loader_cld = SystemDictionary::register_loader(h_loader, CHECK_NULL); + return loader_cld->modules()->unnamed_module(); + } + return module_entry; +} + +void java_lang_reflect_Module::set_module_entry(oop module, ModuleEntry* module_entry) { + assert(_module_entry_offset != -1, "Uninitialized module_entry_offset"); + assert(module != NULL, "module can't be null"); + assert(module->is_oop(), "module must be oop"); + module->address_field_put(_module_entry_offset, (address)module_entry); +} + Handle sun_reflect_ConstantPool::create(TRAPS) { assert(Universe::is_fully_initialized(), "Need to find another solution to the reflection problem"); Klass* k = SystemDictionary::reflect_ConstantPool_klass(); @@ -3352,6 +3494,7 @@ oop java_security_AccessControlContext::create(objArrayHandle context, bool isPr bool java_lang_ClassLoader::offsets_computed = false; int java_lang_ClassLoader::_loader_data_offset = -1; int java_lang_ClassLoader::parallelCapable_offset = -1; +int java_lang_ClassLoader::unnamedModule_offset = -1; ClassLoaderData** java_lang_ClassLoader::loader_data_addr(oop loader) { assert(loader != NULL && loader->is_oop(), "loader must be oop"); @@ -3371,6 +3514,9 @@ void java_lang_ClassLoader::compute_offsets() { compute_optional_offset(parallelCapable_offset, k1, vmSymbols::parallelCapable_name(), vmSymbols::concurrenthashmap_signature()); + compute_offset(unnamedModule_offset, + k1, vmSymbols::unnamedModule_name(), vmSymbols::module_signature()); + CLASSLOADER_INJECTED_FIELDS(INJECTED_FIELD_COMPUTE_OFFSET); } @@ -3438,6 +3584,10 @@ oop java_lang_ClassLoader::non_reflection_class_loader(oop loader) { return loader; } +oop java_lang_ClassLoader::unnamedModule(oop loader) { + assert(is_instance(loader), "loader must be oop"); + return loader->obj_field(unnamedModule_offset); +} // Support for java_lang_System int java_lang_System::in_offset_in_bytes() { @@ -3470,11 +3620,13 @@ int java_lang_Class::_array_klass_offset; int java_lang_Class::_oop_size_offset; int java_lang_Class::_static_oop_field_count_offset; int java_lang_Class::_class_loader_offset; +int java_lang_Class::_module_offset; int java_lang_Class::_protection_domain_offset; int java_lang_Class::_component_mirror_offset; int java_lang_Class::_init_lock_offset; int java_lang_Class::_signers_offset; GrowableArray* java_lang_Class::_fixup_mirror_list = NULL; +GrowableArray* java_lang_Class::_fixup_module_field_list = NULL; int java_lang_Throwable::backtrace_offset; int java_lang_Throwable::detailMessage_offset; int java_lang_Throwable::cause_offset; @@ -3534,6 +3686,8 @@ int java_lang_StackTraceElement::declaringClass_offset; int java_lang_StackTraceElement::methodName_offset; int java_lang_StackTraceElement::fileName_offset; int java_lang_StackTraceElement::lineNumber_offset; +int java_lang_StackTraceElement::moduleName_offset; +int java_lang_StackTraceElement::moduleVersion_offset; int java_lang_StackFrameInfo::_declaringClass_offset; int java_lang_StackFrameInfo::_memberName_offset; int java_lang_StackFrameInfo::_bci_offset; @@ -3575,6 +3729,14 @@ void java_lang_StackTraceElement::set_lineNumber(oop element, int value) { element->int_field_put(lineNumber_offset, value); } +void java_lang_StackTraceElement::set_moduleName(oop element, oop value) { + element->obj_field_put(moduleName_offset, value); +} + +void java_lang_StackTraceElement::set_moduleVersion(oop element, oop value) { + element->obj_field_put(moduleVersion_offset, value); +} + // Support for java_lang_StackFrameInfo void java_lang_StackFrameInfo::set_declaringClass(oop element, oop value) { element->obj_field_put(_declaringClass_offset, value); @@ -3713,6 +3875,8 @@ void JavaClasses::compute_hard_coded_offsets() { java_lang_System::static_security_offset = java_lang_System::hc_static_security_offset * x; // java_lang_StackTraceElement + java_lang_StackTraceElement::moduleName_offset = java_lang_StackTraceElement::hc_moduleName_offset * x + header; + java_lang_StackTraceElement::moduleVersion_offset = java_lang_StackTraceElement::hc_moduleVersion_offset * x + header; java_lang_StackTraceElement::declaringClass_offset = java_lang_StackTraceElement::hc_declaringClass_offset * x + header; java_lang_StackTraceElement::methodName_offset = java_lang_StackTraceElement::hc_methodName_offset * x + header; java_lang_StackTraceElement::fileName_offset = java_lang_StackTraceElement::hc_fileName_offset * x + header; @@ -3752,6 +3916,7 @@ void JavaClasses::compute_offsets() { sun_reflect_ConstantPool::compute_offsets(); sun_reflect_UnsafeStaticFieldAccessorImpl::compute_offsets(); java_lang_reflect_Parameter::compute_offsets(); + java_lang_reflect_Module::compute_offsets(); java_lang_StackFrameInfo::compute_offsets(); java_lang_LiveStackFrameInfo::compute_offsets(); @@ -3899,7 +4064,7 @@ void JavaClasses::check_offsets() { // java.lang.ClassLoader - CHECK_OFFSET("java/lang/ClassLoader", java_lang_ClassLoader, parent, "Ljava/lang/ClassLoader;"); + CHECK_OFFSET("java/lang/ClassLoader", java_lang_ClassLoader, parent, "Ljava/lang/ClassLoader;"); // java.lang.System diff --git a/hotspot/src/share/vm/classfile/javaClasses.hpp b/hotspot/src/share/vm/classfile/javaClasses.hpp index 9f7c2d1233a..6c487534018 100644 --- a/hotspot/src/share/vm/classfile/javaClasses.hpp +++ b/hotspot/src/share/vm/classfile/javaClasses.hpp @@ -210,12 +210,14 @@ class java_lang_Class : AllStatic { static int _init_lock_offset; static int _signers_offset; static int _class_loader_offset; + static int _module_offset; static int _component_mirror_offset; static bool offsets_computed; static int classRedefinedCount_offset; static GrowableArray* _fixup_mirror_list; + static GrowableArray* _fixup_module_field_list; static void set_init_lock(oop java_class, oop init_lock); static void set_protection_domain(oop java_class, oop protection_domain); @@ -226,10 +228,13 @@ class java_lang_Class : AllStatic { static void compute_offsets(); // Instance creation - static void create_mirror(KlassHandle k, Handle class_loader, + static void create_mirror(KlassHandle k, Handle class_loader, Handle module, Handle protection_domain, TRAPS); static void fixup_mirror(KlassHandle k, TRAPS); static oop create_basic_type_mirror(const char* basic_type_name, BasicType type, TRAPS); + + static void fixup_module_field(KlassHandle k, Handle module); + // Conversion static Klass* as_Klass(oop java_class); static void set_klass(oop java_class, Klass* klass); @@ -267,18 +272,29 @@ class java_lang_Class : AllStatic { static void set_signers(oop java_class, objArrayOop signers); static oop class_loader(oop java_class); + static void set_module(oop java_class, oop module); + static oop module(oop java_class); static int oop_size(oop java_class); static void set_oop_size(oop java_class, int size); static int static_oop_field_count(oop java_class); static void set_static_oop_field_count(oop java_class, int size); + static GrowableArray* fixup_mirror_list() { return _fixup_mirror_list; } static void set_fixup_mirror_list(GrowableArray* v) { _fixup_mirror_list = v; } + + static GrowableArray* fixup_module_field_list() { + return _fixup_module_field_list; + } + static void set_fixup_module_field_list(GrowableArray* v) { + _fixup_module_field_list = v; + } + // Debugging friend class JavaClasses; friend class InstanceKlass; // verification code accesses offsets @@ -758,6 +774,39 @@ class java_lang_reflect_Parameter { friend class JavaClasses; }; +#define MODULE_INJECTED_FIELDS(macro) \ + macro(java_lang_reflect_Module, module_entry, intptr_signature, false) + +class java_lang_reflect_Module { + private: + static int loader_offset; + static int name_offset; + static int _module_entry_offset; + static void compute_offsets(); + + public: + // Allocation + static Handle create(Handle loader, Handle module_name, TRAPS); + + // Testers + static bool is_subclass(Klass* klass) { + return klass->is_subclass_of(SystemDictionary::reflect_Module_klass()); + } + static bool is_instance(oop obj); + + // Accessors + static oop loader(oop module); + static void set_loader(oop module, oop value); + + static oop name(oop module); + static void set_name(oop module, oop value); + + static ModuleEntry* module_entry(oop module, TRAPS); + static void set_module_entry(oop module, ModuleEntry* module_entry); + + friend class JavaClasses; +}; + // Interface to sun.reflect.ConstantPool objects class sun_reflect_ConstantPool { private: @@ -1203,6 +1252,7 @@ class java_lang_ClassLoader : AllStatic { static bool offsets_computed; static int parent_offset; static int parallelCapable_offset; + static int unnamedModule_offset; public: static void compute_offsets(); @@ -1227,6 +1277,8 @@ class java_lang_ClassLoader : AllStatic { } static bool is_instance(oop obj); + static oop unnamedModule(oop loader); + // Debugging friend class JavaClasses; friend class ClassFileParser; // access to number_of_fake_fields @@ -1266,12 +1318,16 @@ class java_lang_System : AllStatic { class java_lang_StackTraceElement: AllStatic { private: enum { - hc_declaringClass_offset = 0, - hc_methodName_offset = 1, - hc_fileName_offset = 2, - hc_lineNumber_offset = 3 + hc_moduleName_offset = 0, + hc_moduleVersion_offset = 1, + hc_declaringClass_offset = 2, + hc_methodName_offset = 3, + hc_fileName_offset = 4, + hc_lineNumber_offset = 5 }; + static int moduleName_offset; + static int moduleVersion_offset; static int declaringClass_offset; static int methodName_offset; static int fileName_offset; @@ -1279,6 +1335,8 @@ class java_lang_StackTraceElement: AllStatic { public: // Setters + static void set_moduleName(oop element, oop value); + static void set_moduleVersion(oop element, oop value); static void set_declaringClass(oop element, oop value); static void set_methodName(oop element, oop value); static void set_fileName(oop element, oop value); @@ -1456,8 +1514,8 @@ class InjectedField { CLASSLOADER_INJECTED_FIELDS(macro) \ MEMBERNAME_INJECTED_FIELDS(macro) \ CALLSITECONTEXT_INJECTED_FIELDS(macro) \ - STACKFRAMEINFO_INJECTED_FIELDS(macro) - + STACKFRAMEINFO_INJECTED_FIELDS(macro) \ + MODULE_INJECTED_FIELDS(macro) // Interface to hard-coded offset checking diff --git a/hotspot/src/share/vm/classfile/javaClasses.inline.hpp b/hotspot/src/share/vm/classfile/javaClasses.inline.hpp index 05deb981ba1..1aba5b8f8c1 100644 --- a/hotspot/src/share/vm/classfile/javaClasses.inline.hpp +++ b/hotspot/src/share/vm/classfile/javaClasses.inline.hpp @@ -171,6 +171,10 @@ inline bool java_lang_invoke_DirectMethodHandle::is_instance(oop obj) { +inline bool java_lang_reflect_Module::is_instance(oop obj) { + return obj != NULL && is_subclass(obj->klass()); +} + inline int Backtrace::merge_bci_and_version(int bci, int version) { // only store u2 for version, checking for overflow. if (version > USHRT_MAX || version < 0) version = USHRT_MAX; diff --git a/hotspot/src/share/vm/classfile/jimage.hpp b/hotspot/src/share/vm/classfile/jimage.hpp index 90f6dad9fd1..e538ac805f8 100644 --- a/hotspot/src/share/vm/classfile/jimage.hpp +++ b/hotspot/src/share/vm/classfile/jimage.hpp @@ -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. * * This code is free software; you can redistribute it and/or modify it @@ -22,7 +22,7 @@ * */ -#include "jni.h" +#include "prims/jni.h" // Opaque reference to a JImage file. class JImageFile; @@ -35,6 +35,8 @@ typedef jlong JImageLocationRef; // JImage Error Codes +// Resource was not found +#define JIMAGE_NOT_FOUND (0) // The image file is not prefixed with 0xCAFEDADA #define JIMAGE_BAD_MAGIC (-1) // The image file does not have a compatible (translatable) version @@ -55,7 +57,7 @@ typedef jlong JImageLocationRef; * * Ex. * jint error; - * JImageFile* jimage = (*JImageOpen)(JAVA_HOME "lib/modules/bootmodules.jimage", &error); + * JImageFile* jimage = (*JImageOpen)(JAVA_HOME "lib/modules", &error); * if (image == NULL) { * tty->print_cr("JImage failed to open: %d", error); * ... diff --git a/hotspot/src/share/vm/classfile/klassFactory.cpp b/hotspot/src/share/vm/classfile/klassFactory.cpp index 033d9bfb660..11b07f6b24a 100644 --- a/hotspot/src/share/vm/classfile/klassFactory.cpp +++ b/hotspot/src/share/vm/classfile/klassFactory.cpp @@ -103,11 +103,15 @@ instanceKlassHandle KlassFactory::create_from_stream(ClassFileStream* stream, assert(loader_data != NULL, "invariant"); assert(THREAD->is_Java_thread(), "must be a JavaThread"); + bool changed_by_loadhook = false; + ResourceMark rm; HandleMark hm; JvmtiCachedClassFileData* cached_class_file = NULL; + ClassFileStream* old_stream = stream; + stream = prologue(stream, name, loader_data, @@ -125,8 +129,8 @@ instanceKlassHandle KlassFactory::create_from_stream(ClassFileStream* stream, ClassFileParser::BROADCAST, // publicity level CHECK_NULL); - instanceKlassHandle result = parser.create_instance_klass(CHECK_NULL); - assert(result == parser.create_instance_klass(THREAD), "invariant"); + instanceKlassHandle result = parser.create_instance_klass(old_stream != stream, CHECK_NULL); + assert(result == parser.create_instance_klass(old_stream != stream, THREAD), "invariant"); if (result.is_null()) { return NULL; diff --git a/hotspot/src/share/vm/classfile/moduleEntry.cpp b/hotspot/src/share/vm/classfile/moduleEntry.cpp new file mode 100644 index 00000000000..e86ed6acabf --- /dev/null +++ b/hotspot/src/share/vm/classfile/moduleEntry.cpp @@ -0,0 +1,405 @@ +/* + * 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. + * + */ + +#include "precompiled.hpp" +#include "classfile/classLoaderData.hpp" +#include "classfile/javaClasses.hpp" +#include "classfile/moduleEntry.hpp" +#include "logging/log.hpp" +#include "memory/resourceArea.hpp" +#include "oops/symbol.hpp" +#include "prims/jni.h" +#include "runtime/handles.inline.hpp" +#include "runtime/safepoint.hpp" +#include "trace/traceMacros.hpp" +#include "utilities/events.hpp" +#include "utilities/growableArray.hpp" +#include "utilities/hashtable.inline.hpp" + +ModuleEntry* ModuleEntryTable::_javabase_module = NULL; + + +void ModuleEntry::set_location(Symbol* location) { + if (_location != NULL) { + // _location symbol's refcounts are managed by ModuleEntry, + // must decrement the old one before updating. + _location->decrement_refcount(); + } + + _location = location; + + if (location != NULL) { + location->increment_refcount(); + } +} + +void ModuleEntry::set_version(Symbol* version) { + if (_version != NULL) { + // _version symbol's refcounts are managed by ModuleEntry, + // must decrement the old one before updating. + _version->decrement_refcount(); + } + + _version = version; + + if (version != NULL) { + version->increment_refcount(); + } +} + +// Returns the shared ProtectionDomain +Handle ModuleEntry::shared_protection_domain() { + return Handle(JNIHandles::resolve(_pd)); +} + +// Set the shared ProtectionDomain atomically +void ModuleEntry::set_shared_protection_domain(ClassLoaderData *loader_data, + Handle pd_h) { + // Create a JNI handle for the shared ProtectionDomain and save it atomically. + // If someone beats us setting the _pd cache, the created JNI handle is destroyed. + jobject obj = loader_data->add_handle(pd_h); + if (Atomic::cmpxchg_ptr(obj, &_pd, NULL) != NULL) { + loader_data->remove_handle(obj); + } +} + +// Returns true if this module can read module m +bool ModuleEntry::can_read(ModuleEntry* m) const { + assert(m != NULL, "No module to lookup in this module's reads list"); + + // Unnamed modules read everyone and all modules + // read java.base. If either of these conditions + // hold, readability has been established. + if (!this->is_named() || + (m == ModuleEntryTable::javabase_module())) { + return true; + } + + MutexLocker m1(Module_lock); + if (!has_reads()) { + return false; + } else { + return _reads->contains(m); + } +} + +// Add a new module to this module's reads list +void ModuleEntry::add_read(ModuleEntry* m) { + MutexLocker m1(Module_lock); + if (m == NULL) { + set_can_read_all_unnamed(); + } else { + if (_reads == NULL) { + // Lazily create a module's reads list + _reads = new (ResourceObj::C_HEAP, mtClass)GrowableArray(MODULE_READS_SIZE, true); + } + _reads->append_if_missing(m); + } +} + +bool ModuleEntry::has_reads() const { + assert_locked_or_safepoint(Module_lock); + return ((_reads != NULL) && !_reads->is_empty()); +} + +// Purge dead module entries out of reads list. +void ModuleEntry::purge_reads() { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint"); + if (has_reads()) { + // Go backwards because this removes entries that are dead. + int len = _reads->length(); + for (int idx = len - 1; idx >= 0; idx--) { + ModuleEntry* module_idx = _reads->at(idx); + ClassLoaderData* cld = module_idx->loader(); + if (cld->is_unloading()) { + _reads->delete_at(idx); + } + } + } +} + +void ModuleEntry::module_reads_do(ModuleClosure* const f) { + assert_locked_or_safepoint(Module_lock); + assert(f != NULL, "invariant"); + + if (has_reads()) { + int reads_len = _reads->length(); + for (int i = 0; i < reads_len; ++i) { + f->do_module(_reads->at(i)); + } + } +} + +void ModuleEntry::delete_reads() { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint"); + delete _reads; + _reads = NULL; +} + +ModuleEntryTable::ModuleEntryTable(int table_size) + : Hashtable(table_size, sizeof(ModuleEntry)), _unnamed_module(NULL) +{ +} + +ModuleEntryTable::~ModuleEntryTable() { + assert_locked_or_safepoint(Module_lock); + + // Walk through all buckets and all entries in each bucket, + // freeing each entry. + for (int i = 0; i < table_size(); ++i) { + for (ModuleEntry* m = bucket(i); m != NULL;) { + ModuleEntry* to_remove = m; + // read next before freeing. + m = m->next(); + + ResourceMark rm; + log_debug(modules)("ModuleEntryTable: deleting module: %s", to_remove->name() != NULL ? + to_remove->name()->as_C_string() : UNNAMED_MODULE); + + // Clean out the C heap allocated reads list first before freeing the entry + to_remove->delete_reads(); + if (to_remove->name() != NULL) { + to_remove->name()->decrement_refcount(); + } + if (to_remove->version() != NULL) { + to_remove->version()->decrement_refcount(); + } + if (to_remove->location() != NULL) { + to_remove->location()->decrement_refcount(); + } + + // Unlink from the Hashtable prior to freeing + unlink_entry(to_remove); + FREE_C_HEAP_ARRAY(char, to_remove); + } + } + assert(number_of_entries() == 0, "should have removed all entries"); + assert(new_entry_free_list() == NULL, "entry present on ModuleEntryTable's free list"); + free_buckets(); +} + +void ModuleEntryTable::create_unnamed_module(ClassLoaderData* loader_data) { + assert_locked_or_safepoint(Module_lock); + + // Each ModuleEntryTable has exactly one unnamed module + if (loader_data->is_the_null_class_loader_data()) { + // For the boot loader, the java.lang.reflect.Module for the unnamed module + // is not known until a call to JVM_SetBootLoaderUnnamedModule is made. At + // this point initially create the ModuleEntry for the unnamed module. + _unnamed_module = new_entry(0, Handle(NULL), NULL, NULL, NULL, loader_data); + } else { + // For all other class loaders the java.lang.reflect.Module for their + // corresponding unnamed module can be found in the java.lang.ClassLoader object. + oop module = java_lang_ClassLoader::unnamedModule(loader_data->class_loader()); + _unnamed_module = new_entry(0, Handle(module), NULL, NULL, NULL, loader_data); + + // Store pointer to the ModuleEntry in the unnamed module's java.lang.reflect.Module + // object. + java_lang_reflect_Module::set_module_entry(module, _unnamed_module); + } + + // Add to bucket 0, no name to hash on + add_entry(0, _unnamed_module); +} + +ModuleEntry* ModuleEntryTable::new_entry(unsigned int hash, Handle module_handle, Symbol* name, + Symbol* version, Symbol* location, + ClassLoaderData* loader_data) { + assert_locked_or_safepoint(Module_lock); + ModuleEntry* entry = (ModuleEntry*) NEW_C_HEAP_ARRAY(char, entry_size(), mtClass); + + // Initialize everything BasicHashtable would + entry->set_next(NULL); + entry->set_hash(hash); + entry->set_literal(name); + + // Initialize fields specific to a ModuleEntry + entry->init(); + if (name != NULL) { + name->increment_refcount(); + } else { + // Unnamed modules can read all other unnamed modules. + entry->set_can_read_all_unnamed(); + } + + if (!module_handle.is_null()) { + entry->set_module(loader_data->add_handle(module_handle)); + } + + entry->set_loader(loader_data); + entry->set_version(version); + entry->set_location(location); + + TRACE_INIT_MODULE_ID(entry); + + return entry; +} + +void ModuleEntryTable::add_entry(int index, ModuleEntry* new_entry) { + assert_locked_or_safepoint(Module_lock); + Hashtable::add_entry(index, (HashtableEntry*)new_entry); +} + +ModuleEntry* ModuleEntryTable::locked_create_entry_or_null(Handle module_handle, + Symbol* module_name, + Symbol* module_version, + Symbol* module_location, + ClassLoaderData* loader_data) { + assert(module_name != NULL, "ModuleEntryTable locked_create_entry_or_null should never be called for unnamed module."); + assert_locked_or_safepoint(Module_lock); + // Check if module already exists. + if (lookup_only(module_name) != NULL) { + return NULL; + } else { + ModuleEntry* entry = new_entry(compute_hash(module_name), module_handle, module_name, + module_version, module_location, loader_data); + add_entry(index_for(module_name), entry); + return entry; + } +} + +// lookup_only by Symbol* to find a ModuleEntry. +ModuleEntry* ModuleEntryTable::lookup_only(Symbol* name) { + if (name == NULL) { + // Return this table's unnamed module + return unnamed_module(); + } + int index = index_for(name); + for (ModuleEntry* m = bucket(index); m != NULL; m = m->next()) { + if (m->name()->fast_compare(name) == 0) { + return m; + } + } + return NULL; +} + +// Remove dead modules from all other alive modules' reads list. +// This should only occur at class unloading. +void ModuleEntryTable::purge_all_module_reads() { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint"); + for (int i = 0; i < table_size(); i++) { + for (ModuleEntry* entry = bucket(i); + entry != NULL; + entry = entry->next()) { + entry->purge_reads(); + } + } +} + +void ModuleEntryTable::finalize_javabase(Handle module_handle, Symbol* version, Symbol* location) { + assert_locked_or_safepoint(Module_lock); + ClassLoaderData* boot_loader_data = ClassLoaderData::the_null_class_loader_data(); + ModuleEntryTable* module_table = boot_loader_data->modules(); + + assert(module_table != NULL, "boot loader's ModuleEntryTable not defined"); + + if (module_handle.is_null()) { + fatal("Unable to finalize module definition for java.base"); + } + + // Set java.lang.reflect.Module, version and location for java.base + ModuleEntry* jb_module = javabase_module(); + assert(jb_module != NULL, "java.base ModuleEntry not defined"); + jb_module->set_module(boot_loader_data->add_handle(module_handle)); + jb_module->set_version(version); + jb_module->set_location(location); + // Store pointer to the ModuleEntry for java.base in the java.lang.reflect.Module object. + java_lang_reflect_Module::set_module_entry(module_handle(), jb_module); +} + +void ModuleEntryTable::patch_javabase_entries(Handle module_handle) { + if (module_handle.is_null()) { + fatal("Unable to patch the module field of classes loaded prior to java.base's definition, invalid java.lang.reflect.Module"); + } + + // Do the fixups for the basic primitive types + java_lang_Class::set_module(Universe::int_mirror(), module_handle()); + java_lang_Class::set_module(Universe::float_mirror(), module_handle()); + java_lang_Class::set_module(Universe::double_mirror(), module_handle()); + java_lang_Class::set_module(Universe::byte_mirror(), module_handle()); + java_lang_Class::set_module(Universe::bool_mirror(), module_handle()); + java_lang_Class::set_module(Universe::char_mirror(), module_handle()); + java_lang_Class::set_module(Universe::long_mirror(), module_handle()); + java_lang_Class::set_module(Universe::short_mirror(), module_handle()); + java_lang_Class::set_module(Universe::void_mirror(), module_handle()); + + // Do the fixups for classes that have already been created. + GrowableArray * list = java_lang_Class::fixup_module_field_list(); + int list_length = list->length(); + for (int i = 0; i < list_length; i++) { + Klass* k = list->at(i); + assert(k->is_klass(), "List should only hold classes"); + Thread* THREAD = Thread::current(); + KlassHandle kh(THREAD, k); + java_lang_Class::fixup_module_field(kh, module_handle); + } + + delete java_lang_Class::fixup_module_field_list(); + java_lang_Class::set_fixup_module_field_list(NULL); +} + +#ifndef PRODUCT +void ModuleEntryTable::print() { + tty->print_cr("Module Entry Table (table_size=%d, entries=%d)", + table_size(), number_of_entries()); + for (int i = 0; i < table_size(); i++) { + for (ModuleEntry* probe = bucket(i); + probe != NULL; + probe = probe->next()) { + probe->print(); + } + } +} + +void ModuleEntry::print() { + ResourceMark rm; + tty->print_cr("entry "PTR_FORMAT" name %s module "PTR_FORMAT" loader %s version %s location %s strict %s next "PTR_FORMAT, + p2i(this), + name() == NULL ? UNNAMED_MODULE : name()->as_C_string(), + p2i(module()), + loader()->loader_name(), + version() != NULL ? version()->as_C_string() : "NULL", + location() != NULL ? location()->as_C_string() : "NULL", + BOOL_TO_STR(!can_read_all_unnamed()), p2i(next())); +} +#endif + +void ModuleEntryTable::verify() { + int element_count = 0; + for (int i = 0; i < table_size(); i++) { + for (ModuleEntry* probe = bucket(i); + probe != NULL; + probe = probe->next()) { + probe->verify(); + element_count++; + } + } + guarantee(number_of_entries() == element_count, + "Verify of Module Entry Table failed"); + debug_only(verify_lookup_length((double)number_of_entries() / table_size())); +} + +void ModuleEntry::verify() { + guarantee(loader() != NULL, "A module entry must be associated with a loader."); +} diff --git a/hotspot/src/share/vm/classfile/moduleEntry.hpp b/hotspot/src/share/vm/classfile/moduleEntry.hpp new file mode 100644 index 00000000000..d67251c1869 --- /dev/null +++ b/hotspot/src/share/vm/classfile/moduleEntry.hpp @@ -0,0 +1,230 @@ +/* + * 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. + * + */ + +#ifndef SHARE_VM_CLASSFILE_MODULEENTRY_HPP +#define SHARE_VM_CLASSFILE_MODULEENTRY_HPP + +#include "classfile/classLoaderData.hpp" +#include "classfile/vmSymbols.hpp" +#include "oops/symbol.hpp" +#include "prims/jni.h" +#include "runtime/mutexLocker.hpp" +#include "trace/traceMacros.hpp" +#include "utilities/growableArray.hpp" +#include "utilities/hashtable.hpp" + +#define UNNAMED_MODULE "Unnamed Module" + +class ModuleClosure; + +// A ModuleEntry describes a module that has been defined by a call to JVM_DefineModule. +// It contains: +// - Symbol* containing the module's name. +// - pointer to the java.lang.reflect.Module for this module. +// - ClassLoaderData*, class loader of this module. +// - a growable array containg other module entries that this module can read. +// - a flag indicating if this module can read all unnamed modules. +// +// The Mutex Module_lock is shared between ModuleEntry and PackageEntry, to lock either +// data structure. +class ModuleEntry : public HashtableEntry { +private: + jobject _module; // java.lang.reflect.Module + jobject _pd; // java.security.ProtectionDomain, cached + // for shared classes from this module + ClassLoaderData* _loader; + GrowableArray* _reads; // list of modules that are readable by this module + Symbol* _version; // module version number + Symbol* _location; // module location + bool _can_read_all_unnamed; + bool _has_default_read_edges; // JVMTI redefine/retransform support + TRACE_DEFINE_TRACE_ID_FIELD; + enum {MODULE_READS_SIZE = 101}; // Initial size of list of modules that the module can read. + +public: + void init() { + _module = NULL; + _loader = NULL; + _pd = NULL; + _reads = NULL; + _version = NULL; + _location = NULL; + _can_read_all_unnamed = false; + _has_default_read_edges = false; + } + + Symbol* name() const { return literal(); } + void set_name(Symbol* n) { set_literal(n); } + + jobject module() const { return _module; } + void set_module(jobject j) { _module = j; } + + // The shared ProtectionDomain reference is set once the VM loads a shared class + // originated from the current Module. The referenced ProtectionDomain object is + // created by the ClassLoader when loading a class (shared or non-shared) from the + // Module for the first time. This ProtectionDomain object is used for all + // classes from the Module loaded by the same ClassLoader. + Handle shared_protection_domain(); + void set_shared_protection_domain(ClassLoaderData *loader_data, + Handle pd); + + ClassLoaderData* loader() const { return _loader; } + void set_loader(ClassLoaderData* l) { _loader = l; } + + Symbol* version() const { return _version; } + void set_version(Symbol* version); + + Symbol* location() const { return _location; } + void set_location(Symbol* location); + + bool can_read(ModuleEntry* m) const; + bool has_reads() const; + void add_read(ModuleEntry* m); + + bool is_named() const { return (literal() != NULL); } + + bool can_read_all_unnamed() const { + assert(is_named() || _can_read_all_unnamed == true, + "unnamed modules can always read all unnamed modules"); + return _can_read_all_unnamed; + } + + // Modules can only go from strict to loose. + void set_can_read_all_unnamed() { _can_read_all_unnamed = true; } + + bool has_default_read_edges() const { + return _has_default_read_edges; + } + + // Sets true and returns the previous value. + bool set_has_default_read_edges() { + MutexLocker ml(Module_lock); + bool prev = _has_default_read_edges; + _has_default_read_edges = true; + return prev; + } + + ModuleEntry* next() const { + return (ModuleEntry*)HashtableEntry::next(); + } + ModuleEntry** next_addr() { + return (ModuleEntry**)HashtableEntry::next_addr(); + } + + // iteration support for readability + void module_reads_do(ModuleClosure* const f); + + TRACE_DEFINE_TRACE_ID_METHODS; + + // Purge dead weak references out of reads list when any given class loader is unloaded. + void purge_reads(); + void delete_reads(); + + void print() PRODUCT_RETURN; + void verify(); +}; + +// Iterator interface +class ModuleClosure: public StackObj { + public: + virtual void do_module(ModuleEntry* const module) = 0; +}; + + +// The ModuleEntryTable is a Hashtable containing a list of all modules defined +// by a particular class loader. Each module is represented as a ModuleEntry node. +// +// Each ModuleEntryTable contains a _javabase_module field which allows for the +// creation of java.base's ModuleEntry very early in bootstrapping before the +// corresponding JVM_DefineModule call for java.base occurs during module system +// initialization. Setting up java.base's ModuleEntry early enables classes, +// loaded prior to the module system being initialized to be created with their +// PackageEntry node's correctly pointing at java.base's ModuleEntry. No class +// outside of java.base is allowed to be loaded pre-module system initialization. +// +// The ModuleEntryTable's lookup is lock free. +// +class ModuleEntryTable : public Hashtable { + friend class VMStructs; +public: + enum Constants { + _moduletable_entry_size = 109 // number of entries in module entry table + }; + +private: + static ModuleEntry* _javabase_module; + ModuleEntry* _unnamed_module; + + ModuleEntry* new_entry(unsigned int hash, Handle module_handle, Symbol* name, Symbol* version, + Symbol* location, ClassLoaderData* class_loader); + void add_entry(int index, ModuleEntry* new_entry); + + int entry_size() const { return BasicHashtable::entry_size(); } + + ModuleEntry** bucket_addr(int i) { + return (ModuleEntry**)Hashtable::bucket_addr(i); + } + + static unsigned int compute_hash(Symbol* name) { return ((name == NULL) ? 0 : (unsigned int)(name->identity_hash())); } + int index_for(Symbol* name) const { return hash_to_index(compute_hash(name)); } + +public: + ModuleEntryTable(int table_size); + ~ModuleEntryTable(); + + ModuleEntry* bucket(int i) { + return (ModuleEntry*)Hashtable::bucket(i); + } + + // Create module in loader's module entry table, if already exists then + // return null. Assume Module_lock has been locked by caller. + ModuleEntry* locked_create_entry_or_null(Handle module_handle, + Symbol* module_name, + Symbol* module_version, + Symbol* module_location, + ClassLoaderData* loader_data); + + // Only lookup module within loader's module entry table. The table read is lock-free. + ModuleEntry* lookup_only(Symbol* name); + + // purge dead weak references out of reads list + void purge_all_module_reads(); + + // Special handling for unnamed module, one per class loader's ModuleEntryTable + void create_unnamed_module(ClassLoaderData* loader_data); + ModuleEntry* unnamed_module() { return _unnamed_module; } + + // Special handling for java.base + static ModuleEntry* javabase_module() { return _javabase_module; } + static void set_javabase_module(ModuleEntry* java_base) { _javabase_module = java_base; } + static bool javabase_defined() { return ((_javabase_module != NULL) && + (_javabase_module->module() != NULL)); } + static void finalize_javabase(Handle module_handle, Symbol* version, Symbol* location); + static void patch_javabase_entries(Handle module_handle); + + void print() PRODUCT_RETURN; + void verify(); +}; + +#endif // SHARE_VM_CLASSFILE_MODULEENTRY_HPP diff --git a/hotspot/src/share/vm/classfile/modules.cpp b/hotspot/src/share/vm/classfile/modules.cpp new file mode 100644 index 00000000000..a4a23d44e49 --- /dev/null +++ b/hotspot/src/share/vm/classfile/modules.cpp @@ -0,0 +1,964 @@ +/* +* 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. +* +*/ + +#include "precompiled.hpp" +#include "classfile/classFileParser.hpp" +#include "classfile/classLoader.hpp" +#include "classfile/classLoaderData.inline.hpp" +#include "classfile/javaAssertions.hpp" +#include "classfile/javaClasses.hpp" +#include "classfile/javaClasses.inline.hpp" +#include "classfile/moduleEntry.hpp" +#include "classfile/modules.hpp" +#include "classfile/packageEntry.hpp" +#include "classfile/stringTable.hpp" +#include "classfile/symbolTable.hpp" +#include "classfile/vmSymbols.hpp" +#include "logging/log.hpp" +#include "oops/instanceKlass.hpp" +#include "oops/objArrayKlass.hpp" +#include "oops/objArrayOop.inline.hpp" +#include "runtime/arguments.hpp" +#include "runtime/handles.inline.hpp" +#include "runtime/javaCalls.hpp" +#include "runtime/reflection.hpp" +#include "utilities/utf8.hpp" + +static bool verify_module_name(char *module_name) { + if (module_name == NULL) return false; + int len = (int)strlen(module_name); + return (len > 0 && len <= Symbol::max_length() && + UTF8::is_legal_utf8((unsigned char *)module_name, len, false) && + ClassFileParser::verify_unqualified_name(module_name, len, + ClassFileParser::LegalModule)); +} + +bool Modules::verify_package_name(char *package_name) { + if (package_name == NULL) return false; + int len = (int)strlen(package_name); + return (len > 0 && len <= Symbol::max_length() && + UTF8::is_legal_utf8((unsigned char *)package_name, len, false) && + ClassFileParser::verify_unqualified_name(package_name, len, + ClassFileParser::LegalClass)); +} + +static char* get_module_name(oop module, TRAPS) { + oop name_oop = java_lang_reflect_Module::name(module); + if (name_oop == NULL) { + THROW_MSG_NULL(vmSymbols::java_lang_NullPointerException(), "Null module name"); + } + char* module_name = java_lang_String::as_utf8_string(name_oop); + if (!verify_module_name(module_name)) { + THROW_MSG_NULL(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Invalid module name: %s", + module_name != NULL ? module_name : "NULL")); + } + return module_name; +} + +static const char* get_module_version(jstring version) { + if (version == NULL) { + return NULL; + } + return java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(version)); +} + +static ModuleEntryTable* get_module_entry_table(Handle h_loader, TRAPS) { + // This code can be called during start-up, before the classLoader's classLoader data got + // created. So, call register_loader() to make sure the classLoader data gets created. + ClassLoaderData *loader_cld = SystemDictionary::register_loader(h_loader, CHECK_NULL); + return loader_cld->modules(); +} + +static PackageEntryTable* get_package_entry_table(Handle h_loader, TRAPS) { + // This code can be called during start-up, before the classLoader's classLoader data got + // created. So, call register_loader() to make sure the classLoader data gets created. + ClassLoaderData *loader_cld = SystemDictionary::register_loader(h_loader, CHECK_NULL); + return loader_cld->packages(); +} + +static ModuleEntry* get_module_entry(jobject module, TRAPS) { + Handle module_h(THREAD, JNIHandles::resolve(module)); + if (!java_lang_reflect_Module::is_instance(module_h())) { + THROW_MSG_NULL(vmSymbols::java_lang_IllegalArgumentException(), "Bad module object"); + } + return java_lang_reflect_Module::module_entry(module_h(), CHECK_NULL); +} + +static PackageEntry* get_package_entry(ModuleEntry* module_entry, jstring package, TRAPS) { + ResourceMark rm(THREAD); + if (package == NULL) return NULL; + const char *package_name = java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(package)); + if (package_name == NULL) return NULL; + TempNewSymbol pkg_symbol = SymbolTable::new_symbol(package_name, CHECK_NULL); + PackageEntryTable* package_entry_table = module_entry->loader()->packages(); + assert(package_entry_table != NULL, "Unexpected null package entry table"); + return package_entry_table->lookup_only(pkg_symbol); +} + +static PackageEntry* get_package_entry_by_name(Symbol* package, + Handle h_loader, + TRAPS) { + if (package != NULL) { + ResourceMark rm(THREAD); + if (Modules::verify_package_name(package->as_C_string())) { + PackageEntryTable* const package_entry_table = + get_package_entry_table(h_loader, CHECK_NULL); + assert(package_entry_table != NULL, "Unexpected null package entry table"); + return package_entry_table->lookup_only(package); + } + } + return NULL; +} + +// Check if -Xpatch: was specified. If so, prepend each /module_name, +// if it exists, to bootpath so boot loader can find the class files. Also, if +// using exploded modules, append /modules/module_name, if it exists, +// to bootpath so that its class files can be found by the boot loader. +static void add_to_boot_loader_list(char *module_name, TRAPS) { + // java.base should be handled by argument parsing. + assert(strcmp(module_name, "java.base") != 0, "Unexpected java.base module name"); + char file_sep = os::file_separator()[0]; + size_t module_len = strlen(module_name); + + // If -Xpatch is set then add /module_name paths. + char** patch_dirs = Arguments::patch_dirs(); + if (patch_dirs != NULL) { + int dir_count = Arguments::patch_dirs_count(); + for (int x = 0; x < dir_count; x++) { + // Really shouldn't be NULL, but check can't hurt + if (patch_dirs[x] != NULL) { + size_t len = strlen(patch_dirs[x]); + if (len != 0) { // Ignore empty strings. + len = len + module_len + 2; + char* prefix_path = NEW_C_HEAP_ARRAY(char, len, mtInternal); + jio_snprintf(prefix_path, len, "%s%c%s", patch_dirs[x], file_sep, module_name); + + // See if Xpatch module path exists. + struct stat st; + if ((os::stat(prefix_path, &st) != 0)) { + FREE_C_HEAP_ARRAY(char, prefix_path); + } else { + { + HandleMark hm; + Handle loader_lock = Handle(THREAD, SystemDictionary::system_loader_lock()); + ObjectLocker ol(loader_lock, THREAD); + ClassLoader::prepend_to_list(prefix_path); + } + log_info(classload)("opened: -Xpatch %s", prefix_path); + } + } + } + } + } + + // If "modules" jimage does not exist then assume exploded form + // ${java.home}/modules/ + char* path = NULL; + if (!ClassLoader::has_jimage()) { + const char* home = Arguments::get_java_home(); + size_t len = strlen(home) + module_len + 32; + path = NEW_C_HEAP_ARRAY(char, len, mtInternal); + jio_snprintf(path, len, "%s%cmodules%c%s", home, file_sep, file_sep, module_name); + struct stat st; + // See if exploded module path exists. + if ((os::stat(path, &st) != 0)) { + FREE_C_HEAP_ARRAY(char, path); + path = NULL; + } + } + + if (path != NULL) { + HandleMark hm; + Handle loader_lock = Handle(THREAD, SystemDictionary::system_loader_lock()); + ObjectLocker ol(loader_lock, THREAD); + + log_info(classload)("opened: %s", path); + ClassLoader::add_to_list(path); + } +} + +bool Modules::is_package_defined(Symbol* package, Handle h_loader, TRAPS) { + PackageEntry* res = get_package_entry_by_name(package, h_loader, CHECK_false); + return res != NULL; +} + +static void define_javabase_module(jobject module, jstring version, + jstring location, jobjectArray packages, TRAPS) { + ResourceMark rm(THREAD); + + Handle module_handle(THREAD, JNIHandles::resolve(module)); + + // Obtain java.base's module version + const char* module_version = get_module_version(version); + TempNewSymbol version_symbol; + if (module_version != NULL) { + version_symbol = SymbolTable::new_symbol(module_version, CHECK); + } else { + version_symbol = NULL; + } + + // Obtain java.base's location + const char* module_location = NULL; + TempNewSymbol location_symbol = NULL; + if (location != NULL) { + module_location = + java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(location)); + if (module_location != NULL) { + location_symbol = SymbolTable::new_symbol(module_location, CHECK); + } + } + + objArrayOop packages_oop = objArrayOop(JNIHandles::resolve(packages)); + objArrayHandle packages_h(THREAD, packages_oop); + int num_packages = (packages_h == NULL ? 0 : packages_h->length()); + + // Check that the list of packages has no duplicates and that the + // packages are syntactically ok. + GrowableArray* pkg_list = new GrowableArray(num_packages); + for (int x = 0; x < num_packages; x++) { + oop string_obj = packages_h->obj_at(x); + + if (string_obj == NULL || !string_obj->is_a(SystemDictionary::String_klass())) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "Bad package name for module: java.base"); + } + char *package_name = java_lang_String::as_utf8_string(string_obj); + if (!Modules::verify_package_name(package_name)) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Invalid package name: %s for module: java.base", package_name)); + } + Symbol* pkg_symbol = SymbolTable::new_symbol(package_name, CHECK); + // append_if_missing() returns FALSE if entry already exists. + if (!pkg_list->append_if_missing(pkg_symbol)) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Duplicate package name: %s for module java.base", + package_name)); + } + } + + // Validate java_base's loader is the boot loader. + oop loader = java_lang_reflect_Module::loader(module_handle()); + if (loader != NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "Class loader must be the boot class loader"); + } + Handle h_loader = Handle(THREAD, loader); + + // Ensure the boot loader's PackageEntryTable has been created + PackageEntryTable* package_table = get_package_entry_table(h_loader, CHECK); + assert(pkg_list->length() == 0 || package_table != NULL, "Bad package_table"); + + // Ensure java.base's ModuleEntry has been created + assert(ModuleEntryTable::javabase_module() != NULL, "No ModuleEntry for java.base"); + + { + MutexLocker m1(Module_lock, THREAD); + + // Verify that all java.base packages created during bootstrapping are in + // pkg_list. If any are not in pkg_list, than a non-java.base class was + // loaded erroneously pre java.base module definition. + package_table->verify_javabase_packages(pkg_list); + + // loop through and add any new packages for java.base + PackageEntry* pkg; + for (int x = 0; x < pkg_list->length(); x++) { + // Some of java.base's packages were added early in bootstrapping, ignore duplicates. + if (package_table->lookup_only(pkg_list->at(x)) == NULL) { + pkg = package_table->locked_create_entry_or_null(pkg_list->at(x), ModuleEntryTable::javabase_module()); + assert(pkg != NULL, "Unable to create a java.base package entry"); + } + // Unable to have a GrowableArray of TempNewSymbol. Must decrement the refcount of + // the Symbol* that was created above for each package. The refcount was incremented + // by SymbolTable::new_symbol and as well by the PackageEntry creation. + pkg_list->at(x)->decrement_refcount(); + } + + // Finish defining java.base's ModuleEntry + ModuleEntryTable::finalize_javabase(module_handle, version_symbol, location_symbol); + } + + log_debug(modules)("define_javabase_module(): Definition of module: java.base," + " version: %s, location: %s, package #: %d", + module_version != NULL ? module_version : "NULL", + module_location != NULL ? module_location : "NULL", + pkg_list->length()); + + // packages defined to java.base + for (int x = 0; x < pkg_list->length(); x++) { + log_trace(modules)("define_javabase_module(): creation of package %s for module java.base", + (pkg_list->at(x))->as_C_string()); + } + + // Patch any previously loaded classes' module field with java.base's jlr.Module. + ModuleEntryTable::patch_javabase_entries(module_handle); +} + +void Modules::define_module(jobject module, jstring version, + jstring location, jobjectArray packages, TRAPS) { + ResourceMark rm(THREAD); + + if (module == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), "Null module object"); + } + Handle module_handle(THREAD, JNIHandles::resolve(module)); + if (!java_lang_reflect_Module::is_subclass(module_handle->klass())) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "module is not a subclass of java.lang.reflect.Module"); + } + + char* module_name = get_module_name(module_handle(), CHECK); + if (module_name == NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "Module name cannot be null"); + } + + // Special handling of java.base definition + if (strcmp(module_name, "java.base") == 0) { + define_javabase_module(module, version, location, packages, CHECK); + return; + } + + const char* module_version = get_module_version(version); + + objArrayOop packages_oop = objArrayOop(JNIHandles::resolve(packages)); + objArrayHandle packages_h(THREAD, packages_oop); + int num_packages = (packages_h == NULL ? 0 : packages_h->length()); + + // Check that the list of packages has no duplicates and that the + // packages are syntactically ok. + GrowableArray* pkg_list = new GrowableArray(num_packages); + for (int x = 0; x < num_packages; x++) { + oop string_obj = packages_h->obj_at(x); + + if (string_obj == NULL || !string_obj->is_a(SystemDictionary::String_klass())) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Bad package name for module: %s", module_name)); + } + char *package_name = java_lang_String::as_utf8_string(string_obj); + if (!verify_package_name(package_name)) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Invalid package name: %s for module: %s", + package_name, module_name)); + } + Symbol* pkg_symbol = SymbolTable::new_symbol(package_name, CHECK); + // append_if_missing() returns FALSE if entry already exists. + if (!pkg_list->append_if_missing(pkg_symbol)) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Duplicate package name: %s for module %s", + package_name, module_name)); + } + } + + oop loader = java_lang_reflect_Module::loader(module_handle()); + // Make sure loader is not the sun.reflect.DelegatingClassLoader. + if (loader != java_lang_ClassLoader::non_reflection_class_loader(loader)) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "Class loader is an invalid delegating class loader"); + } + Handle h_loader = Handle(THREAD, loader); + + // Check that loader is a subclass of java.lang.ClassLoader. + if (loader != NULL && !java_lang_ClassLoader::is_subclass(h_loader->klass())) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "Class loader is not a subclass of java.lang.ClassLoader"); + } + + ModuleEntryTable* module_table = get_module_entry_table(h_loader, CHECK); + assert(module_table != NULL, "module entry table shouldn't be null"); + + // Create symbol* entry for module name. + TempNewSymbol module_symbol = SymbolTable::new_symbol(module_name, CHECK); + + int dupl_pkg_index = -1; + bool dupl_modules = false; + + // Create symbol* entry for module version. + TempNewSymbol version_symbol; + if (module_version != NULL) { + version_symbol = SymbolTable::new_symbol(module_version, CHECK); + } else { + version_symbol = NULL; + } + + // Create symbol* entry for module location. + const char* module_location = NULL; + TempNewSymbol location_symbol = NULL; + if (location != NULL) { + module_location = + java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(location)); + if (module_location != NULL) { + location_symbol = SymbolTable::new_symbol(module_location, CHECK); + } + } + + ClassLoaderData* loader_data = ClassLoaderData::class_loader_data_or_null(h_loader()); + assert(loader_data != NULL, "class loader data shouldn't be null"); + + PackageEntryTable* package_table = NULL; + { + MutexLocker ml(Module_lock, THREAD); + + if (num_packages > 0) { + package_table = get_package_entry_table(h_loader, CHECK); + assert(package_table != NULL, "Missing package_table"); + + // Check that none of the packages exist in the class loader's package table. + for (int x = 0; x < pkg_list->length(); x++) { + if (package_table->lookup_only(pkg_list->at(x)) != NULL) { + // This could be because the module was already defined. If so, + // report that error instead of the package error. + if (module_table->lookup_only(module_symbol) != NULL) { + dupl_modules = true; + } else { + dupl_pkg_index = x; + } + break; + } + } + } // if (num_packages > 0)... + + // Add the module and its packages. + if (!dupl_modules && dupl_pkg_index == -1) { + // Create the entry for this module in the class loader's module entry table. + + ModuleEntry* module_entry = module_table->locked_create_entry_or_null(module_handle, module_symbol, + version_symbol, location_symbol, loader_data); + + if (module_entry == NULL) { + dupl_modules = true; + } else { + // Add the packages. + assert(pkg_list->length() == 0 || package_table != NULL, "Bad package table"); + PackageEntry* pkg; + for (int y = 0; y < pkg_list->length(); y++) { + pkg = package_table->locked_create_entry_or_null(pkg_list->at(y), module_entry); + assert(pkg != NULL, "Unable to create a module's package entry"); + + // Unable to have a GrowableArray of TempNewSymbol. Must decrement the refcount of + // the Symbol* that was created above for each package. The refcount was incremented + // by SymbolTable::new_symbol and as well by the PackageEntry creation. + pkg_list->at(y)->decrement_refcount(); + } + + // Store pointer to ModuleEntry record in java.lang.reflect.Module object. + java_lang_reflect_Module::set_module_entry(module_handle(), module_entry); + } + } + } // Release the lock + + // any errors ? + if (dupl_modules) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Module %s is already defined", module_name)); + } + if (dupl_pkg_index != -1) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Package %s for module %s already exists for class loader", + pkg_list->at(dupl_pkg_index)->as_C_string(), module_name)); + } + + if (log_is_enabled(Debug, modules)) { + outputStream* logst = LogHandle(modules)::debug_stream(); + logst->print("define_module(): creation of module: %s, version: %s, location: %s, ", + module_name, module_version != NULL ? module_version : "NULL", + module_location != NULL ? module_location : "NULL"); + loader_data->print_value_on(logst); + logst->print_cr(", package #: %d", pkg_list->length()); + for (int y = 0; y < pkg_list->length(); y++) { + log_trace(modules)("define_module(): creation of package %s for module %s", + (pkg_list->at(y))->as_C_string(), module_name); + } + } + + if (loader == NULL && !Universe::is_module_initialized()) { + // Now that the module is defined, if it is in the bootloader, make sure that + // its classes can be found. Check if -Xpatch: was specified. If + // so prepend /module_name, if it exists, to bootpath. Also, if using + // exploded modules, prepend /modules/module_name, if it exists, + // to bootpath. + add_to_boot_loader_list(module_name, CHECK); + } +} + +void Modules::set_bootloader_unnamed_module(jobject module, TRAPS) { + ResourceMark rm(THREAD); + + if (module == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), "Null module object"); + } + Handle module_handle(THREAD, JNIHandles::resolve(module)); + if (!java_lang_reflect_Module::is_subclass(module_handle->klass())) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "module is not a subclass of java.lang.reflect.Module"); + } + + // Ensure that this is an unnamed module + oop name = java_lang_reflect_Module::name(module_handle()); + if (name != NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "boot loader's unnamed module's java.lang.reflect.Module has a name"); + } + + // Validate java_base's loader is the boot loader. + oop loader = java_lang_reflect_Module::loader(module_handle()); + if (loader != NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "Class loader must be the boot class loader"); + } + Handle h_loader = Handle(THREAD, loader); + + log_debug(modules)("set_bootloader_unnamed_module(): recording unnamed module for boot loader"); + + // Ensure the boot loader's PackageEntryTable has been created + ModuleEntryTable* module_table = get_module_entry_table(h_loader, CHECK); + + // Set java.lang.reflect.Module for the boot loader's unnamed module + ModuleEntry* unnamed_module = module_table->unnamed_module(); + assert(unnamed_module != NULL, "boot loader's unnamed ModuleEntry not defined"); + unnamed_module->set_module(ClassLoaderData::the_null_class_loader_data()->add_handle(module_handle)); + // Store pointer to the ModuleEntry in the unnamed module's java.lang.reflect.Module object. + java_lang_reflect_Module::set_module_entry(module_handle(), unnamed_module); +} + +void Modules::add_module_exports(jobject from_module, jstring package, jobject to_module, TRAPS) { + if (package == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), + "package is null"); + } + if (from_module == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), + "from_module is null"); + } + ModuleEntry* from_module_entry = get_module_entry(from_module, CHECK); + if (from_module_entry == NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "from_module cannot be found"); + } + + // All packages in unnamed are exported by default. + if (!from_module_entry->is_named()) return; + + ModuleEntry* to_module_entry; + if (to_module == NULL) { + to_module_entry = NULL; // It's an unqualified export. + } else { + to_module_entry = get_module_entry(to_module, CHECK); + if (to_module_entry == NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "to_module is invalid"); + } + } + + PackageEntry *package_entry = get_package_entry(from_module_entry, package, CHECK); + ResourceMark rm(THREAD); + if (package_entry == NULL) { + const char *package_name = java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(package)); + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Package %s not found in from_module %s", + package_name != NULL ? package_name : "", + from_module_entry->name()->as_C_string())); + } + if (package_entry->module() != from_module_entry) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Package: %s found in module %s, not in from_module: %s", + package_entry->name()->as_C_string(), + package_entry->module()->name()->as_C_string(), + from_module_entry->name()->as_C_string())); + } + + log_debug(modules)("add_module_exports(): package %s in module %s is exported to module %s", + package_entry->name()->as_C_string(), + from_module_entry->name()->as_C_string(), + to_module_entry == NULL ? "NULL" : + to_module_entry->is_named() ? + to_module_entry->name()->as_C_string() : UNNAMED_MODULE); + + // Do nothing if modules are the same or if package is already exported unqualifiedly. + if (from_module_entry != to_module_entry && !package_entry->is_unqual_exported()) { + package_entry->set_exported(to_module_entry); + } +} + + +void Modules::add_module_exports_qualified(jobject from_module, jstring package, + jobject to_module, TRAPS) { + if (to_module == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), + "to_module is null"); + } + add_module_exports(from_module, package, to_module, CHECK); +} + +void Modules::add_reads_module(jobject from_module, jobject to_module, TRAPS) { + if (from_module == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), + "from_module is null"); + } + + ModuleEntry* from_module_entry = get_module_entry(from_module, CHECK); + if (from_module_entry == NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "from_module is not valid"); + } + + ModuleEntry* to_module_entry; + if (to_module != NULL) { + to_module_entry = get_module_entry(to_module, CHECK); + if (to_module_entry == NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "to_module is invalid"); + } + } else { + to_module_entry = NULL; + } + + ResourceMark rm(THREAD); + log_debug(modules)("add_reads_module(): Adding read from module %s to module %s", + from_module_entry->is_named() ? + from_module_entry->name()->as_C_string() : UNNAMED_MODULE, + to_module_entry == NULL ? "all unnamed" : + (to_module_entry->is_named() ? + to_module_entry->name()->as_C_string() : UNNAMED_MODULE)); + + // if modules are the same or if from_module is unnamed then no need to add the read. + if (from_module_entry != to_module_entry && from_module_entry->is_named()) { + from_module_entry->add_read(to_module_entry); + } +} + +jboolean Modules::can_read_module(jobject asking_module, jobject target_module, TRAPS) { + if (asking_module == NULL) { + THROW_MSG_(vmSymbols::java_lang_NullPointerException(), + "asking_module is null", JNI_FALSE); + } + + ModuleEntry* asking_module_entry = get_module_entry(asking_module, CHECK_false); + if (asking_module_entry == NULL) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + "asking_module is invalid", JNI_FALSE); + } + + // Calling can_read_all_unnamed() with NULL tests if a module is loose. + if (target_module == NULL) { + return asking_module_entry->can_read_all_unnamed(); + } + + ModuleEntry* target_module_entry = get_module_entry(target_module, CHECK_false); + if (target_module_entry == NULL) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + "target_module is invalid", JNI_FALSE); + } + + ResourceMark rm(THREAD); + log_debug(modules)("can_read_module(): module %s trying to read module %s, allowed = %s", + asking_module_entry->is_named() ? + asking_module_entry->name()->as_C_string() : UNNAMED_MODULE, + target_module_entry->is_named() ? + target_module_entry->name()->as_C_string() : UNNAMED_MODULE, + BOOL_TO_STR(asking_module_entry == target_module_entry || + (asking_module_entry->can_read_all_unnamed() && + !target_module_entry->is_named()) || + asking_module_entry->can_read(target_module_entry))); + + // Return true if: + // 1. the modules are the same, or + // 2. the asking_module is unnamed (because unnamed modules read everybody), or + // 3. the asking_module is loose and the target module is unnamed, or + // 4. if can_read() returns true. + if (asking_module_entry == target_module_entry || + (asking_module_entry->can_read_all_unnamed() && !target_module_entry->is_named())) { + return true; + } + return asking_module_entry->can_read(target_module_entry); +} + +jboolean Modules::is_exported_to_module(jobject from_module, jstring package, + jobject to_module, TRAPS) { + if (package == NULL) { + THROW_MSG_(vmSymbols::java_lang_NullPointerException(), + "package is null", JNI_FALSE); + } + if (from_module == NULL) { + THROW_MSG_(vmSymbols::java_lang_NullPointerException(), + "from_module is null", JNI_FALSE); + } + ModuleEntry* from_module_entry = get_module_entry(from_module, CHECK_false); + if (from_module_entry == NULL) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + "from_module is invalid", JNI_FALSE); + } + ModuleEntry* to_module_entry; + if (to_module == NULL) { + THROW_MSG_(vmSymbols::java_lang_NullPointerException(), + "to_module is null", JNI_FALSE); + } + to_module_entry = get_module_entry(to_module, CHECK_false); + if (to_module_entry == NULL) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + "to_module is invalid", JNI_FALSE); + } + + PackageEntry *package_entry = get_package_entry(from_module_entry, package, + CHECK_false); + ResourceMark rm(THREAD); + if (package_entry == NULL) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Package not found in from_module: %s", + from_module_entry->is_named() ? + from_module_entry->name()->as_C_string() : UNNAMED_MODULE), + JNI_FALSE); + } + if (package_entry->module() != from_module_entry) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Package: %s found in module %s, not in from_module: %s", + package_entry->name()->as_C_string(), + package_entry->module()->is_named() ? + package_entry->module()->name()->as_C_string() : UNNAMED_MODULE, + from_module_entry->is_named() ? + from_module_entry->name()->as_C_string() : UNNAMED_MODULE), + JNI_FALSE); + } + + log_debug(modules)("is_exported_to_module: package %s from module %s checking" + " if exported to module %s, exported? = %s", + package_entry->name()->as_C_string(), + from_module_entry->is_named() ? + from_module_entry->name()->as_C_string() : UNNAMED_MODULE, + to_module_entry->is_named() ? + to_module_entry->name()->as_C_string() : UNNAMED_MODULE, + BOOL_TO_STR(!from_module_entry->is_named() || + package_entry->is_unqual_exported() || + from_module_entry == to_module_entry || + package_entry->is_qexported_to(to_module_entry))); + + // Return true if: + // 1. from_module is unnamed because unnamed modules export all their packages (by default), or + // 2. if the package is unqualifiedly exported, or + // 3. if the modules are the same, or + // 4. if the package is exported to to_module + return (!from_module_entry->is_named() || + package_entry->is_unqual_exported() || + from_module_entry == to_module_entry || + package_entry->is_qexported_to(to_module_entry)); +} + +// This method is called by JFR and JNI. +jobject Modules::get_module(jclass clazz, TRAPS) { + assert(ModuleEntryTable::javabase_defined(), "Attempt to call get_module before java.base is defined"); + + if (clazz == NULL) { + THROW_MSG_(vmSymbols::java_lang_NullPointerException(), + "class is null", JNI_FALSE); + } + oop mirror = JNIHandles::resolve_non_null(clazz); + if (mirror == NULL) { + log_debug(modules)("get_module(): no mirror, returning NULL"); + return NULL; + } + if (!java_lang_Class::is_instance(mirror)) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + "Invalid class", JNI_FALSE); + } + + oop module = java_lang_Class::module(mirror); + + assert(module != NULL, "java.lang.Class module field not set"); + assert(java_lang_reflect_Module::is_subclass(module->klass()), "Module is not a java.lang.reflect.Module"); + + if (log_is_enabled(Debug, modules)) { + ResourceMark rm(THREAD); + outputStream* logst = LogHandle(modules)::debug_stream(); + Klass* klass = java_lang_Class::as_Klass(mirror); + oop module_name = java_lang_reflect_Module::name(module); + if (module_name != NULL) { + logst->print("get_module(): module "); + java_lang_String::print(module_name, tty); + } else { + logst->print("get_module(): Unamed Module"); + } + if (klass != NULL) { + logst->print_cr(" for class %s", klass->external_name()); + } else { + logst->print_cr(" for primitive class"); + } + } + + return JNIHandles::make_local(THREAD, module); +} + + +jobject Modules::get_module_by_package_name(jobject loader, jstring package, TRAPS) { + ResourceMark rm(THREAD); + assert(ModuleEntryTable::javabase_defined(), + "Attempt to call get_module_from_pkg before java.base is defined"); + + if (NULL == package) { + THROW_MSG_(vmSymbols::java_lang_NullPointerException(), + "package is null", JNI_FALSE); + } + const char* package_str = + java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(package)); + if (NULL == package_str) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + "Invalid package", JNI_FALSE); + } + + Handle h_loader (THREAD, JNIHandles::resolve(loader)); + // Check that loader is a subclass of java.lang.ClassLoader. + if (loader != NULL && !java_lang_ClassLoader::is_subclass(h_loader->klass())) { + THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), + "Class loader is not a subclass of java.lang.ClassLoader", JNI_FALSE); + } + + if (strlen(package_str) == 0) { + // Return the unnamed module + ModuleEntryTable* module_table = get_module_entry_table(h_loader, CHECK_NULL); + if (NULL == module_table) return NULL; + const ModuleEntry* const unnamed_module = module_table->unnamed_module(); + return JNIHandles::make_local(THREAD, JNIHandles::resolve(unnamed_module->module())); + + } else { + TempNewSymbol package_sym = SymbolTable::new_symbol(package_str, CHECK_NULL); + return get_module(package_sym, h_loader, CHECK_NULL); + } + return NULL; +} + + +// This method is called by JFR and by the above method. +jobject Modules::get_module(Symbol* package_name, Handle h_loader, TRAPS) { + const PackageEntry* const pkg_entry = + get_package_entry_by_name(package_name, h_loader, THREAD); + const ModuleEntry* const module_entry = (pkg_entry != NULL ? pkg_entry->module() : NULL); + + if (module_entry != NULL && + module_entry->module() != NULL) { + return JNIHandles::make_local(THREAD, JNIHandles::resolve(module_entry->module())); + } + + return NULL; +} + +void Modules::add_module_package(jobject module, jstring package, TRAPS) { + ResourceMark rm(THREAD); + + if (module == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), + "module is null"); + } + if (package == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), + "package is null"); + } + ModuleEntry* module_entry = get_module_entry(module, CHECK); + if (module_entry == NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "module is invalid"); + } + if (!module_entry->is_named()) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "module cannot be an unnamed module"); + } + char *package_name = java_lang_String::as_utf8_string( + JNIHandles::resolve_non_null(package)); + if (package_name == NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), "Bad package"); + } + if (!verify_package_name(package_name)) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Invalid package name: %s", package_name)); + } + + log_debug(modules)("add_module_package(): Adding package %s to module %s", + package_name, module_entry->name()->as_C_string()); + + TempNewSymbol pkg_symbol = SymbolTable::new_symbol(package_name, CHECK); + PackageEntryTable* package_table = module_entry->loader()->packages(); + assert(package_table != NULL, "Missing package_table"); + + bool pkg_exists = false; + { + MutexLocker ml(Module_lock, THREAD); + + // Check that the package does not exist in the class loader's package table. + if (!package_table->lookup_only(pkg_symbol)) { + PackageEntry* pkg = package_table->locked_create_entry_or_null(pkg_symbol, module_entry); + assert(pkg != NULL, "Unable to create a module's package entry"); + } else { + pkg_exists = true; + } + } + if (pkg_exists) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Package %s already exists for class loader", package_name)); + } +} + +// Export package in module to all unnamed modules. +void Modules::add_module_exports_to_all_unnamed(jobject module, jstring package, TRAPS) { + if (module == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), + "module is null"); + } + if (package == NULL) { + THROW_MSG(vmSymbols::java_lang_NullPointerException(), + "package is null"); + } + ModuleEntry* module_entry = get_module_entry(module, CHECK); + if (module_entry == NULL) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + "module is invalid"); + } + + if (module_entry->is_named()) { // No-op for unnamed module. + PackageEntry *package_entry = get_package_entry(module_entry, package, CHECK); + ResourceMark rm(THREAD); + if (package_entry == NULL) { + const char *package_name = java_lang_String::as_utf8_string(JNIHandles::resolve_non_null(package)); + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Package %s not found in module %s", + package_name != NULL ? package_name : "", + module_entry->name()->as_C_string())); + } + if (package_entry->module() != module_entry) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), + err_msg("Package: %s found in module %s, not in module: %s", + package_entry->name()->as_C_string(), + package_entry->module()->name()->as_C_string(), + module_entry->name()->as_C_string())); + } + + log_debug(modules)("add_module_exports_to_all_unnamed(): package %s in module" + " %s is exported to all unnamed modules", + package_entry->name()->as_C_string(), + module_entry->name()->as_C_string()); + + // Mark package as exported to all unnamed modules, unless already + // unqualifiedly exported. + if (!package_entry->is_unqual_exported()) { + package_entry->set_is_exported_allUnnamed(); + } + } +} diff --git a/hotspot/src/share/vm/classfile/modules.hpp b/hotspot/src/share/vm/classfile/modules.hpp new file mode 100644 index 00000000000..d8abd7552bc --- /dev/null +++ b/hotspot/src/share/vm/classfile/modules.hpp @@ -0,0 +1,151 @@ +/* +* 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. +* +*/ + +#ifndef SHARE_VM_CLASSFILE_MODULES_HPP +#define SHARE_VM_CLASSFILE_MODULES_HPP + +#include "memory/allocation.hpp" +#include "runtime/handles.hpp" + +class Symbol; + +class Modules : AllStatic { + +public: + // define_module defines a module containing the specified packages. It binds the + // module to its class loader by creating the ModuleEntry record in the + // ClassLoader's ModuleEntry table, and creates PackageEntry records in the class + // loader's PackageEntry table. As in JVM_DefineClass the jstring format for all + // package names must use "/" and not "." + // + // IllegalArgumentExceptions are thrown for the following : + // * Module's Class loader is not a subclass of java.lang.ClassLoader + // * Module's Class loader already has a module with that name + // * Module's Class loader has already defined types for any of the module's packages + // * Module_name is syntactically bad + // * Packages contains an illegal package name + // * Packages contains a duplicate package name + // * A package already exists in another module for this class loader + // * Module is an unnamed module + // NullPointerExceptions are thrown if module is null. + static void define_module(jobject module, jstring version, + jstring location, jobjectArray packages, TRAPS); + + // Provides the java.lang.reflect.Module for the unnamed module defined + // to the boot loader. + // + // IllegalArgumentExceptions are thrown for the following : + // * Module has a name + // * Module is not a subclass of java.lang.reflect.Module + // * Module's class loader is not the boot loader + // NullPointerExceptions are thrown if module is null. + static void set_bootloader_unnamed_module(jobject module, TRAPS); + + // This either does a qualified export of package in module from_module to module + // to_module or, if to_module is null, does an unqualified export of package. + // The format for the package name must use "/' not ".". + // + // Error conditions causing IlegalArgumentException to be throw : + // * Module from_module does not exist + // * Module to_module is not null and does not exist + // * Package is not syntactically correct + // * Package is not defined for from_module's class loader + // * Package is not in module from_module. + static void add_module_exports(jobject from_module, jstring package, jobject to_module, TRAPS); + + // This does a qualified export of package in module from_module to module + // to_module. The format for the package name must use "/' not ".". + // + // Error conditions causing IlegalArgumentException to be throw : + // * Module from_module does not exist + // * Module to_module does not exist + // * Package is not syntactically correct + // * Package is not defined for from_module's class loader + // * Package is not in module from_module. + static void add_module_exports_qualified(jobject from_module, jstring package, jobject to_module, TRAPS); + + // add_reads_module adds module to_module to the list of modules that from_module + // can read. If from_module is the same as to_module then this is a no-op. + // If to_module is null then from_module is marked as a loose module (meaning that + // from_module can read all current and future unnamed modules). + // An IllegalArgumentException is thrown if from_module is null or either (non-null) + // module does not exist. + static void add_reads_module(jobject from_module, jobject to_module, TRAPS); + + // can_read_module returns TRUE if module asking_module can read module target_module, + // or if they are the same module, or if the asking_module is loose and target_module + // is null. + // + // Throws IllegalArgumentException if: + // * either asking_module or target_module is not a java.lang.reflect.Module + static jboolean can_read_module(jobject asking_module, jobject target_module, TRAPS); + + // If package is valid then this returns TRUE if module from_module exports + // package to module to_module, if from_module and to_module are the same + // module, or if package is exported without qualification. + // + // IllegalArgumentException is throw if: + // * Either to_module or from_module does not exist + // * package is syntactically incorrect + // * package is not in from_module + static jboolean is_exported_to_module(jobject from_module, jstring package, jobject to_module, TRAPS); + + // Return the java.lang.reflect.Module object for this class object. + static jobject get_module(jclass clazz, TRAPS); + + // Return the java.lang.reflect.Module object for this class loader and package. + // Returns NULL if the class loader has not loaded any classes in the package. + // The package should contain /'s, not .'s, as in java/lang, not java.lang. + // NullPointerException is thrown if package is null. + // IllegalArgumentException is thrown if loader is neither null nor a subtype of + // java/lang/ClassLoader. + static jobject get_module_by_package_name(jobject loader, jstring package, TRAPS); + + // If package is defined by loader, return the + // java.lang.reflect.Module object for the module in which the package is defined. + // Returns NULL if package is invalid or not defined by loader. + static jobject get_module(Symbol* package_name, Handle h_loader, TRAPS); + + // This adds package to module. + // It throws IllegalArgumentException if: + // * Module is bad + // * Module is unnamed + // * Package is not syntactically correct + // * Package is already defined for module's class loader. + static void add_module_package(jobject module, jstring package, TRAPS); + + // Marks the specified package as exported to all unnamed modules. + // If either module or package is null then NullPointerException is thrown. + // If module or package is bad, or module is unnamed, or package is not in + // module then IllegalArgumentException is thrown. + static void add_module_exports_to_all_unnamed(jobject module, jstring package, TRAPS); + + // Return TRUE if package_name is syntactically valid, false otherwise. + static bool verify_package_name(char *package_name); + + // Return TRUE iff package is defined by loader + static bool is_package_defined(Symbol* package_name, Handle h_loader, TRAPS); +}; + +#endif // SHARE_VM_CLASSFILE_MODULES_HPP diff --git a/hotspot/src/share/vm/classfile/packageEntry.cpp b/hotspot/src/share/vm/classfile/packageEntry.cpp new file mode 100644 index 00000000000..b3117d483c1 --- /dev/null +++ b/hotspot/src/share/vm/classfile/packageEntry.cpp @@ -0,0 +1,322 @@ +/* + * 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. + * + */ + +#include "precompiled.hpp" +#include "classfile/moduleEntry.hpp" +#include "classfile/packageEntry.hpp" +#include "memory/resourceArea.hpp" +#include "oops/symbol.hpp" +#include "runtime/handles.inline.hpp" +#include "trace/traceMacros.hpp" +#include "utilities/events.hpp" +#include "utilities/growableArray.hpp" +#include "utilities/hashtable.inline.hpp" + +// Return true if this package is exported to m. +bool PackageEntry::is_qexported_to(ModuleEntry* m) const { + assert(m != NULL, "No module to lookup in this package's qualified exports list"); + MutexLocker m1(Module_lock); + if (!_is_exported) { + return false; + } else if (_is_exported_allUnnamed && !m->is_named()) { + return true; + } else if (_qualified_exports == NULL) { + return false; + } else { + return _qualified_exports->contains(m); + } +} + +// Add a module to the package's qualified export list. +void PackageEntry::add_qexport(ModuleEntry* m) { + assert_locked_or_safepoint(Module_lock); + assert(_is_exported == true, "Adding a qualified export to a package that is not exported"); + if (_qualified_exports == NULL) { + // Lazily create a package's qualified exports list. + // Initial size is small, do not anticipate export lists to be large. + _qualified_exports = + new (ResourceObj::C_HEAP, mtClass) GrowableArray(QUAL_EXP_SIZE, true); + } + _qualified_exports->append_if_missing(m); +} + +// Set the package's exported state based on the value of the ModuleEntry. +void PackageEntry::set_exported(ModuleEntry* m) { + MutexLocker m1(Module_lock); + if (is_unqual_exported()) { + // An exception could be thrown, but choose to simply ignore. + // Illegal to convert an unqualified exported package to be qualifiedly exported + return; + } + + if (m == NULL) { + // NULL indicates the package is being unqualifiedly exported + if (_is_exported && _qualified_exports != NULL) { + // Legit to transition a package from being qualifiedly exported + // to unqualified. Clean up the qualified lists at the next + // safepoint. + _exported_pending_delete = _qualified_exports; + } + + // Mark package as unqualifiedly exported + set_unqual_exported(); + + } else { + // Add the exported module + _is_exported = true; + add_qexport(m); + } +} + +// Remove dead module entries within the package's exported list. +void PackageEntry::purge_qualified_exports() { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint"); + if (_qualified_exports != NULL) { + // Go backwards because this removes entries that are dead. + int len = _qualified_exports->length(); + for (int idx = len - 1; idx >= 0; idx--) { + ModuleEntry* module_idx = _qualified_exports->at(idx); + ClassLoaderData* cld = module_idx->loader(); + if (cld->is_unloading()) { + _qualified_exports->delete_at(idx); + } + } + } +} + +void PackageEntry::delete_qualified_exports() { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint"); + if (_exported_pending_delete != NULL) { + // If a transition occurred from qualified to unqualified, the _qualified_exports + // field should have been NULL'ed out. + assert(_qualified_exports == NULL, "Package's exported pending delete, exported list should not be active"); + delete _exported_pending_delete; + } + + if (_qualified_exports != NULL) { + delete _qualified_exports; + } + + _exported_pending_delete = NULL; + _qualified_exports = NULL; +} + +PackageEntryTable::PackageEntryTable(int table_size) + : Hashtable(table_size, sizeof(PackageEntry)) +{ +} + +PackageEntryTable::~PackageEntryTable() { + assert_locked_or_safepoint(Module_lock); + + // Walk through all buckets and all entries in each bucket, + // freeing each entry. + for (int i = 0; i < table_size(); ++i) { + for (PackageEntry* p = bucket(i); p != NULL;) { + PackageEntry* to_remove = p; + // read next before freeing. + p = p->next(); + + // Clean out the C heap allocated qualified exports list first before freeing the entry + to_remove->delete_qualified_exports(); + to_remove->name()->decrement_refcount(); + + // Unlink from the Hashtable prior to freeing + unlink_entry(to_remove); + FREE_C_HEAP_ARRAY(char, to_remove); + } + } + assert(number_of_entries() == 0, "should have removed all entries"); + assert(new_entry_free_list() == NULL, "entry present on PackageEntryTable's free list"); + free_buckets(); +} + +PackageEntry* PackageEntryTable::new_entry(unsigned int hash, Symbol* name, ModuleEntry* module) { + assert_locked_or_safepoint(Module_lock); + PackageEntry* entry = (PackageEntry*) NEW_C_HEAP_ARRAY(char, entry_size(), mtClass); + + // Initialize everything BasicHashtable would + entry->set_next(NULL); + entry->set_hash(hash); + entry->set_literal(name); + + TRACE_INIT_PACKAGE_ID(entry); + + // Initialize fields specific to a PackageEntry + entry->init(); + entry->name()->increment_refcount(); + if (!module->is_named()) { + // Set the exported state to true because all packages + // within the unnamed module are unqualifiedly exported + entry->set_exported(true); + } + entry->set_module(module); + return entry; +} + +void PackageEntryTable::add_entry(int index, PackageEntry* new_entry) { + assert_locked_or_safepoint(Module_lock); + Hashtable::add_entry(index, (HashtableEntry*)new_entry); +} + +// Create package in loader's package entry table and return the entry. +// If entry already exists, return null. Assume Module lock was taken by caller. +PackageEntry* PackageEntryTable::locked_create_entry_or_null(Symbol* name, ModuleEntry* module) { + assert_locked_or_safepoint(Module_lock); + // Check if package already exists. Return NULL if it does. + if (lookup_only(name) != NULL) { + return NULL; + } else { + PackageEntry* entry = new_entry(compute_hash(name), name, module); + add_entry(index_for(name), entry); + return entry; + } +} + +PackageEntry* PackageEntryTable::lookup(Symbol* name, ModuleEntry* module) { + PackageEntry* p = lookup_only(name); + if (p != NULL) { + return p; + } else { + // If not found, add to table. Grab the PackageEntryTable lock first. + MutexLocker ml(Module_lock); + + // Since look-up was done lock-free, we need to check if another thread beat + // us in the race to insert the package. + PackageEntry* test = lookup_only(name); + if (test != NULL) { + // A race occurred and another thread introduced the package. + return test; + } else { + assert(module != NULL, "module should never be null"); + PackageEntry* entry = new_entry(compute_hash(name), name, module); + add_entry(index_for(name), entry); + return entry; + } + } +} + +PackageEntry* PackageEntryTable::lookup_only(Symbol* name) { + int index = index_for(name); + for (PackageEntry* p = bucket(index); p != NULL; p = p->next()) { + if (p->name()->fast_compare(name) == 0) { + return p; + } + } + return NULL; +} + +// Called when a define module for java.base is being processed. +// Verify the packages loaded thus far are in java.base's package list. +void PackageEntryTable::verify_javabase_packages(GrowableArray *pkg_list) { + for (int i = 0; i < table_size(); i++) { + for (PackageEntry* entry = bucket(i); + entry != NULL; + entry = entry->next()) { + ModuleEntry* m = entry->module(); + Symbol* module_name = (m == NULL ? NULL : m->name()); + if (module_name != NULL && + (module_name->fast_compare(vmSymbols::java_base()) == 0) && + !pkg_list->contains(entry->name())) { + ResourceMark rm; + vm_exit_during_initialization("A non-java.base package was loaded prior to module system initialization", entry->name()->as_C_string()); + } + } + } + +} + +// Remove dead entries from all packages' exported list +void PackageEntryTable::purge_all_package_exports() { + assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint"); + for (int i = 0; i < table_size(); i++) { + for (PackageEntry* entry = bucket(i); + entry != NULL; + entry = entry->next()) { + if (entry->exported_pending_delete()) { + // exported list is pending deletion due to a transition + // from qualified to unqualified + entry->delete_qualified_exports(); + } else if (entry->is_qual_exported()) { + entry->purge_qualified_exports(); + } + } + } +} + +#ifndef PRODUCT +void PackageEntryTable::print() { + tty->print_cr("Package Entry Table (table_size=%d, entries=%d)", + table_size(), number_of_entries()); + for (int i = 0; i < table_size(); i++) { + for (PackageEntry* probe = bucket(i); + probe != NULL; + probe = probe->next()) { + probe->print(); + } + } +} + +void PackageEntry::print() { + ResourceMark rm; + tty->print_cr("package entry "PTR_FORMAT" name %s module %s classpath_index " + INT32_FORMAT " is_exported %d is_exported_allUnnamed %d " "next "PTR_FORMAT, + p2i(this), name()->as_C_string(), + (module()->is_named() ? module()->name()->as_C_string() : UNNAMED_MODULE), + _classpath_index, _is_exported, _is_exported_allUnnamed, p2i(next())); +} +#endif + +void PackageEntryTable::verify() { + int element_count = 0; + for (int index = 0; index < table_size(); index++) { + for (PackageEntry* probe = bucket(index); + probe != NULL; + probe = probe->next()) { + probe->verify(); + element_count++; + } + } + guarantee(number_of_entries() == element_count, + "Verify of Package Entry Table failed"); + debug_only(verify_lookup_length((double)number_of_entries() / table_size())); +} + +void PackageEntry::verify() { + guarantee(name() != NULL, "A package entry must have a corresponding symbol name."); +} + +// iteration of qualified exports +void PackageEntry::package_exports_do(ModuleClosure* const f) { + assert_locked_or_safepoint(Module_lock); + assert(f != NULL, "invariant"); + + if (is_qual_exported()) { + int qe_len = _qualified_exports->length(); + + for (int i = 0; i < qe_len; ++i) { + f->do_module(_qualified_exports->at(i)); + } + } +} diff --git a/hotspot/src/share/vm/classfile/packageEntry.hpp b/hotspot/src/share/vm/classfile/packageEntry.hpp new file mode 100644 index 00000000000..368609c21b6 --- /dev/null +++ b/hotspot/src/share/vm/classfile/packageEntry.hpp @@ -0,0 +1,202 @@ +/* + * 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. + * + */ + +#ifndef SHARE_VM_CLASSFILE_PACKAGEENTRY_HPP +#define SHARE_VM_CLASSFILE_PACKAGEENTRY_HPP + +#include "classfile/moduleEntry.hpp" +#include "oops/symbol.hpp" +#include "utilities/growableArray.hpp" +#include "utilities/hashtable.hpp" + +// A PackageEntry basically represents a Java package. It contains: +// - Symbol* containing the package's name. +// - ModuleEntry* for this package's containing module. +// - a flag indicating if package is exported, either qualifiedly or +// unqualifiedly. +// - a flag indicating if this package is exported to all unnamed modules. +// - a growable array containing other module entries that this +// package is exported to. +// +// Packages that are: +// - not exported: _qualified_exports = NULL && _is_exported is false +// - qualified exports: (_qualified_exports != NULL || _is_exported_allUnnamed is true) && _is_exported is true +// - unqualified exports: (_qualified_exports = NULL && _is_exported_allUnnamed is false) && _is_exported is true +// +// The Mutex Module_lock is shared between ModuleEntry and PackageEntry, to lock either +// data structure. +class PackageEntry : public HashtableEntry { +private: + ModuleEntry* _module; + // Used to indicate for packages with classes loaded by the boot loader that + // a class in that package has been loaded. And, for packages with classes + // loaded by the boot loader from -Xbootclasspath/a in an unnamed module, it + // indicates from which class path entry. + s2 _classpath_index; + bool _is_exported; + bool _is_exported_allUnnamed; + GrowableArray* _exported_pending_delete; // transitioned from qualified to unqualified, delete at safepoint + GrowableArray* _qualified_exports; + TRACE_DEFINE_TRACE_ID_FIELD; + + // Initial size of a package entry's list of qualified exports. + enum {QUAL_EXP_SIZE = 43}; + +public: + void init() { + _module = NULL; + _classpath_index = -1; + _is_exported = false; + _is_exported_allUnnamed = false; + _exported_pending_delete = NULL; + _qualified_exports = NULL; + } + + // package name + Symbol* name() const { return literal(); } + void set_name(Symbol* n) { set_literal(n); } + + // the module containing the package definition + ModuleEntry* module() const { return _module; } + void set_module(ModuleEntry* m) { _module = m; } + + // package's export state + bool is_exported() const { return _is_exported; } // qualifiedly or unqualifiedly exported + bool is_qual_exported() const { + return (_is_exported && (_qualified_exports != NULL || _is_exported_allUnnamed)); + } + bool is_unqual_exported() const { + return (_is_exported && (_qualified_exports == NULL && !_is_exported_allUnnamed)); + } + void set_unqual_exported() { + _is_exported = true; + _is_exported_allUnnamed = false; + _qualified_exports = NULL; + } + bool exported_pending_delete() const { return (_exported_pending_delete != NULL); } + + void set_exported(bool e) { _is_exported = e; } + void set_exported(ModuleEntry* m); + + void set_is_exported_allUnnamed() { + if (!is_unqual_exported()) { + _is_exported_allUnnamed = true; + _is_exported = true; + } + } + bool is_exported_allUnnamed() const { + assert(_is_exported || !_is_exported_allUnnamed, + "is_allUnnamed set without is_exported being set"); + return _is_exported_allUnnamed; + } + + void set_classpath_index(s2 classpath_index) { + _classpath_index = classpath_index; + } + s2 classpath_index() const { return _classpath_index; } + + bool has_loaded_class() const { return _classpath_index != -1; } + + // returns true if the package is defined in the unnamed module + bool in_unnamed_module() const { return !_module->is_named(); } + + // returns true if the package specifies m as a qualified export + bool is_qexported_to(ModuleEntry* m) const; + + // add the module to the package's qualified exports + void add_qexport(ModuleEntry* m); + + PackageEntry* next() const { + return (PackageEntry*)HashtableEntry::next(); + } + + PackageEntry** next_addr() { + return (PackageEntry**)HashtableEntry::next_addr(); + } + + // iteration of qualified exports + void package_exports_do(ModuleClosure* const f); + + TRACE_DEFINE_TRACE_ID_METHODS; + + // Purge dead weak references out of exported list when any given class loader is unloaded. + void purge_qualified_exports(); + void delete_qualified_exports(); + + void print() PRODUCT_RETURN; + void verify(); +}; + +// The PackageEntryTable is a Hashtable containing a list of all packages defined +// by a particular class loader. Each package is represented as a PackageEntry node. +// The PackageEntryTable's lookup is lock free. +// +class PackageEntryTable : public Hashtable { + friend class VMStructs; +public: + enum Constants { + _packagetable_entry_size = 1009 // number of entries in package entry table + }; + +private: + PackageEntry* new_entry(unsigned int hash, Symbol* name, ModuleEntry* module); + void add_entry(int index, PackageEntry* new_entry); + + int entry_size() const { return BasicHashtable::entry_size(); } + + PackageEntry** bucket_addr(int i) { + return (PackageEntry**)Hashtable::bucket_addr(i); + } + + static unsigned int compute_hash(Symbol* name) { return (unsigned int)(name->identity_hash()); } + int index_for(Symbol* name) const { return hash_to_index(compute_hash(name)); } + +public: + PackageEntryTable(int table_size); + ~PackageEntryTable(); + + PackageEntry* bucket(int i) { + return (PackageEntry*)Hashtable::bucket(i); + } + + // Create package in loader's package entry table and return the entry. + // If entry already exists, return null. Assume Module lock was taken by caller. + PackageEntry* locked_create_entry_or_null(Symbol* name, ModuleEntry* module); + + // lookup Package with loader's package entry table, if not found add + PackageEntry* lookup(Symbol* name, ModuleEntry* module); + + // Only lookup Package within loader's package entry table. The table read is lock-free. + PackageEntry* lookup_only(Symbol* Package); + + void verify_javabase_packages(GrowableArray *pkg_list); + + // purge dead weak references out of exported list + void purge_all_package_exports(); + + void print() PRODUCT_RETURN; + void verify(); +}; + +#endif // SHARE_VM_CLASSFILE_PACKAGEENTRY_HPP diff --git a/hotspot/src/share/vm/classfile/sharedPathsMiscInfo.cpp b/hotspot/src/share/vm/classfile/sharedPathsMiscInfo.cpp index 40843700bd6..939e136de3c 100644 --- a/hotspot/src/share/vm/classfile/sharedPathsMiscInfo.cpp +++ b/hotspot/src/share/vm/classfile/sharedPathsMiscInfo.cpp @@ -77,7 +77,7 @@ void SharedPathsMiscInfo::print_path(int type, const char* path) { outputStream* out = LogHandle(classpath)::info_stream(); switch (type) { case BOOT: - out->print("Expecting -Dsun.boot.class.path=%s", path); + out->print("Expecting BOOT path=%s", path); break; case NON_EXIST: out->print("Expecting that %s does not exist", path); @@ -126,7 +126,7 @@ bool SharedPathsMiscInfo::check(jint type, const char* path) { switch (type) { case BOOT: if (os::file_name_strcmp(path, Arguments::get_sysclasspath()) != 0) { - return fail("[BOOT classpath mismatch, actual: -Dsun.boot.class.path=", Arguments::get_sysclasspath()); + return fail("[BOOT classpath mismatch, actual =", Arguments::get_sysclasspath()); } break; case NON_EXIST: // fall-through diff --git a/hotspot/src/share/vm/classfile/systemDictionary.cpp b/hotspot/src/share/vm/classfile/systemDictionary.cpp index 549182e82cf..71f176bd988 100644 --- a/hotspot/src/share/vm/classfile/systemDictionary.cpp +++ b/hotspot/src/share/vm/classfile/systemDictionary.cpp @@ -32,6 +32,7 @@ #include "classfile/javaClasses.inline.hpp" #include "classfile/klassFactory.hpp" #include "classfile/loaderConstraints.hpp" +#include "classfile/packageEntry.hpp" #include "classfile/placeholders.hpp" #include "classfile/resolutionErrors.hpp" #include "classfile/stringTable.hpp" @@ -51,6 +52,7 @@ #include "oops/objArrayKlass.hpp" #include "oops/objArrayOop.inline.hpp" #include "oops/oop.inline.hpp" +#include "oops/symbol.hpp" #include "oops/typeArrayKlass.hpp" #include "prims/jvmtiEnvBase.hpp" #include "prims/methodHandles.hpp" @@ -171,13 +173,13 @@ bool SystemDictionary::is_parallelDefine(Handle class_loader) { } /** - * Returns true if the passed class loader is the extension class loader. + * Returns true if the passed class loader is the platform class loader. */ -bool SystemDictionary::is_ext_class_loader(Handle class_loader) { +bool SystemDictionary::is_platform_class_loader(Handle class_loader) { if (class_loader.is_null()) { return false; } - return (class_loader->klass()->name() == vmSymbols::sun_misc_Launcher_ExtClassLoader()); + return (class_loader->klass() == SystemDictionary::jdk_internal_loader_ClassLoaders_PlatformClassLoader_klass()); } // ---------------------------------------------------------------------------- @@ -1144,6 +1146,7 @@ Klass* SystemDictionary::resolve_from_stream(Symbol* class_name, const char* pkg = "java/"; if (!HAS_PENDING_EXCEPTION && !class_loader.is_null() && + !SystemDictionary::is_platform_class_loader(class_loader) && parsed_name != NULL && !strncmp((const char*)parsed_name->bytes(), pkg, strlen(pkg))) { // It is illegal to define classes in the "java." package from @@ -1236,13 +1239,88 @@ instanceKlassHandle SystemDictionary::load_shared_class( instanceKlassHandle ik (THREAD, find_shared_class(class_name)); // Make sure we only return the boot class for the NULL classloader. if (ik.not_null() && - SharedClassUtil::is_shared_boot_class(ik()) && class_loader.is_null()) { + ik->is_shared_boot_class() && class_loader.is_null()) { Handle protection_domain; return load_shared_class(ik, class_loader, protection_domain, THREAD); } return instanceKlassHandle(); } +// Check if a shared class can be loaded by the specific classloader: +// +// NULL classloader: +// - Module class from "modules" jimage. ModuleEntry must be defined in the classloader. +// - Class from -Xbootclasspath/a. The class has no defined PackageEntry, or must +// be defined in an unnamed module. +bool SystemDictionary::is_shared_class_visible(Symbol* class_name, + instanceKlassHandle ik, + Handle class_loader, TRAPS) { + int path_index = ik->shared_classpath_index(); + SharedClassPathEntry* ent = + (SharedClassPathEntry*)FileMapInfo::shared_classpath(path_index); + if (!Universe::is_module_initialized()) { + assert(ent->is_jrt(), + "Loading non-bootstrap classes before the module system is initialized"); + assert(class_loader.is_null(), "sanity"); + return true; + } + // Get the pkg_entry from the classloader + TempNewSymbol pkg_name = NULL; + PackageEntry* pkg_entry = NULL; + ModuleEntry* mod_entry = NULL; + int length = 0; + ClassLoaderData* loader_data = class_loader_data(class_loader); + const jbyte* pkg_string = InstanceKlass::package_from_name(class_name, length); + if (pkg_string != NULL) { + pkg_name = SymbolTable::new_symbol((const char*)pkg_string, + length, CHECK_(false)); + if (loader_data != NULL) { + pkg_entry = loader_data->packages()->lookup_only(pkg_name); + } + if (pkg_entry != NULL) { + mod_entry = pkg_entry->module(); + } + } + + if (class_loader.is_null()) { + // The NULL classloader can load archived class originated from the + // "modules" jimage and the -Xbootclasspath/a. For class from the + // "modules" jimage, the PackageEntry/ModuleEntry must be defined + // by the NULL classloader. + if (mod_entry != NULL) { + // PackageEntry/ModuleEntry is found in the classloader. Check if the + // ModuleEntry's location agrees with the archived class' origination. + if (ent->is_jrt() && mod_entry->location()->starts_with("jrt:")) { + return true; // Module class from the "module" jimage + } + } + + // If the archived class is not from the "module" jimage, the class can be + // loaded by the NULL classloader if + // + // 1. the class is from the unamed package + // 2. or, the class is not from a module defined in the NULL classloader + // 3. or, the class is from an unamed module + if (!ent->is_jrt() && ik->is_shared_boot_class()) { + // the class is from the -Xbootclasspath/a + if (pkg_string == NULL || + pkg_entry == NULL || + pkg_entry->in_unnamed_module()) { + assert(mod_entry == NULL || + mod_entry == loader_data->modules()->unnamed_module(), + "the unnamed module is not defined in the classloader"); + return true; + } + } + return false; + } else { + bool res = SystemDictionaryShared::is_shared_class_visible_for_classloader( + ik, class_loader, pkg_string, pkg_name, + pkg_entry, mod_entry, CHECK_(false)); + return res; + } +} + instanceKlassHandle SystemDictionary::load_shared_class(instanceKlassHandle ik, Handle class_loader, Handle protection_domain, TRAPS) { @@ -1250,6 +1328,12 @@ instanceKlassHandle SystemDictionary::load_shared_class(instanceKlassHandle ik, instanceKlassHandle nh = instanceKlassHandle(); // null Handle Symbol* class_name = ik->name(); + bool visible = is_shared_class_visible( + class_name, ik, class_loader, CHECK_(nh)); + if (!visible) { + return nh; + } + // Found the class, now load the superclass and interfaces. If they // are shared, add them to the main system dictionary and reset // their hierarchy references (supers, subs, and interfaces). @@ -1303,12 +1387,20 @@ instanceKlassHandle SystemDictionary::load_shared_class(instanceKlassHandle ik, } if (log_is_enabled(Info, classload)) { - ik()->print_loading_log(LogLevel::Info, loader_data, NULL); + ik()->print_loading_log(LogLevel::Info, loader_data, NULL, NULL); } // No 'else' here as logging levels are not mutually exclusive if (log_is_enabled(Debug, classload)) { - ik()->print_loading_log(LogLevel::Debug, loader_data, NULL); + ik()->print_loading_log(LogLevel::Debug, loader_data, NULL, NULL); + } + + // For boot loader, ensure that GetSystemPackage knows that a class in this + // package was loaded. + if (class_loader.is_null()) { + int path_index = ik->shared_classpath_index(); + ResourceMark rm; + ClassLoader::add_package(ik->name()->as_C_string(), path_index, THREAD); } if (DumpLoadedClassList != NULL && classlist_file->is_open()) { @@ -1329,7 +1421,68 @@ instanceKlassHandle SystemDictionary::load_shared_class(instanceKlassHandle ik, instanceKlassHandle SystemDictionary::load_instance_class(Symbol* class_name, Handle class_loader, TRAPS) { instanceKlassHandle nh = instanceKlassHandle(); // null Handle + if (class_loader.is_null()) { + int length = 0; + PackageEntry* pkg_entry = NULL; + bool search_only_bootloader_append = false; + ClassLoaderData *loader_data = class_loader_data(class_loader); + + // Find the package in the boot loader's package entry table. + const jbyte* pkg_string = InstanceKlass::package_from_name(class_name, length); + if (pkg_string != NULL) { + TempNewSymbol pkg_name = SymbolTable::new_symbol((const char*)pkg_string, length, CHECK_(nh)); + pkg_entry = loader_data->packages()->lookup_only(pkg_name); + } + + // Prior to attempting to load the class, enforce the boot loader's + // visibility boundaries. + if (!Universe::is_module_initialized()) { + // During bootstrapping, prior to module initialization, any + // class attempting to be loaded must be checked against the + // java.base packages in the boot loader's PackageEntryTable. + // No class outside of java.base is allowed to be loaded during + // this bootstrapping window. + if (!DumpSharedSpaces) { + if (pkg_entry == NULL || pkg_entry->in_unnamed_module()) { + // Class is either in the unnamed package or in + // a named package within the unnamed module. Either + // case is outside of java.base, do not attempt to + // load the class post java.base definition. If + // java.base has not been defined, let the class load + // and its package will be checked later by + // ModuleEntryTable::verify_javabase_packages. + if (ModuleEntryTable::javabase_defined()) { + return nh; + } + } else { + // Check that the class' package is defined within java.base. + ModuleEntry* mod_entry = pkg_entry->module(); + Symbol* mod_entry_name = mod_entry->name(); + if (mod_entry_name->fast_compare(vmSymbols::java_base()) != 0) { + return nh; + } + } + } + } else { + assert(!DumpSharedSpaces, "Archive dumped after module system initialization"); + // After the module system has been initialized, check if the class' + // package is in a module defined to the boot loader. + if (pkg_string == NULL || pkg_entry == NULL || pkg_entry->in_unnamed_module()) { + // Class is either in the unnamed package, in a named package + // within a module not defined to the boot loader or in a + // a named package within the unnamed module. In all cases, + // limit visibility to search for the class only in the boot + // loader's append path. + search_only_bootloader_append = true; + } + } + + // Prior to bootstrapping's module initialization, never load a class outside + // of the boot loader's module path + assert(Universe::is_module_initialized() || DumpSharedSpaces || + !search_only_bootloader_append, + "Attempt to load a class outside of boot loader's module path"); // Search the shared system dictionary for classes preloaded into the // shared spaces. @@ -1344,7 +1497,7 @@ instanceKlassHandle SystemDictionary::load_instance_class(Symbol* class_name, Ha if (k.is_null()) { // Use VM class loader PerfTraceTime vmtimer(ClassLoader::perf_sys_classload_time()); - k = ClassLoader::load_class(class_name, CHECK_(nh)); + k = ClassLoader::load_class(class_name, search_only_bootloader_append, CHECK_(nh)); } // find_or_define_instance_class may return a different InstanceKlass @@ -1669,7 +1822,7 @@ Klass* SystemDictionary::find_class(Symbol* class_name, ClassLoaderData* loader_ } -// Get the next class in the diictionary. +// Get the next class in the dictionary. Klass* SystemDictionary::try_get_next_class() { return dictionary()->try_get_next_class(); } @@ -1940,6 +2093,11 @@ void SystemDictionary::initialize_wk_klasses_until(WKID limit_id, WKID &start_id void SystemDictionary::initialize_preloaded_classes(TRAPS) { assert(WK_KLASS(Object_klass) == NULL, "preloaded classes should only be initialized once"); + + // Create the ModuleEntry for java.base. This call needs to be done here, + // after vmSymbols::initialize() is called but before any classes are pre-loaded. + ClassLoader::create_javabase(); + // Preload commonly used klasses WKID scan = FIRST_WKID; // first do Object, then String, Class diff --git a/hotspot/src/share/vm/classfile/systemDictionary.hpp b/hotspot/src/share/vm/classfile/systemDictionary.hpp index 628895f1e44..1e148322348 100644 --- a/hotspot/src/share/vm/classfile/systemDictionary.hpp +++ b/hotspot/src/share/vm/classfile/systemDictionary.hpp @@ -135,6 +135,7 @@ class SymbolPropertyTable; do_klass(Properties_klass, java_util_Properties, Pre ) \ do_klass(reflect_AccessibleObject_klass, java_lang_reflect_AccessibleObject, Pre ) \ do_klass(reflect_Field_klass, java_lang_reflect_Field, Pre ) \ + do_klass(reflect_Module_klass, java_lang_reflect_Module, Pre ) \ do_klass(reflect_Parameter_klass, java_lang_reflect_Parameter, Opt ) \ do_klass(reflect_Method_klass, java_lang_reflect_Method, Pre ) \ do_klass(reflect_Constructor_klass, java_lang_reflect_Constructor, Pre ) \ @@ -167,15 +168,17 @@ class SymbolPropertyTable; do_klass(StringBuffer_klass, java_lang_StringBuffer, Pre ) \ do_klass(StringBuilder_klass, java_lang_StringBuilder, Pre ) \ do_klass(internal_Unsafe_klass, jdk_internal_misc_Unsafe, Pre ) \ + do_klass(module_Modules_klass, jdk_internal_module_Modules, Pre ) \ \ /* support for CDS */ \ do_klass(ByteArrayInputStream_klass, java_io_ByteArrayInputStream, Pre ) \ do_klass(File_klass, java_io_File, Pre ) \ - do_klass(URLClassLoader_klass, java_net_URLClassLoader, Pre ) \ do_klass(URL_klass, java_net_URL, Pre ) \ do_klass(Jar_Manifest_klass, java_util_jar_Manifest, Pre ) \ - do_klass(sun_misc_Launcher_klass, sun_misc_Launcher, Pre ) \ + do_klass(jdk_internal_loader_ClassLoaders_AppClassLoader_klass, jdk_internal_loader_ClassLoaders_AppClassLoader, Pre ) \ + do_klass(jdk_internal_loader_ClassLoaders_PlatformClassLoader_klass, jdk_internal_loader_ClassLoaders_PlatformClassLoader, Pre ) \ do_klass(CodeSource_klass, java_security_CodeSource, Pre ) \ + do_klass(ParseUtil_klass, sun_net_www_ParseUtil, Pre ) \ \ do_klass(StackTraceElement_klass, java_lang_StackTraceElement, Opt ) \ \ @@ -639,6 +642,8 @@ protected: static instanceKlassHandle find_or_define_instance_class(Symbol* class_name, Handle class_loader, instanceKlassHandle k, TRAPS); + static bool is_shared_class_visible(Symbol* class_name, instanceKlassHandle ik, + Handle class_loader, TRAPS); static instanceKlassHandle load_shared_class(instanceKlassHandle ik, Handle class_loader, Handle protection_domain, @@ -653,7 +658,7 @@ public: static instanceKlassHandle load_shared_class(Symbol* class_name, Handle class_loader, TRAPS); - static bool is_ext_class_loader(Handle class_loader); + static bool is_platform_class_loader(Handle class_loader); protected: static Klass* find_shared_class(Symbol* class_name); diff --git a/hotspot/src/share/vm/classfile/systemDictionaryShared.hpp b/hotspot/src/share/vm/classfile/systemDictionaryShared.hpp index 8ada71482c4..50b5dd80920 100644 --- a/hotspot/src/share/vm/classfile/systemDictionaryShared.hpp +++ b/hotspot/src/share/vm/classfile/systemDictionaryShared.hpp @@ -44,6 +44,16 @@ public: oop class_loader = loader_data->class_loader(); return (class_loader == NULL); } + static bool is_shared_class_visible_for_classloader( + instanceKlassHandle ik, + Handle class_loader, + const jbyte* pkg_string, + Symbol* pkg_name, + PackageEntry* pkg_entry, + ModuleEntry* mod_entry, + TRAPS) { + return false; + } static Klass* dump_time_resolve_super_or_fail(Symbol* child_name, Symbol* class_name, diff --git a/hotspot/src/share/vm/classfile/vmSymbols.hpp b/hotspot/src/share/vm/classfile/vmSymbols.hpp index 9323389f5e3..111989451fb 100644 --- a/hotspot/src/share/vm/classfile/vmSymbols.hpp +++ b/hotspot/src/share/vm/classfile/vmSymbols.hpp @@ -49,10 +49,12 @@ // Mapping function names to values. New entries should be added below. #define VM_SYMBOLS_DO(template, do_alias) \ - /* commonly used class names */ \ + /* commonly used class, package, module names */ \ + template(java_base, "java.base") \ template(java_lang_System, "java/lang/System") \ template(java_lang_Object, "java/lang/Object") \ template(java_lang_Class, "java/lang/Class") \ + template(java_lang_Package, "java/lang/Package") \ template(java_lang_String, "java/lang/String") \ template(java_lang_StringLatin1, "java/lang/StringLatin1") \ template(java_lang_StringUTF16, "java/lang/StringUTF16") \ @@ -87,6 +89,7 @@ template(java_lang_reflect_Method, "java/lang/reflect/Method") \ template(java_lang_reflect_Constructor, "java/lang/reflect/Constructor") \ template(java_lang_reflect_Field, "java/lang/reflect/Field") \ + template(java_lang_reflect_Module, "java/lang/reflect/Module") \ template(java_lang_reflect_Parameter, "java/lang/reflect/Parameter") \ template(java_lang_reflect_Array, "java/lang/reflect/Array") \ template(java_lang_StringBuffer, "java/lang/StringBuffer") \ @@ -97,7 +100,6 @@ template(java_security_CodeSource, "java/security/CodeSource") \ template(java_security_ProtectionDomain, "java/security/ProtectionDomain") \ template(java_security_SecureClassLoader, "java/security/SecureClassLoader") \ - template(java_net_URLClassLoader, "java/net/URLClassLoader") \ template(java_net_URL, "java/net/URL") \ template(java_util_jar_Manifest, "java/util/jar/Manifest") \ template(impliesCreateAccessControlContext_name, "impliesCreateAccessControlContext") \ @@ -116,17 +118,25 @@ template(java_util_Hashtable, "java/util/Hashtable") \ template(java_lang_Compiler, "java/lang/Compiler") \ template(jdk_internal_misc_Signal, "jdk/internal/misc/Signal") \ - template(sun_misc_Launcher, "sun/misc/Launcher") \ template(java_lang_AssertionStatusDirectives, "java/lang/AssertionStatusDirectives") \ template(getBootClassPathEntryForClass_name, "getBootClassPathEntryForClass") \ template(sun_misc_PostVMInitHook, "sun/misc/PostVMInitHook") \ - template(sun_misc_Launcher_ExtClassLoader, "sun/misc/Launcher$ExtClassLoader") \ + template(sun_net_www_ParseUtil, "sun/net/www/ParseUtil") \ + \ + template(jdk_internal_loader_ClassLoaders_AppClassLoader, "jdk/internal/loader/ClassLoaders$AppClassLoader") \ + template(jdk_internal_loader_ClassLoaders_PlatformClassLoader, "jdk/internal/loader/ClassLoaders$PlatformClassLoader") \ \ /* Java runtime version access */ \ template(java_lang_VersionProps, "java/lang/VersionProps") \ template(java_runtime_name_name, "java_runtime_name") \ template(java_runtime_version_name, "java_runtime_version") \ \ + /* system initialization */ \ + template(initPhase1_name, "initPhase1") \ + template(initPhase2_name, "initPhase2") \ + template(initPhase3_name, "initPhase3") \ + template(java_lang_reflect_module_init_signature, "(Ljava/lang/ClassLoader;Ljava/lang/String;)V") \ + \ /* class file format tags */ \ template(tag_source_file, "SourceFile") \ template(tag_inner_classes, "InnerClasses") \ @@ -360,7 +370,6 @@ template(run_finalization_name, "runFinalization") \ template(run_finalizers_on_exit_name, "runFinalizersOnExit") \ template(dispatchUncaughtException_name, "dispatchUncaughtException") \ - template(initializeSystemClass_name, "initializeSystemClass") \ template(loadClass_name, "loadClass") \ template(loadClassInternal_name, "loadClassInternal") \ template(get_name, "get") \ @@ -446,14 +455,22 @@ template(signers_name, "signers_name") \ template(loader_data_name, "loader_data") \ template(vmdependencies_name, "vmdependencies") \ + template(loader_name, "loader") \ + template(module_name, "module") \ + template(getModule_name, "getModule") \ + template(addReads_name, "addReads") \ + template(addReads_signature, "(Ljava/lang/reflect/Module;Ljava/lang/reflect/Module;)V") \ template(input_stream_void_signature, "(Ljava/io/InputStream;)V") \ - template(getFileURL_name, "getFileURL") \ - template(getFileURL_signature, "(Ljava/io/File;)Ljava/net/URL;") \ - template(definePackageInternal_name, "definePackageInternal") \ - template(definePackageInternal_signature, "(Ljava/lang/String;Ljava/util/jar/Manifest;Ljava/net/URL;)V") \ + template(definePackage_name, "definePackage") \ + template(definePackage_signature, "(Ljava/lang/String;Ljava/lang/reflect/Module;)Ljava/lang/Package;") \ + template(defineOrCheckPackage_name, "defineOrCheckPackage") \ + template(defineOrCheckPackage_signature, "(Ljava/lang/String;Ljava/util/jar/Manifest;Ljava/net/URL;)Ljava/lang/Package;") \ + template(fileToEncodedURL_name, "fileToEncodedURL") \ + template(fileToEncodedURL_signature, "(Ljava/io/File;)Ljava/net/URL;") \ template(getProtectionDomain_name, "getProtectionDomain") \ template(getProtectionDomain_signature, "(Ljava/security/CodeSource;)Ljava/security/ProtectionDomain;") \ template(url_code_signer_array_void_signature, "(Ljava/net/URL;[Ljava/security/CodeSigner;)V") \ + template(module_entry_name, "module_entry") \ \ /* non-intrinsic name/signature pairs: */ \ template(register_method_name, "register") \ @@ -531,6 +548,7 @@ template(void_class_signature, "()Ljava/lang/Class;") \ template(void_class_array_signature, "()[Ljava/lang/Class;") \ template(void_string_signature, "()Ljava/lang/String;") \ + template(void_module_signature, "()Ljava/lang/reflect/Module;") \ template(object_array_object_signature, "([Ljava/lang/Object;)Ljava/lang/Object;") \ template(object_object_array_object_signature, "(Ljava/lang/Object;[Ljava/lang/Object;)Ljava/lang/Object;")\ template(exception_void_signature, "(Ljava/lang/Exception;)V") \ @@ -550,6 +568,7 @@ template(reference_signature, "Ljava/lang/ref/Reference;") \ template(sun_misc_Cleaner_signature, "Lsun/misc/Cleaner;") \ template(executable_signature, "Ljava/lang/reflect/Executable;") \ + template(module_signature, "Ljava/lang/reflect/Module;") \ template(concurrenthashmap_signature, "Ljava/util/concurrent/ConcurrentHashMap;") \ template(String_StringBuilder_signature, "(Ljava/lang/String;)Ljava/lang/StringBuilder;") \ template(int_StringBuilder_signature, "(I)Ljava/lang/StringBuilder;") \ @@ -574,6 +593,9 @@ /* used to identify class loaders handling parallel class loading */ \ template(parallelCapable_name, "parallelLockMap") \ \ + /* used to return a class loader's unnamed module */ \ + template(unnamedModule_name, "unnamedModule") \ + \ /* JVM monitoring and management support */ \ template(java_lang_StackTraceElement_array, "[Ljava/lang/StackTraceElement;") \ template(java_lang_management_ThreadState, "java/lang/management/ThreadState") \ @@ -632,7 +654,10 @@ template(addThreadDumpForSynchronizers_signature, "(Ljava/lang/management/ThreadInfo;[Ljava/lang/Object;)V") \ \ /* JVMTI/java.lang.instrument support and VM Attach mechanism */ \ + template(jdk_internal_module_Modules, "jdk/internal/module/Modules") \ template(sun_misc_VMSupport, "sun/misc/VMSupport") \ + template(transformedByAgent_name, "transformedByAgent") \ + template(transformedByAgent_signature, "(Ljava/lang/reflect/Module;)V") \ template(appendToClassPathForInstrumentation_name, "appendToClassPathForInstrumentation") \ do_alias(appendToClassPathForInstrumentation_signature, string_void_signature) \ template(serializePropertiesToByteArray_name, "serializePropertiesToByteArray") \ @@ -1063,11 +1088,6 @@ do_name( isCompileConstant_name, "isCompileConstant") \ do_alias( isCompileConstant_signature, object_boolean_signature) \ \ - do_class(sun_hotspot_WhiteBox, "sun/hotspot/WhiteBox") \ - do_intrinsic(_deoptimize, sun_hotspot_WhiteBox, deoptimize_name, deoptimize_signature, F_R) \ - do_name( deoptimize_name, "deoptimize") \ - do_alias( deoptimize_signature, void_method_signature) \ - \ /* unsafe memory references (there are a lot of them...) */ \ do_signature(getObject_signature, "(Ljava/lang/Object;J)Ljava/lang/Object;") \ do_signature(putObject_signature, "(Ljava/lang/Object;JLjava/lang/Object;)V") \ diff --git a/hotspot/src/share/vm/interpreter/bytecodeInterpreter.cpp b/hotspot/src/share/vm/interpreter/bytecodeInterpreter.cpp index 83e6a2146f7..983b0b71ad4 100644 --- a/hotspot/src/share/vm/interpreter/bytecodeInterpreter.cpp +++ b/hotspot/src/share/vm/interpreter/bytecodeInterpreter.cpp @@ -2258,10 +2258,8 @@ run: // Decrement counter at checkcast. BI_PROFILE_SUBTYPECHECK_FAILED(objKlass); ResourceMark rm(THREAD); - const char* objName = objKlass->external_name(); - const char* klassName = klassOf->external_name(); char* message = SharedRuntime::generate_class_cast_message( - objName, klassName); + objKlass, klassOf); VM_JAVA_ERROR(vmSymbols::java_lang_ClassCastException(), message, note_classCheck_trap); } // Profile checkcast with null_seen and receiver. diff --git a/hotspot/src/share/vm/interpreter/interpreterRuntime.cpp b/hotspot/src/share/vm/interpreter/interpreterRuntime.cpp index 839eec62153..acc45335158 100644 --- a/hotspot/src/share/vm/interpreter/interpreterRuntime.cpp +++ b/hotspot/src/share/vm/interpreter/interpreterRuntime.cpp @@ -384,7 +384,7 @@ IRT_ENTRY(void, InterpreterRuntime::throw_ClassCastException( ResourceMark rm(thread); char* message = SharedRuntime::generate_class_cast_message( - thread, obj->klass()->external_name()); + thread, obj->klass()); if (ProfileTraps) { note_trap(thread, Deoptimization::Reason_class_check, CHECK); diff --git a/hotspot/src/share/vm/interpreter/linkResolver.cpp b/hotspot/src/share/vm/interpreter/linkResolver.cpp index 7e3c60b185d..5c7e9b13d35 100644 --- a/hotspot/src/share/vm/interpreter/linkResolver.cpp +++ b/hotspot/src/share/vm/interpreter/linkResolver.cpp @@ -273,18 +273,25 @@ void LinkInfo::print() { // Klass resolution void LinkResolver::check_klass_accessability(KlassHandle ref_klass, KlassHandle sel_klass, TRAPS) { - if (!Reflection::verify_class_access(ref_klass(), - sel_klass(), - true)) { + Reflection::VerifyClassAccessResults vca_result = + Reflection::verify_class_access(ref_klass(), sel_klass(), true); + if (vca_result != Reflection::ACCESS_OK) { ResourceMark rm(THREAD); - Exceptions::fthrow( - THREAD_AND_LOCATION, - vmSymbols::java_lang_IllegalAccessError(), - "tried to access class %s from class %s", - sel_klass->external_name(), - ref_klass->external_name() - ); - return; + char* msg = Reflection::verify_class_access_msg(ref_klass(), sel_klass(), vca_result); + if (msg == NULL) { + Exceptions::fthrow( + THREAD_AND_LOCATION, + vmSymbols::java_lang_IllegalAccessError(), + "failed to access class %s from class %s", + sel_klass->external_name(), + ref_klass->external_name()); + } else { + // Use module specific message returned by verify_class_access_msg(). + Exceptions::fthrow( + THREAD_AND_LOCATION, + vmSymbols::java_lang_IllegalAccessError(), + "%s", msg); + } } } diff --git a/hotspot/src/share/vm/jvmci/jvmciEnv.cpp b/hotspot/src/share/vm/jvmci/jvmciEnv.cpp index b19a39603fa..68d4a05bc7f 100644 --- a/hotspot/src/share/vm/jvmci/jvmciEnv.cpp +++ b/hotspot/src/share/vm/jvmci/jvmciEnv.cpp @@ -77,7 +77,9 @@ bool JVMCIEnv::check_klass_accessibility(KlassHandle accessing_klass, KlassHandl resolved_klass = ObjArrayKlass::cast(resolved_klass())->bottom_klass(); } if (resolved_klass->is_instance_klass()) { - return Reflection::verify_class_access(accessing_klass(), resolved_klass(), true); + Reflection::VerifyClassAccessResults result = + Reflection::verify_class_access(accessing_klass(), resolved_klass(), true); + return result == Reflection::ACCESS_OK; } return true; } diff --git a/hotspot/src/share/vm/logging/logTag.hpp b/hotspot/src/share/vm/logging/logTag.hpp index d292c8339e9..c7fb0a08325 100644 --- a/hotspot/src/share/vm/logging/logTag.hpp +++ b/hotspot/src/share/vm/logging/logTag.hpp @@ -62,6 +62,7 @@ LOG_TAG(logging) \ LOG_TAG(marking) \ LOG_TAG(metaspace) \ + LOG_TAG(modules) \ LOG_TAG(monitorinflation) \ LOG_TAG(os) \ LOG_TAG(phases) \ diff --git a/hotspot/src/share/vm/memory/filemap.cpp b/hotspot/src/share/vm/memory/filemap.cpp index 1b21736315c..0f6415e49e3 100644 --- a/hotspot/src/share/vm/memory/filemap.cpp +++ b/hotspot/src/share/vm/memory/filemap.cpp @@ -226,12 +226,21 @@ void FileMapInfo::allocate_classpath_entry_table() { SharedClassUtil::update_shared_classpath(cpe, ent, st.st_mtime, st.st_size, THREAD); } else { struct stat st; - if ((os::stat(name, &st) == 0) && ((st.st_mode & S_IFDIR) == S_IFDIR)) { - if (!os::dir_is_empty(name)) { - ClassLoader::exit_with_path_failure("Cannot have non-empty directory in archived classpaths", name); + if (os::stat(name, &st) == 0) { + if (cpe->is_jrt()) { + // it's the "modules" jimage + ent->_timestamp = st.st_mtime; + ent->_filesize = st.st_size; + } else if ((st.st_mode & S_IFDIR) == S_IFDIR) { + if (!os::dir_is_empty(name)) { + ClassLoader::exit_with_path_failure( + "Cannot have non-empty directory in archived classpaths", name); + } + ent->_filesize = -1; } - ent->_filesize = -1; - } else { + } + if (ent->_filesize == 0) { + // unknown ent->_filesize = -2; } } @@ -282,7 +291,7 @@ bool FileMapInfo::validate_classpath_entry_table() { fail_continue("directory is not empty: %s", name); ok = false; } - } else if (ent->is_jar()) { + } else if (ent->is_jar_or_bootimage()) { if (ent->_timestamp != st.st_mtime || ent->_filesize != st.st_size) { ok = false; @@ -291,7 +300,7 @@ bool FileMapInfo::validate_classpath_entry_table() { "Timestamp mismatch" : "File size mismatch"); } else { - fail_continue("A jar file is not the one used while building" + fail_continue("A jar/jimage file is not the one used while building" " the shared archive file: %s", name); } } @@ -871,6 +880,11 @@ bool FileMapInfo::FileMapHeader::validate() { return false; } + if (Arguments::patch_dirs() != NULL) { + FileMapInfo::fail_continue("The shared archive file cannot be used with -Xpatch."); + return false; + } + if (_version != current_version()) { FileMapInfo::fail_continue("The shared archive file is the wrong version."); return false; diff --git a/hotspot/src/share/vm/memory/filemap.hpp b/hotspot/src/share/vm/memory/filemap.hpp index 9752e5ddf75..7cb179434b9 100644 --- a/hotspot/src/share/vm/memory/filemap.hpp +++ b/hotspot/src/share/vm/memory/filemap.hpp @@ -44,14 +44,20 @@ class Metaspace; class SharedClassPathEntry VALUE_OBJ_CLASS_SPEC { public: const char *_name; - time_t _timestamp; // jar timestamp, 0 if is directory or other - long _filesize; // jar file size, -1 if is directory, -2 if other - bool is_jar() { + time_t _timestamp; // jar/jimage timestamp, 0 if is directory or other + long _filesize; // jar/jimage file size, -1 if is directory, -2 if other + + // The _timestamp only gets set for jar files and "modules" jimage. + bool is_jar_or_bootimage() { return _timestamp != 0; } bool is_dir() { return _filesize == -1; } + + bool is_jrt() { + return ClassLoader::is_jrt(_name); + } }; class FileMapInfo : public CHeapObj { diff --git a/hotspot/src/share/vm/memory/metaspaceShared.cpp b/hotspot/src/share/vm/memory/metaspaceShared.cpp index 3f10488bf7f..dc4fc082967 100644 --- a/hotspot/src/share/vm/memory/metaspaceShared.cpp +++ b/hotspot/src/share/vm/memory/metaspaceShared.cpp @@ -603,14 +603,7 @@ void VM_PopulateDumpSharedSpace::doit() { SystemDictionary::reverse(); SystemDictionary::copy_buckets(&md_top, md_end); - ClassLoader::verify(); - ClassLoader::copy_package_info_buckets(&md_top, md_end); - ClassLoader::verify(); - SystemDictionary::copy_table(&md_top, md_end); - ClassLoader::verify(); - ClassLoader::copy_package_info_table(&md_top, md_end); - ClassLoader::verify(); // Write the other data to the output array. WriteClosure wc(md_top, md_end); @@ -716,8 +709,7 @@ void VM_PopulateDumpSharedSpace::doit() { } -void MetaspaceShared::link_one_shared_class(Klass* obj, TRAPS) { - Klass* k = obj; +void MetaspaceShared::link_one_shared_class(Klass* k, TRAPS) { if (k->is_instance_klass()) { InstanceKlass* ik = InstanceKlass::cast(k); // Link the class to cause the bytecodes to be rewritten and the @@ -734,6 +726,16 @@ void MetaspaceShared::check_one_shared_class(Klass* k) { } } +void MetaspaceShared::check_shared_class_loader_type(Klass* k) { + if (k->is_instance_klass()) { + InstanceKlass* ik = InstanceKlass::cast(k); + u2 loader_type = ik->loader_type(); + ResourceMark rm; + guarantee(loader_type != 0, + "Class loader type is not set for this class %s", ik->name()->as_C_string()); + } +} + void MetaspaceShared::link_and_cleanup_shared_classes(TRAPS) { // We need to iterate because verification may cause additional classes // to be loaded. @@ -765,6 +767,7 @@ void MetaspaceShared::link_and_cleanup_shared_classes(TRAPS) { } void MetaspaceShared::prepare_for_dumping() { + Arguments::check_unsupported_dumping_properties(); ClassLoader::initialize_shared_path(); FileMapInfo::allocate_classpath_entry_table(); } @@ -901,7 +904,7 @@ bool MetaspaceShared::try_link_class(InstanceKlass* ik, TRAPS) { assert(DumpSharedSpaces, "should only be called during dumping"); if (ik->init_state() < InstanceKlass::linked) { bool saved = BytecodeVerificationLocal; - if (!SharedClassUtil::is_shared_boot_class(ik)) { + if (!(ik->is_shared_boot_class())) { // The verification decision is based on BytecodeVerificationRemote // for non-system classes. Since we are using the NULL classloader // to load non-system classes during dumping, we need to temporarily @@ -1089,36 +1092,14 @@ void MetaspaceShared::initialize_shared_spaces() { number_of_entries); buffer += sharedDictionaryLen; - // Create the package info table using the bucket array at this spot in - // the misc data space. Since the package info table is never - // modified, this region (of mapped pages) will be (effectively, if - // not explicitly) read-only. - - int pkgInfoLen = *(intptr_t*)buffer; - buffer += sizeof(intptr_t); - number_of_entries = *(intptr_t*)buffer; - buffer += sizeof(intptr_t); - ClassLoader::create_package_info_table((HashtableBucket*)buffer, pkgInfoLen, - number_of_entries); - buffer += pkgInfoLen; - ClassLoader::verify(); - // The following data in the shared misc data region are the linked // list elements (HashtableEntry objects) for the shared dictionary - // and package info table. + // table. int len = *(intptr_t*)buffer; // skip over shared dictionary entries buffer += sizeof(intptr_t); buffer += len; - len = *(intptr_t*)buffer; // skip over package info table entries - buffer += sizeof(intptr_t); - buffer += len; - - len = *(intptr_t*)buffer; // skip over package info table char[] arrays. - buffer += sizeof(intptr_t); - buffer += len; - intptr_t* array = (intptr_t*)buffer; ReadClosure rc(&array); serialize(&rc); diff --git a/hotspot/src/share/vm/memory/metaspaceShared.hpp b/hotspot/src/share/vm/memory/metaspaceShared.hpp index dac447c9e05..c78f78122cf 100644 --- a/hotspot/src/share/vm/memory/metaspaceShared.hpp +++ b/hotspot/src/share/vm/memory/metaspaceShared.hpp @@ -208,6 +208,7 @@ class MetaspaceShared : AllStatic { static bool try_link_class(InstanceKlass* ik, TRAPS); static void link_one_shared_class(Klass* obj, TRAPS); static void check_one_shared_class(Klass* obj); + static void check_shared_class_loader_type(Klass* obj); static void link_and_cleanup_shared_classes(TRAPS); static int count_class(const char* classlist_file); diff --git a/hotspot/src/share/vm/memory/universe.cpp b/hotspot/src/share/vm/memory/universe.cpp index 3a7945768e8..60a60e2feed 100644 --- a/hotspot/src/share/vm/memory/universe.cpp +++ b/hotspot/src/share/vm/memory/universe.cpp @@ -155,6 +155,7 @@ uintptr_t Universe::_verify_oop_bits = (uintptr_t) -1; int Universe::_base_vtable_size = 0; bool Universe::_bootstrapping = false; +bool Universe::_module_initialized = false; bool Universe::_fully_initialized = false; size_t Universe::_heap_capacity_at_last_gc; @@ -667,7 +668,6 @@ jint universe_init() { } else { SymbolTable::create_table(); StringTable::create_table(); - ClassLoader::create_package_info_table(); if (DumpSharedSpaces) { MetaspaceShared::prepare_for_dumping(); @@ -886,6 +886,10 @@ void universe2_init() { Universe::genesis(CATCH); } +// Set after initialization of the module runtime, call_initModuleRuntime +void universe_post_module_init() { + Universe::_module_initialized = true; +} bool universe_post_init() { assert(!is_init_completed(), "Error: initialization not yet completed!"); diff --git a/hotspot/src/share/vm/memory/universe.hpp b/hotspot/src/share/vm/memory/universe.hpp index fe10e849121..340e87fb706 100644 --- a/hotspot/src/share/vm/memory/universe.hpp +++ b/hotspot/src/share/vm/memory/universe.hpp @@ -111,6 +111,7 @@ class Universe: AllStatic { friend jint universe_init(); friend void universe2_init(); friend bool universe_post_init(); + friend void universe_post_module_init(); private: // Known classes in the VM @@ -205,6 +206,7 @@ class Universe: AllStatic { // Initialization static bool _bootstrapping; // true during genesis + static bool _module_initialized; // true after call_initPhase2 called static bool _fully_initialized; // true after universe_init and initialize_vtables called // the array of preallocated errors with backtraces @@ -436,6 +438,7 @@ class Universe: AllStatic { // Testers static bool is_bootstrapping() { return _bootstrapping; } + static bool is_module_initialized() { return _module_initialized; } static bool is_fully_initialized() { return _fully_initialized; } static inline bool element_type_should_be_aligned(BasicType type); diff --git a/hotspot/src/share/vm/oops/arrayKlass.cpp b/hotspot/src/share/vm/oops/arrayKlass.cpp index 2278ce92471..50d5e041d84 100644 --- a/hotspot/src/share/vm/oops/arrayKlass.cpp +++ b/hotspot/src/share/vm/oops/arrayKlass.cpp @@ -95,11 +95,18 @@ ArrayKlass::ArrayKlass(Symbol* name) : // Initialization of vtables and mirror object is done separatly from base_create_array_klass, // since a GC can happen. At this point all instance variables of the ArrayKlass must be setup. -void ArrayKlass::complete_create_array_klass(ArrayKlass* k, KlassHandle super_klass, TRAPS) { +void ArrayKlass::complete_create_array_klass(ArrayKlass* k, KlassHandle super_klass, ModuleEntry* module_entry, TRAPS) { ResourceMark rm(THREAD); k->initialize_supers(super_klass(), CHECK); k->vtable()->initialize_vtable(false, CHECK); - java_lang_Class::create_mirror(k, Handle(THREAD, k->class_loader()), Handle(NULL), CHECK); + + // During bootstrapping, before java.base is defined, the module_entry may not be present yet. + // These classes will be put on a fixup list and their module fields will be patched once + // java.base is defined. + assert((module_entry != NULL) || ((module_entry == NULL) && !ModuleEntryTable::javabase_defined()), + "module entry not available post java.base definition"); + oop module = (module_entry != NULL) ? JNIHandles::resolve(module_entry->module()) : (oop)NULL; + java_lang_Class::create_mirror(k, Handle(THREAD, k->class_loader()), Handle(THREAD, module), Handle(NULL), CHECK); } GrowableArray* ArrayKlass::compute_secondary_supers(int num_extra_slots) { diff --git a/hotspot/src/share/vm/oops/arrayKlass.hpp b/hotspot/src/share/vm/oops/arrayKlass.hpp index b98a092f735..3761229a1ce 100644 --- a/hotspot/src/share/vm/oops/arrayKlass.hpp +++ b/hotspot/src/share/vm/oops/arrayKlass.hpp @@ -113,7 +113,7 @@ class ArrayKlass: public Klass { void array_klasses_do(void f(Klass* k, TRAPS), TRAPS); // Return a handle. - static void complete_create_array_klass(ArrayKlass* k, KlassHandle super_klass, TRAPS); + static void complete_create_array_klass(ArrayKlass* k, KlassHandle super_klass, ModuleEntry* module, TRAPS); // jvm support diff --git a/hotspot/src/share/vm/oops/instanceKlass.cpp b/hotspot/src/share/vm/oops/instanceKlass.cpp index d1297af03df..f526756a0ce 100644 --- a/hotspot/src/share/vm/oops/instanceKlass.cpp +++ b/hotspot/src/share/vm/oops/instanceKlass.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2016, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -1972,8 +1972,9 @@ static void restore_unshareable_in_class(Klass* k, TRAPS) { } void InstanceKlass::restore_unshareable_info(ClassLoaderData* loader_data, Handle protection_domain, TRAPS) { - Klass::restore_unshareable_info(loader_data, protection_domain, CHECK); instanceKlassHandle ik(THREAD, this); + ik->set_package(loader_data, CHECK); + Klass::restore_unshareable_info(loader_data, protection_domain, CHECK); Array* methods = ik->methods(); int num_methods = methods->length(); @@ -2178,26 +2179,135 @@ const char* InstanceKlass::signature_name() const { return dest; } -// different verisons of is_same_class_package -bool InstanceKlass::is_same_class_package(const Klass* class2) const { - const Klass* const class1 = (const Klass* const)this; - oop classloader1 = InstanceKlass::cast(class1)->class_loader(); - const Symbol* const classname1 = class1->name(); +const jbyte* InstanceKlass::package_from_name(const Symbol* name, int& length) { + ResourceMark rm; + length = 0; + if (name == NULL) { + return NULL; + } else { + const jbyte* base_name = name->base(); + const jbyte* last_slash = UTF8::strrchr(base_name, name->utf8_length(), '/'); + if (last_slash == NULL) { + // No package name + return NULL; + } else { + // Skip over '['s + if (*base_name == '[') { + do { + base_name++; + } while (*base_name == '['); + if (*base_name != 'L') { + // Fully qualified class names should not contain a 'L'. + // Set length to -1 to indicate that the package name + // could not be obtained due to an error condition. + // In this situtation, is_same_class_package returns false. + length = -1; + return NULL; + } + } + + // Found the package name, look it up in the symbol table. + length = last_slash - base_name; + assert(length > 0, "Bad length for package name"); + return base_name; + } + } +} + +ModuleEntry* InstanceKlass::module() const { + if (!in_unnamed_package()) { + return _package_entry->module(); + } + const Klass* host = host_klass(); + if (host == NULL) { + return class_loader_data()->modules()->unnamed_module(); + } + return host->class_loader_data()->modules()->unnamed_module(); +} + +void InstanceKlass::set_package(ClassLoaderData* loader_data, TRAPS) { + int length = 0; + const jbyte* base_name = package_from_name(name(), length); + + if (base_name != NULL && loader_data != NULL) { + TempNewSymbol pkg_name = SymbolTable::new_symbol((const char*)base_name, length, CHECK); + + // Find in class loader's package entry table. + _package_entry = loader_data->packages()->lookup_only(pkg_name); + + // If the package name is not found in the loader's package + // entry table, it is an indication that the package has not + // been defined. Consider it defined within the unnamed module. + if (_package_entry == NULL) { + ResourceMark rm; + + if (!ModuleEntryTable::javabase_defined()) { + // Before java.base is defined during bootstrapping, define all packages in + // the java.base module. If a non-java.base package is erroneously placed + // in the java.base module it will be caught later when java.base + // is defined by ModuleEntryTable::verify_javabase_packages check. + assert(ModuleEntryTable::javabase_module() != NULL, "java.base module is NULL"); + _package_entry = loader_data->packages()->lookup(pkg_name, ModuleEntryTable::javabase_module()); + } else { + assert(loader_data->modules()->unnamed_module() != NULL, "unnamed module is NULL"); + _package_entry = loader_data->packages()->lookup(pkg_name, + loader_data->modules()->unnamed_module()); + } + + // A package should have been successfully created + assert(_package_entry != NULL, "Package entry for class %s not found, loader %s", + name()->as_C_string(), loader_data->loader_name()); + } + + if (log_is_enabled(Debug, modules)) { + ResourceMark rm; + ModuleEntry* m = _package_entry->module(); + log_trace(modules)("Setting package: class: %s, package: %s, loader: %s, module: %s", + external_name(), + pkg_name->as_C_string(), + loader_data->loader_name(), + (m->is_named() ? m->name()->as_C_string() : UNNAMED_MODULE)); + } + } else { + ResourceMark rm; + log_trace(modules)("Setting package: class: %s, package: unnamed, loader: %s, module: %s", + external_name(), + (loader_data != NULL) ? loader_data->loader_name() : "NULL", + UNNAMED_MODULE); + } +} + + +// different versions of is_same_class_package + +bool InstanceKlass::is_same_class_package(const Klass* class2) const { + oop classloader1 = this->class_loader(); + PackageEntry* classpkg1 = this->package(); if (class2->is_objArray_klass()) { class2 = ObjArrayKlass::cast(class2)->bottom_klass(); } + oop classloader2; + PackageEntry* classpkg2; if (class2->is_instance_klass()) { - classloader2 = InstanceKlass::cast(class2)->class_loader(); + classloader2 = class2->class_loader(); + classpkg2 = InstanceKlass::cast(class2)->package(); } else { assert(class2->is_typeArray_klass(), "should be type array"); classloader2 = NULL; + classpkg2 = NULL; } - const Symbol* classname2 = class2->name(); - return InstanceKlass::is_same_class_package(classloader1, classname1, - classloader2, classname2); + // Same package is determined by comparing class loader + // and package entries. Both must be the same. This rule + // applies even to classes that are defined in the unnamed + // package, they still must have the same class loader. + if ((classloader1 == classloader2) && (classpkg1 == classpkg2)) { + return true; + } + + return false; } bool InstanceKlass::is_same_class_package(oop other_class_loader, @@ -2225,43 +2335,24 @@ bool InstanceKlass::is_same_class_package(oop class_loader1, const Symbol* class // The Symbol*'s are in UTF8 encoding. Since we only need to check explicitly // for ASCII characters ('/', 'L', '['), we can keep them in UTF8 encoding. // Otherwise, we just compare jbyte values between the strings. - const jbyte *name1 = class_name1->base(); - const jbyte *name2 = class_name2->base(); + int length1 = 0; + int length2 = 0; + const jbyte *name1 = package_from_name(class_name1, length1); + const jbyte *name2 = package_from_name(class_name2, length2); - const jbyte *last_slash1 = UTF8::strrchr(name1, class_name1->utf8_length(), '/'); - const jbyte *last_slash2 = UTF8::strrchr(name2, class_name2->utf8_length(), '/'); + if ((length1 < 0) || (length2 < 0)) { + // error occurred parsing package name. + return false; + } - if ((last_slash1 == NULL) || (last_slash2 == NULL)) { + if ((name1 == NULL) || (name2 == NULL)) { // One of the two doesn't have a package. Only return true // if the other one also doesn't have a package. - return last_slash1 == last_slash2; - } else { - // Skip over '['s - if (*name1 == '[') { - do { - name1++; - } while (*name1 == '['); - if (*name1 != 'L') { - // Something is terribly wrong. Shouldn't be here. - return false; - } - } - if (*name2 == '[') { - do { - name2++; - } while (*name2 == '['); - if (*name2 != 'L') { - // Something is terribly wrong. Shouldn't be here. - return false; - } - } - - // Check that package part is identical - int length1 = last_slash1 - name1; - int length2 = last_slash2 - name2; - - return UTF8::equal(name1, length1, name2, length2); + return name1 == name2; } + + // Check that package part is identical + return UTF8::equal(name1, length1, name2, length2); } } @@ -2300,7 +2391,7 @@ bool InstanceKlass::is_same_package_member_impl(const InstanceKlass* class1, if (!class2->is_instance_klass()) return false; // must be in same package before we try anything else - if (!class1->is_same_class_package(class2->class_loader(), class2->name())) + if (!class1->is_same_class_package(class2)) return false; // As long as there is an outer1.getEnclosingClass, @@ -2908,6 +2999,7 @@ const char* InstanceKlass::internal_name() const { void InstanceKlass::print_loading_log(LogLevel::type type, ClassLoaderData* loader_data, + const char* module_name, const ClassFileStream* cfs) const { ResourceMark rm; outputStream* log; @@ -2928,7 +3020,11 @@ void InstanceKlass::print_loading_log(LogLevel::type type, // Source if (cfs != NULL) { if (cfs->source() != NULL) { - log->print(" source: %s", cfs->source()); + if (module_name != NULL) { + log->print(" source: jrt:/%s", module_name); + } else { + log->print(" source: %s", cfs->source()); + } } else if (loader_data == ClassLoaderData::the_null_class_loader_data()) { Thread* THREAD = Thread::current(); Klass* caller = diff --git a/hotspot/src/share/vm/oops/instanceKlass.hpp b/hotspot/src/share/vm/oops/instanceKlass.hpp index 47924e5d03b..4fdcc440e09 100644 --- a/hotspot/src/share/vm/oops/instanceKlass.hpp +++ b/hotspot/src/share/vm/oops/instanceKlass.hpp @@ -25,8 +25,11 @@ #ifndef SHARE_VM_OOPS_INSTANCEKLASS_HPP #define SHARE_VM_OOPS_INSTANCEKLASS_HPP +#include "classfile/classLoader.hpp" #include "classfile/classLoaderData.hpp" +#include "classfile/packageEntry.hpp" #include "gc/shared/specialized_oop_closures.hpp" +#include "classfile/moduleEntry.hpp" #include "logging/logLevel.hpp" #include "memory/referenceType.hpp" #include "oops/annotations.hpp" @@ -140,6 +143,8 @@ class InstanceKlass: public Klass { protected: // Annotations for this class Annotations* _annotations; + // Package this class is defined in + PackageEntry* _package_entry; // Array classes holding elements of this class. Klass* _array_klasses; // Constant pool for this class. @@ -199,16 +204,22 @@ class InstanceKlass: public Klass { // Start after _misc_kind field. enum { - _misc_rewritten = 1 << 2, // methods rewritten. - _misc_has_nonstatic_fields = 1 << 3, // for sizing with UseCompressedOops - _misc_should_verify_class = 1 << 4, // allow caching of preverification - _misc_is_anonymous = 1 << 5, // has embedded _host_klass field - _misc_is_contended = 1 << 6, // marked with contended annotation - _misc_has_default_methods = 1 << 7, // class/superclass/implemented interfaces has default methods - _misc_declares_default_methods = 1 << 8, // directly declares default methods (any access) - _misc_has_been_redefined = 1 << 9, // class has been redefined - _misc_is_scratch_class = 1 << 10 // class is the redefined scratch class + _misc_rewritten = 1 << 2, // methods rewritten. + _misc_has_nonstatic_fields = 1 << 3, // for sizing with UseCompressedOops + _misc_should_verify_class = 1 << 4, // allow caching of preverification + _misc_is_anonymous = 1 << 5, // has embedded _host_klass field + _misc_is_contended = 1 << 6, // marked with contended annotation + _misc_has_default_methods = 1 << 7, // class/superclass/implemented interfaces has default methods + _misc_declares_default_methods = 1 << 8, // directly declares default methods (any access) + _misc_has_been_redefined = 1 << 9, // class has been redefined + _misc_is_scratch_class = 1 << 10, // class is the redefined scratch class + _misc_is_shared_boot_class = 1 << 11, // defining class loader is boot class loader + _misc_is_shared_platform_class = 1 << 12, // defining class loader is platform class loader + _misc_is_shared_app_class = 1 << 13 // defining class loader is app class loader }; + u2 loader_type_bits() { + return _misc_is_shared_boot_class|_misc_is_shared_platform_class|_misc_is_shared_app_class; + } u2 _misc_flags; u2 _minor_version; // minor version number of class file u2 _major_version; // major version number of class file @@ -290,6 +301,39 @@ class InstanceKlass: public Klass { friend class SystemDictionary; public: + u2 loader_type() { + return _misc_flags & loader_type_bits(); + } + + bool is_shared_boot_class() const { + return (_misc_flags & _misc_is_shared_boot_class) != 0; + } + bool is_shared_platform_class() const { + return (_misc_flags & _misc_is_shared_platform_class) != 0; + } + bool is_shared_app_class() const { + return (_misc_flags & _misc_is_shared_app_class) != 0; + } + + void set_class_loader_type(jshort loader_type) { + assert(( _misc_flags & loader_type_bits()) == 0, + "Should only be called once for each class."); + switch (loader_type) { + case ClassLoader::BOOT_LOADER: + _misc_flags |= _misc_is_shared_boot_class; + break; + case ClassLoader::PLATFORM_LOADER: + _misc_flags |= _misc_is_shared_platform_class; + break; + case ClassLoader::APP_LOADER: + _misc_flags |= _misc_is_shared_app_class; + break; + default: + ShouldNotReachHere(); + break; + } + } + bool has_nonstatic_fields() const { return (_misc_flags & _misc_has_nonstatic_fields) != 0; } @@ -395,6 +439,11 @@ class InstanceKlass: public Klass { bool is_override(const methodHandle& super_method, Handle targetclassloader, Symbol* targetclassname, TRAPS); // package + PackageEntry* package() const { return _package_entry; } + ModuleEntry* module() const; + bool in_unnamed_package() const { return (_package_entry == NULL); } + void set_package(PackageEntry* p) { _package_entry = p; } + void set_package(ClassLoaderData* loader_data, TRAPS); bool is_same_class_package(const Klass* class2) const; bool is_same_class_package(oop classloader2, const Symbol* classname2) const; static bool is_same_class_package(oop class_loader1, @@ -1030,6 +1079,7 @@ public: // Naming const char* signature_name() const; + static const jbyte* package_from_name(const Symbol* name, int& length); // GC specific object visitors // @@ -1247,7 +1297,8 @@ public: void oop_verify_on(oop obj, outputStream* st); // Logging - void print_loading_log(LogLevel::type type, ClassLoaderData* loader_data, const ClassFileStream* cfs) const; + void print_loading_log(LogLevel::type type, ClassLoaderData* loader_data, + const char* module_name, const ClassFileStream* cfs) const; }; // for adding methods diff --git a/hotspot/src/share/vm/oops/klass.cpp b/hotspot/src/share/vm/oops/klass.cpp index 3cedd3470f0..8e3267868b8 100644 --- a/hotspot/src/share/vm/oops/klass.cpp +++ b/hotspot/src/share/vm/oops/klass.cpp @@ -512,7 +512,21 @@ void Klass::restore_unshareable_info(ClassLoaderData* loader_data, Handle protec // gotten an OOM later but keep the mirror if it was created. if (java_mirror() == NULL) { Handle loader = loader_data->class_loader(); - java_lang_Class::create_mirror(this, loader, protection_domain, CHECK); + ModuleEntry* module_entry = NULL; + Klass* k = this; + if (k->is_objArray_klass()) { + k = ObjArrayKlass::cast(k)->bottom_klass(); + } + // Obtain klass' module. + if (k->is_instance_klass()) { + InstanceKlass* ik = (InstanceKlass*) k; + module_entry = ik->module(); + } else { + module_entry = ModuleEntryTable::javabase_module(); + } + // Obtain java.lang.reflect.Module, if available + Handle module_handle(THREAD, ((module_entry != NULL) ? JNIHandles::resolve(module_entry->module()) : (oop)NULL)); + java_lang_Class::create_mirror(this, loader, module_handle, protection_domain, CHECK); } } diff --git a/hotspot/src/share/vm/oops/method.cpp b/hotspot/src/share/vm/oops/method.cpp index 7620ea9f985..d309e9aaed8 100644 --- a/hotspot/src/share/vm/oops/method.cpp +++ b/hotspot/src/share/vm/oops/method.cpp @@ -1329,7 +1329,7 @@ vmSymbols::SID Method::klass_id_for_intrinsics(const Klass* holder) { // exception: the AES intrinsics come from lib/ext/sunjce_provider.jar // which does not use the class default class loader so we check for its loader here const InstanceKlass* ik = InstanceKlass::cast(holder); - if ((ik->class_loader() != NULL) && !SystemDictionary::is_ext_class_loader(ik->class_loader())) { + if ((ik->class_loader() != NULL) && !SystemDictionary::is_platform_class_loader(ik->class_loader())) { return vmSymbols::NO_SID; // regardless of name, no intrinsics here } diff --git a/hotspot/src/share/vm/oops/objArrayKlass.cpp b/hotspot/src/share/vm/oops/objArrayKlass.cpp index bd696d5cc0f..8750ac73736 100644 --- a/hotspot/src/share/vm/oops/objArrayKlass.cpp +++ b/hotspot/src/share/vm/oops/objArrayKlass.cpp @@ -135,8 +135,18 @@ Klass* ObjArrayKlass::allocate_objArray_klass(ClassLoaderData* loader_data, // GC walks these as strong roots. loader_data->add_class(oak); + // The array is defined in the module of its bottom class + Klass* bottom_klass = oak->bottom_klass(); + ModuleEntry* module; + if (bottom_klass->is_instance_klass()) { + module = InstanceKlass::cast(bottom_klass)->module(); + } else { + module = ModuleEntryTable::javabase_module(); + } + assert(module != NULL, "No module entry for array"); + // Call complete_create_array_klass after all instance variables has been initialized. - ArrayKlass::complete_create_array_klass(oak, super_klass, CHECK_0); + ArrayKlass::complete_create_array_klass(oak, super_klass, module, CHECK_0); return oak; } diff --git a/hotspot/src/share/vm/oops/typeArrayKlass.cpp b/hotspot/src/share/vm/oops/typeArrayKlass.cpp index f8ec9ee822e..3fa977001a0 100644 --- a/hotspot/src/share/vm/oops/typeArrayKlass.cpp +++ b/hotspot/src/share/vm/oops/typeArrayKlass.cpp @@ -70,7 +70,7 @@ TypeArrayKlass* TypeArrayKlass::create_klass(BasicType type, null_loader_data->add_class(ak); // Call complete_create_array_klass after all instance variables have been initialized. - complete_create_array_klass(ak, ak->super(), CHECK_NULL); + complete_create_array_klass(ak, ak->super(), ModuleEntryTable::javabase_module(), CHECK_NULL); return ak; } diff --git a/hotspot/src/share/vm/opto/library_call.cpp b/hotspot/src/share/vm/opto/library_call.cpp index 74d6f881e8d..cbcc5ae017d 100644 --- a/hotspot/src/share/vm/opto/library_call.cpp +++ b/hotspot/src/share/vm/opto/library_call.cpp @@ -319,8 +319,6 @@ class LibraryCallKit : public GraphKit { bool inline_profileBoolean(); bool inline_isCompileConstant(); - - bool inline_deoptimize(); }; //---------------------------make_vm_intrinsic---------------------------- @@ -821,9 +819,6 @@ bool LibraryCallKit::try_to_inline(int predicate) { case vmIntrinsics::_hasNegatives: return inline_hasNegatives(); - case vmIntrinsics::_deoptimize: - return inline_deoptimize(); - default: // If you get here, it may be that someone has added a new intrinsic // to the list in vmSymbols.hpp without implementing it here. @@ -6754,12 +6749,3 @@ bool LibraryCallKit::inline_isCompileConstant() { set_result(n->is_Con() ? intcon(1) : intcon(0)); return true; } - -bool LibraryCallKit::inline_deoptimize() { - assert(WhiteBoxAPI, ""); - PreserveReexecuteState preexecs(this); - jvms()->set_should_reexecute(false); - uncommon_trap(Deoptimization::Reason_intrinsic, - Deoptimization::Action_none); - return true; -} diff --git a/hotspot/src/share/vm/precompiled/precompiled.hpp b/hotspot/src/share/vm/precompiled/precompiled.hpp index bd79327acc2..f3e0fea113f 100644 --- a/hotspot/src/share/vm/precompiled/precompiled.hpp +++ b/hotspot/src/share/vm/precompiled/precompiled.hpp @@ -58,6 +58,9 @@ # include "classfile/classFileStream.hpp" # include "classfile/classLoader.hpp" # include "classfile/javaClasses.hpp" +# include "classfile/moduleEntry.hpp" +# include "classfile/modules.hpp" +# include "classfile/packageEntry.hpp" # include "classfile/symbolTable.hpp" # include "classfile/systemDictionary.hpp" # include "classfile/vmSymbols.hpp" diff --git a/hotspot/src/share/vm/prims/jni.cpp b/hotspot/src/share/vm/prims/jni.cpp index 40e3c92668a..5f08fcd2c1f 100644 --- a/hotspot/src/share/vm/prims/jni.cpp +++ b/hotspot/src/share/vm/prims/jni.cpp @@ -28,7 +28,9 @@ #include "classfile/altHashing.hpp" #include "classfile/classFileStream.hpp" #include "classfile/classLoader.hpp" +#include "classfile/javaClasses.hpp" #include "classfile/javaClasses.inline.hpp" +#include "classfile/modules.hpp" #include "classfile/symbolTable.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -3443,6 +3445,47 @@ JNI_LEAF(jint, jni_GetJavaVM(JNIEnv *env, JavaVM **vm)) return JNI_OK; JNI_END + +JNI_ENTRY(jobject, jni_GetModule(JNIEnv* env, jclass clazz)) + JNIWrapper("GetModule"); + return Modules::get_module(clazz, THREAD); +JNI_END + + +JNI_ENTRY(void, jni_AddModuleReads(JNIEnv* env, jobject m1, jobject m2)) + JNIWrapper("AddModuleReads"); + if (m1 == NULL || m2 == NULL) { + THROW(vmSymbols::java_lang_NullPointerException()); + } + JavaValue result(T_VOID); + Handle m1_h(THREAD, JNIHandles::resolve(m1)); + if (!java_lang_reflect_Module::is_instance(m1_h())) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), "Bad m1 object"); + } + Handle m2_h(THREAD, JNIHandles::resolve(m2)); + if (!java_lang_reflect_Module::is_instance(m2_h())) { + THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), "Bad m2 object"); + } + JavaCalls::call_static(&result, + KlassHandle(THREAD, SystemDictionary::module_Modules_klass()), + vmSymbols::addReads_name(), + vmSymbols::addReads_signature(), + m1_h, + m2_h, + THREAD); +JNI_END + + +JNI_ENTRY(jboolean, jni_CanReadModule(JNIEnv* env, jobject m1, jobject m2)) + JNIWrapper("CanReadModule"); + if (m1 == NULL || m2 == NULL) { + THROW_(vmSymbols::java_lang_NullPointerException(), JNI_FALSE); + } + jboolean res = Modules::can_read_module(m1, m2, CHECK_false); + return res; +JNI_END + + // Structure containing all jni functions struct JNINativeInterface_ jni_NativeInterface = { NULL, @@ -3722,7 +3765,13 @@ struct JNINativeInterface_ jni_NativeInterface = { // New 1_6 features - jni_GetObjectRefType + jni_GetObjectRefType, + + // Module features + + jni_GetModule, + jni_AddModuleReads, + jni_CanReadModule }; diff --git a/hotspot/src/share/vm/prims/jni.h b/hotspot/src/share/vm/prims/jni.h index 564499f83da..96996885a07 100644 --- a/hotspot/src/share/vm/prims/jni.h +++ b/hotspot/src/share/vm/prims/jni.h @@ -765,6 +765,17 @@ struct JNINativeInterface_ { jobjectRefType (JNICALL *GetObjectRefType) (JNIEnv* env, jobject obj); + + /* Module Features */ + + jobject (JNICALL *GetModule) + (JNIEnv* env, jclass clazz); + + void (JNICALL *AddModuleReads) + (JNIEnv* env, jobject m1, jobject m2); + + jboolean (JNICALL *CanReadModule) + (JNIEnv* env, jobject m1, jobject m2); }; /* @@ -1857,6 +1868,20 @@ struct JNIEnv_ { return functions->GetObjectRefType(this, obj); } + /* Module Features */ + + jobject GetModule(jclass clazz) { + return functions->GetModule(this, clazz); + } + + void AddModuleReads(jobject fromModule, jobject sourceModule) { + functions->AddModuleReads(this, fromModule, sourceModule); + } + + jboolean CanReadModule(jobject askingModule, jobject sourceModule) { + return functions->CanReadModule(this, askingModule, sourceModule); + } + #endif /* __cplusplus */ }; diff --git a/hotspot/src/share/vm/prims/jniCheck.cpp b/hotspot/src/share/vm/prims/jniCheck.cpp index 8ba2212ac78..d6b703574d0 100644 --- a/hotspot/src/share/vm/prims/jniCheck.cpp +++ b/hotspot/src/share/vm/prims/jniCheck.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 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 @@ -1989,7 +1989,48 @@ JNI_ENTRY_CHECKED(jint, return result; JNI_END +JNI_ENTRY_CHECKED(jobject, + checked_jni_GetModule(JNIEnv *env, + jclass clazz)) + functionEnter(thr); + IN_VM( + jniCheck::validate_class(thr, clazz, false); + ) + jobject result = UNCHECKED()->GetModule(env,clazz); + functionExit(thr); + return result; +JNI_END +JNI_ENTRY_CHECKED(void, + checked_jni_AddModuleReads(JNIEnv *env, + jobject fromModule, + jobject sourceModule)) + functionEnter(thr); + IN_VM( + jniCheck::validate_object(thr, fromModule); + if (sourceModule != NULL) { + jniCheck::validate_object(thr, sourceModule); + } + ) + UNCHECKED()->AddModuleReads(env,fromModule,sourceModule); + functionExit(thr); +JNI_END + +JNI_ENTRY_CHECKED(jboolean, + checked_jni_CanReadModule(JNIEnv *env, + jobject askingModule, + jobject sourceModule)) + functionEnter(thr); + IN_VM( + jniCheck::validate_object(thr, askingModule); + if (sourceModule != NULL) { + jniCheck::validate_object(thr, sourceModule); + } + ) + jboolean result = UNCHECKED()->CanReadModule(env,askingModule,sourceModule); + functionExit(thr); + return result; +JNI_END /* * Structure containing all checked jni functions @@ -2272,7 +2313,13 @@ struct JNINativeInterface_ checked_jni_NativeInterface = { // New 1.6 Features - checked_jni_GetObjectRefType + checked_jni_GetObjectRefType, + + // Module Features + + checked_jni_GetModule, + checked_jni_AddModuleReads, + checked_jni_CanReadModule }; diff --git a/hotspot/src/share/vm/prims/jvm.cpp b/hotspot/src/share/vm/prims/jvm.cpp index 2067afd58d5..093efe3bab1 100644 --- a/hotspot/src/share/vm/prims/jvm.cpp +++ b/hotspot/src/share/vm/prims/jvm.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2016, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -25,8 +25,12 @@ #include "precompiled.hpp" #include "classfile/classFileStream.hpp" #include "classfile/classLoader.hpp" +#include "classfile/classLoaderData.inline.hpp" #include "classfile/javaAssertions.hpp" #include "classfile/javaClasses.inline.hpp" +#include "classfile/moduleEntry.hpp" +#include "classfile/modules.hpp" +#include "classfile/packageEntry.hpp" #include "classfile/stringTable.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" @@ -1047,6 +1051,58 @@ JVM_ENTRY(jclass, JVM_FindLoadedClass(JNIEnv *env, jobject loader, jstring name) (jclass) JNIHandles::make_local(env, k->java_mirror()); JVM_END +// Module support ////////////////////////////////////////////////////////////////////////////// + +JVM_ENTRY(void, JVM_DefineModule(JNIEnv *env, jobject module, jstring version, jstring location, + jobjectArray packages)) + JVMWrapper("JVM_DefineModule"); + Modules::define_module(module, version, location, packages, CHECK); +JVM_END + +JVM_ENTRY(void, JVM_SetBootLoaderUnnamedModule(JNIEnv *env, jobject module)) + JVMWrapper("JVM_SetBootLoaderUnnamedModule"); + Modules::set_bootloader_unnamed_module(module, CHECK); +JVM_END + +JVM_ENTRY(void, JVM_AddModuleExports(JNIEnv *env, jobject from_module, jstring package, jobject to_module)) + JVMWrapper("JVM_AddModuleExports"); + Modules::add_module_exports_qualified(from_module, package, to_module, CHECK); +JVM_END + +JVM_ENTRY(void, JVM_AddModuleExportsToAllUnnamed(JNIEnv *env, jobject from_module, jstring package)) + JVMWrapper("JVM_AddModuleExportsToAllUnnamed"); + Modules::add_module_exports_to_all_unnamed(from_module, package, CHECK); +JVM_END + +JVM_ENTRY(void, JVM_AddModuleExportsToAll(JNIEnv *env, jobject from_module, jstring package)) + JVMWrapper("JVM_AddModuleExportsToAll"); + Modules::add_module_exports(from_module, package, NULL, CHECK); +JVM_END + +JVM_ENTRY (void, JVM_AddReadsModule(JNIEnv *env, jobject from_module, jobject source_module)) + JVMWrapper("JVM_AddReadsModule"); + Modules::add_reads_module(from_module, source_module, CHECK); +JVM_END + +JVM_ENTRY(jboolean, JVM_CanReadModule(JNIEnv *env, jobject asking_module, jobject source_module)) + JVMWrapper("JVM_CanReadModule"); + return Modules::can_read_module(asking_module, source_module, THREAD); +JVM_END + +JVM_ENTRY(jboolean, JVM_IsExportedToModule(JNIEnv *env, jobject from_module, jstring package, jobject to_module)) + JVMWrapper("JVM_IsExportedToModule"); + return Modules::is_exported_to_module(from_module, package, to_module, THREAD); +JVM_END + +JVM_ENTRY (void, JVM_AddModulePackage(JNIEnv *env, jobject module, jstring package)) + JVMWrapper("JVM_AddModulePackage"); + Modules::add_module_package(module, package, CHECK); +JVM_END + +JVM_ENTRY (jobject, JVM_GetModuleByPackageName(JNIEnv *env, jobject loader, jstring package)) + JVMWrapper("JVM_GetModuleByPackageName"); + return Modules::get_module_by_package_name(loader, package, THREAD); +JVM_END // Reflection support ////////////////////////////////////////////////////////////////////////////// diff --git a/hotspot/src/share/vm/prims/jvm.h b/hotspot/src/share/vm/prims/jvm.h index af12c25bf1d..31ff94c5743 100644 --- a/hotspot/src/share/vm/prims/jvm.h +++ b/hotspot/src/share/vm/prims/jvm.h @@ -416,6 +416,41 @@ JVM_DefineClassWithSource(JNIEnv *env, const char *name, jobject loader, const jbyte *buf, jsize len, jobject pd, const char *source); +/* + * Module support funcions + */ + +JNIEXPORT void JNICALL +JVM_DefineModule(JNIEnv *env, jobject module, jstring version, jstring location, + jobjectArray packages); + +JNIEXPORT void JNICALL +JVM_SetBootLoaderUnnamedModule(JNIEnv *env, jobject module); + +JNIEXPORT void JNICALL +JVM_AddModuleExports(JNIEnv *env, jobject from_module, jstring package, jobject to_module); + +JNIEXPORT void JNICALL +JVM_AddModuleExportsToAllUnnamed(JNIEnv *env, jobject from_module, jstring package); + +JNIEXPORT void JNICALL +JVM_AddModuleExportsToAll(JNIEnv *env, jobject from_module, jstring package); + +JNIEXPORT void JNICALL +JVM_AddReadsModule(JNIEnv *env, jobject from_module, jobject source_module); + +JNIEXPORT jboolean JNICALL +JVM_CanReadModule(JNIEnv *env, jobject asking_module, jobject source_module); + +JNIEXPORT jboolean JNICALL +JVM_IsExportedToModule(JNIEnv *env, jobject from_module, jstring package, jobject to_module); + +JNIEXPORT void JNICALL +JVM_AddModulePackage(JNIEnv* env, jobject module, jstring package); + +JNIEXPORT jobject JNICALL +JVM_GetModuleByPackageName(JNIEnv* env, jobject loader, jstring package); + /* * Reflection support functions */ @@ -920,6 +955,7 @@ JVM_IsSameClassPackage(JNIEnv *env, jclass class1, jclass class2); #define JVM_ACC_SYNTHETIC 0x1000 /* compiler-generated class, method or field */ #define JVM_ACC_ANNOTATION 0x2000 /* annotation type */ #define JVM_ACC_ENUM 0x4000 /* field is declared as element of enum */ +#define JVM_ACC_MODULE 0x8000 /* module-info class file */ #define JVM_ACC_PUBLIC_BIT 0 #define JVM_ACC_PRIVATE_BIT 1 diff --git a/hotspot/src/share/vm/prims/jvmti.xml b/hotspot/src/share/vm/prims/jvmti.xml index 98a0a0640aa..f6a18c771ac 100644 --- a/hotspot/src/share/vm/prims/jvmti.xml +++ b/hotspot/src/share/vm/prims/jvmti.xml @@ -1,7 +1,7 @@ + majorversion="9" + minorversion="0" + microversion="0"> <tm>JVM</tm> Tool Interface @@ -863,6 +863,23 @@ Agent_OnUnload_L(JavaVM *vm) to be instrumented by the use of wrapper methods. + + Agents that instrument code in named modules may need to arrange for those + modules to read other modules. If code is instrumented to invoke a method + in a support class in another module, then the module of the instrumented + code should read the module of the supporting class. Furthermore, the + supporting class will only be accessible to the instrumented code if + it is public and in a package that is exported by its module. + Agents can use the JNI functions CanReadModule and + AddModuleReads to test and update a module to read another. +

+ As an aid to agents that deploy supporting classes on the search path of + the bootstrap class loader, or the search path of the class loader that + loads the main class, the Java virtual machine arranges for the module + of classes transformed by the event to + read the unnamed module of both class loaders. + + uses modified UTF-8 to encode character strings. This is the same encoding used by JNI. @@ -1880,10 +1897,6 @@ jvmtiEnv *jvmti; jvmtiThreadInfo On return, filled with information describing the specified thread. -

- For JDK 1.1 implementations that don't - recognize context class loaders, - the context_class_loader field will be NULL. @@ -6460,6 +6473,43 @@ class C2 extends C1 implements I2 { + + + + + + + Get All Modules + + Return an array of all modules loaded in the virtual machine. + The number of modules in the array is returned via + module_count_ptr, and the array itself via + modules_ptr. +

+ + new + + + + + + + On return, points to the number of returned modules. + + + + + + On return, points to an array of references, one + for each module. + + + + + + + + @@ -6508,9 +6558,6 @@ class C2 extends C1 implements I2 { either by defining it directly or by delegation to another class loader. See .

- For JDK version 1.1 implementations that don't - recognize the distinction between initiating and defining class loaders, - this function should return all classes loaded in the virtual machine. The number of classes in the array is returned via class_count_ptr, and the array itself via classes_ptr. @@ -9941,6 +9988,12 @@ myInit() { See . + + + Can generate the VMStart event early. + See . + + @@ -12351,7 +12404,7 @@ myInit() { - This event is sent when the VM obtains class file data, @@ -12367,9 +12420,9 @@ myInit() { bytecode instrumentation for usage information.

- This event may be sent before the VM is initialized (the primordial - phase). During this time - no VM resources should be created. Some classes might not be compatible + This event may be sent before the VM is initialized (the start + phase). + Some classes might not be compatible with the function (eg. ROMized classes) and this event will not be generated for these classes.

@@ -12378,9 +12431,7 @@ myInit() { using the memory allocation function because the VM is responsible for freeing the new class file data buffer - using . - Note that - is permitted during the primordial phase. + using .

If the agent wishes to modify the class file, it must set new_class_data to point @@ -12427,8 +12478,6 @@ myInit() { The JNI environment of the event (current) thread. - Will be NULL if sent during the primordial - phase. @@ -12496,9 +12545,28 @@ myInit() { The VM initialization event signals the start of the VM. At this time JNI is live but the VM is not yet fully initialized. Once this event is generated, the agent is free to call any JNI function. - This event signals the beginning of the start phase, + This event signals the beginning of the start phase, functions permitted in the start phase may be called.

+ The timing of this event may depend on whether the agent has added the + + can_generate_early_vmstart capability or not. + If the capability has been added then the VM posts the event as early + as possible. The VM is capable of executing bytecode but it may not have + initialized to the point where it can load classes in modules other than + java.base. Agents that do load-time instrumentation in this + phase must take great care when instrumenting code that potentially + executes in this phase. Care should also be taken with JNI + FindClass as it may not be possible to load classes that are + not in the java.base module. + If the capability has not been added then the VM delays posting this + event until it is capable of loading classes in modules other than + java.base or the VM has completed its initialization. + Agents that create more than one JVM TI environment, where the + capability is added to some but not all environments, may observe the + start phase beginning earlier in the JVM TI environments that possess + the capabilty. +

In the case of VM start-up failure, this event will not be sent. jvmdi @@ -12576,7 +12644,7 @@ myInit() { - Sent when a method is compiled and loaded into memory by the VM. @@ -12667,7 +12735,7 @@ myInit() { - Sent when a compiled method is unloaded from memory. @@ -14340,6 +14408,17 @@ typedef void (JNICALL *jvmtiEventVMInit) Added support for statically linked agents. + + Support for modules: + - The majorversion is 9 now + - The ClassFileLoadHook events are not sent during the primordial phase anymore. + - Add new function GetAllModules + + + Support for modules: + - Add new capability can_generate_early_vmstart + - Allow CompiledMethodLoad events at start phase + diff --git a/hotspot/src/share/vm/prims/jvmti.xsl b/hotspot/src/share/vm/prims/jvmti.xsl index 1f975d2c1d4..00a1551d63d 100644 --- a/hotspot/src/share/vm/prims/jvmti.xsl +++ b/hotspot/src/share/vm/prims/jvmti.xsl @@ -1,6 +1,6 @@ - */ + */ - /* AUTOMATICALLY GENERATED FILE - DO NOT EDIT */ + /* AUTOMATICALLY GENERATED FILE - DO NOT EDIT */ - // AUTOMATICALLY GENERATED FILE - DO NOT EDIT + // AUTOMATICALLY GENERATED FILE - DO NOT EDIT diff --git a/hotspot/src/share/vm/prims/jvmtiManageCapabilities.cpp b/hotspot/src/share/vm/prims/jvmtiManageCapabilities.cpp index b614b5acea0..a9ce01c93d1 100644 --- a/hotspot/src/share/vm/prims/jvmtiManageCapabilities.cpp +++ b/hotspot/src/share/vm/prims/jvmtiManageCapabilities.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 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 @@ -133,9 +133,7 @@ jvmtiCapabilities JvmtiManageCapabilities::init_onload_capabilities() { jc.can_get_owned_monitor_info = 1; jc.can_get_owned_monitor_stack_depth_info = 1; jc.can_get_current_contended_monitor = 1; - // jc.can_get_monitor_info = 1; - jc.can_tag_objects = 1; // TODO: this should have been removed - jc.can_generate_object_free_events = 1; // TODO: this should have been removed + jc.can_generate_early_vmstart = 1; return jc; } @@ -454,6 +452,8 @@ void JvmtiManageCapabilities:: print(const jvmtiCapabilities* cap) { tty->print_cr("can_generate_resource_exhaustion_heap_events"); if (cap->can_generate_resource_exhaustion_threads_events) tty->print_cr("can_generate_resource_exhaustion_threads_events"); + if (cap->can_generate_early_vmstart) + tty->print_cr("can_generate_early_vmstart"); } #endif diff --git a/hotspot/src/share/vm/prims/methodHandles.cpp b/hotspot/src/share/vm/prims/methodHandles.cpp index 38b68a40422..de66179e45b 100644 --- a/hotspot/src/share/vm/prims/methodHandles.cpp +++ b/hotspot/src/share/vm/prims/methodHandles.cpp @@ -1177,9 +1177,9 @@ JVM_ENTRY(jobject, MHN_resolve_Mem(JNIEnv *env, jobject igcls, jobject mname_jh, if (reference_klass != NULL && reference_klass->is_instance_klass()) { // Emulate LinkResolver::check_klass_accessability. Klass* caller = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(caller_jh)); - if (!Reflection::verify_class_access(caller, - reference_klass, - true)) { + if (Reflection::verify_class_access(caller, + reference_klass, + true) != Reflection::ACCESS_OK) { THROW_MSG_NULL(vmSymbols::java_lang_InternalError(), reference_klass->external_name()); } } diff --git a/hotspot/src/share/vm/prims/nativeLookup.cpp b/hotspot/src/share/vm/prims/nativeLookup.cpp index c89e4f10c32..1ebcaeef6ac 100644 --- a/hotspot/src/share/vm/prims/nativeLookup.cpp +++ b/hotspot/src/share/vm/prims/nativeLookup.cpp @@ -41,7 +41,9 @@ #include "runtime/sharedRuntime.hpp" #include "runtime/signature.hpp" #include "utilities/macros.hpp" - +#if INCLUDE_TRACE +#include "trace/traceMacros.hpp" +#endif static void mangle_name_on(outputStream* st, Symbol* name, int begin, int end) { char* bytes = (char*)name->bytes() + begin; @@ -131,6 +133,9 @@ static JNINativeMethod lookup_special_native_methods[] = { { CC"Java_jdk_vm_ci_runtime_JVMCI_initializeRuntime", NULL, FN_PTR(JVM_GetJVMCIRuntime) }, { CC"Java_jdk_vm_ci_hotspot_CompilerToVM_registerNatives", NULL, FN_PTR(JVM_RegisterJVMCINatives) }, #endif +#if INCLUDE_TRACE + { CC"Java_jdk_jfr_internal_JVM_registerNatives", NULL, TRACE_REGISTER_NATIVES }, +#endif }; static address lookup_special_native(char* jni_name) { diff --git a/hotspot/src/share/vm/prims/whitebox.cpp b/hotspot/src/share/vm/prims/whitebox.cpp index 776df026f7d..c6d4c41163a 100644 --- a/hotspot/src/share/vm/prims/whitebox.cpp +++ b/hotspot/src/share/vm/prims/whitebox.cpp @@ -27,6 +27,7 @@ #include #include "classfile/classLoaderData.hpp" +#include "classfile/modules.hpp" #include "classfile/stringTable.hpp" #include "code/codeCache.hpp" #include "compiler/methodMatcher.hpp" @@ -1250,6 +1251,43 @@ WB_ENTRY(void, WB_FreeMetaspace(JNIEnv* env, jobject wb, jobject class_loader, j MetadataFactory::free_array(cld, (Array*)(uintptr_t)addr); WB_END +WB_ENTRY(void, WB_DefineModule(JNIEnv* env, jobject o, jobject module, jstring version, jstring location, + jobjectArray packages)) + Modules::define_module(module, version, location, packages, CHECK); +WB_END + +WB_ENTRY(void, WB_AddModuleExports(JNIEnv* env, jobject o, jobject from_module, jstring package, jobject to_module)) + Modules::add_module_exports_qualified(from_module, package, to_module, CHECK); +WB_END + +WB_ENTRY(void, WB_AddModuleExportsToAllUnnamed(JNIEnv* env, jobject o, jclass module, jstring package)) + Modules::add_module_exports_to_all_unnamed(module, package, CHECK); +WB_END + +WB_ENTRY(void, WB_AddModuleExportsToAll(JNIEnv* env, jobject o, jclass module, jstring package)) + Modules::add_module_exports(module, package, NULL, CHECK); +WB_END + +WB_ENTRY(void, WB_AddReadsModule(JNIEnv* env, jobject o, jobject from_module, jobject source_module)) + Modules::add_reads_module(from_module, source_module, CHECK); +WB_END + +WB_ENTRY(jboolean, WB_CanReadModule(JNIEnv* env, jobject o, jobject asking_module, jobject source_module)) + return Modules::can_read_module(asking_module, source_module, THREAD); +WB_END + +WB_ENTRY(jboolean, WB_IsExportedToModule(JNIEnv* env, jobject o, jobject from_module, jstring package, jobject to_module)) + return Modules::is_exported_to_module(from_module, package, to_module, THREAD); +WB_END + +WB_ENTRY(void, WB_AddModulePackage(JNIEnv* env, jobject o, jclass module, jstring package)) + Modules::add_module_package(module, package, CHECK); +WB_END + +WB_ENTRY(jobject, WB_GetModuleByPackageName(JNIEnv* env, jobject o, jobject loader, jstring package)) + return Modules::get_module_by_package_name(loader, package, THREAD); +WB_END + WB_ENTRY(jlong, WB_IncMetaspaceCapacityUntilGC(JNIEnv* env, jobject wb, jlong inc)) if (inc < 0) { THROW_MSG_0(vmSymbols::java_lang_IllegalArgumentException(), @@ -1278,7 +1316,6 @@ WB_ENTRY(jlong, WB_MetaspaceCapacityUntilGC(JNIEnv* env, jobject wb)) WB_END - WB_ENTRY(void, WB_AssertMatchingSafepointCalls(JNIEnv* env, jobject o, jboolean mutexSafepointValue, jboolean attemptedNoSafepointValue)) Monitor::SafepointCheckRequired sfpt_check_required = mutexSafepointValue ? Monitor::_safepoint_check_always : @@ -1287,10 +1324,6 @@ WB_ENTRY(void, WB_AssertMatchingSafepointCalls(JNIEnv* env, jobject o, jboolean attemptedNoSafepointValue == JNI_TRUE); WB_END -WB_ENTRY(jboolean, WB_IsSharedClass(JNIEnv* env, jobject wb, jclass clazz)) - return (jboolean)MetaspaceShared::is_in_shared_space(java_lang_Class::as_Klass(JNIHandles::resolve_non_null(clazz))); -WB_END - WB_ENTRY(jboolean, WB_IsMonitorInflated(JNIEnv* env, jobject wb, jobject obj)) oop obj_oop = JNIHandles::resolve(obj); return (jboolean) obj_oop->mark()->has_monitor(); @@ -1417,6 +1450,10 @@ WB_ENTRY(jboolean, WB_IsShared(JNIEnv* env, jobject wb, jobject obj)) return MetaspaceShared::is_in_shared_space((void*)obj_oop); WB_END +WB_ENTRY(jboolean, WB_IsSharedClass(JNIEnv* env, jobject wb, jclass clazz)) + return (jboolean)MetaspaceShared::is_in_shared_space(java_lang_Class::as_Klass(JNIHandles::resolve_non_null(clazz))); +WB_END + WB_ENTRY(jboolean, WB_AreSharedStringsIgnored(JNIEnv* env)) return StringTable::shared_string_ignored(); WB_END @@ -1528,10 +1565,9 @@ static JNINativeMethod methods[] = { {CC"runMemoryUnitTests", CC"()V", (void*)&WB_RunMemoryUnitTests}, {CC"readFromNoaccessArea",CC"()V", (void*)&WB_ReadFromNoaccessArea}, {CC"stressVirtualSpaceResize",CC"(JJJ)I", (void*)&WB_StressVirtualSpaceResize}, - {CC"isSharedClass", CC"(Ljava/lang/Class;)Z", (void*)&WB_IsSharedClass }, #if INCLUDE_ALL_GCS {CC"g1InConcurrentMark", CC"()Z", (void*)&WB_G1InConcurrentMark}, - {CC"g1IsHumongous0", CC"(Ljava/lang/Object;)Z",(void*)&WB_G1IsHumongous }, + {CC"g1IsHumongous0", CC"(Ljava/lang/Object;)Z", (void*)&WB_G1IsHumongous }, {CC"g1BelongsToHumongousRegion0", CC"(J)Z", (void*)&WB_G1BelongsToHumongousRegion}, {CC"g1BelongsToFreeRegion0", CC"(J)Z", (void*)&WB_G1BelongsToFreeRegion}, {CC"g1NumMaxRegions", CC"()J", (void*)&WB_G1NumMaxRegions }, @@ -1649,6 +1685,24 @@ static JNINativeMethod methods[] = { {CC"getCodeBlob", CC"(J)[Ljava/lang/Object;",(void*)&WB_GetCodeBlob }, {CC"getThreadStackSize", CC"()J", (void*)&WB_GetThreadStackSize }, {CC"getThreadRemainingStackSize", CC"()J", (void*)&WB_GetThreadRemainingStackSize }, + {CC"DefineModule", CC"(Ljava/lang/Object;Ljava/lang/String;Ljava/lang/String;[Ljava/lang/Object;)V", + (void*)&WB_DefineModule }, + {CC"AddModuleExports", CC"(Ljava/lang/Object;Ljava/lang/String;Ljava/lang/Object;)V", + (void*)&WB_AddModuleExports }, + {CC"AddReadsModule", CC"(Ljava/lang/Object;Ljava/lang/Object;)V", + (void*)&WB_AddReadsModule }, + {CC"CanReadModule", CC"(Ljava/lang/Object;Ljava/lang/Object;)Z", + (void*)&WB_CanReadModule }, + {CC"IsExportedToModule", CC"(Ljava/lang/Object;Ljava/lang/String;Ljava/lang/Object;)Z", + (void*)&WB_IsExportedToModule }, + {CC"AddModulePackage", CC"(Ljava/lang/Object;Ljava/lang/String;)V", + (void*)&WB_AddModulePackage }, + {CC"GetModuleByPackageName", CC"(Ljava/lang/Object;Ljava/lang/String;)Ljava/lang/Object;", + (void*)&WB_GetModuleByPackageName }, + {CC"AddModuleExportsToAllUnnamed", CC"(Ljava/lang/Object;Ljava/lang/String;)V", + (void*)&WB_AddModuleExportsToAllUnnamed }, + {CC"AddModuleExportsToAll", CC"(Ljava/lang/Object;Ljava/lang/String;)V", + (void*)&WB_AddModuleExportsToAll }, {CC"assertMatchingSafepointCalls", CC"(ZZ)V", (void*)&WB_AssertMatchingSafepointCalls }, {CC"isMonitorInflated0", CC"(Ljava/lang/Object;)Z", (void*)&WB_IsMonitorInflated }, {CC"forceSafepoint", CC"()V", (void*)&WB_ForceSafepoint }, @@ -1675,6 +1729,7 @@ static JNINativeMethod methods[] = { CC"(Ljava/lang/reflect/Executable;Ljava/lang/String;)Ljava/lang/String;", (void*)&WB_GetMethodStringOption}, {CC"isShared", CC"(Ljava/lang/Object;)Z", (void*)&WB_IsShared }, + {CC"isSharedClass", CC"(Ljava/lang/Class;)Z", (void*)&WB_IsSharedClass }, {CC"areSharedStringsIgnored", CC"()Z", (void*)&WB_AreSharedStringsIgnored }, {CC"clearInlineCaches", CC"()V", (void*)&WB_ClearInlineCaches }, }; diff --git a/hotspot/src/share/vm/runtime/arguments.cpp b/hotspot/src/share/vm/runtime/arguments.cpp index 8af50bdca2b..e6f1f20a648 100644 --- a/hotspot/src/share/vm/runtime/arguments.cpp +++ b/hotspot/src/share/vm/runtime/arguments.cpp @@ -94,6 +94,9 @@ const char* Arguments::_java_vendor_url_bug = DEFAULT_VENDOR_URL_BUG; const char* Arguments::_sun_java_launcher = DEFAULT_JAVA_LAUNCHER; int Arguments::_sun_java_launcher_pid = -1; bool Arguments::_sun_java_launcher_is_altjvm = false; +int Arguments::_patch_dirs_count = 0; +char** Arguments::_patch_dirs = NULL; +int Arguments::_bootclassloader_append_index = -1; // These parameters are reset in method parse_vm_init_args() bool Arguments::_AlwaysCompileLoopMethods = AlwaysCompileLoopMethods; @@ -117,7 +120,9 @@ SystemProperty *Arguments::_sun_boot_library_path = NULL; SystemProperty *Arguments::_java_library_path = NULL; SystemProperty *Arguments::_java_home = NULL; SystemProperty *Arguments::_java_class_path = NULL; -SystemProperty *Arguments::_sun_boot_class_path = NULL; +SystemProperty *Arguments::_jdk_boot_class_path_append = NULL; + +PathString *Arguments::_system_boot_class_path = NULL; char* Arguments::_ext_dirs = NULL; @@ -195,6 +200,12 @@ void Arguments::process_sun_java_launcher_properties(JavaVMInitArgs* args) { // Initialize system properties key and value. void Arguments::init_system_properties() { + + // Set up _system_boot_class_path which is not a property but + // relies heavily on argument processing and the jdk.boot.class.path.append + // property. It is used to store the underlying system boot class path. + _system_boot_class_path = new PathString(NULL); + PropertyList_add(&_system_properties, new SystemProperty("java.vm.specification.name", "Java Virtual Machine Specification", false)); PropertyList_add(&_system_properties, new SystemProperty("java.vm.version", VM_Version::vm_release(), false)); @@ -208,16 +219,19 @@ void Arguments::init_system_properties() { _sun_boot_library_path = new SystemProperty("sun.boot.library.path", NULL, true); _java_library_path = new SystemProperty("java.library.path", NULL, true); _java_home = new SystemProperty("java.home", NULL, true); - _sun_boot_class_path = new SystemProperty("sun.boot.class.path", NULL, true); - _java_class_path = new SystemProperty("java.class.path", "", true); + // jdk.boot.class.path.append is a non-writeable, internal property. + // It can only be set by either: + // - -Xbootclasspath/a: + // - AddToBootstrapClassLoaderSearch during JVMTI OnLoad phase + _jdk_boot_class_path_append = new SystemProperty("jdk.boot.class.path.append", "", false, true); // Add to System Property list. PropertyList_add(&_system_properties, _sun_boot_library_path); PropertyList_add(&_system_properties, _java_library_path); PropertyList_add(&_system_properties, _java_home); PropertyList_add(&_system_properties, _java_class_path); - PropertyList_add(&_system_properties, _sun_boot_class_path); + PropertyList_add(&_system_properties, _jdk_boot_class_path_append); // Set OS specific system properties values os::init_system_properties_values(); @@ -542,19 +556,19 @@ static bool verify_special_jvm_flags() { } #endif -// Constructs the system class path (aka boot class path) from the following -// components, in order: +// Constructs the system boot class path from the following components, in order: // -// prefix // from -Xbootclasspath/p:... -// base // from os::get_system_properties() or -Xbootclasspath= +// prefix // from -Xpatch:... +// base // from os::get_system_properties() // suffix // from -Xbootclasspath/a:... // // This could be AllStatic, but it isn't needed after argument processing is -// complete. -class SysClassPath: public StackObj { +// complete. After argument processing, the combined components are copied +// to Arguments::_system_boot_class_path via a call to Arguments::set_sysclasspath. +class ArgumentBootClassPath: public StackObj { public: - SysClassPath(const char* base); - ~SysClassPath(); + ArgumentBootClassPath(const char* base); + ~ArgumentBootClassPath(); inline void set_base(const char* base); inline void add_prefix(const char* prefix); @@ -562,9 +576,9 @@ public: inline void add_suffix(const char* suffix); inline void reset_path(const char* base); - inline const char* get_base() const { return _items[_scp_base]; } - inline const char* get_prefix() const { return _items[_scp_prefix]; } - inline const char* get_suffix() const { return _items[_scp_suffix]; } + inline const char* get_base() const { return _items[_bcp_base]; } + inline const char* get_prefix() const { return _items[_bcp_prefix]; } + inline const char* get_suffix() const { return _items[_bcp_suffix]; } // Combine all the components into a single c-heap-allocated string; caller // must free the string if/when no longer needed. @@ -580,55 +594,55 @@ private: // Array indices for the items that make up the sysclasspath. All except the // base are allocated in the C heap and freed by this class. enum { - _scp_prefix, // from -Xbootclasspath/p:... - _scp_base, // the default sysclasspath - _scp_suffix, // from -Xbootclasspath/a:... - _scp_nitems // the number of items, must be last. + _bcp_prefix, // was -Xpatch:... + _bcp_base, // the default system boot class path + _bcp_suffix, // from -Xbootclasspath/a:... + _bcp_nitems // the number of items, must be last. }; - const char* _items[_scp_nitems]; + const char* _items[_bcp_nitems]; }; -SysClassPath::SysClassPath(const char* base) { +ArgumentBootClassPath::ArgumentBootClassPath(const char* base) { memset(_items, 0, sizeof(_items)); - _items[_scp_base] = base; + _items[_bcp_base] = base; } -SysClassPath::~SysClassPath() { +ArgumentBootClassPath::~ArgumentBootClassPath() { // Free everything except the base. - for (int i = 0; i < _scp_nitems; ++i) { - if (i != _scp_base) reset_item_at(i); + for (int i = 0; i < _bcp_nitems; ++i) { + if (i != _bcp_base) reset_item_at(i); } } -inline void SysClassPath::set_base(const char* base) { - _items[_scp_base] = base; +inline void ArgumentBootClassPath::set_base(const char* base) { + _items[_bcp_base] = base; } -inline void SysClassPath::add_prefix(const char* prefix) { - _items[_scp_prefix] = add_to_path(_items[_scp_prefix], prefix, true); +inline void ArgumentBootClassPath::add_prefix(const char* prefix) { + _items[_bcp_prefix] = add_to_path(_items[_bcp_prefix], prefix, true); } -inline void SysClassPath::add_suffix_to_prefix(const char* suffix) { - _items[_scp_prefix] = add_to_path(_items[_scp_prefix], suffix, false); +inline void ArgumentBootClassPath::add_suffix_to_prefix(const char* suffix) { + _items[_bcp_prefix] = add_to_path(_items[_bcp_prefix], suffix, false); } -inline void SysClassPath::add_suffix(const char* suffix) { - _items[_scp_suffix] = add_to_path(_items[_scp_suffix], suffix, false); +inline void ArgumentBootClassPath::add_suffix(const char* suffix) { + _items[_bcp_suffix] = add_to_path(_items[_bcp_suffix], suffix, false); } -inline void SysClassPath::reset_item_at(int index) { - assert(index < _scp_nitems && index != _scp_base, "just checking"); +inline void ArgumentBootClassPath::reset_item_at(int index) { + assert(index < _bcp_nitems && index != _bcp_base, "just checking"); if (_items[index] != NULL) { FREE_C_HEAP_ARRAY(char, _items[index]); _items[index] = NULL; } } -inline void SysClassPath::reset_path(const char* base) { +inline void ArgumentBootClassPath::reset_path(const char* base) { // Clear the prefix and suffix. - reset_item_at(_scp_prefix); - reset_item_at(_scp_suffix); + reset_item_at(_bcp_prefix); + reset_item_at(_bcp_suffix); set_base(base); } @@ -637,17 +651,21 @@ inline void SysClassPath::reset_path(const char* base) { // Combine the bootclasspath elements, some of which may be null, into a single // c-heap-allocated string. -char* SysClassPath::combined_path() { - assert(_items[_scp_base] != NULL, "empty default sysclasspath"); +char* ArgumentBootClassPath::combined_path() { + assert(_items[_bcp_base] != NULL, "empty default sysclasspath"); - size_t lengths[_scp_nitems]; + size_t lengths[_bcp_nitems]; size_t total_len = 0; const char separator = *os::path_separator(); // Get the lengths. int i; - for (i = 0; i < _scp_nitems; ++i) { + for (i = 0; i < _bcp_nitems; ++i) { + if (i == _bcp_suffix) { + // Record index of boot loader's append path. + Arguments::set_bootclassloader_append_index((int)total_len); + } if (_items[i] != NULL) { lengths[i] = strlen(_items[i]); // Include space for the separator char (or a NULL for the last item). @@ -659,7 +677,7 @@ char* SysClassPath::combined_path() { // Copy the _items to a single string. char* cp = NEW_C_HEAP_ARRAY(char, total_len, mtInternal); char* cp_tmp = cp; - for (i = 0; i < _scp_nitems; ++i) { + for (i = 0; i < _bcp_nitems; ++i) { if (_items[i] != NULL) { memcpy(cp_tmp, _items[i], lengths[i]); cp_tmp += lengths[i]; @@ -672,7 +690,7 @@ char* SysClassPath::combined_path() { // Note: path must be c-heap-allocated (or NULL); it is freed if non-null. char* -SysClassPath::add_to_path(const char* path, const char* str, bool prepend) { +ArgumentBootClassPath::add_to_path(const char* path, const char* str, bool prepend) { char *cp; assert(str != NULL, "just checking"); @@ -706,7 +724,7 @@ SysClassPath::add_to_path(const char* path, const char* str, bool prepend) { // Scan the directory and append any jar or zip files found to path. // Note: path must be c-heap-allocated (or NULL); it is freed if non-null. -char* SysClassPath::add_jars_to_path(char* path, const char* directory) { +char* ArgumentBootClassPath::add_jars_to_path(char* path, const char* directory) { DIR* dir = os::opendir(directory); if (dir == NULL) return path; @@ -1375,6 +1393,54 @@ bool Arguments::add_property(const char* prop) { return true; } +// sets or adds a module name to the jdk.launcher.addmods property +bool Arguments::append_to_addmods_property(const char* module_name) { + const char* key = "jdk.launcher.addmods"; + const char* old_value = Arguments::get_property(key); + size_t buf_len = strlen(key) + strlen(module_name) + 2; + if (old_value != NULL) { + buf_len += strlen(old_value) + 1; + } + char* new_value = AllocateHeap(buf_len, mtInternal); + if (new_value == NULL) { + return false; + } + if (old_value == NULL) { + jio_snprintf(new_value, buf_len, "%s=%s", key, module_name); + } else { + jio_snprintf(new_value, buf_len, "%s=%s,%s", key, old_value, module_name); + } + bool added = add_property(new_value); + FreeHeap(new_value); + return added; +} + +#if INCLUDE_CDS +void Arguments::check_unsupported_dumping_properties() { + assert(DumpSharedSpaces, "this function is only used with -Xshare:dump"); + const char* unsupported_properties[5] = { "jdk.module.main", + "jdk.module.path", + "jdk.upgrade.module.path", + "jdk.launcher.addmods", + "jdk.launcher.limitmods" }; + const char* unsupported_options[5] = { "-m", + "-modulepath", + "-upgrademodulepath", + "-addmods", + "-limitmods" }; + SystemProperty* sp = system_properties(); + while (sp != NULL) { + for (int i = 0; i < 5; i++) { + if (strcmp(sp->key(), unsupported_properties[i]) == 0) { + vm_exit_during_initialization( + "Cannot use the following option when dumping the shared archive", unsupported_options[i]); + } + } + sp = sp->next(); + } +} +#endif + //=========================================================================================================== // Setting int/mixed/comp mode flags @@ -2553,8 +2619,8 @@ jint Arguments::parse_vm_init_args(const JavaVMInitArgs *java_tool_options_args, const JavaVMInitArgs *java_options_args, const JavaVMInitArgs *cmd_line_args) { // For components of the system classpath. - SysClassPath scp(Arguments::get_sysclasspath()); - bool scp_assembly_required = false; + ArgumentBootClassPath bcp(Arguments::get_sysclasspath()); + bool bcp_assembly_required = false; // Save default settings for some mode flags Arguments::_AlwaysCompileLoopMethods = AlwaysCompileLoopMethods; @@ -2572,13 +2638,13 @@ jint Arguments::parse_vm_init_args(const JavaVMInitArgs *java_tool_options_args, // Parse args structure generated from JAVA_TOOL_OPTIONS environment // variable (if present). jint result = parse_each_vm_init_arg( - java_tool_options_args, &scp, &scp_assembly_required, Flag::ENVIRON_VAR); + java_tool_options_args, &bcp, &bcp_assembly_required, Flag::ENVIRON_VAR); if (result != JNI_OK) { return result; } // Parse args structure generated from the command line flags. - result = parse_each_vm_init_arg(cmd_line_args, &scp, &scp_assembly_required, + result = parse_each_vm_init_arg(cmd_line_args, &bcp, &bcp_assembly_required, Flag::COMMAND_LINE); if (result != JNI_OK) { return result; @@ -2587,13 +2653,13 @@ jint Arguments::parse_vm_init_args(const JavaVMInitArgs *java_tool_options_args, // Parse args structure generated from the _JAVA_OPTIONS environment // variable (if present) (mimics classic VM) result = parse_each_vm_init_arg( - java_options_args, &scp, &scp_assembly_required, Flag::ENVIRON_VAR); + java_options_args, &bcp, &bcp_assembly_required, Flag::ENVIRON_VAR); if (result != JNI_OK) { return result; } // Do final processing now that all arguments have been parsed - result = finalize_vm_init_args(&scp, scp_assembly_required); + result = finalize_vm_init_args(&bcp, bcp_assembly_required); if (result != JNI_OK) { return result; } @@ -2647,8 +2713,8 @@ bool valid_jdwp_agent(char *name, bool is_path) { } jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args, - SysClassPath* scp_p, - bool* scp_assembly_required_p, + ArgumentBootClassPath* bcp_p, + bool* bcp_assembly_required_p, Flag::Flags origin) { // For match_option to return remaining or value part of option string const char* tail; @@ -2700,16 +2766,18 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args, JavaAssertions::setSystemClassDefault(enable); // -bootclasspath: } else if (match_option(option, "-Xbootclasspath:", &tail)) { - scp_p->reset_path(tail); - *scp_assembly_required_p = true; + jio_fprintf(defaultStream::output_stream(), + "-Xbootclasspath is no longer a supported option.\n"); + return JNI_EINVAL; // -bootclasspath/a: } else if (match_option(option, "-Xbootclasspath/a:", &tail)) { - scp_p->add_suffix(tail); - *scp_assembly_required_p = true; + bcp_p->add_suffix(tail); + *bcp_assembly_required_p = true; // -bootclasspath/p: } else if (match_option(option, "-Xbootclasspath/p:", &tail)) { - scp_p->add_prefix(tail); - *scp_assembly_required_p = true; + jio_fprintf(defaultStream::output_stream(), + "-Xbootclasspath/p is no longer a supported option.\n"); + return JNI_EINVAL; // -Xrun } else if (match_option(option, "-Xrun", &tail)) { if (tail != NULL) { @@ -2761,9 +2829,14 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args, "Instrumentation agents are not supported in this VM\n"); return JNI_ERR; #else - if(tail != NULL) { + if (tail != NULL) { char *options = strcpy(NEW_C_HEAP_ARRAY(char, strlen(tail) + 1, mtInternal), tail); add_init_agent("instrument", options, false); + // java agents need module java.instrument. Also -addmods ALL-SYSTEM because + // the java agent is in the unmamed module of the application class loader + if (!Arguments::append_to_addmods_property("java.instrument,ALL-SYSTEM")) { + return JNI_ENOMEM; + } } #endif // !INCLUDE_JVMTI // -Xnoclassgc @@ -3029,12 +3102,50 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args, if (FLAG_SET_CMDLINE(bool, ManagementServer, true) != Flag::SUCCESS) { return JNI_EINVAL; } + // management agent in module java.management + if (!Arguments::append_to_addmods_property("java.management")) { + return JNI_ENOMEM; + } #else jio_fprintf(defaultStream::output_stream(), "-Dcom.sun.management is not supported in this VM.\n"); return JNI_ERR; #endif } + if (match_option(option, "-Djdk.launcher.patch.0=", &tail)) { + // -Xpatch + int dir_count; + char** patch_dirs = os::split_path(tail, &dir_count); + if (patch_dirs == NULL) { + jio_fprintf(defaultStream::output_stream(), + "Bad value for -Xpatch.\n"); + return JNI_ERR; + } + set_patch_dirs(patch_dirs); + set_patch_dirs_count(dir_count); + + // Create a path for each patch dir consisting of dir/java.base. + char file_sep = os::file_separator()[0]; + for (int x = 0; x < dir_count; x++) { + // Really shouldn't be NULL, but check can't hurt + if (patch_dirs[x] != NULL) { + size_t len = strlen(patch_dirs[x]); + if (len != 0) { // Ignore empty strings. + len += 11; // file_sep + "java.base" + null terminator. + char* dir = NEW_C_HEAP_ARRAY(char, len, mtInternal); + jio_snprintf(dir, len, "%s%cjava.base", patch_dirs[x], file_sep); + + // See if Xpatch module path exists. + struct stat st; + if ((os::stat(dir, &st) == 0)) { + bcp_p->add_prefix(dir); + *bcp_assembly_required_p = true; + } + FREE_C_HEAP_ARRAY(char, dir); + } + } + } + } // -Xint } else if (match_option(option, "-Xint")) { set_mode_flags(_int); @@ -3294,6 +3405,18 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args, return JNI_OK; } +// Set property jdk.boot.class.path.append to the contents of the bootclasspath +// that follows either the jimage file or exploded module directories. The +// property will contain -Xbootclasspath/a and/or jvmti appended additions. +void Arguments::set_jdkbootclasspath_append() { + char *sysclasspath = get_sysclasspath(); + assert(sysclasspath != NULL, "NULL sysclasspath"); + int bcp_a_idx = bootclassloader_append_index(); + if (bcp_a_idx != -1 && bcp_a_idx < (int)strlen(sysclasspath)) { + _jdk_boot_class_path_append->set_value(sysclasspath + bcp_a_idx); + } +} + // Remove all empty paths from the app classpath (if IgnoreEmptyClassPaths is enabled) // // This is necessary because some apps like to specify classpath like -cp foo.jar:${XYZ}:bar.jar @@ -3329,7 +3452,7 @@ void Arguments::fix_appclasspath() { // Keep replacing ";;" -> ";" until we have no more ";;" (windows) } - _java_class_path->set_value(copy); + _java_class_path->set_writeable_value(copy); FreeHeap(copy); // a copy was made by set_value, so don't need this anymore } } @@ -3380,7 +3503,7 @@ static int check_non_empty_dirs(const char* path) { return nonEmptyDirs; } -jint Arguments::finalize_vm_init_args(SysClassPath* scp_p, bool scp_assembly_required) { +jint Arguments::finalize_vm_init_args(ArgumentBootClassPath* bcp_p, bool bcp_assembly_required) { // check if the default lib/endorsed directory exists; if so, error char path[JVM_MAXPATHLEN]; const char* fileSep = os::file_separator(); @@ -3416,11 +3539,16 @@ jint Arguments::finalize_vm_init_args(SysClassPath* scp_p, bool scp_assembly_req return JNI_ERR; } - if (scp_assembly_required) { + if (bcp_assembly_required) { // Assemble the bootclasspath elements into the final path. - char *combined_path = scp_p->combined_path(); + char *combined_path = bcp_p->combined_path(); Arguments::set_sysclasspath(combined_path); FREE_C_HEAP_ARRAY(char, combined_path); + } else { + // At this point in sysclasspath processing anything + // added would be considered in the boot loader's append path. + // Record this index, including +1 for the file separator character. + Arguments::set_bootclassloader_append_index(((int)strlen(Arguments::get_sysclasspath()))+1); } // This must be done after all arguments have been processed. @@ -3766,6 +3894,11 @@ jint Arguments::parse_options_buffer(const char* name, char* buffer, const size_ void Arguments::set_shared_spaces_flags() { if (DumpSharedSpaces) { + if (Arguments::patch_dirs() != NULL) { + vm_exit_during_initialization( + "Cannot use the following option when dumping the shared archive", "-Xpatch"); + } + if (RequireSharedSpaces) { warning("Cannot dump shared archive while using shared archive"); } @@ -4455,7 +4588,7 @@ void Arguments::PropertyList_unique_add(SystemProperty** plist, const char* k, c if (append) { prop->append_value(v); } else { - prop->set_value(v); + prop->set_writeable_value(v); } return; } diff --git a/hotspot/src/share/vm/runtime/arguments.hpp b/hotspot/src/share/vm/runtime/arguments.hpp index 8fb95492c61..67a9661b8ae 100644 --- a/hotspot/src/share/vm/runtime/arguments.hpp +++ b/hotspot/src/share/vm/runtime/arguments.hpp @@ -43,37 +43,30 @@ extern "C" { } // Forward declarations +class ArgumentBootClassPath; -class SysClassPath; - -// Element describing System and User (-Dkey=value flags) defined property. - -class SystemProperty: public CHeapObj { - private: - char* _key; +// PathString is used as the underlying value container for a +// SystemProperty and for the string that represents the system +// boot class path, Arguments::_system_boot_class_path. +class PathString : public CHeapObj { + protected: char* _value; - SystemProperty* _next; - bool _writeable; - bool writeable() { return _writeable; } - public: - // Accessors - const char* key() const { return _key; } char* value() const { return _value; } - SystemProperty* next() const { return _next; } - void set_next(SystemProperty* next) { _next = next; } + bool set_value(const char *value) { - if (writeable()) { - if (_value != NULL) { - FreeHeap(_value); - } - _value = AllocateHeap(strlen(value)+1, mtInternal); - if (_value != NULL) { - strcpy(_value, value); - } - return true; + if (_value != NULL) { + FreeHeap(_value); } - return false; + _value = AllocateHeap(strlen(value)+1, mtInternal); + assert(_value != NULL, "Unable to allocate space for new path value"); + if (_value != NULL) { + strcpy(_value, value); + } else { + // not able to allocate + return false; + } + return true; } void append_value(const char *value) { @@ -85,6 +78,7 @@ class SystemProperty: public CHeapObj { len += strlen(_value); } sp = AllocateHeap(len+2, mtInternal); + assert(sp != NULL, "Unable to allocate space for new append path value"); if (sp != NULL) { if (_value != NULL) { strcpy(sp, _value); @@ -100,20 +94,61 @@ class SystemProperty: public CHeapObj { } // Constructor - SystemProperty(const char* key, const char* value, bool writeable) { - if (key == NULL) { - _key = NULL; - } else { - _key = AllocateHeap(strlen(key)+1, mtInternal); - strcpy(_key, key); - } + PathString(const char* value) { if (value == NULL) { _value = NULL; } else { _value = AllocateHeap(strlen(value)+1, mtInternal); strcpy(_value, value); } + } +}; + +// Element describing System and User (-Dkey=value flags) defined property. +// +// An internal SystemProperty is one that has been removed in +// jdk.internal.VM.saveAndRemoveProperties, like jdk.boot.class.path.append. +// +class SystemProperty : public PathString { + private: + char* _key; + SystemProperty* _next; + bool _internal; + bool _writeable; + bool writeable() { return _writeable; } + + public: + // Accessors + char* value() const { return PathString::value(); } + const char* key() const { return _key; } + bool internal() const { return _internal; } + SystemProperty* next() const { return _next; } + void set_next(SystemProperty* next) { _next = next; } + + // A system property should only have its value set + // via an external interface if it is a writeable property. + // The internal, non-writeable property jdk.boot.class.path.append + // is the only exception to this rule. It can be set externally + // via -Xbootclasspath/a or JVMTI OnLoad phase call to AddToBootstrapClassLoaderSearch. + // In those cases for jdk.boot.class.path.append, the base class + // set_value and append_value methods are called directly. + bool set_writeable_value(const char *value) { + if (writeable()) { + return set_value(value); + } + return false; + } + + // Constructor + SystemProperty(const char* key, const char* value, bool writeable, bool internal = false) : PathString(value) { + if (key == NULL) { + _key = NULL; + } else { + _key = AllocateHeap(strlen(key)+1, mtInternal); + strcpy(_key, key); + } _next = NULL; + _internal = internal; _writeable = writeable; } }; @@ -273,7 +308,13 @@ class Arguments : AllStatic { static SystemProperty *_java_library_path; static SystemProperty *_java_home; static SystemProperty *_java_class_path; - static SystemProperty *_sun_boot_class_path; + static SystemProperty *_jdk_boot_class_path_append; + + // The constructed value of the system class path after + // argument processing and JVMTI OnLoad additions via + // calls to AddToBootstrapClassLoaderSearch. This is the + // final form before ClassLoader::setup_bootstrap_search(). + static PathString *_system_boot_class_path; // temporary: to emit warning if the default ext dirs are not empty. // remove this variable when the warning is no longer needed. @@ -298,7 +339,7 @@ class Arguments : AllStatic { // Value of the conservative maximum heap alignment needed static size_t _conservative_max_heap_alignment; - static uintx _min_heap_size; + static uintx _min_heap_size; // -Xrun arguments static AgentLibraryList _libraryList; @@ -323,6 +364,17 @@ class Arguments : AllStatic { static void set_java_compiler(bool arg) { _java_compiler = arg; } static bool java_compiler() { return _java_compiler; } + // Capture the index location of -Xbootclasspath\a within sysclasspath. + // Used when setting up the bootstrap search path in order to + // mark the boot loader's append path observability boundary. + static int _bootclassloader_append_index; + + // -Xpatch flag + static char** _patch_dirs; + static int _patch_dirs_count; + static void set_patch_dirs(char** dirs) { _patch_dirs = dirs; } + static void set_patch_dirs_count(int count) { _patch_dirs_count = count; } + // -Xdebug flag static bool _xdebug_mode; static void set_xdebug_mode(bool arg) { _xdebug_mode = arg; } @@ -373,6 +425,9 @@ class Arguments : AllStatic { // System properties static bool add_property(const char* prop); + // Miscellaneous system property setter + static bool append_to_addmods_property(const char* module_name); + // Aggressive optimization flags. static jint set_aggressive_opts_flags(); @@ -406,8 +461,8 @@ class Arguments : AllStatic { static jint parse_vm_init_args(const JavaVMInitArgs *java_tool_options_args, const JavaVMInitArgs *java_options_args, const JavaVMInitArgs *cmd_line_args); - static jint parse_each_vm_init_arg(const JavaVMInitArgs* args, SysClassPath* scp_p, bool* scp_assembly_required_p, Flag::Flags origin); - static jint finalize_vm_init_args(SysClassPath* scp_p, bool scp_assembly_required); + static jint parse_each_vm_init_arg(const JavaVMInitArgs* args, ArgumentBootClassPath* bcp_p, bool* bcp_assembly_required_p, Flag::Flags origin); + static jint finalize_vm_init_args(ArgumentBootClassPath* bcp_p, bool bcp_assembly_required); static bool is_bad_option(const JavaVMOption* option, jboolean ignore, const char* option_type); static bool is_bad_option(const JavaVMOption* option, jboolean ignore) { @@ -569,6 +624,18 @@ class Arguments : AllStatic { static size_t min_heap_size() { return _min_heap_size; } static void set_min_heap_size(size_t v) { _min_heap_size = v; } + // -Xbootclasspath/a + static int bootclassloader_append_index() { + return _bootclassloader_append_index; + } + static void set_bootclassloader_append_index(int value) { + _bootclassloader_append_index = value; + } + + // -Xpatch + static char** patch_dirs() { return _patch_dirs; } + static int patch_dirs_count() { return _patch_dirs_count; } + // -Xrun static AgentLibrary* libraries() { return _libraryList.first(); } static bool init_libraries_at_startup() { return !_libraryList.is_empty(); } @@ -625,24 +692,35 @@ class Arguments : AllStatic { static void set_java_home(const char *value) { _java_home->set_value(value); } static void set_library_path(const char *value) { _java_library_path->set_value(value); } static void set_ext_dirs(char *value) { _ext_dirs = os::strdup_check_oom(value); } - static void set_sysclasspath(const char *value) { _sun_boot_class_path->set_value(value); } - static void append_sysclasspath(const char *value) { _sun_boot_class_path->append_value(value); } + + // Set up of the underlying system boot class path + static void set_jdkbootclasspath_append(); + static void set_sysclasspath(const char *value) { + _system_boot_class_path->set_value(value); + set_jdkbootclasspath_append(); + } + static void append_sysclasspath(const char *value) { + _system_boot_class_path->append_value(value); + set_jdkbootclasspath_append(); + } static char* get_java_home() { return _java_home->value(); } static char* get_dll_dir() { return _sun_boot_library_path->value(); } - static char* get_sysclasspath() { return _sun_boot_class_path->value(); } + static char* get_sysclasspath() { return _system_boot_class_path->value(); } static char* get_ext_dirs() { return _ext_dirs; } static char* get_appclasspath() { return _java_class_path->value(); } static void fix_appclasspath(); // Operation modi - static Mode mode() { return _mode; } + static Mode mode() { return _mode; } static bool is_interpreter_only() { return mode() == _int; } // Utility: copies src into buf, replacing "%%" with "%" and "%p" with pid. static bool copy_expand_pid(const char* src, size_t srclen, char* buf, size_t buflen); + + static void check_unsupported_dumping_properties() NOT_CDS_RETURN; }; bool Arguments::gc_selected() { diff --git a/hotspot/src/share/vm/runtime/frame.cpp b/hotspot/src/share/vm/runtime/frame.cpp index 1a72cec46a8..b2a321687b7 100644 --- a/hotspot/src/share/vm/runtime/frame.cpp +++ b/hotspot/src/share/vm/runtime/frame.cpp @@ -642,6 +642,13 @@ void frame::print_on_error(outputStream* st, char* buf, int buflen, bool verbose m->name_and_sig_as_C_string(buf, buflen); st->print("j %s", buf); st->print("+%d", this->interpreter_frame_bci()); + ModuleEntry* module = m->method_holder()->module(); + if (module->is_named()) { + module->name()->as_C_string(buf, buflen); + st->print(" %s", buf); + module->version()->as_C_string(buf, buflen); + st->print("@%s", buf); + } } else { st->print("j " PTR_FORMAT, p2i(pc())); } @@ -662,6 +669,13 @@ void frame::print_on_error(outputStream* st, char* buf, int buflen, bool verbose st->print("J %d%s %s ", nm->compile_id(), (nm->is_osr_method() ? "%" : ""), ((nm->compiler() != NULL) ? nm->compiler()->name() : "")); + ModuleEntry* module = m->method_holder()->module(); + if (module->is_named()) { + module->name()->as_C_string(buf, buflen); + st->print(" %s", buf); + module->version()->as_C_string(buf, buflen); + st->print("@%s", buf); + } st->print("%s (%d bytes) @ " PTR_FORMAT " [" PTR_FORMAT "+" INTPTR_FORMAT "]", buf, m->code_size(), p2i(_pc), p2i(_cb->code_begin()), _pc - _cb->code_begin()); #if INCLUDE_JVMCI diff --git a/hotspot/src/share/vm/runtime/jniHandles.cpp b/hotspot/src/share/vm/runtime/jniHandles.cpp index 8e4636bf640..3d44d322a4c 100644 --- a/hotspot/src/share/vm/runtime/jniHandles.cpp +++ b/hotspot/src/share/vm/runtime/jniHandles.cpp @@ -467,6 +467,14 @@ jobject JNIHandleBlock::allocate_handle(oop obj) { return allocate_handle(obj); // retry } +void JNIHandleBlock::release_handle(jobject h) { + if (h != NULL) { + assert(chain_contains(h), "does not contain the JNI handle"); + // Mark the handle as deleted, allocate will reuse it + *((oop*)h) = JNIHandles::deleted_handle(); + } +} + void JNIHandleBlock::rebuild_free_list() { assert(_allocate_before_rebuild == 0 && _free_list == NULL, "just checking"); diff --git a/hotspot/src/share/vm/runtime/jniHandles.hpp b/hotspot/src/share/vm/runtime/jniHandles.hpp index 069a1f35cfa..7a39655bfae 100644 --- a/hotspot/src/share/vm/runtime/jniHandles.hpp +++ b/hotspot/src/share/vm/runtime/jniHandles.hpp @@ -138,6 +138,9 @@ class JNIHandleBlock : public CHeapObj { // Handle allocation jobject allocate_handle(oop obj); + // Release Handle + void release_handle(jobject); + // Block allocation and block free list management static JNIHandleBlock* allocate_block(Thread* thread = NULL); static void release_block(JNIHandleBlock* block, Thread* thread = NULL); diff --git a/hotspot/src/share/vm/runtime/mutexLocker.cpp b/hotspot/src/share/vm/runtime/mutexLocker.cpp index fa1aea26a55..7066ae21c1d 100644 --- a/hotspot/src/share/vm/runtime/mutexLocker.cpp +++ b/hotspot/src/share/vm/runtime/mutexLocker.cpp @@ -39,7 +39,7 @@ Mutex* Patching_lock = NULL; Monitor* SystemDictionary_lock = NULL; -Mutex* PackageTable_lock = NULL; +Mutex* Module_lock = NULL; Mutex* CompiledIC_lock = NULL; Mutex* InlineCacheBuffer_lock = NULL; Mutex* VMStatistic_lock = NULL; @@ -206,7 +206,7 @@ void mutex_init() { def(JmethodIdCreation_lock , Mutex , leaf, true, Monitor::_safepoint_check_always); // used for creating jmethodIDs. def(SystemDictionary_lock , Monitor, leaf, true, Monitor::_safepoint_check_always); // lookups done by VM thread - def(PackageTable_lock , Mutex , leaf, false, Monitor::_safepoint_check_always); + def(Module_lock , Mutex , leaf+2, true, Monitor::_safepoint_check_always); def(InlineCacheBuffer_lock , Mutex , leaf, true, Monitor::_safepoint_check_always); def(VMStatistic_lock , Mutex , leaf, false, Monitor::_safepoint_check_always); def(ExpandHeap_lock , Mutex , leaf, true, Monitor::_safepoint_check_always); // Used during compilation by VM thread diff --git a/hotspot/src/share/vm/runtime/mutexLocker.hpp b/hotspot/src/share/vm/runtime/mutexLocker.hpp index eccd3ca7314..f60329725cd 100644 --- a/hotspot/src/share/vm/runtime/mutexLocker.hpp +++ b/hotspot/src/share/vm/runtime/mutexLocker.hpp @@ -32,7 +32,7 @@ extern Mutex* Patching_lock; // a lock used to guard code patching of compiled code extern Monitor* SystemDictionary_lock; // a lock on the system dictionary -extern Mutex* PackageTable_lock; // a lock on the class loader package table +extern Mutex* Module_lock; // a lock on module and package related data structures extern Mutex* CompiledIC_lock; // a lock used to guard compiled IC patching and access extern Mutex* InlineCacheBuffer_lock; // a lock used to guard the InlineCacheBuffer extern Mutex* VMStatistic_lock; // a lock used to guard statistics count increment diff --git a/hotspot/src/share/vm/runtime/os.cpp b/hotspot/src/share/vm/runtime/os.cpp index c482d2135dd..016b6e3d84f 100644 --- a/hotspot/src/share/vm/runtime/os.cpp +++ b/hotspot/src/share/vm/runtime/os.cpp @@ -1193,67 +1193,14 @@ char* os::format_boot_path(const char* format_string, return formatted_path; } -// returns a PATH of all entries in the given directory that do not start with a '.' -static char* expand_entries_to_path(char* directory, char fileSep, char pathSep) { - DIR* dir = os::opendir(directory); - if (dir == NULL) return NULL; - - char* path = NULL; - size_t path_len = 0; // path length including \0 terminator - - size_t directory_len = strlen(directory); - struct dirent *entry; - char* dbuf = NEW_C_HEAP_ARRAY(char, os::readdir_buf_size(directory), mtInternal); - while ((entry = os::readdir(dir, (dirent *) dbuf)) != NULL) { - const char* name = entry->d_name; - if (name[0] == '.') continue; - - size_t name_len = strlen(name); - size_t needed = directory_len + name_len + 2; - size_t new_len = path_len + needed; - if (path == NULL) { - path = NEW_C_HEAP_ARRAY(char, new_len, mtInternal); - } else { - path = REALLOC_C_HEAP_ARRAY(char, path, new_len, mtInternal); - } - if (path == NULL) - break; - - // append directoryname - char* p = path; - if (path_len > 0) { - p += (path_len -1); - *p = pathSep; - p++; - } - - strcpy(p, directory); - p += directory_len; - - *p = fileSep; - p++; - - strcpy(p, name); - p += name_len; - - path_len = new_len; - } - - FREE_C_HEAP_ARRAY(char, dbuf); - os::closedir(dir); - - return path; -} - bool os::set_boot_path(char fileSep, char pathSep) { const char* home = Arguments::get_java_home(); int home_len = (int)strlen(home); - char* sysclasspath = NULL; struct stat st; - // modular image if bootmodules.jimage exists - char* jimage = format_boot_path("%/lib/modules/" BOOT_IMAGE_NAME, home, home_len, fileSep, pathSep); + // modular image if "modules" jimage exists + char* jimage = format_boot_path("%/lib/" MODULES_IMAGE_NAME, home, home_len, fileSep, pathSep); if (jimage == NULL) return false; bool has_jimage = (os::stat(jimage, &st) == 0); if (has_jimage) { @@ -1264,23 +1211,16 @@ bool os::set_boot_path(char fileSep, char pathSep) { FREE_C_HEAP_ARRAY(char, jimage); // check if developer build with exploded modules - char* modules_dir = format_boot_path("%/modules", home, home_len, fileSep, pathSep); - if (os::stat(modules_dir, &st) == 0) { - if ((st.st_mode & S_IFDIR) == S_IFDIR) { - sysclasspath = expand_entries_to_path(modules_dir, fileSep, pathSep); - } + char* base_classes = format_boot_path("%/modules/java.base", home, home_len, fileSep, pathSep); + if (base_classes == NULL) return false; + if (os::stat(base_classes, &st) == 0) { + Arguments::set_sysclasspath(base_classes); + FREE_C_HEAP_ARRAY(char, base_classes); + return true; } - FREE_C_HEAP_ARRAY(char, modules_dir); + FREE_C_HEAP_ARRAY(char, base_classes); - // fallback to classes - if (sysclasspath == NULL) - sysclasspath = format_boot_path("%/classes", home, home_len, fileSep, pathSep); - - if (sysclasspath == NULL) return false; - Arguments::set_sysclasspath(sysclasspath); - FREE_C_HEAP_ARRAY(char, sysclasspath); - - return true; + return false; } /* diff --git a/hotspot/src/share/vm/runtime/os.hpp b/hotspot/src/share/vm/runtime/os.hpp index 128c73c3194..7b71ca5228d 100644 --- a/hotspot/src/share/vm/runtime/os.hpp +++ b/hotspot/src/share/vm/runtime/os.hpp @@ -778,6 +778,8 @@ class os: AllStatic { // Amount beyond the callee frame size that we bang the stack. static int extra_bang_size_in_bytes(); + static char** split_path(const char* path, int* n); + // Extensions #include "runtime/os_ext.hpp" @@ -993,7 +995,6 @@ class os: AllStatic { char fileSep, char pathSep); static bool set_boot_path(char fileSep, char pathSep); - static char** split_path(const char* path, int* n); }; diff --git a/hotspot/src/share/vm/runtime/reflection.cpp b/hotspot/src/share/vm/runtime/reflection.cpp index 7eda3c51393..b60c63a1ad8 100644 --- a/hotspot/src/share/vm/runtime/reflection.cpp +++ b/hotspot/src/share/vm/runtime/reflection.cpp @@ -24,6 +24,8 @@ #include "precompiled.hpp" #include "classfile/javaClasses.hpp" +#include "classfile/moduleEntry.hpp" +#include "classfile/packageEntry.hpp" #include "classfile/stringTable.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/verifier.hpp" @@ -452,25 +454,193 @@ static bool can_relax_access_check_for(const Klass* accessor, return false; } -bool Reflection::verify_class_access(const Klass* current_class, - const Klass* new_class, - bool classloader_only) { +/* + Type Accessibility check for public types: Callee Type T is accessible to Caller Type S if: + + Callee T in Callee T in package PT, + unnamed module runtime module MT + ------------------------------------------------------------------------------------------------ + + Caller S in package If MS is loose: YES If same classloader/package (PS == PT): YES + PS, runtime module MS If MS can read T's If same runtime module: (MS == MT): YES + unnamed module: YES + Else if (MS can read MT (Establish readability) && + MT exports PT to MS or to all modules): YES + + ------------------------------------------------------------------------------------------------ + Caller S in unnamed YES Readability exists because unnamed module + module UM "reads" all modules + if (MT exports PT to UM or to all modules): YES + + ------------------------------------------------------------------------------------------------ + + Note: a loose module is a module that can read all current and future unnamed modules. +*/ +Reflection::VerifyClassAccessResults Reflection::verify_class_access( + const Klass* current_class, const Klass* new_class, bool classloader_only) { + // Verify that current_class can access new_class. If the classloader_only // flag is set, we automatically allow any accesses in which current_class // doesn't have a classloader. if ((current_class == NULL) || (current_class == new_class) || - (new_class->is_public()) || is_same_class_package(current_class, new_class)) { - return true; + return ACCESS_OK; } // Allow all accesses from sun/reflect/MagicAccessorImpl subclasses to // succeed trivially. if (current_class->is_subclass_of(SystemDictionary::reflect_MagicAccessorImpl_klass())) { - return true; + return ACCESS_OK; } - return can_relax_access_check_for(current_class, new_class, classloader_only); + // module boundaries + if (new_class->is_public()) { + // Ignore modules for DumpSharedSpaces because we do not have any package + // or module information for modules other than java.base. + if (DumpSharedSpaces) { + return ACCESS_OK; + } + + // Find the module entry for current_class, the accessor + ModuleEntry* module_from = InstanceKlass::cast(current_class)->module(); + // Find the module entry for new_class, the accessee + if (new_class->is_objArray_klass()) { + new_class = ObjArrayKlass::cast(new_class)->bottom_klass(); + } + if (!new_class->is_instance_klass()) { + // Everyone can read a typearray. + assert (new_class->is_typeArray_klass(), "Unexpected klass type"); + return ACCESS_OK; + } + ModuleEntry* module_to = InstanceKlass::cast(new_class)->module(); + + // both in same (possibly unnamed) module + if (module_from == module_to) { + return ACCESS_OK; + } + + // Acceptable access to a type in an unamed module. Note that since + // unnamed modules can read all unnamed modules, this also handles the + // case where module_from is also unnamed but in a different class loader. + if (!module_to->is_named() && + (module_from->can_read_all_unnamed() || module_from->can_read(module_to))) { + return ACCESS_OK; + } + + // Establish readability, check if module_from is allowed to read module_to. + if (!module_from->can_read(module_to)) { + return MODULE_NOT_READABLE; + } + + PackageEntry* package_to = InstanceKlass::cast(new_class)->package(); + assert(package_to != NULL, "can not obtain new_class' package"); + + // Once readability is established, if module_to exports T unqualifiedly, + // (to all modules), than whether module_from is in the unnamed module + // or not does not matter, access is allowed. + if (package_to->is_unqual_exported()) { + return ACCESS_OK; + } + + // Access is allowed if both 1 & 2 hold: + // 1. Readability, module_from can read module_to (established above). + // 2. Either module_to exports T to module_from qualifiedly. + // or + // module_to exports T to all unnamed modules and module_from is unnamed. + // or + // module_to exports T unqualifiedly to all modules (checked above). + if (!package_to->is_qexported_to(module_from)) { + return TYPE_NOT_EXPORTED; + } + return ACCESS_OK; + } + + if (can_relax_access_check_for(current_class, new_class, classloader_only)) { + return ACCESS_OK; + } + return OTHER_PROBLEM; +} + +// Return an error message specific to the specified Klass*'s and result. +// This function must be called from within a block containing a ResourceMark. +char* Reflection::verify_class_access_msg(const Klass* current_class, + const Klass* new_class, + VerifyClassAccessResults result) { + assert(result != ACCESS_OK, "must be failure result"); + char * msg = NULL; + if (result != OTHER_PROBLEM && new_class != NULL && current_class != NULL) { + // Find the module entry for current_class, the accessor + ModuleEntry* module_from = InstanceKlass::cast(current_class)->module(); + const char * module_from_name = module_from->is_named() ? module_from->name()->as_C_string() : UNNAMED_MODULE; + const char * current_class_name = current_class->external_name(); + + // Find the module entry for new_class, the accessee + ModuleEntry* module_to = NULL; + if (new_class->is_objArray_klass()) { + new_class = ObjArrayKlass::cast(new_class)->bottom_klass(); + } + if (new_class->is_instance_klass()) { + module_to = InstanceKlass::cast(new_class)->module(); + } else { + module_to = ModuleEntryTable::javabase_module(); + } + const char * module_to_name = module_to->is_named() ? module_to->name()->as_C_string() : UNNAMED_MODULE; + const char * new_class_name = new_class->external_name(); + + if (result == MODULE_NOT_READABLE) { + assert(module_from->is_named(), "Unnamed modules can read all modules"); + if (module_to->is_named()) { + size_t len = 100 + strlen(current_class_name) + 2*strlen(module_from_name) + + strlen(new_class_name) + 2*strlen(module_to_name); + msg = NEW_RESOURCE_ARRAY(char, len); + jio_snprintf(msg, len - 1, + "class %s (in module %s) cannot access class %s (in module %s) because module %s does not read module %s", + current_class_name, module_from_name, new_class_name, + module_to_name, module_from_name, module_to_name); + } else { + jobject jlrm = module_to->module(); + assert(jlrm != NULL, "Null jlrm in module_to ModuleEntry"); + intptr_t identity_hash = JNIHandles::resolve(jlrm)->identity_hash(); + size_t len = 160 + strlen(current_class_name) + 2*strlen(module_from_name) + + strlen(new_class_name) + 2*sizeof(uintx); + msg = NEW_RESOURCE_ARRAY(char, len); + jio_snprintf(msg, len - 1, + "class %s (in module %s) cannot access class %s (in unnamed module @" SIZE_FORMAT_HEX ") because module %s does not read unnamed module @" SIZE_FORMAT_HEX, + current_class_name, module_from_name, new_class_name, uintx(identity_hash), + module_from_name, uintx(identity_hash)); + } + + } else if (result == TYPE_NOT_EXPORTED) { + assert(InstanceKlass::cast(new_class)->package() != NULL, + "Unnamed packages are always exported"); + const char * package_name = + InstanceKlass::cast(new_class)->package()->name()->as_klass_external_name(); + assert(module_to->is_named(), "Unnamed modules export all packages"); + if (module_from->is_named()) { + size_t len = 118 + strlen(current_class_name) + 2*strlen(module_from_name) + + strlen(new_class_name) + 2*strlen(module_to_name) + strlen(package_name); + msg = NEW_RESOURCE_ARRAY(char, len); + jio_snprintf(msg, len - 1, + "class %s (in module %s) cannot access class %s (in module %s) because module %s does not export %s to module %s", + current_class_name, module_from_name, new_class_name, + module_to_name, module_to_name, package_name, module_from_name); + } else { + jobject jlrm = module_from->module(); + assert(jlrm != NULL, "Null jlrm in module_from ModuleEntry"); + intptr_t identity_hash = JNIHandles::resolve(jlrm)->identity_hash(); + size_t len = 170 + strlen(current_class_name) + strlen(new_class_name) + + 2*strlen(module_to_name) + strlen(package_name) + 2*sizeof(uintx); + msg = NEW_RESOURCE_ARRAY(char, len); + jio_snprintf(msg, len - 1, + "class %s (in unnamed module @" SIZE_FORMAT_HEX ") cannot access class %s (in module %s) because module %s does not export %s to unnamed module @" SIZE_FORMAT_HEX, + current_class_name, uintx(identity_hash), new_class_name, module_to_name, + module_to_name, package_name, uintx(identity_hash)); + } + } else { + ShouldNotReachHere(); + } + } // result != OTHER_PROBLEM... + return msg; } bool Reflection::verify_field_access(const Klass* current_class, diff --git a/hotspot/src/share/vm/runtime/reflection.hpp b/hotspot/src/share/vm/runtime/reflection.hpp index 920cf4c22a3..736e21f9f6b 100644 --- a/hotspot/src/share/vm/runtime/reflection.hpp +++ b/hotspot/src/share/vm/runtime/reflection.hpp @@ -53,6 +53,14 @@ class Reflection: public AllStatic { MAX_DIM = 255 }; + // Results returned by verify_class_access() + enum VerifyClassAccessResults { + ACCESS_OK = 0, + MODULE_NOT_READABLE = 1, + TYPE_NOT_EXPORTED = 2, + OTHER_PROBLEM = 3 + }; + // Boxing. Returns boxed value of appropriate type. Throws IllegalArgumentException. static oop box(jvalue* v, BasicType type, TRAPS); // Unboxing. Returns type code and sets value. @@ -73,9 +81,14 @@ class Reflection: public AllStatic { static arrayOop reflect_new_multi_array(oop element_mirror, typeArrayOop dimensions, TRAPS); // Verification - static bool verify_class_access(const Klass* current_class, - const Klass* new_class, - bool classloader_only); + static VerifyClassAccessResults verify_class_access(const Klass* current_class, + const Klass* new_class, + bool classloader_only); + // Return an error message specific to the specified Klass*'s and result. + // This function must be called from within a block containing a ResourceMark. + static char* verify_class_access_msg(const Klass* current_class, + const Klass* new_class, + const VerifyClassAccessResults result); static bool verify_field_access(const Klass* current_class, const Klass* resolved_class, diff --git a/hotspot/src/share/vm/runtime/sharedRuntime.cpp b/hotspot/src/share/vm/runtime/sharedRuntime.cpp index 4adec0c45ca..fe697685022 100644 --- a/hotspot/src/share/vm/runtime/sharedRuntime.cpp +++ b/hotspot/src/share/vm/runtime/sharedRuntime.cpp @@ -39,6 +39,8 @@ #include "interpreter/interpreterRuntime.hpp" #include "logging/log.hpp" #include "memory/universe.inline.hpp" +#include "oops/klass.hpp" +#include "oops/objArrayKlass.hpp" #include "oops/oop.inline.hpp" #include "prims/forte.hpp" #include "prims/jvmtiExport.hpp" @@ -1915,28 +1917,58 @@ JRT_ENTRY(void, SharedRuntime::slow_arraycopy_C(oopDesc* src, jint src_pos, } JRT_END +// The caller of generate_class_cast_message() (or one of its callers) +// must use a ResourceMark in order to correctly free the result. char* SharedRuntime::generate_class_cast_message( - JavaThread* thread, const char* objName) { + JavaThread* thread, Klass* caster_klass) { // Get target class name from the checkcast instruction vframeStream vfst(thread, true); assert(!vfst.at_end(), "Java frame must exist"); Bytecode_checkcast cc(vfst.method(), vfst.method()->bcp_from(vfst.bci())); - Klass* targetKlass = vfst.method()->constants()->klass_at( + Klass* target_klass = vfst.method()->constants()->klass_at( cc.index(), thread); - return generate_class_cast_message(objName, targetKlass->external_name()); + return generate_class_cast_message(caster_klass, target_klass); } char* SharedRuntime::generate_class_cast_message( - const char* objName, const char* targetKlassName, const char* desc) { - size_t msglen = strlen(objName) + strlen(desc) + strlen(targetKlassName) + 1; + Klass* caster_klass, Klass* target_klass) { + + const char* caster_klass_name = caster_klass->external_name(); + Klass* c_klass = caster_klass->is_objArray_klass() ? + ObjArrayKlass::cast(caster_klass)->bottom_klass() : caster_klass; + ModuleEntry* caster_module; + const char* caster_module_name; + if (c_klass->is_instance_klass()) { + caster_module = InstanceKlass::cast(c_klass)->module(); + caster_module_name = caster_module->is_named() ? + caster_module->name()->as_C_string() : UNNAMED_MODULE; + } else { + caster_module_name = "java.base"; + } + const char* target_klass_name = target_klass->external_name(); + Klass* t_klass = target_klass->is_objArray_klass() ? + ObjArrayKlass::cast(target_klass)->bottom_klass() : target_klass; + ModuleEntry* target_module; + const char* target_module_name; + if (t_klass->is_instance_klass()) { + target_module = InstanceKlass::cast(t_klass)->module(); + target_module_name = target_module->is_named() ? + target_module->name()->as_C_string(): UNNAMED_MODULE; + } else { + target_module_name = "java.base"; + } + + size_t msglen = strlen(caster_klass_name) + strlen(caster_module_name) + + strlen(target_klass_name) + strlen(target_module_name) + 50; char* message = NEW_RESOURCE_ARRAY(char, msglen); if (NULL == message) { // Shouldn't happen, but don't cause even more problems if it does - message = const_cast(objName); + message = const_cast(caster_klass_name); } else { - jio_snprintf(message, msglen, "%s%s%s", objName, desc, targetKlassName); + jio_snprintf(message, msglen, "%s (in module: %s) cannot be cast to %s (in module: %s)", + caster_klass_name, caster_module_name, target_klass_name, target_module_name); } return message; } diff --git a/hotspot/src/share/vm/runtime/sharedRuntime.hpp b/hotspot/src/share/vm/runtime/sharedRuntime.hpp index f00b911d048..10c94e6c366 100644 --- a/hotspot/src/share/vm/runtime/sharedRuntime.hpp +++ b/hotspot/src/share/vm/runtime/sharedRuntime.hpp @@ -293,22 +293,21 @@ class SharedRuntime: AllStatic { // Fill in the "X cannot be cast to a Y" message for ClassCastException // // @param thr the current thread - // @param name the name of the class of the object attempted to be cast + // @param caster_klass the class of the object we are casting // @return the dynamically allocated exception message (must be freed // by the caller using a resource mark) // // BCP must refer to the current 'checkcast' opcode for the frame // on top of the stack. - // The caller (or one of it's callers) must use a ResourceMark + // The caller (or one of its callers) must use a ResourceMark // in order to correctly free the result. // - static char* generate_class_cast_message(JavaThread* thr, const char* name); + static char* generate_class_cast_message(JavaThread* thr, Klass* caster_klass); // Fill in the "X cannot be cast to a Y" message for ClassCastException // - // @param name the name of the class of the object attempted to be cast - // @param klass the name of the target klass attempt - // @param gripe the specific kind of problem being reported + // @param caster_klass the class of the object we are casting + // @param target_klass the target klass attempt // @return the dynamically allocated exception message (must be freed // by the caller using a resource mark) // @@ -317,8 +316,7 @@ class SharedRuntime: AllStatic { // The caller (or one of it's callers) must use a ResourceMark // in order to correctly free the result. // - static char* generate_class_cast_message(const char* name, const char* klass, - const char* gripe = " cannot be cast to "); + static char* generate_class_cast_message(Klass* caster_klass, Klass* target_klass); // Resolves a call site- may patch in the destination of the call into the // compiled code. diff --git a/hotspot/src/share/vm/runtime/statSampler.cpp b/hotspot/src/share/vm/runtime/statSampler.cpp index 3e693d997da..cc88f88bead 100644 --- a/hotspot/src/share/vm/runtime/statSampler.cpp +++ b/hotspot/src/share/vm/runtime/statSampler.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 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 @@ -238,7 +238,6 @@ static const char* property_counters_us[] = { // unstable interface, unsupported counters static const char* property_counters_uu[] = { - "sun.boot.class.path", "sun.boot.library.path", NULL }; diff --git a/hotspot/src/share/vm/runtime/thread.cpp b/hotspot/src/share/vm/runtime/thread.cpp index 9d6531de9ef..a7b9f41f9b1 100644 --- a/hotspot/src/share/vm/runtime/thread.cpp +++ b/hotspot/src/share/vm/runtime/thread.cpp @@ -25,6 +25,7 @@ #include "precompiled.hpp" #include "classfile/classLoader.hpp" #include "classfile/javaClasses.hpp" +#include "classfile/moduleEntry.hpp" #include "classfile/systemDictionary.hpp" #include "classfile/vmSymbols.hpp" #include "code/codeCache.hpp" @@ -118,6 +119,9 @@ #include "runtime/rtmLocking.hpp" #endif +// Initialization after module runtime initialization +void universe_post_module_init(); // must happen after call_initPhase2 + #ifdef DTRACE_ENABLED // Only bother with this argument setup if dtrace is available @@ -997,15 +1001,6 @@ static oop create_initial_thread(Handle thread_group, JavaThread* thread, return thread_oop(); } -static void call_initializeSystemClass(TRAPS) { - Klass* k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_System(), true, CHECK); - instanceKlassHandle klass (THREAD, k); - - JavaValue result(T_VOID); - JavaCalls::call_static(&result, klass, vmSymbols::initializeSystemClass_name(), - vmSymbols::void_method_signature(), CHECK); -} - char java_runtime_name[128] = ""; char java_runtime_version[128] = ""; @@ -3364,6 +3359,62 @@ void Threads::threads_do(ThreadClosure* tc) { // If CompilerThreads ever become non-JavaThreads, add them here } +// The system initialization in the library has three phases. +// +// Phase 1: java.lang.System class initialization +// java.lang.System is a primordial class loaded and initialized +// by the VM early during startup. java.lang.System. +// only does registerNatives and keeps the rest of the class +// initialization work later until thread initialization completes. +// +// System.initPhase1 initializes the system properties, the static +// fields in, out, and err. Set up java signal handlers, OS-specific +// system settings, and thread group of the main thread. +static void call_initPhase1(TRAPS) { + Klass* k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_System(), true, CHECK); + instanceKlassHandle klass (THREAD, k); + + JavaValue result(T_VOID); + JavaCalls::call_static(&result, klass, vmSymbols::initPhase1_name(), + vmSymbols::void_method_signature(), CHECK); +} + +// Phase 2. Module system initialization +// This will initialize the module system. Only java.base classes +// can be loaded until phase 2 completes. +// +// Call System.initPhase2 after the compiler initialization and jsr292 +// classes get initialized because module initialization runs a lot of java +// code, that for performance reasons, should be compiled. Also, this will +// enable the startup code to use lambda and other language features in this +// phase and onward. +// +// After phase 2, The VM will begin search classes from -Xbootclasspath/a. +static void call_initPhase2(TRAPS) { + Klass* k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_System(), true, CHECK); + instanceKlassHandle klass (THREAD, k); + + JavaValue result(T_VOID); + JavaCalls::call_static(&result, klass, vmSymbols::initPhase2_name(), + vmSymbols::void_method_signature(), CHECK); + universe_post_module_init(); +} + +// Phase 3. final setup - set security manager, system class loader and TCCL +// +// This will instantiate and set the security manager, set the system class +// loader as well as the thread context class loader. The security manager +// and system class loader may be a custom class loaded from -Xbootclasspath/a, +// other modules or the application's classpath. +static void call_initPhase3(TRAPS) { + Klass* k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_System(), true, CHECK); + instanceKlassHandle klass (THREAD, k); + + JavaValue result(T_VOID); + JavaCalls::call_static(&result, klass, vmSymbols::initPhase3_name(), + vmSymbols::void_method_signature(), CHECK); +} + void Threads::initialize_java_lang_classes(JavaThread* main_thread, TRAPS) { TraceStartupTime timer("Initialize java.lang classes"); @@ -3391,10 +3442,15 @@ void Threads::initialize_java_lang_classes(JavaThread* main_thread, TRAPS) { java_lang_Thread::set_thread_status(thread_object, java_lang_Thread::RUNNABLE); + // The VM creates objects of this class. + initialize_class(vmSymbols::java_lang_reflect_Module(), CHECK); + // The VM preresolves methods to these classes. Make sure that they get initialized initialize_class(vmSymbols::java_lang_reflect_Method(), CHECK); initialize_class(vmSymbols::java_lang_ref_Finalizer(), CHECK); - call_initializeSystemClass(CHECK); + + // Phase 1 of the system initialization in the library, java.lang.System class initialization + call_initPhase1(CHECK); // get the Java runtime name after java.lang.System is initialized JDK_Version::set_runtime_name(get_java_runtime_name(THREAD)); @@ -3612,10 +3668,10 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) { // Always call even when there are not JVMTI environments yet, since environments // may be attached late and JVMTI must track phases of VM execution - JvmtiExport::enter_start_phase(); + JvmtiExport::enter_early_start_phase(); // Notify JVMTI agents that VM has started (JNI is up) - nop if no agents. - JvmtiExport::post_vm_start(); + JvmtiExport::post_early_vm_start(); initialize_java_lang_classes(main_thread, CHECK_JNI_ERR); @@ -3643,12 +3699,9 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) { Management::record_vm_init_completed(); #endif // INCLUDE_MANAGEMENT - // Compute system loader. Note that this has to occur after set_init_completed, since - // valid exceptions may be thrown in the process. // Note that we do not use CHECK_0 here since we are inside an EXCEPTION_MARK and // set_init_completed has just been called, causing exceptions not to be shortcut // anymore. We call vm_exit_during_initialization directly instead. - SystemDictionary::compute_java_system_loader(CHECK_(JNI_ERR)); #if INCLUDE_ALL_GCS // Support for ConcurrentMarkSweep. This should be cleaned up @@ -3663,10 +3716,6 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) { } #endif // INCLUDE_ALL_GCS - // Always call even when there are not JVMTI environments yet, since environments - // may be attached late and JVMTI must track phases of VM execution - JvmtiExport::enter_live_phase(); - // Signal Dispatcher needs to be started before VMInit event is posted os::signal_init(); @@ -3685,13 +3734,6 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) { create_vm_init_libraries(); } - // Notify JVMTI agents that VM initialization is complete - nop if no agents. - JvmtiExport::post_vm_initialized(); - - if (TRACE_START() != JNI_OK) { - vm_exit_during_initialization("Failed to start tracing backend."); - } - if (CleanChunkPoolAsync) { Chunk::start_chunk_pool_cleaner_task(); } @@ -3716,6 +3758,34 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) { // (see SystemDictionary::find_method_handle_intrinsic). initialize_jsr292_core_classes(CHECK_JNI_ERR); + // This will initialize the module system. Only java.base classes can be + // loaded until phase 2 completes + call_initPhase2(CHECK_JNI_ERR); + + // Always call even when there are not JVMTI environments yet, since environments + // may be attached late and JVMTI must track phases of VM execution + JvmtiExport::enter_start_phase(); + + // Notify JVMTI agents that VM has started (JNI is up) - nop if no agents. + JvmtiExport::post_vm_start(); + + // Final system initialization including security manager and system class loader + call_initPhase3(CHECK_JNI_ERR); + + // cache the system class loader + SystemDictionary::compute_java_system_loader(CHECK_(JNI_ERR)); + + // Always call even when there are not JVMTI environments yet, since environments + // may be attached late and JVMTI must track phases of VM execution + JvmtiExport::enter_live_phase(); + + // Notify JVMTI agents that VM initialization is complete - nop if no agents. + JvmtiExport::post_vm_initialized(); + + if (TRACE_START() != JNI_OK) { + vm_exit_during_initialization("Failed to start tracing backend."); + } + #if INCLUDE_MANAGEMENT Management::initialize(THREAD); diff --git a/hotspot/src/share/vm/services/jmm.h b/hotspot/src/share/vm/services/jmm.h index 0362e794d10..b6497cd8982 100644 --- a/hotspot/src/share/vm/services/jmm.h +++ b/hotspot/src/share/vm/services/jmm.h @@ -59,7 +59,6 @@ typedef struct { unsigned int isThreadContentionMonitoringSupported : 1; unsigned int isCurrentThreadCpuTimeSupported : 1; unsigned int isOtherThreadCpuTimeSupported : 1; - unsigned int isBootClassPathSupported : 1; unsigned int isObjectMonitorUsageSupported : 1; unsigned int isSynchronizerUsageSupported : 1; unsigned int isThreadAllocatedMemorySupported : 1; diff --git a/hotspot/src/share/vm/services/management.cpp b/hotspot/src/share/vm/services/management.cpp index 892ce92fe7e..6de924a21ab 100644 --- a/hotspot/src/share/vm/services/management.cpp +++ b/hotspot/src/share/vm/services/management.cpp @@ -121,7 +121,6 @@ void Management::init() { _optional_support.isOtherThreadCpuTimeSupported = 0; } - _optional_support.isBootClassPathSupported = 1; _optional_support.isObjectMonitorUsageSupported = 1; #if INCLUDE_SERVICES // This depends on the heap inspector diff --git a/hotspot/src/share/vm/trace/traceBackend.cpp b/hotspot/src/share/vm/trace/traceBackend.cpp new file mode 100644 index 00000000000..a93077405e2 --- /dev/null +++ b/hotspot/src/share/vm/trace/traceBackend.cpp @@ -0,0 +1,28 @@ +/* +* 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. +* +*/ + +#include "precompiled.hpp" +#include "prims/jni.h" + +extern "C" void JNICALL trace_register_natives(JNIEnv*, jclass) {} diff --git a/hotspot/src/share/vm/trace/traceDataTypes.hpp b/hotspot/src/share/vm/trace/traceDataTypes.hpp index b51343d9575..5df07eb1bbe 100644 --- a/hotspot/src/share/vm/trace/traceDataTypes.hpp +++ b/hotspot/src/share/vm/trace/traceDataTypes.hpp @@ -57,6 +57,10 @@ enum ReservedEvent { typedef enum ReservedEvent ReservedEvent; +typedef u8 traceid; + +class ModuleEntry; +class PackageEntry; class Symbol; #endif // SHARE_VM_TRACE_TRACEDATATYPES_HPP diff --git a/hotspot/src/share/vm/trace/traceMacros.hpp b/hotspot/src/share/vm/trace/traceMacros.hpp index c67024ae3f0..23a155748ce 100644 --- a/hotspot/src/share/vm/trace/traceMacros.hpp +++ b/hotspot/src/share/vm/trace/traceMacros.hpp @@ -32,10 +32,14 @@ typedef u8 traceid; #define TRACE_KLASS_CREATION(k, p, t) #define TRACE_INIT_KLASS_ID(k) +#define TRACE_INIT_MODULE_ID(m) +#define TRACE_INIT_PACKAGE_ID(p) #define TRACE_INIT_THREAD_ID(td) #define TRACE_DATA TraceThreadData #define THREAD_TRACE_ID(thread) ((traceid)thread->osthread()->thread_id()) +extern "C" void JNICALL trace_register_natives(JNIEnv*, jclass); +#define TRACE_REGISTER_NATIVES ((void*)((address_word)(&trace_register_natives))) #define TRACE_START() JNI_OK #define TRACE_INITIALIZE() JNI_OK diff --git a/hotspot/src/share/vm/trace/tracetypes.xml b/hotspot/src/share/vm/trace/tracetypes.xml index 52d6e9a6b07..c4c3ddbf959 100644 --- a/hotspot/src/share/vm/trace/tracetypes.xml +++ b/hotspot/src/share/vm/trace/tracetypes.xml @@ -78,6 +78,7 @@ Now we can use the content + data type in declaring event fields. type="U8" builtin_type="CLASS"> + @@ -174,6 +175,21 @@ Now we can use the content + data type in declaring event fields. type="U8" jvm_type="INFLATECAUSE"> + + + + + + + + + + + + + @@ -286,6 +302,12 @@ Now we can use the content + data type in declaring event fields. + + + + next()) { // Print in two stages to avoid problems with long // keys/values. + assert(p->key() != NULL, "p->key() is NULL"); text->print_raw(p->key()); text->put('='); + assert(p->value() != NULL, "p->value() is NULL"); text->print_raw_cr(p->value()); } xs->tail("properties"); diff --git a/hotspot/src/share/vm/utilities/utf8.cpp b/hotspot/src/share/vm/utilities/utf8.cpp index d7a6043097a..93ad3e23d89 100644 --- a/hotspot/src/share/vm/utilities/utf8.cpp +++ b/hotspot/src/share/vm/utilities/utf8.cpp @@ -333,6 +333,68 @@ jint UTF8::get_supplementary_character(const unsigned char* str) { + ((str[4] & 0x0f) << 6) + (str[5] & 0x3f); } +bool UTF8::is_legal_utf8(const unsigned char* buffer, int length, + bool version_leq_47) { + int i = 0; + int count = length >> 2; + for (int k=0; k= 128 (highest bit 1) for v == 0 or v >= 128. + unsigned char res = b0 | b0 - 1 | + b1 | b1 - 1 | + b2 | b2 - 1 | + b3 | b3 - 1; + if (res >= 128) break; + i += 4; + } + for(; i < length; i++) { + unsigned short c; + // no embedded zeros + if (buffer[i] == 0) return false; + if(buffer[i] < 128) { + continue; + } + if ((i + 5) < length) { // see if it's legal supplementary character + if (UTF8::is_supplementary_character(&buffer[i])) { + c = UTF8::get_supplementary_character(&buffer[i]); + i += 5; + continue; + } + } + switch (buffer[i] >> 4) { + default: break; + case 0x8: case 0x9: case 0xA: case 0xB: case 0xF: + return false; + case 0xC: case 0xD: // 110xxxxx 10xxxxxx + c = (buffer[i] & 0x1F) << 6; + i++; + if ((i < length) && ((buffer[i] & 0xC0) == 0x80)) { + c += buffer[i] & 0x3F; + if (version_leq_47 || c == 0 || c >= 0x80) { + break; + } + } + return false; + case 0xE: // 1110xxxx 10xxxxxx 10xxxxxx + c = (buffer[i] & 0xF) << 12; + i += 2; + if ((i < length) && ((buffer[i-1] & 0xC0) == 0x80) && ((buffer[i] & 0xC0) == 0x80)) { + c += ((buffer[i-1] & 0x3F) << 6) + (buffer[i] & 0x3F); + if (version_leq_47 || c >= 0x800) { + break; + } + } + return false; + } // end of switch + } // end of for + return true; +} + //------------------------------------------------------------------------------------- bool UNICODE::is_latin1(jchar c) { diff --git a/hotspot/src/share/vm/utilities/utf8.hpp b/hotspot/src/share/vm/utilities/utf8.hpp index 161eb410f40..1a30cb456fc 100644 --- a/hotspot/src/share/vm/utilities/utf8.hpp +++ b/hotspot/src/share/vm/utilities/utf8.hpp @@ -73,6 +73,9 @@ class UTF8 : AllStatic { static bool equal(const jbyte* base1, int length1, const jbyte* base2,int length2); static bool is_supplementary_character(const unsigned char* str); static jint get_supplementary_character(const unsigned char* str); + + static bool is_legal_utf8(const unsigned char* buffer, int length, + bool version_leq_47); }; diff --git a/hotspot/test/TEST.ROOT b/hotspot/test/TEST.ROOT index 8dd7c84e9ee..55979b0a554 100644 --- a/hotspot/test/TEST.ROOT +++ b/hotspot/test/TEST.ROOT @@ -32,8 +32,8 @@ keys=cte_test jcmd nmt regression gc stress groups=TEST.groups [closed/TEST.groups] requires.properties=sun.arch.data.model -# Tests using jtreg 4.1 b12 features -requiredVersion=4.1 b12 +# Tests using jtreg 4.2 b01 features +requiredVersion=4.2 b01 # Path to libraries in the topmost test directory. This is needed so @library # does not need ../../ notation to reach them diff --git a/hotspot/test/compiler/dependencies/MonomorphicObjectCall/TestMonomorphicObjectCall.java b/hotspot/test/compiler/dependencies/MonomorphicObjectCall/TestMonomorphicObjectCall.java index f4109d65810..24a8222f23f 100644 --- a/hotspot/test/compiler/dependencies/MonomorphicObjectCall/TestMonomorphicObjectCall.java +++ b/hotspot/test/compiler/dependencies/MonomorphicObjectCall/TestMonomorphicObjectCall.java @@ -22,6 +22,8 @@ */ import java.io.File; +import java.nio.file.Files; +import java.nio.file.Paths; import java.util.ArrayList; import java.util.Collections; @@ -35,11 +37,11 @@ import jdk.test.lib.*; * @modules java.base/sun.misc * java.management * java.base/jdk.internal + * @ignore 8132924 * @compile -XDignore.symbol.file java/lang/Object.java TestMonomorphicObjectCall.java * @run main TestMonomorphicObjectCall */ public class TestMonomorphicObjectCall { - final static String testClasses = System.getProperty("test.classes") + File.separator; private static void callFinalize(Object object) throws Throwable { // Call modified version of java.lang.Object::finalize() that is @@ -50,6 +52,9 @@ public class TestMonomorphicObjectCall { public static void main(String[] args) throws Throwable { if (args.length == 0) { + byte[] bytecode = Files.readAllBytes(Paths.get(System.getProperty("test.classes") + File.separator + + "java" + File.separator + "lang" + File.separator + "Object.class")); + ClassFileInstaller.writeClassToDisk("java.lang.Object", bytecode, "mods/java.base"); // Execute new instance with modified java.lang.Object executeTestJvm(); } else { @@ -62,7 +67,7 @@ public class TestMonomorphicObjectCall { // Execute test with modified version of java.lang.Object // in -Xbootclasspath. String[] vmOpts = new String[] { - "-Xbootclasspath/p:" + testClasses, + "-Xpatch:mods", "-Xcomp", "-XX:+IgnoreUnrecognizedVMOptions", "-XX:-VerifyDependencies", diff --git a/hotspot/test/compiler/jsr292/CallSiteDepContextTest.java b/hotspot/test/compiler/jsr292/CallSiteDepContextTest.java index e650aa558c1..78ee7131476 100644 --- a/hotspot/test/compiler/jsr292/CallSiteDepContextTest.java +++ b/hotspot/test/compiler/jsr292/CallSiteDepContextTest.java @@ -24,13 +24,25 @@ /** * @test * @bug 8057967 + * @modules java.base/jdk.internal.org.objectweb.asm + * @library patches + * @build java.base/java.lang.invoke.MethodHandleHelper * @run main/bootclasspath -Xbatch -XX:+IgnoreUnrecognizedVMOptions -Xlog:classunload * -XX:+PrintCompilation -XX:+TraceDependencies -XX:+TraceReferenceGC - * -verbose:gc java.lang.invoke.CallSiteDepContextTest + * -verbose:gc compiler.jsr292.CallSiteDepContextTest */ -package java.lang.invoke; -import java.lang.ref.*; +package compiler.jsr292; + +import java.lang.invoke.CallSite; +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandleHelper; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; +import java.lang.invoke.MutableCallSite; +import java.lang.ref.PhantomReference; +import java.lang.ref.Reference; +import java.lang.ref.ReferenceQueue; import java.lang.reflect.Field; import jdk.internal.org.objectweb.asm.*; @@ -40,10 +52,10 @@ import static jdk.internal.org.objectweb.asm.Opcodes.*; public class CallSiteDepContextTest { static final Unsafe UNSAFE = Unsafe.getUnsafe(); - static final MethodHandles.Lookup LOOKUP = MethodHandles.Lookup.IMPL_LOOKUP; + static final MethodHandles.Lookup LOOKUP = MethodHandleHelper.IMPL_LOOKUP; static final String CLASS_NAME = "java/lang/invoke/Test"; static final String METHOD_NAME = "m"; - static final MethodType TYPE = MethodType.methodType(int.class); + static final MethodType TYPE = MethodType.methodType(int.class); static MutableCallSite mcs; static MethodHandle bsmMH; @@ -77,7 +89,8 @@ public class CallSiteDepContextTest { mv = cw.visitMethod(ACC_PUBLIC | ACC_STATIC, METHOD_NAME, TYPE.toMethodDescriptorString(), null, null); mv.visitCode(); Handle bsm = new Handle(H_INVOKESTATIC, - "java/lang/invoke/CallSiteDepContextTest", "bootstrap", + CallSiteDepContextTest.class.getName().replace(".", "/"), + "bootstrap", bsmMH.type().toMethodDescriptorString()); mv.visitInvokeDynamicInsn("methodName", TYPE.toMethodDescriptorString(), bsm); mv.visitInsn(IRETURN); @@ -99,9 +112,9 @@ public class CallSiteDepContextTest { } } - public static void testHiddenDepField() throws Exception { + public static void testHiddenDepField() { try { - Field f = MethodHandleNatives.CallSiteContext.class.getDeclaredField("vmdependencies"); + Field f = MethodHandleHelper.MHN_CALL_SITE_CONTEXT_CLASS.getDeclaredField("vmdependencies"); throw new AssertionError("Context.dependencies field should be hidden"); } catch(NoSuchFieldException e) { /* expected */ } } @@ -111,8 +124,8 @@ public class CallSiteDepContextTest { Class cls2 = UNSAFE.defineAnonymousClass(Object.class, getClassFile("CS_2"), null); MethodHandle[] mhs = new MethodHandle[] { - LOOKUP.findStatic(cls1, METHOD_NAME, TYPE), - LOOKUP.findStatic(cls2, METHOD_NAME, TYPE) + LOOKUP.findStatic(cls1, METHOD_NAME, TYPE), + LOOKUP.findStatic(cls2, METHOD_NAME, TYPE) }; mcs = new MutableCallSite(LOOKUP.findStatic(T.class, "f1", TYPE)); @@ -198,3 +211,4 @@ public class CallSiteDepContextTest { System.out.println("TEST PASSED"); } } + diff --git a/hotspot/test/compiler/jsr292/NonInlinedCall/Agent.java b/hotspot/test/compiler/jsr292/NonInlinedCall/Agent.java index 70665d5f604..e9f2f46a8f4 100644 --- a/hotspot/test/compiler/jsr292/NonInlinedCall/Agent.java +++ b/hotspot/test/compiler/jsr292/NonInlinedCall/Agent.java @@ -20,28 +20,39 @@ * or visit www.oracle.com if you need additional information or have any * questions. */ + +package compiler.jsr292.NonInlinedCall; + import java.io.File; import java.io.PrintStream; import java.util.Arrays; +import jdk.test.lib.JDKToolLauncher; +import jdk.test.lib.OutputAnalyzer; + public class Agent { public static void main(String[] args) throws Exception { String jarName = args[0]; String className = args[1]; String manifestName = "manifest.mf"; - System.out.println("Creating "+manifestName); + System.out.println("Creating " + manifestName); try (PrintStream out = new PrintStream(new File(manifestName))) { out.println("Premain-Class: " + className); out.println("Can-Redefine-Classes: true"); } - System.out.println("Building "+jarName); - String[] jarArgs = new String[] {"-cfm", jarName, manifestName }; - System.out.println("Running jar " + Arrays.toString(jarArgs)); - sun.tools.jar.Main jarTool = new sun.tools.jar.Main(System.out, System.err, "jar"); - if (!jarTool.run(jarArgs)) { - throw new Error("jar failed: args=" + Arrays.toString(args)); - } + System.out.println("Building " + jarName); + JDKToolLauncher jar = JDKToolLauncher + .create("jar") + .addToolArg("-cfm") + .addToolArg(jarName) + .addToolArg(manifestName); + + System.out.println("Running jar " + Arrays.toString(jar.getCommand())); + + ProcessBuilder pb = new ProcessBuilder(jar.getCommand()); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldHaveExitValue(0); } } diff --git a/hotspot/test/compiler/jsr292/NonInlinedCall/GCTest.java b/hotspot/test/compiler/jsr292/NonInlinedCall/GCTest.java index 49ce05490fd..fc705850e00 100644 --- a/hotspot/test/compiler/jsr292/NonInlinedCall/GCTest.java +++ b/hotspot/test/compiler/jsr292/NonInlinedCall/GCTest.java @@ -24,29 +24,38 @@ /* * @test * @bug 8072008 - * @library /testlibrary /test/lib - * @compile GCTest.java NonInlinedReinvoker.java - * @run main ClassFileInstaller sun.hotspot.WhiteBox - * sun.hotspot.WhiteBox$WhiteBoxPermission - * java.lang.invoke.GCTest - * java.lang.invoke.GCTest$T - * java.lang.invoke.NonInlinedReinvoker - * jdk.test.lib.Asserts - * @run main/othervm -Xbootclasspath/a:. -XX:+IgnoreUnrecognizedVMOptions - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI - * -Xbatch -XX:-TieredCompilation -XX:CICompilerCount=1 - * java.lang.invoke.GCTest + * @library /testlibrary /test/lib ../patches + * @modules java.base/jdk.internal.vm.annotation + * @build java.base/java.lang.invoke.MethodHandleHelper + * @build sun.hotspot.WhiteBox + * @run main/bootclasspath -XX:+IgnoreUnrecognizedVMOptions + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -Xbatch -XX:-TieredCompilation -XX:CICompilerCount=1 + * -XX:+FoldStableValues + * compiler.jsr292.NonInlinedCall.GCTest */ -package java.lang.invoke; -import sun.hotspot.WhiteBox; +package compiler.jsr292.NonInlinedCall; + +import java.lang.invoke.MethodHandleHelper; +import java.lang.invoke.MethodHandleHelper.NonInlinedReinvoker; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodHandle; + +import java.lang.invoke.MethodType; +import java.lang.ref.PhantomReference; +import java.lang.ref.Reference; +import java.lang.ref.ReferenceQueue; + import jdk.internal.vm.annotation.DontInline; import jdk.internal.vm.annotation.Stable; -import java.lang.ref.*; + +import sun.hotspot.WhiteBox; + import static jdk.test.lib.Asserts.*; public class GCTest { - static final MethodHandles.Lookup LOOKUP = MethodHandles.Lookup.IMPL_LOOKUP; + static final MethodHandles.Lookup LOOKUP = MethodHandleHelper.IMPL_LOOKUP; static class T { static int f1() { return 0; } @@ -54,15 +63,15 @@ public class GCTest { } static @Stable MethodHandle mh; - static PhantomReference lform; + static PhantomReference lform; - static final ReferenceQueue rq = new ReferenceQueue<>(); + static final ReferenceQueue rq = new ReferenceQueue<>(); static final WhiteBox WB = WhiteBox.getWhiteBox(); @DontInline static int invokeBasic() { try { - return (int) mh.invokeBasic(); + return MethodHandleHelper.invokeBasicI(mh); } catch (Throwable e) { throw new Error(e); } @@ -80,7 +89,7 @@ public class GCTest { LOOKUP.findStatic(T.class, "f1", MethodType.methodType(int.class))); // Monitor LambdaForm GC - lform = new PhantomReference<>(mh.form, rq); + lform = new PhantomReference<>(MethodHandleHelper.getLambdaForm(mh), rq); test(0); WB.clearInlineCaches(); diff --git a/hotspot/test/compiler/jsr292/NonInlinedCall/InvokeTest.java b/hotspot/test/compiler/jsr292/NonInlinedCall/InvokeTest.java index 90724b371a8..225db6018ce 100644 --- a/hotspot/test/compiler/jsr292/NonInlinedCall/InvokeTest.java +++ b/hotspot/test/compiler/jsr292/NonInlinedCall/InvokeTest.java @@ -24,30 +24,33 @@ /* * @test * @bug 8072008 - * @library /testlibrary /test/lib - * @compile InvokeTest.java NonInlinedReinvoker.java - * @run main ClassFileInstaller sun.hotspot.WhiteBox - * sun.hotspot.WhiteBox$WhiteBoxPermission - * java.lang.invoke.InvokeTest - * java.lang.invoke.InvokeTest$T - * java.lang.invoke.InvokeTest$P1 - * java.lang.invoke.InvokeTest$P2 - * java.lang.invoke.InvokeTest$I - * java.lang.invoke.NonInlinedReinvoker - * jdk.test.lib.Asserts - * @run main/othervm -Xbootclasspath/a:. -XX:+IgnoreUnrecognizedVMOptions - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI - * -Xbatch -XX:-TieredCompilation -XX:CICompilerCount=1 - * java.lang.invoke.InvokeTest + * @library /testlibrary /test/lib / ../patches + * @modules java.base/jdk.internal.vm.annotation + * @build java.base/java.lang.invoke.MethodHandleHelper + * @build sun.hotspot.WhiteBox + * @build compiler.jsr292.NonInlinedCall.InvokeTest + * @run main/bootclasspath -XX:+IgnoreUnrecognizedVMOptions + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -Xbatch -XX:-TieredCompilation -XX:CICompilerCount=1 + * compiler.jsr292.NonInlinedCall.InvokeTest */ -package java.lang.invoke; + +package compiler.jsr292.NonInlinedCall; + +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandleHelper; +import java.lang.invoke.MethodHandleHelper.NonInlinedReinvoker; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; + +import jdk.internal.vm.annotation.DontInline; import sun.hotspot.WhiteBox; -import jdk.internal.vm.annotation.DontInline; + import static jdk.test.lib.Asserts.*; public class InvokeTest { - static MethodHandles.Lookup LOOKUP = MethodHandles.Lookup.IMPL_LOOKUP; + static MethodHandles.Lookup LOOKUP = MethodHandleHelper.IMPL_LOOKUP; static final MethodHandle virtualMH; // invokevirtual T.f1 static final MethodHandle staticMH; // invokestatic T.f2 @@ -136,7 +139,7 @@ public class InvokeTest { @DontInline static void invokeBasic() { try { - Class cls = (Class)basicMH.invokeBasic(); + Class cls = (Class)MethodHandleHelper.invokeBasicL(basicMH); assertEquals(cls, T.class); } catch (Throwable e) { throw new Error(e); diff --git a/hotspot/test/compiler/jsr292/NonInlinedCall/RedefineTest.java b/hotspot/test/compiler/jsr292/NonInlinedCall/RedefineTest.java index 51481e87bb4..6607ec19c6c 100644 --- a/hotspot/test/compiler/jsr292/NonInlinedCall/RedefineTest.java +++ b/hotspot/test/compiler/jsr292/NonInlinedCall/RedefineTest.java @@ -24,35 +24,46 @@ /* * @test * @bug 8072008 - * @library /testlibrary /test/lib - * @compile -XDignore.symbol.file RedefineTest.java Agent.java + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.misc + * java.base/jdk.internal.vm.annotation + * @library /testlibrary /test/lib / ../patches + * @build sun.hotspot.WhiteBox + * @build java.base/java.lang.invoke.MethodHandleHelper + * @build compiler.jsr292.NonInlinedCall.RedefineTest + * @run main compiler.jsr292.NonInlinedCall.Agent agent.jar compiler.jsr292.NonInlinedCall.RedefineTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * java.lang.invoke.RedefineTest - * Agent - * @run main Agent agent.jar java.lang.invoke.RedefineTest - * @run main/othervm -Xbootclasspath/a:. -javaagent:agent.jar - * -XX:+IgnoreUnrecognizedVMOptions - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI - * -Xbatch -XX:-TieredCompilation -XX:CICompilerCount=1 - * java.lang.invoke.RedefineTest + * compiler.jsr292.NonInlinedCall.RedefineTest + * @run main/bootclasspath -javaagent:agent.jar + * -XX:+IgnoreUnrecognizedVMOptions + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -Xbatch -XX:-TieredCompilation -XX:CICompilerCount=1 + * compiler.jsr292.NonInlinedCall.RedefineTest */ -package java.lang.invoke; + +package compiler.jsr292.NonInlinedCall; import sun.hotspot.WhiteBox; -import sun.misc.Unsafe; -import jdk.internal.org.objectweb.asm.*; -import jdk.internal.vm.annotation.DontInline; + import java.lang.instrument.ClassDefinition; import java.lang.instrument.Instrumentation; +import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodHandleHelper; +import java.lang.invoke.MethodType; + +import jdk.internal.misc.Unsafe; +import jdk.internal.vm.annotation.DontInline; +import jdk.internal.org.objectweb.asm.*; import static jdk.internal.org.objectweb.asm.Opcodes.*; public class RedefineTest { - static final MethodHandles.Lookup LOOKUP = MethodHandles.Lookup.IMPL_LOOKUP; + static final MethodHandles.Lookup LOOKUP = MethodHandleHelper.IMPL_LOOKUP; static final Unsafe UNSAFE = Unsafe.getUnsafe(); - static final String NAME = "java/lang/invoke/RedefineTest$T"; + static final String NAME = "compiler/jsr292/NonInlinedCall/RedefineTest$T"; static Class getClass(int r) { byte[] classFile = getClassFile(r); diff --git a/hotspot/test/compiler/jsr292/patches/java.base/java/lang/invoke/MethodHandleHelper.java b/hotspot/test/compiler/jsr292/patches/java.base/java/lang/invoke/MethodHandleHelper.java new file mode 100644 index 00000000000..a4732624a3a --- /dev/null +++ b/hotspot/test/compiler/jsr292/patches/java.base/java/lang/invoke/MethodHandleHelper.java @@ -0,0 +1,92 @@ +/* + * 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 java.lang.invoke; + +import java.lang.invoke.MethodHandles.Lookup; +import jdk.internal.vm.annotation.DontInline; +import jdk.internal.vm.annotation.ForceInline; +/** + * Helper class to inject into java.lang.invoke that provides access to + * package-private methods in this package. + */ + +public class MethodHandleHelper { + + private MethodHandleHelper() { } + + public static final Lookup IMPL_LOOKUP = Lookup.IMPL_LOOKUP; + public static final Class MHN_CALL_SITE_CONTEXT_CLASS + = MethodHandleNatives.CallSiteContext.class; + + public static void customize(MethodHandle mh) { + mh.customize(); + } + + @ForceInline + public static Object invokeBasicL(MethodHandle mh) throws Throwable { + return mh.invokeBasic(); + } + + @ForceInline + public static int invokeBasicI(MethodHandle mh) throws Throwable { + return (int) mh.invokeBasic(); + } + + public static MethodHandle varargsArray(int nargs) { + return MethodHandleImpl.varargsArray(nargs); + } + + public static MethodHandle varargsArray(Class arrayType, int nargs) { + return MethodHandleImpl.varargsArray(arrayType, nargs); + } + + public static LambdaForm getLambdaForm(MethodHandle mh) { + return mh.form; + } + + public static class NonInlinedReinvoker extends DelegatingMethodHandle { + private final MethodHandle target; + + private NonInlinedReinvoker(MethodHandle target, LambdaForm lf) { + super(target.type(), lf); + this.target = target; + } + @Override + public MethodHandle getTarget() { + return target; + } + + @Override + public MethodHandle asTypeUncached(MethodType newType) { + return asTypeCache = target.asType(newType); + } + + public static MethodHandle make(MethodHandle target) { + LambdaForm lform = DelegatingMethodHandle.makeReinvokerForm( + target, -1, DelegatingMethodHandle.class, "reinvoker.dontInline", + /*forceInline=*/false, DelegatingMethodHandle.NF_getTarget, null); + return new NonInlinedReinvoker(target, lform); + } + } +} diff --git a/hotspot/test/compiler/jvmci/SecurityRestrictionsTest.java b/hotspot/test/compiler/jvmci/SecurityRestrictionsTest.java index e74fc43ca01..993cbba8a18 100644 --- a/hotspot/test/compiler/jvmci/SecurityRestrictionsTest.java +++ b/hotspot/test/compiler/jvmci/SecurityRestrictionsTest.java @@ -27,25 +27,26 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ./common/CompilerToVMHelper.java - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -Xbootclasspath/a:. + * @library common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @run main/othervm -XX:+UnlockExperimentalVMOptions * -XX:+EnableJVMCI * compiler.jvmci.SecurityRestrictionsTest * NO_SEC_MAN - * @run main/othervm -XX:+UnlockExperimentalVMOptions -Xbootclasspath/a:. + * @run main/othervm -XX:+UnlockExperimentalVMOptions * -XX:+EnableJVMCI * compiler.jvmci.SecurityRestrictionsTest * NO_PERM - * @run main/othervm -XX:+UnlockExperimentalVMOptions -Xbootclasspath/a:. + * @run main/othervm -XX:+UnlockExperimentalVMOptions * -XX:+EnableJVMCI * compiler.jvmci.SecurityRestrictionsTest * ALL_PERM - * @run main/othervm -XX:+UnlockExperimentalVMOptions -Xbootclasspath/a:. + * @run main/othervm -XX:+UnlockExperimentalVMOptions * -XX:+EnableJVMCI * compiler.jvmci.SecurityRestrictionsTest * NO_JVMCI_ACCESS_PERM - * @run main/othervm -XX:+UnlockExperimentalVMOptions -Xbootclasspath/a:. + * @run main/othervm -XX:+UnlockExperimentalVMOptions * -XX:-EnableJVMCI * compiler.jvmci.SecurityRestrictionsTest * NO_JVMCI diff --git a/hotspot/test/compiler/jvmci/code/DataPatchTest.java b/hotspot/test/compiler/jvmci/code/DataPatchTest.java index af438a306bf..60e46915202 100644 --- a/hotspot/test/compiler/jvmci/code/DataPatchTest.java +++ b/hotspot/test/compiler/jvmci/code/DataPatchTest.java @@ -24,7 +24,14 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9") & os.arch != "aarch64" - * @compile CodeInstallationTest.java TestAssembler.java amd64/AMD64TestAssembler.java sparc/SPARCTestAssembler.java + * @library / + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.code.site + * jdk.vm.ci/jdk.vm.ci.runtime + * jdk.vm.ci/jdk.vm.ci.amd64 + * jdk.vm.ci/jdk.vm.ci.sparc * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI compiler.jvmci.code.DataPatchTest */ diff --git a/hotspot/test/compiler/jvmci/code/SimpleCodeInstallationTest.java b/hotspot/test/compiler/jvmci/code/SimpleCodeInstallationTest.java index 60def019ad1..6c1377c8935 100644 --- a/hotspot/test/compiler/jvmci/code/SimpleCodeInstallationTest.java +++ b/hotspot/test/compiler/jvmci/code/SimpleCodeInstallationTest.java @@ -24,7 +24,14 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9") & os.arch != "aarch64" - * @compile CodeInstallationTest.java TestAssembler.java amd64/AMD64TestAssembler.java sparc/SPARCTestAssembler.java + * @library / + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.code.site + * jdk.vm.ci/jdk.vm.ci.runtime + * jdk.vm.ci/jdk.vm.ci.amd64 + * jdk.vm.ci/jdk.vm.ci.sparc * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI compiler.jvmci.code.SimpleCodeInstallationTest */ diff --git a/hotspot/test/compiler/jvmci/code/SimpleDebugInfoTest.java b/hotspot/test/compiler/jvmci/code/SimpleDebugInfoTest.java index 041a184bc16..b27802ce47a 100644 --- a/hotspot/test/compiler/jvmci/code/SimpleDebugInfoTest.java +++ b/hotspot/test/compiler/jvmci/code/SimpleDebugInfoTest.java @@ -24,7 +24,14 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9") & os.arch != "aarch64" - * @compile CodeInstallationTest.java DebugInfoTest.java TestAssembler.java amd64/AMD64TestAssembler.java sparc/SPARCTestAssembler.java + * @library / + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.code.site + * jdk.vm.ci/jdk.vm.ci.runtime + * jdk.vm.ci/jdk.vm.ci.amd64 + * jdk.vm.ci/jdk.vm.ci.sparc * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI compiler.jvmci.code.SimpleDebugInfoTest */ diff --git a/hotspot/test/compiler/jvmci/code/VirtualObjectDebugInfoTest.java b/hotspot/test/compiler/jvmci/code/VirtualObjectDebugInfoTest.java index 2f1061adfd0..c06c5076eae 100644 --- a/hotspot/test/compiler/jvmci/code/VirtualObjectDebugInfoTest.java +++ b/hotspot/test/compiler/jvmci/code/VirtualObjectDebugInfoTest.java @@ -24,7 +24,14 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9") & os.arch != "aarch64" - * @compile CodeInstallationTest.java DebugInfoTest.java TestAssembler.java amd64/AMD64TestAssembler.java sparc/SPARCTestAssembler.java + * @library / + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.code.site + * jdk.vm.ci/jdk.vm.ci.runtime + * jdk.vm.ci/jdk.vm.ci.amd64 + * jdk.vm.ci/jdk.vm.ci.sparc * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI compiler.jvmci.code.VirtualObjectDebugInfoTest */ diff --git a/hotspot/test/compiler/jvmci/common/CTVMUtilities.java b/hotspot/test/compiler/jvmci/common/CTVMUtilities.java index 67a1aef7a19..6c3301631a5 100644 --- a/hotspot/test/compiler/jvmci/common/CTVMUtilities.java +++ b/hotspot/test/compiler/jvmci/common/CTVMUtilities.java @@ -24,6 +24,7 @@ package compiler.jvmci.common; import java.io.IOException; +import java.lang.reflect.Module; import java.lang.reflect.Field; import java.lang.reflect.Executable; import java.lang.reflect.Constructor; @@ -88,20 +89,24 @@ public class CTVMUtilities { } public static Map getBciToLineNumber(Executable method) { Map lineNumbers = new TreeMap<>(); + Class aClass = method.getDeclaringClass(); + ClassReader cr; try { - ClassReader cr = new ClassReader(method.getDeclaringClass() - .getName()); - ClassNode cn = new ClassNode(); - cr.accept(cn, ClassReader.EXPAND_FRAMES); - - Map labels = new HashMap<>(); - ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_FRAMES); - ClassVisitor cv = new ClassVisitorForLabels(cw, labels, method); - cr.accept(cv, ClassReader.EXPAND_FRAMES); - labels.forEach((k, v) -> lineNumbers.put(k.getOffset(), v)); + Module aModule = aClass.getModule(); + String name = aClass.getName(); + cr = new ClassReader(aModule.getResourceAsStream( + name.replace('.', '/') + ".class")); } catch (IOException e) { - throw new Error("TEST BUG " + e, e); + throw new Error("TEST BUG: can read " + aClass.getName() + " : " + e, e); } + ClassNode cn = new ClassNode(); + cr.accept(cn, ClassReader.EXPAND_FRAMES); + + Map labels = new HashMap<>(); + ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_FRAMES); + ClassVisitor cv = new ClassVisitorForLabels(cw, labels, method); + cr.accept(cv, ClassReader.EXPAND_FRAMES); + labels.forEach((k, v) -> lineNumbers.put(k.getOffset(), v)); boolean isEmptyMethod = Modifier.isAbstract(method.getModifiers()) || Modifier.isNative(method.getModifiers()); if (lineNumbers.isEmpty() && !isEmptyMethod) { diff --git a/hotspot/test/compiler/jvmci/common/CompilerToVMHelper.java b/hotspot/test/compiler/jvmci/common/patches/jdk.vm.ci/jdk/vm/ci/hotspot/CompilerToVMHelper.java similarity index 100% rename from hotspot/test/compiler/jvmci/common/CompilerToVMHelper.java rename to hotspot/test/compiler/jvmci/common/patches/jdk.vm.ci/jdk/vm/ci/hotspot/CompilerToVMHelper.java diff --git a/hotspot/test/compiler/jvmci/events/MetaAccessWrapper.java b/hotspot/test/compiler/jvmci/common/patches/jdk.vm.ci/jdk/vm/ci/hotspot/MetaAccessWrapper.java similarity index 100% rename from hotspot/test/compiler/jvmci/events/MetaAccessWrapper.java rename to hotspot/test/compiler/jvmci/common/patches/jdk.vm.ci/jdk/vm/ci/hotspot/MetaAccessWrapper.java diff --git a/hotspot/test/compiler/jvmci/common/PublicMetaspaceWrapperObject.java b/hotspot/test/compiler/jvmci/common/patches/jdk.vm.ci/jdk/vm/ci/hotspot/PublicMetaspaceWrapperObject.java similarity index 100% rename from hotspot/test/compiler/jvmci/common/PublicMetaspaceWrapperObject.java rename to hotspot/test/compiler/jvmci/common/patches/jdk.vm.ci/jdk/vm/ci/hotspot/PublicMetaspaceWrapperObject.java diff --git a/hotspot/test/compiler/jvmci/compilerToVM/AllocateCompileIdTest.java b/hotspot/test/compiler/jvmci/compilerToVM/AllocateCompileIdTest.java index 63c009c3df3..3902acf2388 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/AllocateCompileIdTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/AllocateCompileIdTest.java @@ -26,15 +26,21 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.AllocateCompileIdTest * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * -XX:-BackgroundCompilation - * compiler.jvmci.compilerToVM.AllocateCompileIdTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:-BackgroundCompilation + * compiler.jvmci.compilerToVM.AllocateCompileIdTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/CanInlineMethodTest.java b/hotspot/test/compiler/jvmci/compilerToVM/CanInlineMethodTest.java index 64dca91cfb5..b24d25ca005 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/CanInlineMethodTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/CanInlineMethodTest.java @@ -27,14 +27,20 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.CanInlineMethodTest * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.CanInlineMethodTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.CanInlineMethodTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/CollectCountersTest.java b/hotspot/test/compiler/jvmci/compilerToVM/CollectCountersTest.java index be1d594bbd4..81b7bd47d7b 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/CollectCountersTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/CollectCountersTest.java @@ -26,20 +26,20 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib/ - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.CollectCountersTest * @run main/othervm -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI -Xbootclasspath/a:. - * -XX:JVMCICounterSize=0 - * -Dcompiler.jvmci.compilerToVM.CollectCountersTest.expected=0 - * compiler.jvmci.compilerToVM.CollectCountersTest + * -XX:+EnableJVMCI + * -XX:JVMCICounterSize=0 + * -Dcompiler.jvmci.compilerToVM.CollectCountersTest.expected=0 + * compiler.jvmci.compilerToVM.CollectCountersTest * @run main/othervm -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI -Xbootclasspath/a:. - * -XX:JVMCICounterSize=11 - * -Dcompiler.jvmci.compilerToVM.CollectCountersTest.expected=11 - * compiler.jvmci.compilerToVM.CollectCountersTest + * -XX:+EnableJVMCI + * -XX:JVMCICounterSize=11 + * -Dcompiler.jvmci.compilerToVM.CollectCountersTest.expected=11 + * compiler.jvmci.compilerToVM.CollectCountersTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/DebugOutputTest.java b/hotspot/test/compiler/jvmci/compilerToVM/DebugOutputTest.java index d6c22fc5ee1..fe251871e07 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/DebugOutputTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/DebugOutputTest.java @@ -26,12 +26,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run driver compiler.jvmci.compilerToVM.DebugOutputTest + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.DebugOutputTest + * @run main/othervm compiler.jvmci.compilerToVM.DebugOutputTest */ + // as soon as CODETOOLS-7901589 fixed, '@run main/othervm' should be replaced w/ '@run driver' + package compiler.jvmci.compilerToVM; import jdk.vm.ci.hotspot.CompilerToVMHelper; @@ -50,15 +53,13 @@ public class DebugOutputTest { System.out.println(testCase); OutputAnalyzer oa; try { - ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( - /* use test options = */ true, + oa = ProcessTools.executeTestJvmAllArgs( "-XX:+UnlockExperimentalVMOptions", "-XX:+EnableJVMCI", "-Xbootclasspath/a:.", DebugOutputTest.Worker.class.getName(), testCase.name()); - oa = ProcessTools.executeProcess(pb); - } catch (Exception e) { + } catch (Throwable e) { e.printStackTrace(); throw new Error("Problems running child process", e); } diff --git a/hotspot/test/compiler/jvmci/compilerToVM/DisassembleCodeBlobTest.java b/hotspot/test/compiler/jvmci/compilerToVM/DisassembleCodeBlobTest.java index 60a734d68a5..bd39a56e2e5 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/DisassembleCodeBlobTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/DisassembleCodeBlobTest.java @@ -27,16 +27,21 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code * @ignore 8139700 - * @compile ../common/CompilerToVMHelper.java + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.DisassembleCodeBlobTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.DisassembleCodeBlobTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.DisassembleCodeBlobTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java b/hotspot/test/compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java index a87cab4a386..747a9cf6123 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/DoNotInlineOrCompileTest.java @@ -27,14 +27,20 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.DoNotInlineOrCompileTest * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.DoNotInlineOrCompileTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.DoNotInlineOrCompileTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ExecuteInstalledCodeTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ExecuteInstalledCodeTest.java index 839b509af15..fb89c4e5428 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ExecuteInstalledCodeTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ExecuteInstalledCodeTest.java @@ -23,16 +23,21 @@ import java.util.Map; * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / + * @library ../common/patches * @ignore 8139383 - * @compile ../common/CompilerToVMHelper.java + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.ExecuteInstalledCodeTest * @build sun.hotspot.WhiteBox - * compiler.jvmci.compilerToVM.ExecuteInstalledCodeTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.ExecuteInstalledCodeTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.ExecuteInstalledCodeTest */ public class ExecuteInstalledCodeTest { diff --git a/hotspot/test/compiler/jvmci/compilerToVM/FindUniqueConcreteMethodTest.java b/hotspot/test/compiler/jvmci/compilerToVM/FindUniqueConcreteMethodTest.java index 2b39757975c..0cca017515d 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/FindUniqueConcreteMethodTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/FindUniqueConcreteMethodTest.java @@ -26,12 +26,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.FindUniqueConcreteMethodTest - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.FindUniqueConcreteMethodTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.FindUniqueConcreteMethodTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetBytecodeTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetBytecodeTest.java index 586160f58fc..472bc379cb5 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetBytecodeTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetBytecodeTest.java @@ -27,11 +27,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.GetBytecodeTest * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.GetBytecodeTest + * compiler.jvmci.compilerToVM.GetBytecodeTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetClassInitializerTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetClassInitializerTest.java index 6c731a4540f..1b0eacdf3c6 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetClassInitializerTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetClassInitializerTest.java @@ -26,11 +26,12 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.GetClassInitializerTest - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.GetClassInitializerTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.GetClassInitializerTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetConstantPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetConstantPoolTest.java index 4c347964800..3e14a684644 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetConstantPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetConstantPoolTest.java @@ -27,16 +27,19 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java ../common/PublicMetaspaceWrapperObject.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build jdk.vm.ci/jdk.vm.ci.hotspot.PublicMetaspaceWrapperObject + * @build compiler.jvmci.compilerToVM.GetConstantPoolTest * @build sun.hotspot.WhiteBox - * compiler.jvmci.compilerToVM.GetConstantPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * jdk.vm.ci.hotspot.PublicMetaspaceWrapperObject - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions - * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.GetConstantPoolTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.GetConstantPoolTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetExceptionTableTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetExceptionTableTest.java index e8630c10d0f..63f44334e4e 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetExceptionTableTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetExceptionTableTest.java @@ -27,11 +27,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.GetExceptionTableTest * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.GetExceptionTableTest + * compiler.jvmci.compilerToVM.GetExceptionTableTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetImplementorTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetImplementorTest.java index fd88f772cba..1b03ef2e79e 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetImplementorTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetImplementorTest.java @@ -26,12 +26,12 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib/ - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.GetImplementorTest - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.GetImplementorTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.GetImplementorTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetLineNumberTableTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetLineNumberTableTest.java index 8ca2f535bbc..99fcd9fae21 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetLineNumberTableTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetLineNumberTableTest.java @@ -27,11 +27,16 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper + * @library ../common/patches + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.GetLineNumberTableTest * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.GetLineNumberTableTest + * compiler.jvmci.compilerToVM.GetLineNumberTableTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetLocalVariableTableTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetLocalVariableTableTest.java index d967e60a7d1..958b2e48fe4 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetLocalVariableTableTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetLocalVariableTableTest.java @@ -27,15 +27,19 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code * @clean compiler.jvmci.compilerToVM.* * @compile -g DummyInterface.java * @compile -g DummyAbstractClass.java * @compile -g DummyClass.java - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.GetLocalVariableTableTest * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.GetLocalVariableTableTest + * compiler.jvmci.compilerToVM.GetLocalVariableTableTest * @clean compiler.jvmci.compilerToVM.* */ diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetMaxCallTargetOffsetTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetMaxCallTargetOffsetTest.java index b803f550600..c180d41d548 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetMaxCallTargetOffsetTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetMaxCallTargetOffsetTest.java @@ -26,12 +26,12 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib/ - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.GetMaxCallTargetOffsetTest - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.GetMaxCallTargetOffsetTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.GetMaxCallTargetOffsetTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetNextStackFrameTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetNextStackFrameTest.java index 830ff7b066a..44fbb4cdaef 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetNextStackFrameTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetNextStackFrameTest.java @@ -26,11 +26,16 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.GetNextStackFrameTest + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.GetNextStackFrameTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.GetNextStackFrameTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaMethodAtSlotTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaMethodAtSlotTest.java index 3f578e9883d..87afff2df3b 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaMethodAtSlotTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaMethodAtSlotTest.java @@ -27,11 +27,16 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.GetNextStackFrameTest * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.GetResolvedJavaMethodAtSlotTest + * compiler.jvmci.compilerToVM.GetResolvedJavaMethodAtSlotTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaMethodTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaMethodTest.java index 5a98e880f9c..4735af3ee3e 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaMethodTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaMethodTest.java @@ -26,17 +26,18 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java - * ../common/PublicMetaspaceWrapperObject.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * jdk.vm.ci/jdk.vm.ci.hotspot.PublicMetaspaceWrapperObject * @build compiler.jvmci.compilerToVM.GetResolvedJavaMethodTest * @run main ClassFileInstaller * sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * jdk.vm.ci.hotspot.PublicMetaspaceWrapperObject - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI - * compiler.jvmci.compilerToVM.GetResolvedJavaMethodTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.GetResolvedJavaMethodTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaTypeTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaTypeTest.java index 50b1a60acb8..0ddc246aec3 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaTypeTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetResolvedJavaTypeTest.java @@ -26,22 +26,26 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java - * ../common/PublicMetaspaceWrapperObject.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * jdk.vm.ci/jdk.vm.ci.hotspot.PublicMetaspaceWrapperObject * @build compiler.jvmci.compilerToVM.GetResolvedJavaTypeTest + * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller * sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * jdk.vm.ci.hotspot.PublicMetaspaceWrapperObject - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI - * -XX:+UseCompressedOops - * compiler.jvmci.compilerToVM.GetResolvedJavaTypeTest - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI - * -XX:-UseCompressedOops - * compiler.jvmci.compilerToVM.GetResolvedJavaTypeTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:+UseCompressedOops + * compiler.jvmci.compilerToVM.GetResolvedJavaTypeTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:-UseCompressedOops + * compiler.jvmci.compilerToVM.GetResolvedJavaTypeTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetStackTraceElementTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetStackTraceElementTest.java index 5554bc3914b..a6feb931eb0 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetStackTraceElementTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetStackTraceElementTest.java @@ -27,10 +27,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.GetStackTraceElementTest * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. compiler.jvmci.compilerToVM.GetStackTraceElementTest + * compiler.jvmci.compilerToVM.GetStackTraceElementTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetSymbolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetSymbolTest.java index e550282d5b0..9e1f82998ca 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetSymbolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetSymbolTest.java @@ -26,17 +26,16 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.GetSymbolTest - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper - * compiler.jvmci.common.testcases.SingleImplementer - * compiler.jvmci.common.testcases.SingleImplementerInterface - * compiler.jvmci.compilerToVM.GetSymbolTest - * compiler.jvmci.common.CTVMUtilities - * jdk.test.lib.Utils - * jdk.test.lib.Asserts * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. compiler.jvmci.compilerToVM.GetSymbolTest + * compiler.jvmci.compilerToVM.GetSymbolTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/GetVtableIndexForInterfaceTest.java b/hotspot/test/compiler/jvmci/compilerToVM/GetVtableIndexForInterfaceTest.java index f97a44750a6..49fd32128df 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/GetVtableIndexForInterfaceTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/GetVtableIndexForInterfaceTest.java @@ -26,12 +26,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.GetVtableIndexForInterfaceTest - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.GetVtableIndexForInterfaceTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.GetVtableIndexForInterfaceTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/HasCompiledCodeForOSRTest.java b/hotspot/test/compiler/jvmci/compilerToVM/HasCompiledCodeForOSRTest.java index cf839d71059..e45940e3e6a 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/HasCompiledCodeForOSRTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/HasCompiledCodeForOSRTest.java @@ -27,15 +27,21 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.HasCompiledCodeForOSRTest * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * -XX:-BackgroundCompilation - * compiler.jvmci.compilerToVM.HasCompiledCodeForOSRTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:-BackgroundCompilation + * compiler.jvmci.compilerToVM.HasCompiledCodeForOSRTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/HasFinalizableSubclassTest.java b/hotspot/test/compiler/jvmci/compilerToVM/HasFinalizableSubclassTest.java index ce6f1a830f9..066e78448de 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/HasFinalizableSubclassTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/HasFinalizableSubclassTest.java @@ -26,11 +26,12 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.HasFinalizableSubclassTest - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.HasFinalizableSubclassTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.HasFinalizableSubclassTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/InitializeConfigurationTest.java b/hotspot/test/compiler/jvmci/compilerToVM/InitializeConfigurationTest.java index 2a2268ab38c..3de8ec56166 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/InitializeConfigurationTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/InitializeConfigurationTest.java @@ -26,13 +26,12 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.InitializeConfigurationTest - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * compiler.jvmci.compilerToVM.InitializeConfigurationTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.InitializeConfigurationTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/InvalidateInstalledCodeTest.java b/hotspot/test/compiler/jvmci/compilerToVM/InvalidateInstalledCodeTest.java index a8eac1de6cd..b7a0cfd0c19 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/InvalidateInstalledCodeTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/InvalidateInstalledCodeTest.java @@ -27,16 +27,22 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.runtime * @ignore 8139700 - * @compile ../common/CompilerToVMHelper.java + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.InvalidateInstalledCodeTest * @build sun.hotspot.WhiteBox - * compiler.jvmci.compilerToVM.InvalidateInstalledCodeTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.InvalidateInstalledCodeTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.InvalidateInstalledCodeTest */ package compiler.jvmci.compilerToVM; @@ -86,6 +92,7 @@ public class InvalidateInstalledCodeTest { CompilationResult compResult = new CompilationResult(name); // to pass sanity check of default -1 compResult.setTotalFrameSize(0); + compResult.close(); InstalledCode installedCode = CACHE_PROVIDER.installCode( compRequest, compResult, new InstalledCode(name), /* speculationLog = */ null, diff --git a/hotspot/test/compiler/jvmci/compilerToVM/IsMatureTest.java b/hotspot/test/compiler/jvmci/compilerToVM/IsMatureTest.java index a8004268fd2..f6ebb8303a6 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/IsMatureTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/IsMatureTest.java @@ -26,14 +26,17 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java - * @build sun.hotspot.WhiteBox IsMatureTest + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.IsMatureTest + * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions - * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * compiler.jvmci.compilerToVM.IsMatureTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.IsMatureTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/LookupKlassInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/LookupKlassInPoolTest.java index 09893c05008..21be817e7db 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/LookupKlassInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/LookupKlassInPoolTest.java @@ -28,15 +28,21 @@ * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @summary Testing compiler.jvmci.CompilerToVM.lookupKlassInPool method * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.LookupKlassInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions - * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.LookupKlassInPoolTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * compiler.jvmci.compilerToVM.LookupKlassInPoolTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/LookupKlassRefIndexInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/LookupKlassRefIndexInPoolTest.java index e1dbe9f9494..b17763bd3ac 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/LookupKlassRefIndexInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/LookupKlassRefIndexInPoolTest.java @@ -27,12 +27,17 @@ * @bug 8138708 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.LookupKlassRefIndexInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * compiler.jvmci.compilerToVM.LookupKlassRefIndexInPoolTest diff --git a/hotspot/test/compiler/jvmci/compilerToVM/LookupMethodInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/LookupMethodInPoolTest.java index a1e71bb2b3a..1c23c7458c1 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/LookupMethodInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/LookupMethodInPoolTest.java @@ -27,12 +27,17 @@ * @bug 8138708 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.LookupMethodInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * compiler.jvmci.compilerToVM.LookupMethodInPoolTest diff --git a/hotspot/test/compiler/jvmci/compilerToVM/LookupNameAndTypeRefIndexInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/LookupNameAndTypeRefIndexInPoolTest.java index 0eea14aa8d4..16add7012e0 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/LookupNameAndTypeRefIndexInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/LookupNameAndTypeRefIndexInPoolTest.java @@ -27,12 +27,17 @@ * @bug 8138708 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.LookupNameAndTypeRefIndexInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * compiler.jvmci.compilerToVM.LookupNameAndTypeRefIndexInPoolTest diff --git a/hotspot/test/compiler/jvmci/compilerToVM/LookupNameInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/LookupNameInPoolTest.java index 09db06c8a77..9dc255210fb 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/LookupNameInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/LookupNameInPoolTest.java @@ -27,12 +27,17 @@ * @bug 8138708 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.LookupNameInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * compiler.jvmci.compilerToVM.LookupNameInPoolTest diff --git a/hotspot/test/compiler/jvmci/compilerToVM/LookupSignatureInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/LookupSignatureInPoolTest.java index 00f55833891..adcf6cb85f4 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/LookupSignatureInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/LookupSignatureInPoolTest.java @@ -27,12 +27,17 @@ * @bug 8138708 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.LookupSignatureInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * compiler.jvmci.compilerToVM.LookupSignatureInPoolTest diff --git a/hotspot/test/compiler/jvmci/compilerToVM/LookupTypeTest.java b/hotspot/test/compiler/jvmci/compilerToVM/LookupTypeTest.java index 2f6efe1795f..269c5a7d39c 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/LookupTypeTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/LookupTypeTest.java @@ -26,12 +26,12 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.LookupTypeTest - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.LookupTypeTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.LookupTypeTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/MaterializeVirtualObjectTest.java b/hotspot/test/compiler/jvmci/compilerToVM/MaterializeVirtualObjectTest.java index 326f24e92fb..e28174905a2 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/MaterializeVirtualObjectTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/MaterializeVirtualObjectTest.java @@ -26,22 +26,34 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.meta * @ignore 8139703 - * @compile ../common/CompilerToVMHelper.java - * @build sun.hotspot.WhiteBox MaterializeVirtualObjectTest + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.MaterializeVirtualObjectTest + * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions - * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:CompileCommand=exclude,*::check -XX:+DoEscapeAnalysis -Xbatch - * -Dcompiler.jvmci.compilerToVM.MaterializeVirtualObjectTest.invalidate=false - * compiler.jvmci.compilerToVM.MaterializeVirtualObjectTest - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions - * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:CompileCommand=exclude,*::check -XX:+DoEscapeAnalysis -Xbatch - * -Dcompiler.jvmci.compilerToVM.MaterializeVirtualObjectTest.invalidate=true - * compiler.jvmci.compilerToVM.MaterializeVirtualObjectTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:CompileCommand=exclude,*::check + * -XX:+DoEscapeAnalysis + * -Xbatch + * -Dcompiler.jvmci.compilerToVM.MaterializeVirtualObjectTest.invalidate=false + * compiler.jvmci.compilerToVM.MaterializeVirtualObjectTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:CompileCommand=exclude,*::check + * -XX:+DoEscapeAnalysis + * -Xbatch + * -Dcompiler.jvmci.compilerToVM.MaterializeVirtualObjectTest.invalidate=true + * compiler.jvmci.compilerToVM.MaterializeVirtualObjectTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/MethodIsIgnoredBySecurityStackWalkTest.java b/hotspot/test/compiler/jvmci/compilerToVM/MethodIsIgnoredBySecurityStackWalkTest.java index 3d00dae0cd7..b63fa7af5ee 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/MethodIsIgnoredBySecurityStackWalkTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/MethodIsIgnoredBySecurityStackWalkTest.java @@ -27,11 +27,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java - * @run main ClassFileInstaller jdk.vm.ci.hotspot.CompilerToVMHelper + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.MethodIsIgnoredBySecurityStackWalkTest * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.MethodIsIgnoredBySecurityStackWalkTest + * compiler.jvmci.compilerToVM.MethodIsIgnoredBySecurityStackWalkTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ReadUncompressedOopTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ReadUncompressedOopTest.java index 5ff5913d6fe..4ec17c2e71a 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ReadUncompressedOopTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ReadUncompressedOopTest.java @@ -26,22 +26,24 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib/ - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.ReadUncompressedOopTest + * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller * sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI - * -XX:-UseCompressedOops - * compiler.jvmci.compilerToVM.ReadUncompressedOopTest + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:-UseCompressedOops + * compiler.jvmci.compilerToVM.ReadUncompressedOopTest * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI - * -XX:+UseCompressedOops - * compiler.jvmci.compilerToVM.ReadUncompressedOopTest + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:+UseCompressedOops + * compiler.jvmci.compilerToVM.ReadUncompressedOopTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ReprofileTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ReprofileTest.java index bc3e781086d..1e8cb12dd89 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ReprofileTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ReprofileTest.java @@ -27,16 +27,23 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox + * @build compiler.jvmci.compilerToVM.ReprofileTest * @run main ClassFileInstaller * sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * -Xmixed - * compiler.jvmci.compilerToVM.ReprofileTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -Xmixed + * compiler.jvmci.compilerToVM.ReprofileTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ResolveConstantInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ResolveConstantInPoolTest.java index 2c37309923d..285f0ff77af 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ResolveConstantInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ResolveConstantInPoolTest.java @@ -27,14 +27,19 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.ResolveConstantInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions - * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI * compiler.jvmci.compilerToVM.ResolveConstantInPoolTest */ diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ResolveFieldInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ResolveFieldInPoolTest.java index 9e99842110f..2d7145dec15 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ResolveFieldInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ResolveFieldInPoolTest.java @@ -27,12 +27,17 @@ * @bug 8138708 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.ResolveFieldInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * compiler.jvmci.compilerToVM.ResolveFieldInPoolTest diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ResolveMethodTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ResolveMethodTest.java index ba43e6b08ff..ff08b7e53bf 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ResolveMethodTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ResolveMethodTest.java @@ -26,12 +26,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.ResolveMethodTest - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.ResolveMethodTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.ResolveMethodTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ResolvePossiblyCachedConstantInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ResolvePossiblyCachedConstantInPoolTest.java index 6f0af4c2ce1..1e0f269e8cd 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ResolvePossiblyCachedConstantInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ResolvePossiblyCachedConstantInPoolTest.java @@ -27,12 +27,17 @@ * @bug 8138708 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.ResolvePossiblyCachedConstantInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * compiler.jvmci.compilerToVM.ResolvePossiblyCachedConstantInPoolTest diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ResolveTypeInPoolTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ResolveTypeInPoolTest.java index 1e5fe57f369..cad8ccd9f31 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ResolveTypeInPoolTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ResolveTypeInPoolTest.java @@ -28,15 +28,20 @@ * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @summary Testing compiler.jvmci.CompilerToVM.resolveTypeInPool method * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.misc + * java.base/jdk.internal.org.objectweb.asm + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.meta + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build sun.hotspot.WhiteBox * compiler.jvmci.compilerToVM.ResolveTypeInPoolTest * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions - * -XX:+WhiteBoxAPI -XX:+UnlockExperimentalVMOptions - * -XX:+EnableJVMCI compiler.jvmci.compilerToVM.ResolveTypeInPoolTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * compiler.jvmci.compilerToVM.ResolveTypeInPoolTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ShouldDebugNonSafepointsTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ShouldDebugNonSafepointsTest.java index 86f9cef8ec0..69ceeffdf7d 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ShouldDebugNonSafepointsTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ShouldDebugNonSafepointsTest.java @@ -26,22 +26,20 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary /test/lib/ - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest - * @run main ClassFileInstaller - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm - * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions - * -XX:+DebugNonSafepoints - * -Dcompiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest.expected=true - * compiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest - * @run main/othervm - * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions - * -XX:-DebugNonSafepoints - * -Dcompiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest.expected=false - * compiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:+UnlockDiagnosticVMOptions + * -XX:+DebugNonSafepoints + * -Dcompiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest.expected=true + * compiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest + * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * -XX:+UnlockDiagnosticVMOptions + * -XX:-DebugNonSafepoints + * -Dcompiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest.expected=false + * compiler.jvmci.compilerToVM.ShouldDebugNonSafepointsTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/compilerToVM/ShouldInlineMethodTest.java b/hotspot/test/compiler/jvmci/compilerToVM/ShouldInlineMethodTest.java index a034166604a..615b5859df6 100644 --- a/hotspot/test/compiler/jvmci/compilerToVM/ShouldInlineMethodTest.java +++ b/hotspot/test/compiler/jvmci/compilerToVM/ShouldInlineMethodTest.java @@ -27,14 +27,20 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library /testlibrary /test/lib / - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper + * @build compiler.jvmci.compilerToVM.ShouldInlineMethodTest * @build sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * jdk.vm.ci.hotspot.CompilerToVMHelper - * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xbootclasspath/a:. - * compiler.jvmci.compilerToVM.ShouldInlineMethodTest + * @run main/othervm -Xbootclasspath/a:. + * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI + * compiler.jvmci.compilerToVM.ShouldInlineMethodTest */ package compiler.jvmci.compilerToVM; diff --git a/hotspot/test/compiler/jvmci/events/JvmciCreateMetaAccessContextTest.java b/hotspot/test/compiler/jvmci/events/JvmciCreateMetaAccessContextTest.java index 810404bf8ef..c8bfbff88d1 100644 --- a/hotspot/test/compiler/jvmci/events/JvmciCreateMetaAccessContextTest.java +++ b/hotspot/test/compiler/jvmci/events/JvmciCreateMetaAccessContextTest.java @@ -26,7 +26,12 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary - * @compile ./MetaAccessWrapper.java + * @library ../common/patches + * @modules jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci/jdk.vm.ci.hotspot.MetaAccessWrapper * @build compiler.jvmci.common.JVMCIHelpers * compiler.jvmci.events.JvmciCreateMetaAccessContextTest * @run main jdk.test.lib.FileInstaller ../common/services/ ./META-INF/services/ @@ -37,7 +42,6 @@ * compiler.jvmci.common.JVMCIHelpers$EmptyHotspotCompiler * compiler.jvmci.common.JVMCIHelpers$EmptyCompilerFactory * compiler.jvmci.events.JvmciCreateMetaAccessContextTest - * jdk.vm.ci.hotspot.MetaAccessWrapper * jdk.test.lib.Asserts * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * -Xbootclasspath/a:. diff --git a/hotspot/test/compiler/jvmci/events/JvmciNotifyInstallEventTest.java b/hotspot/test/compiler/jvmci/events/JvmciNotifyInstallEventTest.java index 75c808e34cc..5e6f972a883 100644 --- a/hotspot/test/compiler/jvmci/events/JvmciNotifyInstallEventTest.java +++ b/hotspot/test/compiler/jvmci/events/JvmciNotifyInstallEventTest.java @@ -26,7 +26,15 @@ * @bug 8136421 * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") * @library / /testlibrary - * @compile ../common/CompilerToVMHelper.java + * @library ../common/patches + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.org.objectweb.asm.tree + * jdk.vm.ci/jdk.vm.ci.hotspot + * jdk.vm.ci/jdk.vm.ci.code + * jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @ignore 8144964 + * @build jdk.vm.ci/jdk.vm.ci.hotspot.CompilerToVMHelper * @build compiler.jvmci.common.JVMCIHelpers * compiler.jvmci.events.JvmciNotifyInstallEventTest * @run main jdk.test.lib.FileInstaller ../common/services/ ./META-INF/services/ @@ -38,7 +46,6 @@ * compiler.jvmci.events.JvmciNotifyInstallEventTest * compiler.jvmci.common.CTVMUtilities * compiler.jvmci.common.testcases.SimpleClass - * jdk.vm.ci.hotspot.CompilerToVMHelper * jdk.test.lib.Asserts * @run main/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI * -Djvmci.compiler=EmptyCompiler -Xbootclasspath/a:. -Xmixed diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/ConstantTest.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/ConstantTest.java index 5be56d0c33a..761db0e7a06 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/ConstantTest.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/ConstantTest.java @@ -24,10 +24,13 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile ConstantTest.java FieldUniverse.java TypeUniverse.java TestMetaAccessProvider.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.ConstantTest * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.ConstantTest */ - +// * @compile ConstantTest.java FieldUniverse.java TypeUniverse.java TestMetaAccessProvider.java package jdk.vm.ci.runtime.test; import jdk.vm.ci.meta.JavaConstant; diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/RedefineClassTest.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/RedefineClassTest.java index 11392414300..038efdcb2d2 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/RedefineClassTest.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/RedefineClassTest.java @@ -24,7 +24,10 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile RedefineClassTest.java TypeUniverse.java TestMetaAccessProvider.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.RedefineClassTest * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.RedefineClassTest */ diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestConstantReflectionProvider.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestConstantReflectionProvider.java index dbde990b71c..fe31f5907be 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestConstantReflectionProvider.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestConstantReflectionProvider.java @@ -24,7 +24,10 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile TestConstantReflectionProvider.java TypeUniverse.java TestMetaAccessProvider.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.TestConstantReflectionProvider * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.TestConstantReflectionProvider */ diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaField.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaField.java index d4c3a58db5d..46f4e80ac89 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaField.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaField.java @@ -24,7 +24,10 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile TestJavaField.java FieldUniverse.java TypeUniverse.java TestMetaAccessProvider.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.TestJavaField * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.TestJavaField */ diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaMethod.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaMethod.java index 50e7b2ed3ee..460abd992af 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaMethod.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaMethod.java @@ -24,7 +24,10 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile TestJavaMethod.java MethodUniverse.java TypeUniverse.java TestMetaAccessProvider.java NameAndSignature.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.TestJavaMethod * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.TestJavaMethod */ diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaType.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaType.java index 311c96ac5af..72506cf0933 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaType.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestJavaType.java @@ -24,7 +24,10 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile TestJavaType.java TypeUniverse.java TestMetaAccessProvider.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.TestJavaType * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.TestJavaType */ diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestMetaAccessProvider.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestMetaAccessProvider.java index 9c1974035cf..10b2e70c334 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestMetaAccessProvider.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestMetaAccessProvider.java @@ -24,7 +24,10 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile TestMetaAccessProvider.java TypeUniverse.java TestMetaAccessProvider.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.TestMetaAccessProvider * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.TestMetaAccessProvider */ diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaField.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaField.java index fb960c00af7..71e9f188c43 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaField.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaField.java @@ -24,7 +24,10 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile TestResolvedJavaField.java FieldUniverse.java TypeUniverse.java TestMetaAccessProvider.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.TestResolvedJavaField * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.TestResolvedJavaField */ diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaMethod.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaMethod.java index 13dd0999fee..c60a7d45e42 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaMethod.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaMethod.java @@ -24,7 +24,10 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile TestResolvedJavaMethod.java MethodUniverse.java TypeUniverse.java TestMetaAccessProvider.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * @build jdk.vm.ci.runtime.test.TestResolvedJavaMethod * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.TestResolvedJavaMethod */ diff --git a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java index 8c869808338..2011bc0c863 100644 --- a/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java +++ b/hotspot/test/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java @@ -24,7 +24,11 @@ /** * @test * @requires (os.simpleArch == "x64" | os.simpleArch == "sparcv9" | os.simpleArch == "aarch64") - * @compile TestResolvedJavaType.java TypeUniverse.java TestMetaAccessProvider.java NameAndSignature.java + * @library ../../../../../ + * @modules jdk.vm.ci/jdk.vm.ci.meta + * jdk.vm.ci/jdk.vm.ci.runtime + * jdk.vm.ci/jdk.vm.ci.common + * @build jdk.vm.ci.runtime.test.TestResolvedJavaType * @run junit/othervm -XX:+UnlockExperimentalVMOptions -XX:+EnableJVMCI jdk.vm.ci.runtime.test.TestResolvedJavaType */ diff --git a/hotspot/test/compiler/stable/StableConfiguration.java b/hotspot/test/compiler/stable/StableConfiguration.java index be69081640c..db7e9a0f742 100644 --- a/hotspot/test/compiler/stable/StableConfiguration.java +++ b/hotspot/test/compiler/stable/StableConfiguration.java @@ -22,16 +22,17 @@ * or visit www.oracle.com if you need additional information or have any * questions. */ -package java.lang.invoke; + +package compiler.stable; + +import sun.hotspot.WhiteBox; import java.lang.reflect.Method; -import java.util.Properties; -import sun.hotspot.WhiteBox; public class StableConfiguration { static final WhiteBox WB = WhiteBox.getWhiteBox(); - static final boolean isStableEnabled; - static final boolean isServerWithStable; + public static final boolean isStableEnabled; + public static final boolean isServerWithStable; static { Boolean value = WB.getBooleanVMFlag("FoldStableValues"); @@ -60,8 +61,6 @@ public class StableConfiguration { static void get1() { } - - // ::get() is among immediately compiled methods. static boolean get() { try { @@ -80,5 +79,4 @@ public class StableConfiguration { throw new Error(e); } } - } diff --git a/hotspot/test/compiler/stable/TestStableBoolean.java b/hotspot/test/compiler/stable/TestStableBoolean.java index 168ddf48da7..ed8d18b18fa 100644 --- a/hotspot/test/compiler/stable/TestStableBoolean.java +++ b/hotspot/test/compiler/stable/TestStableBoolean.java @@ -26,64 +26,37 @@ /* * @test TestStableBoolean * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableBoolean StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableBoolean - * java/lang/invoke/TestStableBoolean$BooleanStable - * java/lang/invoke/TestStableBoolean$StaticBooleanStable - * java/lang/invoke/TestStableBoolean$VolatileBooleanStable - * java/lang/invoke/TestStableBoolean$BooleanArrayDim1 - * java/lang/invoke/TestStableBoolean$BooleanArrayDim2 - * java/lang/invoke/TestStableBoolean$BooleanArrayDim3 - * java/lang/invoke/TestStableBoolean$BooleanArrayDim4 - * java/lang/invoke/TestStableBoolean$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableBoolean$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableBoolean$NestedStableField - * java/lang/invoke/TestStableBoolean$NestedStableField$A - * java/lang/invoke/TestStableBoolean$NestedStableField1 - * java/lang/invoke/TestStableBoolean$NestedStableField1$A - * java/lang/invoke/TestStableBoolean$NestedStableField2 - * java/lang/invoke/TestStableBoolean$NestedStableField2$A - * java/lang/invoke/TestStableBoolean$NestedStableField3 - * java/lang/invoke/TestStableBoolean$NestedStableField3$A - * java/lang/invoke/TestStableBoolean$DefaultValue - * java/lang/invoke/TestStableBoolean$DefaultStaticValue - * java/lang/invoke/TestStableBoolean$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableBoolean * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableBoolean - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableBoolean - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableBoolean - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableBoolean + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableBoolean + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:-FoldStableValues + * compiler.stable.TestStableBoolean * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 + * -XX:+FoldStableValues + * compiler.stable.TestStableBoolean + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 + * -XX:-FoldStableValues + * compiler.stable.TestStableBoolean */ -package java.lang.invoke; + +package compiler.stable; import jdk.internal.vm.annotation.Stable; - import java.lang.reflect.InvocationTargetException; public class TestStableBoolean { @@ -122,12 +95,13 @@ public class TestStableBoolean { /* ==================================================== */ static class DefaultValue { - public @Stable boolean v; + public @Stable + boolean v; public static final DefaultValue c = new DefaultValue(); public static boolean get() { return c.v; } public static void test() throws Exception { - boolean val1 = get(); + boolean val1 = get(); c.v = true; boolean val2 = get(); assertEquals(val1, false); assertEquals(val2, true); @@ -157,7 +131,7 @@ public class TestStableBoolean { public static final DefaultStaticValue c = new DefaultStaticValue(); public static boolean get() { return c.v; } public static void test() throws Exception { - boolean val1 = get(); + boolean val1 = get(); c.v = true; boolean val2 = get(); assertEquals(val1, false); assertEquals(val2, true); @@ -207,14 +181,14 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new boolean[1]; c.v[0] = true; boolean val1 = get(); - c.v[0] = false; boolean val2 = get(); + c.v[0] = false; boolean val2 = get(); assertEquals(val1, true); assertEquals(val2, (isServerWithStable ? true : false)); } { c.v = new boolean[20]; c.v[10] = true; boolean val1 = get1(); - c.v[10] = false; boolean val2 = get1(); + c.v[10] = false; boolean val2 = get1(); assertEquals(val1, true); assertEquals(val2, (isServerWithStable ? true : false)); } @@ -239,7 +213,7 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new boolean[1][1]; c.v[0][0] = true; boolean val1 = get(); - c.v[0][0] = false; boolean val2 = get(); + c.v[0][0] = false; boolean val2 = get(); assertEquals(val1, true); assertEquals(val2, (isServerWithStable ? true : false)); @@ -277,7 +251,7 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new boolean[1][1][1]; c.v[0][0][0] = true; boolean val1 = get(); - c.v[0][0][0] = false; boolean val2 = get(); + c.v[0][0][0] = false; boolean val2 = get(); assertEquals(val1, true); assertEquals(val2, (isServerWithStable ? true : false)); @@ -325,7 +299,7 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new boolean[1][1][1][1]; c.v[0][0][0][0] = true; boolean val1 = get(); - c.v[0][0][0][0] = false; boolean val2 = get(); + c.v[0][0][0][0] = false; boolean val2 = get(); assertEquals(val1, true); assertEquals(val2, (isServerWithStable ? true : false)); @@ -383,7 +357,7 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new boolean[1]; ((boolean[])c.v)[0] = true; boolean val1 = get(); - ((boolean[])c.v)[0] = false; boolean val2 = get(); + ((boolean[])c.v)[0] = false; boolean val2 = get(); assertEquals(val1, true); assertEquals(val2, false); @@ -410,7 +384,7 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new boolean[1][1]; ((boolean[][])c.v)[0][0] = true; boolean val1 = get(); - ((boolean[][])c.v)[0][0] = false; boolean val2 = get(); + ((boolean[][])c.v)[0][0] = false; boolean val2 = get(); assertEquals(val1, true); assertEquals(val2, false); @@ -418,7 +392,7 @@ public class TestStableBoolean { { c.v = new boolean[1][1]; c.v[0] = new boolean[0]; boolean[] val1 = get1(); - c.v[0] = new boolean[0]; boolean[] val2 = get1(); + c.v[0] = new boolean[0]; boolean[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -446,7 +420,7 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new boolean[1][1][1]; ((boolean[][][])c.v)[0][0][0] = true; boolean val1 = get(); - ((boolean[][][])c.v)[0][0][0] = false; boolean val2 = get(); + ((boolean[][][])c.v)[0][0][0] = false; boolean val2 = get(); assertEquals(val1, true); assertEquals(val2, false); @@ -454,14 +428,14 @@ public class TestStableBoolean { { c.v = new boolean[1][1][1]; c.v[0][0] = new boolean[0]; boolean[] val1 = get1(); - c.v[0][0] = new boolean[0]; boolean[] val2 = get1(); + c.v[0][0] = new boolean[0]; boolean[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new boolean[1][1][1]; c.v[0] = new boolean[0][0]; boolean[][] val1 = get2(); - c.v[0] = new boolean[0][0]; boolean[][] val2 = get2(); + c.v[0] = new boolean[0][0]; boolean[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -491,7 +465,7 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new A(); c.v.a = true; A val1 = get(); - c.v.a = false; A val2 = get(); + c.v.a = false; A val2 = get(); assertEquals(val1.a, false); assertEquals(val2.a, false); @@ -499,7 +473,7 @@ public class TestStableBoolean { { c.v = new A(); c.v.a = true; boolean val1 = get1(); - c.v.a = false; boolean val2 = get1(); + c.v.a = false; boolean val2 = get1(); c.v = new A(); c.v.a = false; boolean val3 = get1(); assertEquals(val1, true); @@ -525,8 +499,8 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = true; c.v.next.a = true; A val1 = get(); - c.v.a = false; c.v.next.a = false; A val2 = get(); + c.v.a = true; c.v.next.a = true; A val1 = get(); + c.v.a = false; c.v.next.a = false; A val2 = get(); assertEquals(val1.a, false); assertEquals(val2.a, false); @@ -534,10 +508,10 @@ public class TestStableBoolean { { c.v = new A(); c.v.next = c.v; - c.v.a = true; boolean val1 = get1(); - c.v.a = false; boolean val2 = get1(); + c.v.a = true; boolean val1 = get1(); + c.v.a = false; boolean val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = false; boolean val3 = get1(); + c.v.a = false; boolean val3 = get1(); assertEquals(val1, true); assertEquals(val2, (isStableEnabled ? true : false)); @@ -563,8 +537,8 @@ public class TestStableBoolean { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = true; boolean val1 = get(); boolean val2 = get1(); - c.v.a = false; boolean val3 = get(); boolean val4 = get1(); + c.v.a = true; boolean val1 = get(); boolean val2 = get1(); + c.v.a = false; boolean val3 = get(); boolean val4 = get1(); assertEquals(val1, true); assertEquals(val3, (isStableEnabled ? true : false)); @@ -594,8 +568,8 @@ public class TestStableBoolean { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = true; boolean val1 = get(); boolean val2 = get1(); - elem.a = false; boolean val3 = get(); boolean val4 = get1(); + elem.a = true; boolean val1 = get(); boolean val2 = get1(); + elem.a = false; boolean val3 = get(); boolean val4 = get1(); assertEquals(val1, true); assertEquals(val3, (isServerWithStable ? true : false)); @@ -632,4 +606,5 @@ public class TestStableBoolean { } } } + } diff --git a/hotspot/test/compiler/stable/TestStableByte.java b/hotspot/test/compiler/stable/TestStableByte.java index 694205e8e55..14ce6e40135 100644 --- a/hotspot/test/compiler/stable/TestStableByte.java +++ b/hotspot/test/compiler/stable/TestStableByte.java @@ -26,64 +26,37 @@ /* * @test TestStableByte * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableByte StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableByte - * java/lang/invoke/TestStableByte$ByteStable - * java/lang/invoke/TestStableByte$StaticByteStable - * java/lang/invoke/TestStableByte$VolatileByteStable - * java/lang/invoke/TestStableByte$ByteArrayDim1 - * java/lang/invoke/TestStableByte$ByteArrayDim2 - * java/lang/invoke/TestStableByte$ByteArrayDim3 - * java/lang/invoke/TestStableByte$ByteArrayDim4 - * java/lang/invoke/TestStableByte$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableByte$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableByte$NestedStableField - * java/lang/invoke/TestStableByte$NestedStableField$A - * java/lang/invoke/TestStableByte$NestedStableField1 - * java/lang/invoke/TestStableByte$NestedStableField1$A - * java/lang/invoke/TestStableByte$NestedStableField2 - * java/lang/invoke/TestStableByte$NestedStableField2$A - * java/lang/invoke/TestStableByte$NestedStableField3 - * java/lang/invoke/TestStableByte$NestedStableField3$A - * java/lang/invoke/TestStableByte$DefaultValue - * java/lang/invoke/TestStableByte$DefaultStaticValue - * java/lang/invoke/TestStableByte$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableByte * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableByte - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableByte - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableByte - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableByte + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableByte + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableByte * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableByte + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableByte */ -package java.lang.invoke; + +package compiler.stable; import jdk.internal.vm.annotation.Stable; - import java.lang.reflect.InvocationTargetException; public class TestStableByte { @@ -127,7 +100,7 @@ public class TestStableByte { public static final DefaultValue c = new DefaultValue(); public static byte get() { return c.v; } public static void test() throws Exception { - byte val1 = get(); + byte val1 = get(); c.v = 1; byte val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1); @@ -157,7 +130,7 @@ public class TestStableByte { public static final DefaultStaticValue c = new DefaultStaticValue(); public static byte get() { return c.v; } public static void test() throws Exception { - byte val1 = get(); + byte val1 = get(); c.v = 1; byte val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1); @@ -207,24 +180,24 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new byte[1]; c.v[0] = 1; byte val1 = get(); - c.v[0] = 2; byte val2 = get(); + c.v[0] = 2; byte val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new byte[1]; c.v[0] = 3; byte val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); } { c.v = new byte[20]; c.v[10] = 1; byte val1 = get1(); - c.v[10] = 2; byte val2 = get1(); + c.v[10] = 2; byte val2 = get1(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new byte[20]; c.v[10] = 3; byte val3 = get1(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); } { @@ -247,17 +220,17 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new byte[1][1]; c.v[0][0] = 1; byte val1 = get(); - c.v[0][0] = 2; byte val2 = get(); + c.v[0][0] = 2; byte val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new byte[1][1]; c.v[0][0] = 3; byte val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new byte[1]; c.v[0][0] = 4; byte val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); } { @@ -287,21 +260,21 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new byte[1][1][1]; c.v[0][0][0] = 1; byte val1 = get(); - c.v[0][0][0] = 2; byte val2 = get(); + c.v[0][0][0] = 2; byte val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new byte[1][1][1]; c.v[0][0][0] = 3; byte val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new byte[1][1]; c.v[0][0][0] = 4; byte val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); c.v[0][0] = new byte[1]; c.v[0][0][0] = 5; byte val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 5)); + : 5)); } { @@ -338,25 +311,25 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new byte[1][1][1][1]; c.v[0][0][0][0] = 1; byte val1 = get(); - c.v[0][0][0][0] = 2; byte val2 = get(); + c.v[0][0][0][0] = 2; byte val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new byte[1][1][1][1]; c.v[0][0][0][0] = 3; byte val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new byte[1][1][1]; c.v[0][0][0][0] = 4; byte val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); c.v[0][0] = new byte[1][1]; c.v[0][0][0][0] = 5; byte val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 5)); + : 5)); c.v[0][0][0] = new byte[1]; c.v[0][0][0][0] = 6; byte val6 = get(); assertEquals(val6, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 6)); + : 6)); } { @@ -399,7 +372,7 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new byte[1]; ((byte[])c.v)[0] = 1; byte val1 = get(); - ((byte[])c.v)[0] = 2; byte val2 = get(); + ((byte[])c.v)[0] = 2; byte val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -426,7 +399,7 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new byte[1][1]; ((byte[][])c.v)[0][0] = 1; byte val1 = get(); - ((byte[][])c.v)[0][0] = 2; byte val2 = get(); + ((byte[][])c.v)[0][0] = 2; byte val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -434,7 +407,7 @@ public class TestStableByte { { c.v = new byte[1][1]; c.v[0] = new byte[0]; byte[] val1 = get1(); - c.v[0] = new byte[0]; byte[] val2 = get1(); + c.v[0] = new byte[0]; byte[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -462,7 +435,7 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new byte[1][1][1]; ((byte[][][])c.v)[0][0][0] = 1; byte val1 = get(); - ((byte[][][])c.v)[0][0][0] = 2; byte val2 = get(); + ((byte[][][])c.v)[0][0][0] = 2; byte val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -470,14 +443,14 @@ public class TestStableByte { { c.v = new byte[1][1][1]; c.v[0][0] = new byte[0]; byte[] val1 = get1(); - c.v[0][0] = new byte[0]; byte[] val2 = get1(); + c.v[0][0] = new byte[0]; byte[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new byte[1][1][1]; c.v[0] = new byte[0][0]; byte[][] val1 = get2(); - c.v[0] = new byte[0][0]; byte[][] val2 = get2(); + c.v[0] = new byte[0][0]; byte[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -507,7 +480,7 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new A(); c.v.a = 1; A val1 = get(); - c.v.a = 2; A val2 = get(); + c.v.a = 2; A val2 = get(); assertEquals(val1.a, 2); assertEquals(val2.a, 2); @@ -515,7 +488,7 @@ public class TestStableByte { { c.v = new A(); c.v.a = 1; byte val1 = get1(); - c.v.a = 2; byte val2 = get1(); + c.v.a = 2; byte val2 = get1(); c.v = new A(); c.v.a = 3; byte val3 = get1(); assertEquals(val1, 1); @@ -541,8 +514,8 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = 1; c.v.next.a = 1; A val1 = get(); - c.v.a = 2; c.v.next.a = 2; A val2 = get(); + c.v.a = 1; c.v.next.a = 1; A val1 = get(); + c.v.a = 2; c.v.next.a = 2; A val2 = get(); assertEquals(val1.a, 2); assertEquals(val2.a, 2); @@ -550,10 +523,10 @@ public class TestStableByte { { c.v = new A(); c.v.next = c.v; - c.v.a = 1; byte val1 = get1(); - c.v.a = 2; byte val2 = get1(); + c.v.a = 1; byte val1 = get1(); + c.v.a = 2; byte val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = 3; byte val3 = get1(); + c.v.a = 3; byte val3 = get1(); assertEquals(val1, 1); assertEquals(val2, (isStableEnabled ? 1 : 2)); @@ -579,8 +552,8 @@ public class TestStableByte { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = 1; byte val1 = get(); byte val2 = get1(); - c.v.a = 2; byte val3 = get(); byte val4 = get1(); + c.v.a = 1; byte val1 = get(); byte val2 = get1(); + c.v.a = 2; byte val3 = get(); byte val4 = get1(); assertEquals(val1, 1); assertEquals(val3, (isStableEnabled ? 1 : 2)); @@ -610,8 +583,8 @@ public class TestStableByte { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = 1; byte val1 = get(); byte val2 = get1(); - elem.a = 2; byte val3 = get(); byte val4 = get1(); + elem.a = 1; byte val1 = get(); byte val2 = get1(); + elem.a = 2; byte val3 = get(); byte val4 = get1(); assertEquals(val1, 1); assertEquals(val3, (isServerWithStable ? 1 : 2)); diff --git a/hotspot/test/compiler/stable/TestStableChar.java b/hotspot/test/compiler/stable/TestStableChar.java index d92dfb67c73..ba8e935fd35 100644 --- a/hotspot/test/compiler/stable/TestStableChar.java +++ b/hotspot/test/compiler/stable/TestStableChar.java @@ -26,65 +26,38 @@ /* * @test TestStableChar * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableChar StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableChar - * java/lang/invoke/TestStableChar$CharStable - * java/lang/invoke/TestStableChar$StaticCharStable - * java/lang/invoke/TestStableChar$VolatileCharStable - * java/lang/invoke/TestStableChar$CharArrayDim1 - * java/lang/invoke/TestStableChar$CharArrayDim2 - * java/lang/invoke/TestStableChar$CharArrayDim3 - * java/lang/invoke/TestStableChar$CharArrayDim4 - * java/lang/invoke/TestStableChar$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableChar$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableChar$NestedStableField - * java/lang/invoke/TestStableChar$NestedStableField$A - * java/lang/invoke/TestStableChar$NestedStableField1 - * java/lang/invoke/TestStableChar$NestedStableField1$A - * java/lang/invoke/TestStableChar$NestedStableField2 - * java/lang/invoke/TestStableChar$NestedStableField2$A - * java/lang/invoke/TestStableChar$NestedStableField3 - * java/lang/invoke/TestStableChar$NestedStableField3$A - * java/lang/invoke/TestStableChar$DefaultValue - * java/lang/invoke/TestStableChar$DefaultStaticValue - * java/lang/invoke/TestStableChar$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableChar * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableChar - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableChar - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableChar - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableChar + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableChar + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableChar * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableChar + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableChar */ -package java.lang.invoke; -import jdk.internal.vm.annotation.Stable; +package compiler.stable; import java.lang.reflect.InvocationTargetException; +import jdk.internal.vm.annotation.Stable; public class TestStableChar { static final boolean isStableEnabled = StableConfiguration.isStableEnabled; @@ -127,7 +100,7 @@ public class TestStableChar { public static final DefaultValue c = new DefaultValue(); public static char get() { return c.v; } public static void test() throws Exception { - char val1 = get(); + char val1 = get(); c.v = 'a'; char val2 = get(); assertEquals(val1, 0); assertEquals(val2, 'a'); @@ -157,7 +130,7 @@ public class TestStableChar { public static final DefaultStaticValue c = new DefaultStaticValue(); public static char get() { return c.v; } public static void test() throws Exception { - char val1 = get(); + char val1 = get(); c.v = 'a'; char val2 = get(); assertEquals(val1, 0); assertEquals(val2, 'a'); @@ -207,24 +180,24 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new char[1]; c.v[0] = 'a'; char val1 = get(); - c.v[0] = 'b'; char val2 = get(); + c.v[0] = 'b'; char val2 = get(); assertEquals(val1, 'a'); assertEquals(val2, (isServerWithStable ? 'a' : 'b')); c.v = new char[1]; c.v[0] = 'c'; char val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'c')); + : 'c')); } { c.v = new char[20]; c.v[10] = 'a'; char val1 = get1(); - c.v[10] = 'b'; char val2 = get1(); + c.v[10] = 'b'; char val2 = get1(); assertEquals(val1, 'a'); assertEquals(val2, (isServerWithStable ? 'a' : 'b')); c.v = new char[20]; c.v[10] = 'c'; char val3 = get1(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'c')); + : 'c')); } { @@ -247,17 +220,17 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new char[1][1]; c.v[0][0] = 'a'; char val1 = get(); - c.v[0][0] = 'b'; char val2 = get(); + c.v[0][0] = 'b'; char val2 = get(); assertEquals(val1, 'a'); assertEquals(val2, (isServerWithStable ? 'a' : 'b')); c.v = new char[1][1]; c.v[0][0] = 'c'; char val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'c')); + : 'c')); c.v[0] = new char[1]; c.v[0][0] = 'd'; char val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'd')); + : 'd')); } { @@ -287,21 +260,21 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new char[1][1][1]; c.v[0][0][0] = 'a'; char val1 = get(); - c.v[0][0][0] = 'b'; char val2 = get(); + c.v[0][0][0] = 'b'; char val2 = get(); assertEquals(val1, 'a'); assertEquals(val2, (isServerWithStable ? 'a' : 'b')); c.v = new char[1][1][1]; c.v[0][0][0] = 'c'; char val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'c')); + : 'c')); c.v[0] = new char[1][1]; c.v[0][0][0] = 'd'; char val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'd')); + : 'd')); c.v[0][0] = new char[1]; c.v[0][0][0] = 'e'; char val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'e')); + : 'e')); } { @@ -338,25 +311,25 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new char[1][1][1][1]; c.v[0][0][0][0] = 'a'; char val1 = get(); - c.v[0][0][0][0] = 'b'; char val2 = get(); + c.v[0][0][0][0] = 'b'; char val2 = get(); assertEquals(val1, 'a'); assertEquals(val2, (isServerWithStable ? 'a' : 'b')); c.v = new char[1][1][1][1]; c.v[0][0][0][0] = 'c'; char val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'c')); + : 'c')); c.v[0] = new char[1][1][1]; c.v[0][0][0][0] = 'd'; char val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'd')); + : 'd')); c.v[0][0] = new char[1][1]; c.v[0][0][0][0] = 'e'; char val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'e')); + : 'e')); c.v[0][0][0] = new char[1]; c.v[0][0][0][0] = 'f'; char val6 = get(); assertEquals(val6, (isStableEnabled ? (isServerWithStable ? 'a' : 'b') - : 'f')); + : 'f')); } { @@ -397,7 +370,7 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new char[1]; ((char[])c.v)[0] = 'a'; char val1 = get(); - ((char[])c.v)[0] = 'b'; char val2 = get(); + ((char[])c.v)[0] = 'b'; char val2 = get(); assertEquals(val1, 'a'); assertEquals(val2, 'b'); @@ -424,7 +397,7 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new char[1][1]; ((char[][])c.v)[0][0] = 'a'; char val1 = get(); - ((char[][])c.v)[0][0] = 'b'; char val2 = get(); + ((char[][])c.v)[0][0] = 'b'; char val2 = get(); assertEquals(val1, 'a'); assertEquals(val2, 'b'); @@ -432,7 +405,7 @@ public class TestStableChar { { c.v = new char[1][1]; c.v[0] = new char[0]; char[] val1 = get1(); - c.v[0] = new char[0]; char[] val2 = get1(); + c.v[0] = new char[0]; char[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -460,7 +433,7 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new char[1][1][1]; ((char[][][])c.v)[0][0][0] = 'a'; char val1 = get(); - ((char[][][])c.v)[0][0][0] = 'b'; char val2 = get(); + ((char[][][])c.v)[0][0][0] = 'b'; char val2 = get(); assertEquals(val1, 'a'); assertEquals(val2, 'b'); @@ -468,14 +441,14 @@ public class TestStableChar { { c.v = new char[1][1][1]; c.v[0][0] = new char[0]; char[] val1 = get1(); - c.v[0][0] = new char[0]; char[] val2 = get1(); + c.v[0][0] = new char[0]; char[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new char[1][1][1]; c.v[0] = new char[0][0]; char[][] val1 = get2(); - c.v[0] = new char[0][0]; char[][] val2 = get2(); + c.v[0] = new char[0][0]; char[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -505,7 +478,7 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new A(); c.v.a = 'a'; A val1 = get(); - c.v.a = 'b'; A val2 = get(); + c.v.a = 'b'; A val2 = get(); assertEquals(val1.a, 'b'); assertEquals(val2.a, 'b'); @@ -513,7 +486,7 @@ public class TestStableChar { { c.v = new A(); c.v.a = 'a'; char val1 = get1(); - c.v.a = 'b'; char val2 = get1(); + c.v.a = 'b'; char val2 = get1(); c.v = new A(); c.v.a = 'c'; char val3 = get1(); assertEquals(val1, 'a'); @@ -539,8 +512,8 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = 'a'; c.v.next.a = 'a'; A val1 = get(); - c.v.a = 'b'; c.v.next.a = 'b'; A val2 = get(); + c.v.a = 'a'; c.v.next.a = 'a'; A val1 = get(); + c.v.a = 'b'; c.v.next.a = 'b'; A val2 = get(); assertEquals(val1.a, 'b'); assertEquals(val2.a, 'b'); @@ -548,10 +521,10 @@ public class TestStableChar { { c.v = new A(); c.v.next = c.v; - c.v.a = 'a'; char val1 = get1(); - c.v.a = 'b'; char val2 = get1(); + c.v.a = 'a'; char val1 = get1(); + c.v.a = 'b'; char val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = 'c'; char val3 = get1(); + c.v.a = 'c'; char val3 = get1(); assertEquals(val1, 'a'); assertEquals(val2, (isStableEnabled ? 'a' : 'b')); @@ -577,8 +550,8 @@ public class TestStableChar { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = 'a'; char val1 = get(); char val2 = get1(); - c.v.a = 'b'; char val3 = get(); char val4 = get1(); + c.v.a = 'a'; char val1 = get(); char val2 = get1(); + c.v.a = 'b'; char val3 = get(); char val4 = get1(); assertEquals(val1, 'a'); assertEquals(val3, (isStableEnabled ? 'a' : 'b')); @@ -608,8 +581,8 @@ public class TestStableChar { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = 'a'; char val1 = get(); char val2 = get1(); - elem.a = 'b'; char val3 = get(); char val4 = get1(); + elem.a = 'a'; char val1 = get(); char val2 = get1(); + elem.a = 'b'; char val3 = get(); char val4 = get1(); assertEquals(val1, 'a'); assertEquals(val3, (isServerWithStable ? 'a' : 'b')); diff --git a/hotspot/test/compiler/stable/TestStableDouble.java b/hotspot/test/compiler/stable/TestStableDouble.java index 5e55a0f8597..c5ef689f365 100644 --- a/hotspot/test/compiler/stable/TestStableDouble.java +++ b/hotspot/test/compiler/stable/TestStableDouble.java @@ -26,65 +26,38 @@ /* * @test TestStableDouble * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableDouble StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableDouble - * java/lang/invoke/TestStableDouble$DoubleStable - * java/lang/invoke/TestStableDouble$StaticDoubleStable - * java/lang/invoke/TestStableDouble$VolatileDoubleStable - * java/lang/invoke/TestStableDouble$DoubleArrayDim1 - * java/lang/invoke/TestStableDouble$DoubleArrayDim2 - * java/lang/invoke/TestStableDouble$DoubleArrayDim3 - * java/lang/invoke/TestStableDouble$DoubleArrayDim4 - * java/lang/invoke/TestStableDouble$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableDouble$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableDouble$NestedStableField - * java/lang/invoke/TestStableDouble$NestedStableField$A - * java/lang/invoke/TestStableDouble$NestedStableField1 - * java/lang/invoke/TestStableDouble$NestedStableField1$A - * java/lang/invoke/TestStableDouble$NestedStableField2 - * java/lang/invoke/TestStableDouble$NestedStableField2$A - * java/lang/invoke/TestStableDouble$NestedStableField3 - * java/lang/invoke/TestStableDouble$NestedStableField3$A - * java/lang/invoke/TestStableDouble$DefaultValue - * java/lang/invoke/TestStableDouble$DefaultStaticValue - * java/lang/invoke/TestStableDouble$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableDouble * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableDouble - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableDouble - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableDouble - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableDouble + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableDouble + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableDouble * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableDouble + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableDouble */ -package java.lang.invoke; -import jdk.internal.vm.annotation.Stable; +package compiler.stable; import java.lang.reflect.InvocationTargetException; +import jdk.internal.vm.annotation.Stable; public class TestStableDouble { static final boolean isStableEnabled = StableConfiguration.isStableEnabled; @@ -127,7 +100,7 @@ public class TestStableDouble { public static final DefaultValue c = new DefaultValue(); public static double get() { return c.v; } public static void test() throws Exception { - double val1 = get(); + double val1 = get(); c.v = 1.0; double val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1.0); @@ -157,7 +130,7 @@ public class TestStableDouble { public static final DefaultStaticValue c = new DefaultStaticValue(); public static double get() { return c.v; } public static void test() throws Exception { - double val1 = get(); + double val1 = get(); c.v = 1.0; double val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1.0); @@ -207,24 +180,24 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new double[1]; c.v[0] = 1.0; double val1 = get(); - c.v[0] = 2.0; double val2 = get(); + c.v[0] = 2.0; double val2 = get(); assertEquals(val1, 1.0); assertEquals(val2, (isServerWithStable ? 1.0 : 2.0)); c.v = new double[1]; c.v[0] = 3.0; double val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 3.0)); + : 3.0)); } { c.v = new double[20]; c.v[10] = 1.0; double val1 = get1(); - c.v[10] = 2.0; double val2 = get1(); + c.v[10] = 2.0; double val2 = get1(); assertEquals(val1, 1.0); assertEquals(val2, (isServerWithStable ? 1.0 : 2.0)); c.v = new double[20]; c.v[10] = 3.0; double val3 = get1(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 3.0)); + : 3.0)); } { @@ -247,17 +220,17 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new double[1][1]; c.v[0][0] = 1.0; double val1 = get(); - c.v[0][0] = 2.0; double val2 = get(); + c.v[0][0] = 2.0; double val2 = get(); assertEquals(val1, 1.0); assertEquals(val2, (isServerWithStable ? 1.0 : 2.0)); c.v = new double[1][1]; c.v[0][0] = 3.0; double val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 3.0)); + : 3.0)); c.v[0] = new double[1]; c.v[0][0] = 4.0; double val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 4.0)); + : 4.0)); } { @@ -287,21 +260,21 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new double[1][1][1]; c.v[0][0][0] = 1.0; double val1 = get(); - c.v[0][0][0] = 2.0; double val2 = get(); + c.v[0][0][0] = 2.0; double val2 = get(); assertEquals(val1, 1.0); assertEquals(val2, (isServerWithStable ? 1.0 : 2.0)); c.v = new double[1][1][1]; c.v[0][0][0] = 3.0; double val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 3.0)); + : 3.0)); c.v[0] = new double[1][1]; c.v[0][0][0] = 4.0; double val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 4.0)); + : 4.0)); c.v[0][0] = new double[1]; c.v[0][0][0] = 5.0; double val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 5.0)); + : 5.0)); } { @@ -338,25 +311,25 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new double[1][1][1][1]; c.v[0][0][0][0] = 1.0; double val1 = get(); - c.v[0][0][0][0] = 2.0; double val2 = get(); + c.v[0][0][0][0] = 2.0; double val2 = get(); assertEquals(val1, 1.0); assertEquals(val2, (isServerWithStable ? 1.0 : 2.0)); c.v = new double[1][1][1][1]; c.v[0][0][0][0] = 3.0; double val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 3.0)); + : 3.0)); c.v[0] = new double[1][1][1]; c.v[0][0][0][0] = 4.0; double val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 4.0)); + : 4.0)); c.v[0][0] = new double[1][1]; c.v[0][0][0][0] = 5.0; double val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 5.0)); + : 5.0)); c.v[0][0][0] = new double[1]; c.v[0][0][0][0] = 6.0; double val6 = get(); assertEquals(val6, (isStableEnabled ? (isServerWithStable ? 1.0 : 2.0) - : 6.0)); + : 6.0)); } { @@ -397,7 +370,7 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new double[1]; ((double[])c.v)[0] = 1.0; double val1 = get(); - ((double[])c.v)[0] = 2.0; double val2 = get(); + ((double[])c.v)[0] = 2.0; double val2 = get(); assertEquals(val1, 1.0); assertEquals(val2, 2.0); @@ -424,7 +397,7 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new double[1][1]; ((double[][])c.v)[0][0] = 1.0; double val1 = get(); - ((double[][])c.v)[0][0] = 2.0; double val2 = get(); + ((double[][])c.v)[0][0] = 2.0; double val2 = get(); assertEquals(val1, 1.0); assertEquals(val2, 2.0); @@ -432,7 +405,7 @@ public class TestStableDouble { { c.v = new double[1][1]; c.v[0] = new double[0]; double[] val1 = get1(); - c.v[0] = new double[0]; double[] val2 = get1(); + c.v[0] = new double[0]; double[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -460,7 +433,7 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new double[1][1][1]; ((double[][][])c.v)[0][0][0] = 1.0; double val1 = get(); - ((double[][][])c.v)[0][0][0] = 2.0; double val2 = get(); + ((double[][][])c.v)[0][0][0] = 2.0; double val2 = get(); assertEquals(val1, 1.0); assertEquals(val2, 2.0); @@ -468,14 +441,14 @@ public class TestStableDouble { { c.v = new double[1][1][1]; c.v[0][0] = new double[0]; double[] val1 = get1(); - c.v[0][0] = new double[0]; double[] val2 = get1(); + c.v[0][0] = new double[0]; double[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new double[1][1][1]; c.v[0] = new double[0][0]; double[][] val1 = get2(); - c.v[0] = new double[0][0]; double[][] val2 = get2(); + c.v[0] = new double[0][0]; double[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -505,7 +478,7 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new A(); c.v.a = 1.0; A val1 = get(); - c.v.a = 2.0; A val2 = get(); + c.v.a = 2.0; A val2 = get(); assertEquals(val1.a, 2.0); assertEquals(val2.a, 2.0); @@ -513,7 +486,7 @@ public class TestStableDouble { { c.v = new A(); c.v.a = 1.0; double val1 = get1(); - c.v.a = 2.0; double val2 = get1(); + c.v.a = 2.0; double val2 = get1(); c.v = new A(); c.v.a = 3.0; double val3 = get1(); assertEquals(val1, 1.0); @@ -539,8 +512,8 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = 1.0; c.v.next.a = 1.0; A val1 = get(); - c.v.a = 2.0; c.v.next.a = 2.0; A val2 = get(); + c.v.a = 1.0; c.v.next.a = 1.0; A val1 = get(); + c.v.a = 2.0; c.v.next.a = 2.0; A val2 = get(); assertEquals(val1.a, 2.0); assertEquals(val2.a, 2.0); @@ -548,10 +521,10 @@ public class TestStableDouble { { c.v = new A(); c.v.next = c.v; - c.v.a = 1.0; double val1 = get1(); - c.v.a = 2.0; double val2 = get1(); + c.v.a = 1.0; double val1 = get1(); + c.v.a = 2.0; double val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = 3.0; double val3 = get1(); + c.v.a = 3.0; double val3 = get1(); assertEquals(val1, 1.0); assertEquals(val2, (isStableEnabled ? 1.0 : 2.0)); @@ -577,8 +550,8 @@ public class TestStableDouble { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = 1.0; double val1 = get(); double val2 = get1(); - c.v.a = 2.0; double val3 = get(); double val4 = get1(); + c.v.a = 1.0; double val1 = get(); double val2 = get1(); + c.v.a = 2.0; double val3 = get(); double val4 = get1(); assertEquals(val1, 1.0); assertEquals(val3, (isStableEnabled ? 1.0 : 2.0)); @@ -608,8 +581,8 @@ public class TestStableDouble { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = 1.0; double val1 = get(); double val2 = get1(); - elem.a = 2.0; double val3 = get(); double val4 = get1(); + elem.a = 1.0; double val1 = get(); double val2 = get1(); + elem.a = 2.0; double val3 = get(); double val4 = get1(); assertEquals(val1, 1.0); assertEquals(val3, (isServerWithStable ? 1.0 : 2.0)); diff --git a/hotspot/test/compiler/stable/TestStableFloat.java b/hotspot/test/compiler/stable/TestStableFloat.java index 04acead22ef..28f3427c47b 100644 --- a/hotspot/test/compiler/stable/TestStableFloat.java +++ b/hotspot/test/compiler/stable/TestStableFloat.java @@ -26,64 +26,37 @@ /* * @test TestStableFloat * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableFloat StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableFloat - * java/lang/invoke/TestStableFloat$FloatStable - * java/lang/invoke/TestStableFloat$StaticFloatStable - * java/lang/invoke/TestStableFloat$VolatileFloatStable - * java/lang/invoke/TestStableFloat$FloatArrayDim1 - * java/lang/invoke/TestStableFloat$FloatArrayDim2 - * java/lang/invoke/TestStableFloat$FloatArrayDim3 - * java/lang/invoke/TestStableFloat$FloatArrayDim4 - * java/lang/invoke/TestStableFloat$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableFloat$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableFloat$NestedStableField - * java/lang/invoke/TestStableFloat$NestedStableField$A - * java/lang/invoke/TestStableFloat$NestedStableField1 - * java/lang/invoke/TestStableFloat$NestedStableField1$A - * java/lang/invoke/TestStableFloat$NestedStableField2 - * java/lang/invoke/TestStableFloat$NestedStableField2$A - * java/lang/invoke/TestStableFloat$NestedStableField3 - * java/lang/invoke/TestStableFloat$NestedStableField3$A - * java/lang/invoke/TestStableFloat$DefaultValue - * java/lang/invoke/TestStableFloat$DefaultStaticValue - * java/lang/invoke/TestStableFloat$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableFloat * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableFloat - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableFloat - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableFloat - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableFloat + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableFloat + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableFloat * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableFloat + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableFloat */ -package java.lang.invoke; + +package compiler.stable; import jdk.internal.vm.annotation.Stable; - import java.lang.reflect.InvocationTargetException; public class TestStableFloat { @@ -127,7 +100,7 @@ public class TestStableFloat { public static final DefaultValue c = new DefaultValue(); public static float get() { return c.v; } public static void test() throws Exception { - float val1 = get(); + float val1 = get(); c.v = 1.0F; float val2 = get(); assertEquals(val1, 0F); assertEquals(val2, 1.0F); @@ -157,7 +130,7 @@ public class TestStableFloat { public static final DefaultStaticValue c = new DefaultStaticValue(); public static float get() { return c.v; } public static void test() throws Exception { - float val1 = get(); + float val1 = get(); c.v = 1.0F; float val2 = get(); assertEquals(val1, 0F); assertEquals(val2, 1.0F); @@ -207,24 +180,24 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new float[1]; c.v[0] = 1.0F; float val1 = get(); - c.v[0] = 2.0F; float val2 = get(); + c.v[0] = 2.0F; float val2 = get(); assertEquals(val1, 1.0F); assertEquals(val2, (isServerWithStable ? 1.0F : 2.0F)); c.v = new float[1]; c.v[0] = 3.0F; float val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 3.0F)); + : 3.0F)); } { c.v = new float[20]; c.v[10] = 1.0F; float val1 = get1(); - c.v[10] = 2.0F; float val2 = get1(); + c.v[10] = 2.0F; float val2 = get1(); assertEquals(val1, 1.0F); assertEquals(val2, (isServerWithStable ? 1.0F : 2.0F)); c.v = new float[20]; c.v[10] = 3.0F; float val3 = get1(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 3.0F)); + : 3.0F)); } { @@ -247,17 +220,17 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new float[1][1]; c.v[0][0] = 1.0F; float val1 = get(); - c.v[0][0] = 2.0F; float val2 = get(); + c.v[0][0] = 2.0F; float val2 = get(); assertEquals(val1, 1.0F); assertEquals(val2, (isServerWithStable ? 1.0F : 2.0F)); c.v = new float[1][1]; c.v[0][0] = 3.0F; float val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 3.0F)); + : 3.0F)); c.v[0] = new float[1]; c.v[0][0] = 4.0F; float val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 4.0F)); + : 4.0F)); } { @@ -287,21 +260,21 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new float[1][1][1]; c.v[0][0][0] = 1.0F; float val1 = get(); - c.v[0][0][0] = 2.0F; float val2 = get(); + c.v[0][0][0] = 2.0F; float val2 = get(); assertEquals(val1, 1.0F); assertEquals(val2, (isServerWithStable ? 1.0F : 2.0F)); c.v = new float[1][1][1]; c.v[0][0][0] = 3.0F; float val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 3.0F)); + : 3.0F)); c.v[0] = new float[1][1]; c.v[0][0][0] = 4.0F; float val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 4.0F)); + : 4.0F)); c.v[0][0] = new float[1]; c.v[0][0][0] = 5.0F; float val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 5.0F)); + : 5.0F)); } { @@ -338,25 +311,25 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new float[1][1][1][1]; c.v[0][0][0][0] = 1.0F; float val1 = get(); - c.v[0][0][0][0] = 2.0F; float val2 = get(); + c.v[0][0][0][0] = 2.0F; float val2 = get(); assertEquals(val1, 1.0F); assertEquals(val2, (isServerWithStable ? 1.0F : 2.0F)); c.v = new float[1][1][1][1]; c.v[0][0][0][0] = 3.0F; float val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 3.0F)); + : 3.0F)); c.v[0] = new float[1][1][1]; c.v[0][0][0][0] = 4.0F; float val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 4.0F)); + : 4.0F)); c.v[0][0] = new float[1][1]; c.v[0][0][0][0] = 5.0F; float val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 5.0F)); + : 5.0F)); c.v[0][0][0] = new float[1]; c.v[0][0][0][0] = 6.0F; float val6 = get(); assertEquals(val6, (isStableEnabled ? (isServerWithStable ? 1.0F : 2.0F) - : 6.0F)); + : 6.0F)); } { @@ -397,7 +370,7 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new float[1]; ((float[])c.v)[0] = 1.0F; float val1 = get(); - ((float[])c.v)[0] = 2.0F; float val2 = get(); + ((float[])c.v)[0] = 2.0F; float val2 = get(); assertEquals(val1, 1.0F); assertEquals(val2, 2.0F); @@ -424,7 +397,7 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new float[1][1]; ((float[][])c.v)[0][0] = 1.0F; float val1 = get(); - ((float[][])c.v)[0][0] = 2.0F; float val2 = get(); + ((float[][])c.v)[0][0] = 2.0F; float val2 = get(); assertEquals(val1, 1.0F); assertEquals(val2, 2.0F); @@ -432,7 +405,7 @@ public class TestStableFloat { { c.v = new float[1][1]; c.v[0] = new float[0]; float[] val1 = get1(); - c.v[0] = new float[0]; float[] val2 = get1(); + c.v[0] = new float[0]; float[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -460,7 +433,7 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new float[1][1][1]; ((float[][][])c.v)[0][0][0] = 1.0F; float val1 = get(); - ((float[][][])c.v)[0][0][0] = 2.0F; float val2 = get(); + ((float[][][])c.v)[0][0][0] = 2.0F; float val2 = get(); assertEquals(val1, 1.0F); assertEquals(val2, 2.0F); @@ -468,14 +441,14 @@ public class TestStableFloat { { c.v = new float[1][1][1]; c.v[0][0] = new float[0]; float[] val1 = get1(); - c.v[0][0] = new float[0]; float[] val2 = get1(); + c.v[0][0] = new float[0]; float[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new float[1][1][1]; c.v[0] = new float[0][0]; float[][] val1 = get2(); - c.v[0] = new float[0][0]; float[][] val2 = get2(); + c.v[0] = new float[0][0]; float[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -505,7 +478,7 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new A(); c.v.a = 1.0F; A val1 = get(); - c.v.a = 2.0F; A val2 = get(); + c.v.a = 2.0F; A val2 = get(); assertEquals(val1.a, 2.0F); assertEquals(val2.a, 2.0F); @@ -513,7 +486,7 @@ public class TestStableFloat { { c.v = new A(); c.v.a = 1.0F; float val1 = get1(); - c.v.a = 2.0F; float val2 = get1(); + c.v.a = 2.0F; float val2 = get1(); c.v = new A(); c.v.a = 3.0F; float val3 = get1(); assertEquals(val1, 1.0F); @@ -539,8 +512,8 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = 1.0F; c.v.next.a = 1.0F; A val1 = get(); - c.v.a = 2.0F; c.v.next.a = 2.0F; A val2 = get(); + c.v.a = 1.0F; c.v.next.a = 1.0F; A val1 = get(); + c.v.a = 2.0F; c.v.next.a = 2.0F; A val2 = get(); assertEquals(val1.a, 2.0F); assertEquals(val2.a, 2.0F); @@ -548,10 +521,10 @@ public class TestStableFloat { { c.v = new A(); c.v.next = c.v; - c.v.a = 1.0F; float val1 = get1(); - c.v.a = 2.0F; float val2 = get1(); + c.v.a = 1.0F; float val1 = get1(); + c.v.a = 2.0F; float val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = 3.0F; float val3 = get1(); + c.v.a = 3.0F; float val3 = get1(); assertEquals(val1, 1.0F); assertEquals(val2, (isStableEnabled ? 1.0F : 2.0F)); @@ -577,8 +550,8 @@ public class TestStableFloat { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = 1.0F; float val1 = get(); float val2 = get1(); - c.v.a = 2.0F; float val3 = get(); float val4 = get1(); + c.v.a = 1.0F; float val1 = get(); float val2 = get1(); + c.v.a = 2.0F; float val3 = get(); float val4 = get1(); assertEquals(val1, 1.0F); assertEquals(val3, (isStableEnabled ? 1.0F : 2.0F)); @@ -608,8 +581,8 @@ public class TestStableFloat { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = 1.0F; float val1 = get(); float val2 = get1(); - elem.a = 2.0F; float val3 = get(); float val4 = get1(); + elem.a = 1.0F; float val1 = get(); float val2 = get1(); + elem.a = 2.0F; float val3 = get(); float val4 = get1(); assertEquals(val1, 1.0F); assertEquals(val3, (isServerWithStable ? 1.0F : 2.0F)); diff --git a/hotspot/test/compiler/stable/TestStableInt.java b/hotspot/test/compiler/stable/TestStableInt.java index 2837bd3d1a5..831afd2bf9d 100644 --- a/hotspot/test/compiler/stable/TestStableInt.java +++ b/hotspot/test/compiler/stable/TestStableInt.java @@ -26,64 +26,37 @@ /* * @test TestStableInt * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableInt StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableInt - * java/lang/invoke/TestStableInt$IntStable - * java/lang/invoke/TestStableInt$StaticIntStable - * java/lang/invoke/TestStableInt$VolatileIntStable - * java/lang/invoke/TestStableInt$IntArrayDim1 - * java/lang/invoke/TestStableInt$IntArrayDim2 - * java/lang/invoke/TestStableInt$IntArrayDim3 - * java/lang/invoke/TestStableInt$IntArrayDim4 - * java/lang/invoke/TestStableInt$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableInt$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableInt$NestedStableField - * java/lang/invoke/TestStableInt$NestedStableField$A - * java/lang/invoke/TestStableInt$NestedStableField1 - * java/lang/invoke/TestStableInt$NestedStableField1$A - * java/lang/invoke/TestStableInt$NestedStableField2 - * java/lang/invoke/TestStableInt$NestedStableField2$A - * java/lang/invoke/TestStableInt$NestedStableField3 - * java/lang/invoke/TestStableInt$NestedStableField3$A - * java/lang/invoke/TestStableInt$DefaultValue - * java/lang/invoke/TestStableInt$DefaultStaticValue - * java/lang/invoke/TestStableInt$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableInt * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableInt - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableInt - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableInt - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableInt + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableInt + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableInt * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableInt + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableInt */ -package java.lang.invoke; + +package compiler.stable; import jdk.internal.vm.annotation.Stable; - import java.lang.reflect.InvocationTargetException; public class TestStableInt { @@ -127,7 +100,7 @@ public class TestStableInt { public static final DefaultValue c = new DefaultValue(); public static int get() { return c.v; } public static void test() throws Exception { - int val1 = get(); + int val1 = get(); c.v = 1; int val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1); @@ -157,7 +130,7 @@ public class TestStableInt { public static final DefaultStaticValue c = new DefaultStaticValue(); public static int get() { return c.v; } public static void test() throws Exception { - int val1 = get(); + int val1 = get(); c.v = 1; int val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1); @@ -207,24 +180,24 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new int[1]; c.v[0] = 1; int val1 = get(); - c.v[0] = 2; int val2 = get(); + c.v[0] = 2; int val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new int[1]; c.v[0] = 3; int val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); } { c.v = new int[20]; c.v[10] = 1; int val1 = get1(); - c.v[10] = 2; int val2 = get1(); + c.v[10] = 2; int val2 = get1(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new int[20]; c.v[10] = 3; int val3 = get1(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); } { @@ -247,17 +220,17 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new int[1][1]; c.v[0][0] = 1; int val1 = get(); - c.v[0][0] = 2; int val2 = get(); + c.v[0][0] = 2; int val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new int[1][1]; c.v[0][0] = 3; int val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new int[1]; c.v[0][0] = 4; int val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); } { @@ -287,21 +260,21 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new int[1][1][1]; c.v[0][0][0] = 1; int val1 = get(); - c.v[0][0][0] = 2; int val2 = get(); + c.v[0][0][0] = 2; int val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new int[1][1][1]; c.v[0][0][0] = 3; int val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new int[1][1]; c.v[0][0][0] = 4; int val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); c.v[0][0] = new int[1]; c.v[0][0][0] = 5; int val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 5)); + : 5)); } { @@ -338,25 +311,25 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new int[1][1][1][1]; c.v[0][0][0][0] = 1; int val1 = get(); - c.v[0][0][0][0] = 2; int val2 = get(); + c.v[0][0][0][0] = 2; int val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new int[1][1][1][1]; c.v[0][0][0][0] = 3; int val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new int[1][1][1]; c.v[0][0][0][0] = 4; int val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); c.v[0][0] = new int[1][1]; c.v[0][0][0][0] = 5; int val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 5)); + : 5)); c.v[0][0][0] = new int[1]; c.v[0][0][0][0] = 6; int val6 = get(); assertEquals(val6, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 6)); + : 6)); } { @@ -397,7 +370,7 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new int[1]; ((int[])c.v)[0] = 1; int val1 = get(); - ((int[])c.v)[0] = 2; int val2 = get(); + ((int[])c.v)[0] = 2; int val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -424,7 +397,7 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new int[1][1]; ((int[][])c.v)[0][0] = 1; int val1 = get(); - ((int[][])c.v)[0][0] = 2; int val2 = get(); + ((int[][])c.v)[0][0] = 2; int val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -432,7 +405,7 @@ public class TestStableInt { { c.v = new int[1][1]; c.v[0] = new int[0]; int[] val1 = get1(); - c.v[0] = new int[0]; int[] val2 = get1(); + c.v[0] = new int[0]; int[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -460,7 +433,7 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new int[1][1][1]; ((int[][][])c.v)[0][0][0] = 1; int val1 = get(); - ((int[][][])c.v)[0][0][0] = 2; int val2 = get(); + ((int[][][])c.v)[0][0][0] = 2; int val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -468,14 +441,14 @@ public class TestStableInt { { c.v = new int[1][1][1]; c.v[0][0] = new int[0]; int[] val1 = get1(); - c.v[0][0] = new int[0]; int[] val2 = get1(); + c.v[0][0] = new int[0]; int[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new int[1][1][1]; c.v[0] = new int[0][0]; int[][] val1 = get2(); - c.v[0] = new int[0][0]; int[][] val2 = get2(); + c.v[0] = new int[0][0]; int[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -505,7 +478,7 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new A(); c.v.a = 1; A val1 = get(); - c.v.a = 2; A val2 = get(); + c.v.a = 2; A val2 = get(); assertEquals(val1.a, 2); assertEquals(val2.a, 2); @@ -513,7 +486,7 @@ public class TestStableInt { { c.v = new A(); c.v.a = 1; int val1 = get1(); - c.v.a = 2; int val2 = get1(); + c.v.a = 2; int val2 = get1(); c.v = new A(); c.v.a = 3; int val3 = get1(); assertEquals(val1, 1); @@ -539,8 +512,8 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = 1; c.v.next.a = 1; A val1 = get(); - c.v.a = 2; c.v.next.a = 2; A val2 = get(); + c.v.a = 1; c.v.next.a = 1; A val1 = get(); + c.v.a = 2; c.v.next.a = 2; A val2 = get(); assertEquals(val1.a, 2); assertEquals(val2.a, 2); @@ -548,10 +521,10 @@ public class TestStableInt { { c.v = new A(); c.v.next = c.v; - c.v.a = 1; int val1 = get1(); - c.v.a = 2; int val2 = get1(); + c.v.a = 1; int val1 = get1(); + c.v.a = 2; int val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = 3; int val3 = get1(); + c.v.a = 3; int val3 = get1(); assertEquals(val1, 1); assertEquals(val2, (isStableEnabled ? 1 : 2)); @@ -577,8 +550,8 @@ public class TestStableInt { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = 1; int val1 = get(); int val2 = get1(); - c.v.a = 2; int val3 = get(); int val4 = get1(); + c.v.a = 1; int val1 = get(); int val2 = get1(); + c.v.a = 2; int val3 = get(); int val4 = get1(); assertEquals(val1, 1); assertEquals(val3, (isStableEnabled ? 1 : 2)); @@ -608,8 +581,8 @@ public class TestStableInt { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = 1; int val1 = get(); int val2 = get1(); - elem.a = 2; int val3 = get(); int val4 = get1(); + elem.a = 1; int val1 = get(); int val2 = get1(); + elem.a = 2; int val3 = get(); int val4 = get1(); assertEquals(val1, 1); assertEquals(val3, (isServerWithStable ? 1 : 2)); diff --git a/hotspot/test/compiler/stable/TestStableLong.java b/hotspot/test/compiler/stable/TestStableLong.java index b6c2fbb0be6..d162e53313e 100644 --- a/hotspot/test/compiler/stable/TestStableLong.java +++ b/hotspot/test/compiler/stable/TestStableLong.java @@ -26,64 +26,37 @@ /* * @test TestStableLong * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableLong StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableLong - * java/lang/invoke/TestStableLong$LongStable - * java/lang/invoke/TestStableLong$StaticLongStable - * java/lang/invoke/TestStableLong$VolatileLongStable - * java/lang/invoke/TestStableLong$LongArrayDim1 - * java/lang/invoke/TestStableLong$LongArrayDim2 - * java/lang/invoke/TestStableLong$LongArrayDim3 - * java/lang/invoke/TestStableLong$LongArrayDim4 - * java/lang/invoke/TestStableLong$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableLong$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableLong$NestedStableField - * java/lang/invoke/TestStableLong$NestedStableField$A - * java/lang/invoke/TestStableLong$NestedStableField1 - * java/lang/invoke/TestStableLong$NestedStableField1$A - * java/lang/invoke/TestStableLong$NestedStableField2 - * java/lang/invoke/TestStableLong$NestedStableField2$A - * java/lang/invoke/TestStableLong$NestedStableField3 - * java/lang/invoke/TestStableLong$NestedStableField3$A - * java/lang/invoke/TestStableLong$DefaultValue - * java/lang/invoke/TestStableLong$DefaultStaticValue - * java/lang/invoke/TestStableLong$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableLong * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableLong - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableLong - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableLong - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableLong + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableLong + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableLong * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableLong + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableLong */ -package java.lang.invoke; + +package compiler.stable; import jdk.internal.vm.annotation.Stable; - import java.lang.reflect.InvocationTargetException; public class TestStableLong { @@ -127,7 +100,7 @@ public class TestStableLong { public static final DefaultValue c = new DefaultValue(); public static long get() { return c.v; } public static void test() throws Exception { - long val1 = get(); + long val1 = get(); c.v = 1L; long val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1L); @@ -157,7 +130,7 @@ public class TestStableLong { public static final DefaultStaticValue c = new DefaultStaticValue(); public static long get() { return c.v; } public static void test() throws Exception { - long val1 = get(); + long val1 = get(); c.v = 1L; long val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1L); @@ -207,24 +180,24 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new long[1]; c.v[0] = 1; long val1 = get(); - c.v[0] = 2; long val2 = get(); + c.v[0] = 2; long val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new long[1]; c.v[0] = 3; long val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); } { c.v = new long[20]; c.v[10] = 1; long val1 = get1(); - c.v[10] = 2; long val2 = get1(); + c.v[10] = 2; long val2 = get1(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new long[20]; c.v[10] = 3; long val3 = get1(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); } { @@ -247,17 +220,17 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new long[1][1]; c.v[0][0] = 1; long val1 = get(); - c.v[0][0] = 2; long val2 = get(); + c.v[0][0] = 2; long val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new long[1][1]; c.v[0][0] = 3; long val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new long[1]; c.v[0][0] = 4; long val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); } { @@ -287,21 +260,21 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new long[1][1][1]; c.v[0][0][0] = 1; long val1 = get(); - c.v[0][0][0] = 2; long val2 = get(); + c.v[0][0][0] = 2; long val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new long[1][1][1]; c.v[0][0][0] = 3; long val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new long[1][1]; c.v[0][0][0] = 4; long val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); c.v[0][0] = new long[1]; c.v[0][0][0] = 5; long val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 5)); + : 5)); } { @@ -338,25 +311,25 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new long[1][1][1][1]; c.v[0][0][0][0] = 1; long val1 = get(); - c.v[0][0][0][0] = 2; long val2 = get(); + c.v[0][0][0][0] = 2; long val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new long[1][1][1][1]; c.v[0][0][0][0] = 3; long val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new long[1][1][1]; c.v[0][0][0][0] = 4; long val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); c.v[0][0] = new long[1][1]; c.v[0][0][0][0] = 5; long val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 5)); + : 5)); c.v[0][0][0] = new long[1]; c.v[0][0][0][0] = 6; long val6 = get(); assertEquals(val6, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 6)); + : 6)); } { @@ -397,7 +370,7 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new long[1]; ((long[])c.v)[0] = 1; long val1 = get(); - ((long[])c.v)[0] = 2; long val2 = get(); + ((long[])c.v)[0] = 2; long val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -424,7 +397,7 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new long[1][1]; ((long[][])c.v)[0][0] = 1; long val1 = get(); - ((long[][])c.v)[0][0] = 2; long val2 = get(); + ((long[][])c.v)[0][0] = 2; long val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -432,7 +405,7 @@ public class TestStableLong { { c.v = new long[1][1]; c.v[0] = new long[0]; long[] val1 = get1(); - c.v[0] = new long[0]; long[] val2 = get1(); + c.v[0] = new long[0]; long[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -460,7 +433,7 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new long[1][1][1]; ((long[][][])c.v)[0][0][0] = 1L; long val1 = get(); - ((long[][][])c.v)[0][0][0] = 2L; long val2 = get(); + ((long[][][])c.v)[0][0][0] = 2L; long val2 = get(); assertEquals(val1, 1L); assertEquals(val2, 2L); @@ -468,14 +441,14 @@ public class TestStableLong { { c.v = new long[1][1][1]; c.v[0][0] = new long[0]; long[] val1 = get1(); - c.v[0][0] = new long[0]; long[] val2 = get1(); + c.v[0][0] = new long[0]; long[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new long[1][1][1]; c.v[0] = new long[0][0]; long[][] val1 = get2(); - c.v[0] = new long[0][0]; long[][] val2 = get2(); + c.v[0] = new long[0][0]; long[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -505,7 +478,7 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new A(); c.v.a = 1; A val1 = get(); - c.v.a = 2; A val2 = get(); + c.v.a = 2; A val2 = get(); assertEquals(val1.a, 2); assertEquals(val2.a, 2); @@ -513,7 +486,7 @@ public class TestStableLong { { c.v = new A(); c.v.a = 1; long val1 = get1(); - c.v.a = 2; long val2 = get1(); + c.v.a = 2; long val2 = get1(); c.v = new A(); c.v.a = 3; long val3 = get1(); assertEquals(val1, 1); @@ -539,8 +512,8 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = 1; c.v.next.a = 1; A val1 = get(); - c.v.a = 2; c.v.next.a = 2; A val2 = get(); + c.v.a = 1; c.v.next.a = 1; A val1 = get(); + c.v.a = 2; c.v.next.a = 2; A val2 = get(); assertEquals(val1.a, 2); assertEquals(val2.a, 2); @@ -548,10 +521,10 @@ public class TestStableLong { { c.v = new A(); c.v.next = c.v; - c.v.a = 1; long val1 = get1(); - c.v.a = 2; long val2 = get1(); + c.v.a = 1; long val1 = get1(); + c.v.a = 2; long val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = 3; long val3 = get1(); + c.v.a = 3; long val3 = get1(); assertEquals(val1, 1); assertEquals(val2, (isStableEnabled ? 1 : 2)); @@ -577,8 +550,8 @@ public class TestStableLong { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = 1; long val1 = get(); long val2 = get1(); - c.v.a = 2; long val3 = get(); long val4 = get1(); + c.v.a = 1; long val1 = get(); long val2 = get1(); + c.v.a = 2; long val3 = get(); long val4 = get1(); assertEquals(val1, 1); assertEquals(val3, (isStableEnabled ? 1 : 2)); @@ -608,8 +581,8 @@ public class TestStableLong { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = 1; long val1 = get(); long val2 = get1(); - elem.a = 2; long val3 = get(); long val4 = get1(); + elem.a = 1; long val1 = get(); long val2 = get1(); + elem.a = 2; long val3 = get(); long val4 = get1(); assertEquals(val1, 1); assertEquals(val3, (isServerWithStable ? 1 : 2)); diff --git a/hotspot/test/compiler/stable/TestStableMemoryBarrier.java b/hotspot/test/compiler/stable/TestStableMemoryBarrier.java index f49b2ad811a..6b413401a8a 100644 --- a/hotspot/test/compiler/stable/TestStableMemoryBarrier.java +++ b/hotspot/test/compiler/stable/TestStableMemoryBarrier.java @@ -28,66 +28,44 @@ * @bug 8139758 * @summary tests memory barrier correctly inserted for stable fields * @library /testlibrary /test/lib + * @modules java.base/jdk.internal.vm.annotation * * @run main/bootclasspath -Xcomp -XX:CompileOnly=::testCompile - * java.lang.invoke.TestStableMemoryBarrier + * compiler.stable.TestStableMemoryBarrier * * @author hui.shi@linaro.org */ -package java.lang.invoke; + +package compiler.stable; import jdk.internal.vm.annotation.Stable; -import java.lang.reflect.InvocationTargetException; - public class TestStableMemoryBarrier { - public static void main(String[] args) throws Exception { - run(NotDominate.class); - + for (int i = 0; i < 1000000; i++) { + NotDominate.testCompile(i); + } } - /* ==================================================== * Stable field initialized in method, but its allocation * doesn't dominate MemBar Release at the end of method. */ - - static class NotDominate{ + private static class NotDominate { public @Stable int v; public static int[] array = new int[100]; + public static NotDominate testCompile(int n) { - if ((n % 2) == 0) return null; - // add a loop here, trigger PhaseIdealLoop::verify_dominance - for (int i = 0; i < 100; i++) { - array[i] = n; - } - NotDominate nm = new NotDominate(); - nm.v = n; - return nm; - } - - public static void test() throws Exception { - for (int i = 0; i < 1000000; i++) - testCompile(i); - } - } - - public static void run(Class test) { - Throwable ex = null; - System.out.print(test.getName()+": "); - try { - test.getMethod("test").invoke(null); - } catch (InvocationTargetException e) { - ex = e.getCause(); - } catch (Throwable e) { - ex = e; - } finally { - if (ex == null) { - System.out.println("PASSED"); - } else { - System.out.println("FAILED"); - ex.printStackTrace(System.out); + if ((n % 2) == 0) return null; + // add a loop here, trigger PhaseIdealLoop::verify_dominance + for (int i = 0; i < 100; i++) { + array[i] = n; } + NotDominate nm = new NotDominate(); + nm.v = n; + return nm; } + + } } + diff --git a/hotspot/test/compiler/stable/TestStableObject.java b/hotspot/test/compiler/stable/TestStableObject.java index b61736b6e32..d16a5441c16 100644 --- a/hotspot/test/compiler/stable/TestStableObject.java +++ b/hotspot/test/compiler/stable/TestStableObject.java @@ -26,65 +26,37 @@ /* * @test TestStableObject * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableObject StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableObject - * java/lang/invoke/TestStableObject$ObjectStable - * java/lang/invoke/TestStableObject$StaticObjectStable - * java/lang/invoke/TestStableObject$VolatileObjectStable - * java/lang/invoke/TestStableObject$ObjectArrayDim1 - * java/lang/invoke/TestStableObject$ObjectArrayDim2 - * java/lang/invoke/TestStableObject$ObjectArrayDim3 - * java/lang/invoke/TestStableObject$ObjectArrayDim4 - * java/lang/invoke/TestStableObject$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableObject$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableObject$NestedStableField - * java/lang/invoke/TestStableObject$NestedStableField$A - * java/lang/invoke/TestStableObject$NestedStableField1 - * java/lang/invoke/TestStableObject$NestedStableField1$A - * java/lang/invoke/TestStableObject$NestedStableField2 - * java/lang/invoke/TestStableObject$NestedStableField2$A - * java/lang/invoke/TestStableObject$NestedStableField3 - * java/lang/invoke/TestStableObject$NestedStableField3$A - * java/lang/invoke/TestStableObject$Values - * java/lang/invoke/TestStableObject$DefaultValue - * java/lang/invoke/TestStableObject$DefaultStaticValue - * java/lang/invoke/TestStableObject$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableObject * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableObject - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableObject - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableObject - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableObject + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableObject + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableObject * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableObject + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableObject */ -package java.lang.invoke; + +package compiler.stable; import jdk.internal.vm.annotation.Stable; - import java.lang.reflect.InvocationTargetException; public class TestStableObject { @@ -130,7 +102,7 @@ public class TestStableObject { public static final DefaultValue c = new DefaultValue(); public static Object get() { return c.v; } public static void test() throws Exception { - Object val1 = get(); + Object val1 = get(); c.v = Values.A; Object val2 = get(); assertEquals(val1, null); assertEquals(val2, Values.A); @@ -160,7 +132,7 @@ public class TestStableObject { public static final DefaultStaticValue c = new DefaultStaticValue(); public static Object get() { return c.v; } public static void test() throws Exception { - Object val1 = get(); + Object val1 = get(); c.v = Values.A; Object val2 = get(); assertEquals(val1, null); assertEquals(val2, Values.A); @@ -210,24 +182,24 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new Object[1]; c.v[0] = Values.A; Object val1 = get(); - c.v[0] = Values.B; Object val2 = get(); + c.v[0] = Values.B; Object val2 = get(); assertEquals(val1, Values.A); assertEquals(val2, (isServerWithStable ? Values.A : Values.B)); c.v = new Object[1]; c.v[0] = Values.C; Object val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.C)); + : Values.C)); } { c.v = new Object[20]; c.v[10] = Values.A; Object val1 = get1(); - c.v[10] = Values.B; Object val2 = get1(); + c.v[10] = Values.B; Object val2 = get1(); assertEquals(val1, Values.A); assertEquals(val2, (isServerWithStable ? Values.A : Values.B)); c.v = new Object[20]; c.v[10] = Values.C; Object val3 = get1(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.C)); + : Values.C)); } { @@ -250,17 +222,17 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new Object[1][1]; c.v[0][0] = Values.A; Object val1 = get(); - c.v[0][0] = Values.B; Object val2 = get(); + c.v[0][0] = Values.B; Object val2 = get(); assertEquals(val1, Values.A); assertEquals(val2, (isServerWithStable ? Values.A : Values.B)); c.v = new Object[1][1]; c.v[0][0] = Values.C; Object val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.C)); + : Values.C)); c.v[0] = new Object[1]; c.v[0][0] = Values.D; Object val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.D)); + : Values.D)); } { @@ -290,21 +262,21 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new Object[1][1][1]; c.v[0][0][0] = Values.A; Object val1 = get(); - c.v[0][0][0] = Values.B; Object val2 = get(); + c.v[0][0][0] = Values.B; Object val2 = get(); assertEquals(val1, Values.A); assertEquals(val2, (isServerWithStable ? Values.A : Values.B)); c.v = new Object[1][1][1]; c.v[0][0][0] = Values.C; Object val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.C)); + : Values.C)); c.v[0] = new Object[1][1]; c.v[0][0][0] = Values.D; Object val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.D)); + : Values.D)); c.v[0][0] = new Object[1]; c.v[0][0][0] = Values.E; Object val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.E)); + : Values.E)); } { @@ -341,25 +313,25 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new Object[1][1][1][1]; c.v[0][0][0][0] = Values.A; Object val1 = get(); - c.v[0][0][0][0] = Values.B; Object val2 = get(); + c.v[0][0][0][0] = Values.B; Object val2 = get(); assertEquals(val1, Values.A); assertEquals(val2, (isServerWithStable ? Values.A : Values.B)); c.v = new Object[1][1][1][1]; c.v[0][0][0][0] = Values.C; Object val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.C)); + : Values.C)); c.v[0] = new Object[1][1][1]; c.v[0][0][0][0] = Values.D; Object val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.D)); + : Values.D)); c.v[0][0] = new Object[1][1]; c.v[0][0][0][0] = Values.E; Object val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.E)); + : Values.E)); c.v[0][0][0] = new Object[1]; c.v[0][0][0][0] = Values.F; Object val6 = get(); assertEquals(val6, (isStableEnabled ? (isServerWithStable ? Values.A : Values.B) - : Values.F)); + : Values.F)); } { @@ -400,7 +372,7 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new Object[1]; ((Object[])c.v)[0] = Values.A; Object val1 = get(); - ((Object[])c.v)[0] = Values.B; Object val2 = get(); + ((Object[])c.v)[0] = Values.B; Object val2 = get(); assertEquals(val1, Values.A); assertEquals(val2, Values.B); @@ -427,7 +399,7 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new Object[1][1]; ((Object[][])c.v)[0][0] = Values.A; Object val1 = get(); - ((Object[][])c.v)[0][0] = Values.B; Object val2 = get(); + ((Object[][])c.v)[0][0] = Values.B; Object val2 = get(); assertEquals(val1, Values.A); assertEquals(val2, Values.B); @@ -435,7 +407,7 @@ public class TestStableObject { { c.v = new Object[1][1]; c.v[0] = new Object[0]; Object[] val1 = get1(); - c.v[0] = new Object[0]; Object[] val2 = get1(); + c.v[0] = new Object[0]; Object[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -463,7 +435,7 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new Object[1][1][1]; ((Object[][][])c.v)[0][0][0] = Values.A; Object val1 = get(); - ((Object[][][])c.v)[0][0][0] = Values.B; Object val2 = get(); + ((Object[][][])c.v)[0][0][0] = Values.B; Object val2 = get(); assertEquals(val1, Values.A); assertEquals(val2, Values.B); @@ -471,14 +443,14 @@ public class TestStableObject { { c.v = new Object[1][1][1]; c.v[0][0] = new Object[0]; Object[] val1 = get1(); - c.v[0][0] = new Object[0]; Object[] val2 = get1(); + c.v[0][0] = new Object[0]; Object[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new Object[1][1][1]; c.v[0] = new Object[0][0]; Object[][] val1 = get2(); - c.v[0] = new Object[0][0]; Object[][] val2 = get2(); + c.v[0] = new Object[0][0]; Object[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -508,7 +480,7 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new A(); c.v.a = Values.A; A val1 = get(); - c.v.a = Values.B; A val2 = get(); + c.v.a = Values.B; A val2 = get(); assertEquals(val1.a, Values.B); assertEquals(val2.a, Values.B); @@ -516,7 +488,7 @@ public class TestStableObject { { c.v = new A(); c.v.a = Values.A; Object val1 = get1(); - c.v.a = Values.B; Object val2 = get1(); + c.v.a = Values.B; Object val2 = get1(); c.v = new A(); c.v.a = Values.C; Object val3 = get1(); assertEquals(val1, Values.A); @@ -542,8 +514,8 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = Values.A; c.v.next.a = Values.A; A val1 = get(); - c.v.a = Values.B; c.v.next.a = Values.B; A val2 = get(); + c.v.a = Values.A; c.v.next.a = Values.A; A val1 = get(); + c.v.a = Values.B; c.v.next.a = Values.B; A val2 = get(); assertEquals(val1.a, Values.B); assertEquals(val2.a, Values.B); @@ -551,10 +523,10 @@ public class TestStableObject { { c.v = new A(); c.v.next = c.v; - c.v.a = Values.A; Object val1 = get1(); - c.v.a = Values.B; Object val2 = get1(); + c.v.a = Values.A; Object val1 = get1(); + c.v.a = Values.B; Object val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = Values.C; Object val3 = get1(); + c.v.a = Values.C; Object val3 = get1(); assertEquals(val1, Values.A); assertEquals(val2, (isStableEnabled ? Values.A : Values.B)); @@ -580,8 +552,8 @@ public class TestStableObject { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = Values.A; Object val1 = get(); Object val2 = get1(); - c.v.a = Values.B; Object val3 = get(); Object val4 = get1(); + c.v.a = Values.A; Object val1 = get(); Object val2 = get1(); + c.v.a = Values.B; Object val3 = get(); Object val4 = get1(); assertEquals(val1, Values.A); assertEquals(val3, (isStableEnabled ? Values.A : Values.B)); @@ -611,8 +583,8 @@ public class TestStableObject { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = Values.A; Object val1 = get(); Object val2 = get1(); - elem.a = Values.B; Object val3 = get(); Object val4 = get1(); + elem.a = Values.A; Object val1 = get(); Object val2 = get1(); + elem.a = Values.B; Object val3 = get(); Object val4 = get1(); assertEquals(val1, Values.A); assertEquals(val3, (isServerWithStable ? Values.A : Values.B)); diff --git a/hotspot/test/compiler/stable/TestStableShort.java b/hotspot/test/compiler/stable/TestStableShort.java index 1b0f127785c..617fb929b4e 100644 --- a/hotspot/test/compiler/stable/TestStableShort.java +++ b/hotspot/test/compiler/stable/TestStableShort.java @@ -26,64 +26,37 @@ /* * @test TestStableShort * @summary tests on stable fields and arrays - * @library /testlibrary /test/lib - * @build TestStableShort StableConfiguration sun.hotspot.WhiteBox - * @run main ClassFileInstaller sun.hotspot.WhiteBox sun.hotspot.WhiteBox$WhiteBoxPermission - * @run main ClassFileInstaller - * java/lang/invoke/StableConfiguration - * java/lang/invoke/TestStableShort - * java/lang/invoke/TestStableShort$ShortStable - * java/lang/invoke/TestStableShort$StaticShortStable - * java/lang/invoke/TestStableShort$VolatileShortStable - * java/lang/invoke/TestStableShort$ShortArrayDim1 - * java/lang/invoke/TestStableShort$ShortArrayDim2 - * java/lang/invoke/TestStableShort$ShortArrayDim3 - * java/lang/invoke/TestStableShort$ShortArrayDim4 - * java/lang/invoke/TestStableShort$ObjectArrayLowerDim0 - * java/lang/invoke/TestStableShort$ObjectArrayLowerDim1 - * java/lang/invoke/TestStableShort$NestedStableField - * java/lang/invoke/TestStableShort$NestedStableField$A - * java/lang/invoke/TestStableShort$NestedStableField1 - * java/lang/invoke/TestStableShort$NestedStableField1$A - * java/lang/invoke/TestStableShort$NestedStableField2 - * java/lang/invoke/TestStableShort$NestedStableField2$A - * java/lang/invoke/TestStableShort$NestedStableField3 - * java/lang/invoke/TestStableShort$NestedStableField3$A - * java/lang/invoke/TestStableShort$DefaultValue - * java/lang/invoke/TestStableShort$DefaultStaticValue - * java/lang/invoke/TestStableShort$ObjectArrayLowerDim2 + * @library /testlibrary /test/lib / + * @modules java.base/jdk.internal.vm.annotation + * @build sun.hotspot.WhiteBox + * @build compiler.stable.TestStableShort * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableShort - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:-TieredCompilation - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableShort - * - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:+FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableShort - * @run main/othervm -Xbootclasspath/a:. - * -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp - * -XX:+TieredCompilation -XX:TieredStopAtLevel=1 - * -XX:-FoldStableValues - * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 - * java.lang.invoke.TestStableShort + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableShort + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableShort * + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableShort + * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xcomp + * -XX:CompileOnly=::get,::get1,::get2,::get3,::get4 + * -XX:-TieredCompilation + * -XX:+FoldStableValues + * compiler.stable.TestStableShort */ -package java.lang.invoke; + +package compiler.stable; import jdk.internal.vm.annotation.Stable; - import java.lang.reflect.InvocationTargetException; public class TestStableShort { @@ -127,7 +100,7 @@ public class TestStableShort { public static final DefaultValue c = new DefaultValue(); public static short get() { return c.v; } public static void test() throws Exception { - short val1 = get(); + short val1 = get(); c.v = 1; short val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1); @@ -157,7 +130,7 @@ public class TestStableShort { public static final DefaultStaticValue c = new DefaultStaticValue(); public static short get() { return c.v; } public static void test() throws Exception { - short val1 = get(); + short val1 = get(); c.v = 1; short val2 = get(); assertEquals(val1, 0); assertEquals(val2, 1); @@ -207,24 +180,24 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new short[1]; c.v[0] = 1; short val1 = get(); - c.v[0] = 2; short val2 = get(); + c.v[0] = 2; short val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new short[1]; c.v[0] = 3; short val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); } { c.v = new short[20]; c.v[10] = 1; short val1 = get1(); - c.v[10] = 2; short val2 = get1(); + c.v[10] = 2; short val2 = get1(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new short[20]; c.v[10] = 3; short val3 = get1(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); } { @@ -247,17 +220,17 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new short[1][1]; c.v[0][0] = 1; short val1 = get(); - c.v[0][0] = 2; short val2 = get(); + c.v[0][0] = 2; short val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new short[1][1]; c.v[0][0] = 3; short val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new short[1]; c.v[0][0] = 4; short val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); } { @@ -287,21 +260,21 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new short[1][1][1]; c.v[0][0][0] = 1; short val1 = get(); - c.v[0][0][0] = 2; short val2 = get(); + c.v[0][0][0] = 2; short val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new short[1][1][1]; c.v[0][0][0] = 3; short val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new short[1][1]; c.v[0][0][0] = 4; short val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); c.v[0][0] = new short[1]; c.v[0][0][0] = 5; short val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 5)); + : 5)); } { @@ -338,25 +311,25 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new short[1][1][1][1]; c.v[0][0][0][0] = 1; short val1 = get(); - c.v[0][0][0][0] = 2; short val2 = get(); + c.v[0][0][0][0] = 2; short val2 = get(); assertEquals(val1, 1); assertEquals(val2, (isServerWithStable ? 1 : 2)); c.v = new short[1][1][1][1]; c.v[0][0][0][0] = 3; short val3 = get(); assertEquals(val3, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 3)); + : 3)); c.v[0] = new short[1][1][1]; c.v[0][0][0][0] = 4; short val4 = get(); assertEquals(val4, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 4)); + : 4)); c.v[0][0] = new short[1][1]; c.v[0][0][0][0] = 5; short val5 = get(); assertEquals(val5, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 5)); + : 5)); c.v[0][0][0] = new short[1]; c.v[0][0][0][0] = 6; short val6 = get(); assertEquals(val6, (isStableEnabled ? (isServerWithStable ? 1 : 2) - : 6)); + : 6)); } { @@ -397,7 +370,7 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new short[1]; ((short[])c.v)[0] = 1; short val1 = get(); - ((short[])c.v)[0] = 2; short val2 = get(); + ((short[])c.v)[0] = 2; short val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -424,7 +397,7 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new short[1][1]; ((short[][])c.v)[0][0] = 1; short val1 = get(); - ((short[][])c.v)[0][0] = 2; short val2 = get(); + ((short[][])c.v)[0][0] = 2; short val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -432,7 +405,7 @@ public class TestStableShort { { c.v = new short[1][1]; c.v[0] = new short[0]; short[] val1 = get1(); - c.v[0] = new short[0]; short[] val2 = get1(); + c.v[0] = new short[0]; short[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -460,7 +433,7 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new short[1][1][1]; ((short[][][])c.v)[0][0][0] = 1; short val1 = get(); - ((short[][][])c.v)[0][0][0] = 2; short val2 = get(); + ((short[][][])c.v)[0][0][0] = 2; short val2 = get(); assertEquals(val1, 1); assertEquals(val2, 2); @@ -468,14 +441,14 @@ public class TestStableShort { { c.v = new short[1][1][1]; c.v[0][0] = new short[0]; short[] val1 = get1(); - c.v[0][0] = new short[0]; short[] val2 = get1(); + c.v[0][0] = new short[0]; short[] val2 = get1(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } { c.v = new short[1][1][1]; c.v[0] = new short[0][0]; short[][] val1 = get2(); - c.v[0] = new short[0][0]; short[][] val2 = get2(); + c.v[0] = new short[0][0]; short[][] val2 = get2(); assertTrue((isServerWithStable ? (val1 == val2) : (val1 != val2))); } @@ -505,7 +478,7 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new A(); c.v.a = 1; A val1 = get(); - c.v.a = 2; A val2 = get(); + c.v.a = 2; A val2 = get(); assertEquals(val1.a, 2); assertEquals(val2.a, 2); @@ -513,7 +486,7 @@ public class TestStableShort { { c.v = new A(); c.v.a = 1; short val1 = get1(); - c.v.a = 2; short val2 = get1(); + c.v.a = 2; short val2 = get1(); c.v = new A(); c.v.a = 3; short val3 = get1(); assertEquals(val1, 1); @@ -539,8 +512,8 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new A(); c.v.next = new A(); c.v.next.next = c.v; - c.v.a = 1; c.v.next.a = 1; A val1 = get(); - c.v.a = 2; c.v.next.a = 2; A val2 = get(); + c.v.a = 1; c.v.next.a = 1; A val1 = get(); + c.v.a = 2; c.v.next.a = 2; A val2 = get(); assertEquals(val1.a, 2); assertEquals(val2.a, 2); @@ -548,10 +521,10 @@ public class TestStableShort { { c.v = new A(); c.v.next = c.v; - c.v.a = 1; short val1 = get1(); - c.v.a = 2; short val2 = get1(); + c.v.a = 1; short val1 = get1(); + c.v.a = 2; short val2 = get1(); c.v = new A(); c.v.next = c.v; - c.v.a = 3; short val3 = get1(); + c.v.a = 3; short val3 = get1(); assertEquals(val1, 1); assertEquals(val2, (isStableEnabled ? 1 : 2)); @@ -577,8 +550,8 @@ public class TestStableShort { public static void test() throws Exception { { c.v = new A(); c.v.left = c.v.right = c.v; - c.v.a = 1; short val1 = get(); short val2 = get1(); - c.v.a = 2; short val3 = get(); short val4 = get1(); + c.v.a = 1; short val1 = get(); short val2 = get1(); + c.v.a = 2; short val3 = get(); short val4 = get1(); assertEquals(val1, 1); assertEquals(val3, (isStableEnabled ? 1 : 2)); @@ -608,8 +581,8 @@ public class TestStableShort { { A elem = new A(); c.v = new A[] { elem, elem }; c.v[0].left = c.v[0].right = c.v; - elem.a = 1; short val1 = get(); short val2 = get1(); - elem.a = 2; short val3 = get(); short val4 = get1(); + elem.a = 1; short val1 = get(); short val2 = get1(); + elem.a = 2; short val3 = get(); short val4 = get1(); assertEquals(val1, 1); assertEquals(val3, (isServerWithStable ? 1 : 2)); diff --git a/hotspot/test/compiler/unsafe/UnsafeGetConstantField.java b/hotspot/test/compiler/unsafe/UnsafeGetConstantField.java index c1d6b057e14..45f08e63d9f 100644 --- a/hotspot/test/compiler/unsafe/UnsafeGetConstantField.java +++ b/hotspot/test/compiler/unsafe/UnsafeGetConstantField.java @@ -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. * * This code is free software; you can redistribute it and/or modify it @@ -26,38 +26,50 @@ /* * @test * @summary tests on constant folding of unsafe get operations - * @library /testlibrary /test/lib + * @library /testlibrary * * @requires vm.flavor != "client" * + * @modules java.base/jdk.internal.org.objectweb.asm + * java.base/jdk.internal.vm.annotation + * java.base/jdk.internal.misc * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions - * -Xbatch -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:+UseUnalignedAccesses - * java.lang.invoke.UnsafeGetConstantField + * -Xbatch -XX:-TieredCompilation + * -XX:+FoldStableValues + * -XX:CompileCommand=dontinline,UnsafeGetConstantField.checkGetAddress() + * -XX:CompileCommand=dontinline,*.test* + * -XX:+UseUnalignedAccesses + * compiler.unsafe.UnsafeGetConstantField + * * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions - * -Xbatch -XX:-TieredCompilation - * -XX:+FoldStableValues - * -XX:-UseUnalignedAccesses - * java.lang.invoke.UnsafeGetConstantField + * -Xbatch -XX:-TieredCompilation + * -XX:+FoldStableValues + * -XX:CompileCommand=dontinline,UnsafeGetConstantField.checkGetAddress() + * -XX:CompileCommand=dontinline,*.test* + * -XX:-UseUnalignedAccesses + * compiler.unsafe.UnsafeGetConstantField */ -package java.lang.invoke; +package compiler.unsafe; -import jdk.internal.vm.annotation.DontInline; -import jdk.internal.vm.annotation.Stable; -import jdk.internal.misc.Unsafe; import jdk.internal.org.objectweb.asm.ClassWriter; import jdk.internal.org.objectweb.asm.FieldVisitor; import jdk.internal.org.objectweb.asm.MethodVisitor; import jdk.internal.org.objectweb.asm.Opcodes; import jdk.internal.org.objectweb.asm.Type; +import jdk.internal.vm.annotation.Stable; import jdk.test.lib.Asserts; +import jdk.internal.misc.Unsafe; + +import java.io.IOException; +import java.lang.reflect.Field; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; import static jdk.internal.org.objectweb.asm.Opcodes.*; public class UnsafeGetConstantField { static final Class THIS_CLASS = UnsafeGetConstantField.class; - static final Unsafe U = Unsafe.getUnsafe(); public static void main(String[] args) { @@ -75,7 +87,7 @@ public class UnsafeGetConstantField { Asserts.assertEquals(checkGetAddress(), cookie); } } - @DontInline + static long checkGetAddress() { return U.getAddress(nativeAddr); } @@ -117,36 +129,65 @@ public class UnsafeGetConstantField { static void runTest(JavaType t, int flags, boolean stable, boolean hasDefaultValue, String postfix) { Generator g = new Generator(t, flags, stable, hasDefaultValue, postfix); Test test = g.generate(); - System.out.printf("type=%s flags=%d stable=%b default=%b post=%s\n", + System.err.printf("type=%s flags=%d stable=%b default=%b post=%s\n", t.typeName, flags, stable, hasDefaultValue, postfix); - // Trigger compilation - for (int i = 0; i < 20_000; i++) { - Asserts.assertEQ(test.testDirect(), test.testUnsafe()); + try { + Object expected = hasDefaultValue ? t.defaultValue : t.value; + // Trigger compilation + for (int i = 0; i < 20_000; i++) { + Asserts.assertEQ(expected, test.testDirect(), "i = "+ i +" direct read returns wrong value"); + Asserts.assertEQ(expected, test.testUnsafe(), "i = "+ i +" unsafe read returns wrong value"); + } + + test.changeToDefault(); + if (!hasDefaultValue && (stable || g.isFinal())) { + Asserts.assertEQ(t.value, test.testDirect(), + "direct read doesn't return prev value"); + // fails for getCharUnaligned due to JDK-8148518 + if (!(t == JavaType.C && "Unaligned".equals(postfix))) { + Asserts.assertEQ(test.testDirect(), test.testUnsafe()); + } + } else { + Asserts.assertEQ(t.defaultValue, test.testDirect(), + "direct read doesn't return default value"); + Asserts.assertEQ(test.testDirect(), test.testUnsafe(), + "direct and unsafe reads return different values"); + } + } catch (Throwable e) { + try { + g.dump(); + } catch (IOException io) { + io.printStackTrace(); + } + throw e; } } - interface Test { + public interface Test { Object testDirect(); Object testUnsafe(); + void changeToDefault(); } enum JavaType { - Z("Boolean", true), - B("Byte", new Byte((byte)-1)), - S("Short", new Short((short)-1)), - C("Char", Character.MAX_VALUE), - I("Int", -1), - J("Long", -1L), - F("Float", -1F), - D("Double", -1D), - L("Object", new Object()); + Z("Boolean", true, false), + B("Byte", new Byte((byte) -1), new Byte((byte) 0)), + S("Short", new Short((short) -1), new Short((short) 0)), + C("Char", Character.MAX_VALUE, '\0'), + I("Int", -1, 0), + J("Long", -1L, 0L), + F("Float", -1F, 0F), + D("Double", -1D, 0D), + L("Object", "", null); String typeName; Object value; + Object defaultValue; String wrapper; - JavaType(String name, Object value) { + JavaType(String name, Object value, Object defaultValue) { this.typeName = name; this.value = value; + this.defaultValue = defaultValue; this.wrapper = internalName(value.getClass()); } @@ -154,7 +195,7 @@ public class UnsafeGetConstantField { if (this == JavaType.L) { return "Ljava/lang/Object;"; } else { - return toString(); + return name(); } } } @@ -177,6 +218,7 @@ public class UnsafeGetConstantField { * } * public Object testDirect() { return t.f; } * public Object testUnsafe() { return U.getInt(t, FIELD_OFFSET); } + * public void changeToDefault() { U.putInt(t, 0, FIELD_OFFSET); } * } */ static class Generator { @@ -190,9 +232,11 @@ public class UnsafeGetConstantField { final boolean hasDefaultValue; final String nameSuffix; + final String name; final String className; final String classDesc; final String fieldDesc; + final byte[] classFile; Generator(JavaType t, int flags, boolean stable, boolean hasDefaultValue, String suffix) { this.type = t; @@ -202,9 +246,11 @@ public class UnsafeGetConstantField { this.nameSuffix = suffix; fieldDesc = type.desc(); - className = String.format("%s$Test%s%s__f=%d__s=%b__d=%b", internalName(THIS_CLASS), type.typeName, - suffix, flags, stable, hasDefaultValue); + name = String.format("Test%s%s__f=%d__s=%b__d=%b", + type.typeName, suffix, flags, stable, hasDefaultValue); + className = "java/lang/invoke/" + name; classDesc = String.format("L%s;", className); + classFile = generateClassFile(); } byte[] generateClassFile() { @@ -228,7 +274,7 @@ public class UnsafeGetConstantField { // Methods { // - MethodVisitor mv = cw.visitMethod(0, "", "()V", null, null); + MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "", "()V", null, null); mv.visitCode(); mv.visitVarInsn(ALOAD, 0); @@ -266,6 +312,30 @@ public class UnsafeGetConstantField { mv.visitEnd(); } + { // public void changeToDefault() { U.putInt(t, FIELD_OFFSET, 0); } + MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "changeToDefault", "()V", null, null); + mv.visitCode(); + getUnsafe(mv); + if (isStatic()) { + mv.visitFieldInsn(GETSTATIC, className, "STATIC_BASE", "Ljava/lang/Object;"); + } else { + mv.visitFieldInsn(GETSTATIC, className, "t", classDesc); + } + mv.visitFieldInsn(GETSTATIC, className, "FIELD_OFFSET", "J"); + + if (type.defaultValue != null) { + mv.visitLdcInsn(type.defaultValue); + } else { + mv.visitInsn(ACONST_NULL); + } + String name = "put" + type.typeName + nameSuffix; + mv.visitMethodInsn(INVOKEVIRTUAL, UNSAFE_NAME, name, "(Ljava/lang/Object;J" + type.desc()+ ")V", false); + mv.visitInsn(RETURN); + + mv.visitMaxs(0, 0); + mv.visitEnd(); + } + { // MethodVisitor mv = cw.visitMethod(ACC_STATIC, "", "()V", null, null); mv.visitCode(); @@ -305,7 +375,6 @@ public class UnsafeGetConstantField { } Test generate() { - byte[] classFile = generateClassFile(); Class c = U.defineClass(className, classFile, 0, classFile.length, THIS_CLASS.getClassLoader(), null); try { return (Test) c.newInstance(); @@ -360,20 +429,14 @@ public class UnsafeGetConstantField { if (!isStatic()) { mv.visitVarInsn(ALOAD, 0); } - switch (type) { - case L: { - mv.visitTypeInsn(NEW, "java/lang/Object"); - mv.visitInsn(DUP); - mv.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "", "()V", false); - - break; - } - default: { - mv.visitLdcInsn(type.value); - break; - } - } + mv.visitLdcInsn(type.value); mv.visitFieldInsn((isStatic() ? PUTSTATIC : PUTFIELD), className, FIELD_NAME, fieldDesc); } + + public void dump() throws IOException { + Path path = Paths.get(".", name + ".class").toAbsolutePath(); + System.err.println("dumping test class to " + path); + Files.write(path, classFile); + } } } diff --git a/hotspot/test/gc/TestSmallHeap.java b/hotspot/test/gc/TestSmallHeap.java index 1289046edba..ffafa078a71 100644 --- a/hotspot/test/gc/TestSmallHeap.java +++ b/hotspot/test/gc/TestSmallHeap.java @@ -31,6 +31,7 @@ * @summary Verify that starting the VM with a small heap works * @library /testlibrary /test/lib * @modules java.management/sun.management + * @ignore 8076621 * @build TestSmallHeap * @run main ClassFileInstaller sun.hotspot.WhiteBox * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xmx2m -XX:+UseParallelGC TestSmallHeap diff --git a/hotspot/test/testlibrary/jdk/test/lib/PerfCounter.java b/hotspot/test/gc/metaspace/PerfCounter.java similarity index 98% rename from hotspot/test/testlibrary/jdk/test/lib/PerfCounter.java rename to hotspot/test/gc/metaspace/PerfCounter.java index fe0bd1da5ab..d39277c910e 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/PerfCounter.java +++ b/hotspot/test/gc/metaspace/PerfCounter.java @@ -21,8 +21,6 @@ * questions. */ -package jdk.test.lib; - import sun.jvmstat.monitor.Monitor; /** diff --git a/hotspot/test/testlibrary/jdk/test/lib/PerfCounters.java b/hotspot/test/gc/metaspace/PerfCounters.java similarity index 98% rename from hotspot/test/testlibrary/jdk/test/lib/PerfCounters.java rename to hotspot/test/gc/metaspace/PerfCounters.java index 5db6a0496bf..d6848f2b7b7 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/PerfCounters.java +++ b/hotspot/test/gc/metaspace/PerfCounters.java @@ -21,13 +21,12 @@ * questions. */ -package jdk.test.lib; - import sun.jvmstat.monitor.Monitor; import sun.jvmstat.monitor.MonitorException; import sun.jvmstat.monitor.MonitoredHost; import sun.jvmstat.monitor.MonitoredVm; import sun.jvmstat.monitor.VmIdentifier; +import jdk.test.lib.ProcessTools; /** * PerfCounters can be used to get a performance counter from the currently diff --git a/hotspot/test/runtime/BadObjectClass/BootstrapRedefine.java b/hotspot/test/runtime/BadObjectClass/BootstrapRedefine.java index fde6feb76fd..bed5f7c510f 100644 --- a/hotspot/test/runtime/BadObjectClass/BootstrapRedefine.java +++ b/hotspot/test/runtime/BadObjectClass/BootstrapRedefine.java @@ -28,7 +28,6 @@ * @library /testlibrary * @modules java.base/sun.misc * java.management - * @compile Object.java * @run main BootstrapRedefine */ @@ -37,8 +36,19 @@ import jdk.test.lib.*; public class BootstrapRedefine { public static void main(String[] args) throws Exception { - String testClasses = System.getProperty("test.classes", "."); - ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-Xbootclasspath/p:" + testClasses, "-version"); + String source = "package java.lang;" + + "public class Object {" + + " void dummy1() { return; }" + + " void dummy2() { return; }" + + " void dummy3() { return; }" + + "}"; + + ClassFileInstaller.writeClassToDisk("java/lang/Object", + InMemoryJavaCompiler.compile("java.lang.Object", source, + "-Xmodule:java.base"), + "mods/java.base"); + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-Xpatch:mods", "-version"); new OutputAnalyzer(pb.start()) .shouldContain("Incompatible definition of java.lang.Object") .shouldHaveExitValue(1); diff --git a/hotspot/test/runtime/BootClassAppendProp/BootClassPathAppend.java b/hotspot/test/runtime/BootClassAppendProp/BootClassPathAppend.java new file mode 100644 index 00000000000..5dba459c4b7 --- /dev/null +++ b/hotspot/test/runtime/BootClassAppendProp/BootClassPathAppend.java @@ -0,0 +1,46 @@ +/* + * 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. + */ + +/* + * @test + * @bug 8087154 + * @summary Uninitialized system property jdk.boot.class.path.append causes SIGSEGV + * @library /testlibrary + * @modules java.base/sun.misc + */ + +import jdk.test.lib.*; + +// Test that system property jdk.boot.class.path.append is initialized. Otherwise, +// -XX:+PrintCompilation does causes a SIGSEGV. +public class BootClassPathAppend { + public static void main(String[] args) throws Exception { + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-XX:+PrintCompilation", "-Xcomp", "-version"); + + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("java.lang.Object"); + output.shouldHaveExitValue(0); + } +} diff --git a/hotspot/test/runtime/BootClassAppendProp/BootClassPathAppendProp.java b/hotspot/test/runtime/BootClassAppendProp/BootClassPathAppendProp.java new file mode 100644 index 00000000000..1d18c3ababd --- /dev/null +++ b/hotspot/test/runtime/BootClassAppendProp/BootClassPathAppendProp.java @@ -0,0 +1,48 @@ +/* + * 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. + */ + +import java.io.File; + +/* + * @test + * @build BootClassPathAppendProp + * @run main/othervm -Xbootclasspath/a:/usr/lib -showversion -Xbootclasspath/a:/i/dont/exist BootClassPathAppendProp + * @run main/othervm -Xpatch:/not/here -Xbootclasspath/a:/i/may/exist BootClassPathAppendProp + * @run main/othervm -Djdk.boot.class.path.append=newdir BootClassPathAppendProp + * @run main/othervm BootClassPathAppendProp + */ + +// Test that property jdk.boot.class.path.append contains only the bootclasspath +// info following the "modules" jimage file. +public class BootClassPathAppendProp { + public static void main(String[] args) throws Exception { + // jdk.boot.class.path.append is a non-writeable, internal property. + // The call to System.getProperty should return null. + if (System.getProperty("jdk.boot.class.path.append") != null) { + throw new RuntimeException("Test failed, jdk.boot.class.path.append has value: " + + System.getProperty("jdk.boot.class.path.append")); + } else { + System.out.println("Test BootClassPathAppendProp passed"); + } + } +} diff --git a/hotspot/test/runtime/BootClassAppendProp/SunBootClassPath.java b/hotspot/test/runtime/BootClassAppendProp/SunBootClassPath.java new file mode 100644 index 00000000000..e18b2853d9f --- /dev/null +++ b/hotspot/test/runtime/BootClassAppendProp/SunBootClassPath.java @@ -0,0 +1,39 @@ +/* + * 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. + */ + +/* + * @test + * @summary Make sure property sun.boot.class.path is null starting with JDK-9. + */ + +// Test that the value of property sun.boot.class.path is null. +public class SunBootClassPath { + public static void main(String[] args) throws Exception { + if (System.getProperty("sun.boot.class.path") != null) { + throw new RuntimeException("Test failed, sun.boot.class.path has value: " + + System.getProperty("sun.boot.class.path")); + } else { + System.out.println("Test SunBootClassPath passed"); + } + } +} diff --git a/hotspot/test/runtime/CommandLine/OptionsValidation/TestOptionsWithRanges.java b/hotspot/test/runtime/CommandLine/OptionsValidation/TestOptionsWithRanges.java index 27ccc5d546b..dcbc3535027 100644 --- a/hotspot/test/runtime/CommandLine/OptionsValidation/TestOptionsWithRanges.java +++ b/hotspot/test/runtime/CommandLine/OptionsValidation/TestOptionsWithRanges.java @@ -27,8 +27,9 @@ * @library /testlibrary /runtime/CommandLine/OptionsValidation/common * @modules java.base/sun.misc * java.management - * jdk.attach - * jdk.management/sun.tools.attach + * jdk.attach/sun.tools.attach + * jdk.jvmstat/sun.jvmstat.monitor + * @build jdk.test.lib.* TestOptionsWithRanges * @run main/othervm/timeout=900 TestOptionsWithRanges */ diff --git a/hotspot/test/runtime/CommandLine/OptionsValidation/TestOptionsWithRangesDynamic.java b/hotspot/test/runtime/CommandLine/OptionsValidation/TestOptionsWithRangesDynamic.java index 517f3a16266..66c84a20bb4 100644 --- a/hotspot/test/runtime/CommandLine/OptionsValidation/TestOptionsWithRangesDynamic.java +++ b/hotspot/test/runtime/CommandLine/OptionsValidation/TestOptionsWithRangesDynamic.java @@ -26,9 +26,8 @@ * @summary Test writeable VM Options with ranges. * @library /testlibrary /runtime/CommandLine/OptionsValidation/common * @modules java.base/sun.misc + * jdk.attach/sun.tools.attach * java.management - * jdk.attach - * jdk.management/sun.tools.attach * @run main/othervm -XX:MinHeapFreeRatio=0 -XX:MaxHeapFreeRatio=100 TestOptionsWithRangesDynamic */ diff --git a/hotspot/test/runtime/SharedArchiveFile/BasicJarBuilder.java b/hotspot/test/runtime/SharedArchiveFile/BasicJarBuilder.java index 8a0b6f783f1..ab8955cb0f7 100644 --- a/hotspot/test/runtime/SharedArchiveFile/BasicJarBuilder.java +++ b/hotspot/test/runtime/SharedArchiveFile/BasicJarBuilder.java @@ -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. * * This code is free software; you can redistribute it and/or modify it @@ -40,8 +40,18 @@ import sun.tools.jar.Main; public class BasicJarBuilder { private static final String classDir = System.getProperty("test.classes"); + public static void build(boolean classesInWorkDir, String jarName, + String ...classNames) throws Exception { + + if (classesInWorkDir) { + createSimpleJar(".", classDir + File.separator + jarName + ".jar", classNames); + } else { + build(jarName, classNames); + } + } + public static void build(String jarName, String ...classNames) throws Exception { - createSimpleJar(".", classDir + File.separator + jarName + ".jar", + createSimpleJar(classDir, classDir + File.separator + jarName + ".jar", classNames); } diff --git a/hotspot/test/runtime/SharedArchiveFile/BootAppendTests.java b/hotspot/test/runtime/SharedArchiveFile/BootAppendTests.java new file mode 100644 index 00000000000..3b69f956465 --- /dev/null +++ b/hotspot/test/runtime/SharedArchiveFile/BootAppendTests.java @@ -0,0 +1,252 @@ +/* + * 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. + */ + +/** + * @test + * @summary Testing -Xbootclasspath/a support for CDS + * @library /testlibrary + * @modules java.base/jdk.internal.misc + * java.management + * jdk.jartool/sun.tools.jar + * jdk.jvmstat/sun.jvmstat.monitor + * @ignore 8150683 + * @compile javax/sound/sampled/MyClass.jasm + * @compile org/omg/CORBA/Context.jasm + * @compile nonjdk/myPackage/MyClass.java + * @build jdk.test.lib.* LoadClass + * @run main ClassFileInstaller LoadClass + * @run main/othervm BootAppendTests + */ + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.PrintStream; + +import java.nio.file.Path; +import java.nio.file.Paths; + +import jdk.test.lib.ProcessTools; +import jdk.test.lib.OutputAnalyzer; + +public class BootAppendTests { + private static final String APP_CLASS = "LoadClass"; + private static final String BOOT_APPEND_MODULE_CLASS = "javax/sound/sampled/MyClass"; + private static final String BOOT_APPEND_DUPLICATE_MODULE_CLASS = "org/omg/CORBA/Context"; + private static final String BOOT_APPEND_CLASS = "nonjdk/myPackage/MyClass"; + private static final String BOOT_APPEND_MODULE_CLASS_NAME = + BOOT_APPEND_MODULE_CLASS.replace('/', '.'); + private static final String BOOT_APPEND_DUPLICATE_MODULE_CLASS_NAME = + BOOT_APPEND_DUPLICATE_MODULE_CLASS.replace('/', '.'); + private static final String BOOT_APPEND_CLASS_NAME = + BOOT_APPEND_CLASS.replace('/', '.'); + private static final String[] ARCHIVE_CLASSES = + {BOOT_APPEND_MODULE_CLASS, BOOT_APPEND_DUPLICATE_MODULE_CLASS, BOOT_APPEND_CLASS}; + + private static final String modes[] = {"on", "off"}; + + private static String appJar; + private static String bootAppendJar; + + public static void main(String... args) throws Exception { + dumpArchive(); + testBootAppendModuleClass(); + testBootAppendDuplicateModuleClass(); + testBootAppendExcludedModuleClass(); + testBootAppendDuplicateExcludedModuleClass(); + testBootAppendClass(); + } + + static void dumpArchive() throws Exception { + // create the classlist + File classlist = new File(new File(System.getProperty("test.classes", ".")), + "BootAppendTest.classlist"); + FileOutputStream fos = new FileOutputStream(classlist); + PrintStream ps = new PrintStream(fos); + for (String s : ARCHIVE_CLASSES) { + ps.println(s); + } + ps.close(); + fos.close(); + + // build jar files + BasicJarBuilder.build(true, "app", APP_CLASS); + appJar = BasicJarBuilder.getTestJar("app.jar"); + BasicJarBuilder.build("bootAppend", + BOOT_APPEND_MODULE_CLASS, BOOT_APPEND_DUPLICATE_MODULE_CLASS, BOOT_APPEND_CLASS); + bootAppendJar = BasicJarBuilder.getTestJar("bootAppend.jar"); + + // dump + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-XX:+UnlockDiagnosticVMOptions", + "-XX:SharedArchiveFile=./BootAppendTests.jsa", + "-XX:SharedClassListFile=" + classlist.getPath(), + "-XX:+PrintSharedSpaces", + "-Xbootclasspath/a:" + bootAppendJar, + "-Xshare:dump"); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("Loading classes to share") + .shouldHaveExitValue(0); + + // Make sure all the classes were successfully archived. + for (String archiveClass : ARCHIVE_CLASSES) { + output.shouldNotContain("Preload Warning: Cannot find " + archiveClass); + } + } + + // Test #1: If a class on -Xbootclasspath/a is from a package defined in + // bootmodules, the class is not loaded at runtime. + // Verify the behavior is the same when the class is archived + // with CDS enabled at runtime. + // + // The javax.sound.sampled package is defined in the java.desktop module. + // The archived javax.sound.sampled.MyClass from the -Xbootclasspath/a + // should not be loaded at runtime. + public static void testBootAppendModuleClass() throws Exception { + for (String mode : modes) { + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-XX:+UnlockDiagnosticVMOptions", + "-XX:SharedArchiveFile=./BootAppendTests.jsa", + "-cp", appJar, + "-Xbootclasspath/a:" + bootAppendJar, + "-Xshare:" + mode, + APP_CLASS, + BOOT_APPEND_MODULE_CLASS_NAME); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("java.lang.ClassNotFoundException: javax.sound.sampled.MyClass"); + } + } + + // Test #2: If a class on -Xbootclasspath/a has the same fully qualified + // name as a class defined in boot modules, the class is not loaded + // from -Xbootclasspath/a. Verify the behavior is the same at runtime + // when CDS is enabled. + // + // The org.omg.CORBA.Context is a boot module class. The class on + // the -Xbootclasspath/a path that has the same fully-qualified name + // should not be loaded at runtime when CDS is enabled. + // The one from the boot modules should be loaded instead. + public static void testBootAppendDuplicateModuleClass() throws Exception { + for (String mode : modes) { + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-XX:+UnlockDiagnosticVMOptions", + "-XX:SharedArchiveFile=./BootAppendTests.jsa", + "-XX:+TraceClassLoading", + "-cp", appJar, + "-Xbootclasspath/a:" + bootAppendJar, + "-Xshare:" + mode, + APP_CLASS, + BOOT_APPEND_DUPLICATE_MODULE_CLASS_NAME); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("[classload] org.omg.CORBA.Context source: jrt:/java.corba"); + } + } + + // Test #3: If a class on -Xbootclasspath/a is from a package defined in boot modules, + // the class can be loaded from -Xbootclasspath/a when the module is excluded + // using -limitmods. Verify the behavior is the same at runtime when CDS is + // enabled. + // + // The java.desktop module is excluded using -limitmods at runtime, + // javax.sound.sampled.MyClass is archived from -Xbootclasspath/a. It can be + // loaded from the archive at runtime. + public static void testBootAppendExcludedModuleClass() throws Exception { + for (String mode : modes) { + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-XX:+UnlockDiagnosticVMOptions", + "-XX:SharedArchiveFile=./BootAppendTests.jsa", + "-XX:+TraceClassLoading", + "-cp", appJar, + "-Xbootclasspath/a:" + bootAppendJar, + "-limitmods", "java.base", + "-Xshare:" + mode, + APP_CLASS, + BOOT_APPEND_MODULE_CLASS_NAME); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("[classload] javax.sound.sampled.MyClass"); + + // When CDS is enabled, the shared class should be loaded from the archive. + if (mode.equals("on")) { + output.shouldContain("[classload] javax.sound.sampled.MyClass source: shared objects file"); + } + } + } + + // Test #4: If a class on -Xbootclasspath/a has the same fully qualified + // name as a class defined in boot modules, the class is loaded + // from -Xbootclasspath/a when the boot module is excluded using + // -limitmods. Verify the behavior is the same at runtime when CDS is + // enabled. + // + // The org.omg.CORBA.Context is a boot module class. The class + // on -Xbootclasspath/a that has the same fully-qualified name + // as org.omg.CORBA.Context can be loaded at runtime when + // java.corba is excluded. + public static void testBootAppendDuplicateExcludedModuleClass() throws Exception { + for (String mode : modes) { + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-XX:+UnlockDiagnosticVMOptions", + "-XX:SharedArchiveFile=./BootAppendTests.jsa", + "-XX:+TraceClassLoading", + "-cp", appJar, + "-Xbootclasspath/a:" + bootAppendJar, + "-limitmods", "java.base", + "-Xshare:" + mode, + APP_CLASS, + BOOT_APPEND_DUPLICATE_MODULE_CLASS_NAME); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("[classload] org.omg.CORBA.Context"); + output.shouldMatch(".*\\[classload\\] org.omg.CORBA.Context source:.*bootAppend.jar"); + } + } + + // Test #5: If a class on -Xbootclasspath/a is not from named modules, + // the class can be loaded at runtime. Verify the behavior is + // the same at runtime when CDS is enabled. + // + // The nonjdk.myPackage is not defined in named modules. The + // archived nonjdk.myPackage.MyClass from -Xbootclasspath/a + // can be loaded at runtime when CDS is enabled. + public static void testBootAppendClass() throws Exception { + for (String mode : modes) { + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-XX:+UnlockDiagnosticVMOptions", + "-XX:SharedArchiveFile=./BootAppendTests.jsa", + "-XX:+TraceClassLoading", + "-cp", appJar, + "-Xbootclasspath/a:" + bootAppendJar, + "-Xshare:" + mode, + APP_CLASS, + BOOT_APPEND_CLASS_NAME); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("[classload] nonjdk.myPackage.MyClass"); + + // If CDS is enabled, the nonjdk.myPackage.MyClass should be loaded + // from the shared archive. + if (mode.equals("on")) { + output.shouldContain( + "[classload] nonjdk.myPackage.MyClass source: shared objects file"); + } + } + } +} diff --git a/hotspot/test/runtime/SharedArchiveFile/LoadClass.java b/hotspot/test/runtime/SharedArchiveFile/LoadClass.java new file mode 100644 index 00000000000..417f92451ae --- /dev/null +++ b/hotspot/test/runtime/SharedArchiveFile/LoadClass.java @@ -0,0 +1,40 @@ +/* + * 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. + */ + +/* + * @summary Load the class specifiecd by the argument + * Input: className + */ +public class LoadClass { + public static void main(String args[]) { + Class c = null; + try { + c = Class.forName(args[0]); + } catch (ClassNotFoundException e) { + System.out.println(e); + } + if (c != null) { + System.out.println(c + " loaded."); + } + } +} diff --git a/hotspot/test/runtime/SharedArchiveFile/PrintSharedArchiveAndExit.java b/hotspot/test/runtime/SharedArchiveFile/PrintSharedArchiveAndExit.java index 55e35246b96..45f74515339 100644 --- a/hotspot/test/runtime/SharedArchiveFile/PrintSharedArchiveAndExit.java +++ b/hotspot/test/runtime/SharedArchiveFile/PrintSharedArchiveAndExit.java @@ -60,24 +60,6 @@ public class PrintSharedArchiveAndExit { output.shouldNotContain("Usage:"); // Should not print JVM help message output.shouldHaveExitValue(0); // Should report success in error code. - // (2) With an invalid archive (boot class path has been prepended) - pb = ProcessTools.createJavaProcessBuilder( - "-Xbootclasspath/p:foo.jar", - "-XX:+UnlockDiagnosticVMOptions", "-XX:SharedArchiveFile=" + filename, - "-XX:+PrintSharedArchiveAndExit", "-version"); - output = new OutputAnalyzer(pb.start()); - output.shouldContain("archive is invalid"); - output.shouldNotContain("java version"); // Should not print JVM version - output.shouldHaveExitValue(1); // Should report failure in error code. - - pb = ProcessTools.createJavaProcessBuilder( - "-Xbootclasspath/p:foo.jar", - "-XX:+UnlockDiagnosticVMOptions", "-XX:SharedArchiveFile=" + filename, - "-XX:+PrintSharedArchiveAndExit"); - output = new OutputAnalyzer(pb.start()); - output.shouldContain("archive is invalid"); - output.shouldNotContain("Usage:"); // Should not print JVM help message - output.shouldHaveExitValue(1); // Should report failure in error code. } catch (RuntimeException e) { e.printStackTrace(); output.shouldContain("Unable to use shared archive"); diff --git a/hotspot/test/runtime/SharedArchiveFile/SharedStrings.java b/hotspot/test/runtime/SharedArchiveFile/SharedStrings.java index 50906ea5d33..67059bf5715 100644 --- a/hotspot/test/runtime/SharedArchiveFile/SharedStrings.java +++ b/hotspot/test/runtime/SharedArchiveFile/SharedStrings.java @@ -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. * * This code is free software; you can redistribute it and/or modify it @@ -32,6 +32,7 @@ * @library /testlibrary /test/lib * @modules java.base/sun.misc * java.management + * jdk.jartool/sun.tools.jar * @build SharedStringsWb SharedStrings BasicJarBuilder sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox * @run main SharedStrings @@ -40,7 +41,7 @@ import jdk.test.lib.*; public class SharedStrings { public static void main(String[] args) throws Exception { - BasicJarBuilder.build("whitebox", "sun/hotspot/WhiteBox"); + BasicJarBuilder.build(true, "whitebox", "sun/hotspot/WhiteBox"); ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( "-XX:+UnlockDiagnosticVMOptions", diff --git a/hotspot/test/runtime/SharedArchiveFile/javax/sound/sampled/MyClass.jasm b/hotspot/test/runtime/SharedArchiveFile/javax/sound/sampled/MyClass.jasm new file mode 100644 index 00000000000..5e6078cbee2 --- /dev/null +++ b/hotspot/test/runtime/SharedArchiveFile/javax/sound/sampled/MyClass.jasm @@ -0,0 +1,46 @@ +/* + * 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 javax/sound/sampled; + +public class MyClass + version 51:0 +{ + +public Method "":"()V" + stack 1 locals 1 +{ + aload_0; + invokespecial Method java/lang/Object."":"()V"; + return; +} + + +public Method toString:"()Ljava/lang/String;" + stack 1 locals 1 +{ + ldc String "hi"; + areturn; +} + +} diff --git a/hotspot/test/runtime/SharedArchiveFile/nonjdk/myPackage/MyClass.java b/hotspot/test/runtime/SharedArchiveFile/nonjdk/myPackage/MyClass.java new file mode 100644 index 00000000000..265457ca70c --- /dev/null +++ b/hotspot/test/runtime/SharedArchiveFile/nonjdk/myPackage/MyClass.java @@ -0,0 +1,30 @@ +/* + * 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 nonjdk.myPackage; + +public class MyClass { + public String toString() { + return "hi"; + } +} diff --git a/hotspot/test/runtime/SharedArchiveFile/org/omg/CORBA/Context.jasm b/hotspot/test/runtime/SharedArchiveFile/org/omg/CORBA/Context.jasm new file mode 100644 index 00000000000..c3424b9f799 --- /dev/null +++ b/hotspot/test/runtime/SharedArchiveFile/org/omg/CORBA/Context.jasm @@ -0,0 +1,46 @@ +/* + * 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 org/omg/CORBA; + +public class Context + version 51:0 +{ + +public Method "":"()V" + stack 1 locals 1 +{ + aload_0; + invokespecial Method java/lang/Object."":"()V"; + return; +} + + +public Method toString:"()Ljava/lang/String;" + stack 1 locals 1 +{ + ldc String "hi"; + areturn; +} + +} diff --git a/hotspot/test/runtime/getSysPackage/GetSysPkgTest.java b/hotspot/test/runtime/getSysPackage/GetSysPkgTest.java new file mode 100644 index 00000000000..7e3930137e7 --- /dev/null +++ b/hotspot/test/runtime/getSysPackage/GetSysPkgTest.java @@ -0,0 +1,160 @@ +/* + * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @modules java.base/jdk.internal.loader + * java.desktop + * @library /testlibrary + * @run main/othervm GetSysPkgTest + */ + +import java.io.File; +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; +import jdk.test.lib.*; + +// Test that JVM get_system_package() returns the module location for defined packages. +public class GetSysPkgTest { + + private static Object invoke(Method m, Object obj, Object... args) throws Throwable { + try { + return m.invoke(obj, args); + } catch (InvocationTargetException e) { + throw e.getCause(); + } + } + + private static Method findMethod(String name) { + for (Method m : jdk.internal.loader.BootLoader.class.getDeclaredMethods()) { + if (m.getName().equals(name)) { + m.setAccessible(true); + return m; + } + } + throw new RuntimeException("Failed to find method " + name + " in java.lang.reflect.Module"); + } + + // Throw RuntimeException if getSystemPackageLocation() does not return + // the expected location. + static void getPkg(String name, String expected_loc) throws Throwable { + String loc = (String)invoke(findMethod("getSystemPackageLocation"), null, name); + if (loc == null) { + if (expected_loc == null) return; + System.out.println("Expected location: " + expected_loc + + ", for package: " + name + ", got: null"); + } else if (expected_loc == null) { + System.out.println("Expected location: null, for package: " + + name + ", got: " + loc); + } else if (!loc.equals(expected_loc)) { + System.out.println("Expected location: " + + expected_loc + ", for package: " + name + ", got: " + loc); + } else { + return; + } + throw new RuntimeException(); + } + + public static void main(String args[]) throws Throwable { + if (args.length == 0 || !args[0].equals("do_tests")) { + + // Create a package found via -Xbootclasspath/a + String source = "package BootLdr_package; " + + "public class BootLdrPkg { " + + " public int mth() { return 4; } " + + "}"; + byte[] klassbuf = + InMemoryJavaCompiler.compile("BootLdr_package.BootLdrPkg", source); + ClassFileInstaller.writeClassToDisk("BootLdr_package/BootLdrPkg", klassbuf, "bl_dir"); + + // Create a package found via -cp. + source = "package GetSysPkg_package; " + + "public class GetSysClass { " + + " public int mth() { return 4; } " + + "}"; + klassbuf = + InMemoryJavaCompiler.compile("GetSysPkg_package.GetSysClass", source); + ClassFileInstaller.writeClassToDisk("GetSysPkg_package/GetSysClass", klassbuf); + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-Xbootclasspath/a:bl_dir", + "-XaddExports:java.base/jdk.internal.loader=ALL-UNNAMED", "-cp", "." + File.pathSeparator + + System.getProperty("test.classes"), "GetSysPkgTest", "do_tests"); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldHaveExitValue(0); + return; + } + + getPkg("java/lang", "jrt:/java.base"); + getPkg("javax/script", null); // Package not defined + + // Test a package that does not yet have any referenced classes. + // Note: if another class in com/sun/crypto/provider/ happens to get + // loaded or if class PrivateKeyInfo disappears from this package + // then this test will fail. + getPkg("com/sun/crypto/provider", null); + // Now make sure a class in the package is referenced. + Class newClass = Class.forName("com.sun.crypto.provider.PrivateKeyInfo"); + getPkg("com/sun/crypto/provider", "jrt:/java.base"); + + getPkg("java/nio/charset", "jrt:/java.base"); + + // Test a package in a module not owned by boot loader. + Class clss = Class.forName("javax.activation.DataHandler"); + if (clss == null) + throw new RuntimeException("Could not find class javax.activation.DataHandler"); + getPkg("javax/activation", null); // Not owned by boot loader + + // Test a package not in jimage file. + clss = Class.forName("GetSysPkg_package.GetSysClass"); + if (clss == null) + throw new RuntimeException("Could not find class GetSysPkg_package.GetSysClass"); + getPkg("GetSysPkg_package", null); + + // Access a class with a package in a boot loader module other than java.base + clss = Class.forName("java.awt.Button"); + if (clss == null) + throw new RuntimeException("Could not find class java.awt.Button"); + getPkg("java/awt", "jrt:/java.desktop"); + + // Test getting the package location from a class found via -Xbootclasspath/a + clss = Class.forName("BootLdr_package.BootLdrPkg"); + if (clss == null) + throw new RuntimeException("Could not find class BootLdr_package.BootLdrPkg"); + String bootldrPkg = (String)invoke(findMethod("getSystemPackageLocation"), null, "BootLdr_package"); + if (bootldrPkg == null) { + throw new RuntimeException("Expected BootLdr_package to return non-null value"); + } + if (!bootldrPkg.equals("bl_dir")) { + throw new RuntimeException("Expected BootLdr_package to return bl_dir, got: " + bootldrPkg); + } + + // Test when package's class reference is an array. + // Note: if another class in javax/crypto happens to get loaded + // or if class AEADBadTagException disappears from this package + // then this test will fail. + getPkg("javax/crypto", null); + javax.crypto.AEADBadTagException[] blah = new javax.crypto.AEADBadTagException[3]; + getPkg("javax/crypto", "jrt:/java.base"); + + } +} diff --git a/hotspot/test/runtime/logging/ModulesTest.java b/hotspot/test/runtime/logging/ModulesTest.java new file mode 100644 index 00000000000..3547a06062a --- /dev/null +++ b/hotspot/test/runtime/logging/ModulesTest.java @@ -0,0 +1,53 @@ +/* + * 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. + */ + +/* + * @test + * @summary modules=debug should have logging from statements in the code + * @library /testlibrary + * @modules java.base/sun.misc + * java.management + * @build jdk.test.lib.OutputAnalyzer jdk.test.lib.ProcessTools + * @run main ModulesTest + */ + +import jdk.test.lib.OutputAnalyzer; +import jdk.test.lib.ProcessTools; + +public class ModulesTest { + public static void main(String[] args) throws Exception { + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-Xlog:modules=trace", "-version"); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("define_javabase_module(): Definition of module:"); + output.shouldContain("define_javabase_module(): creation of package"); + output.shouldContain("define_module(): creation of module"); + output.shouldContain("define_module(): creation of package"); + output.shouldContain("set_bootloader_unnamed_module(): recording unnamed"); + output.shouldContain("add_module_exports(): package"); + output.shouldContain("add_reads_module(): Adding read from module"); + output.shouldContain("Setting package: class:"); + output.shouldHaveExitValue(0); + } +} + diff --git a/hotspot/test/compiler/jsr292/NonInlinedCall/NonInlinedReinvoker.java b/hotspot/test/runtime/modules/AccModuleTest.java similarity index 56% rename from hotspot/test/compiler/jsr292/NonInlinedCall/NonInlinedReinvoker.java rename to hotspot/test/runtime/modules/AccModuleTest.java index c4c36d3c49d..f1f8ab0bc60 100644 --- a/hotspot/test/compiler/jsr292/NonInlinedCall/NonInlinedReinvoker.java +++ b/hotspot/test/runtime/modules/AccModuleTest.java @@ -20,29 +20,23 @@ * or visit www.oracle.com if you need additional information or have any * questions. */ -package java.lang.invoke; -class NonInlinedReinvoker extends DelegatingMethodHandle { - private final MethodHandle target; +/* + * @test + * @library /testlibrary + * @modules java.base/sun.misc + * @compile acc_module.jcod + * @build AccModuleTest + * @run main AccModuleTest + */ - private NonInlinedReinvoker(MethodHandle target, LambdaForm lf) { - super(target.type(), lf); - this.target = target; - } - @Override - protected MethodHandle getTarget() { - return target; - } +import java.io.File; +import jdk.test.lib.*; - @Override - MethodHandle asTypeUncached(MethodType newType) { - return asTypeCache = target.asType(newType); - } +public class AccModuleTest { - static MethodHandle make(MethodHandle target) { - LambdaForm lform = DelegatingMethodHandle.makeReinvokerForm( - target, -1, DelegatingMethodHandle.class, "reinvoker.dontInline", - /*forceInline=*/false, DelegatingMethodHandle.NF_getTarget, null); - return new NonInlinedReinvoker(target, lform); + public static void main(String args[]) throws Throwable { + System.out.println("Test that ACC_MODULE in class access flags does not cause ClassFormatError"); + Class clss = Class.forName("acc_module"); } } diff --git a/hotspot/test/runtime/modules/AccessCheck/CheckRead.java b/hotspot/test/runtime/modules/AccessCheck/CheckRead.java new file mode 100644 index 00000000000..ec88eef6525 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/CheckRead.java @@ -0,0 +1,138 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can not read module m2, then class p1.c1 + * in module m1 can not access p2.c2 in module m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build CheckRead + * @run main/othervm -Xbootclasspath/a:. CheckRead + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// defines m2 --> packages p2 +// defines m3 --> packages p3 +// +// m1 can not read m2 +// package p2 in m2 is exported to m1 +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2. +// Access denied since m1 can not read m2. +// +public class CheckRead { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publicly defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m3 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m3") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: p2 is exported to m1 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m1") + .build(); + + // Define module: m3 + // Can read: java.base, m2 + // Packages: p3 + // Packages exported: none + ModuleDescriptor descriptor_m3 = + new ModuleDescriptor.Builder("m3") + .requires("java.base") + .requires("m2") + .conceals("p3") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2, descriptor_m3); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + map.put("m3", MySameClassLoader.loader1); + + // Create Layer that contains m1, m2 and m3 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m3") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is exported to m1 but m2 is not readable from m1)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("cannot access")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + CheckRead test = new CheckRead(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/DiffCL_CheckRead.java b/hotspot/test/runtime/modules/AccessCheck/DiffCL_CheckRead.java new file mode 100644 index 00000000000..7ccf918d232 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/DiffCL_CheckRead.java @@ -0,0 +1,138 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can not read module m2, then class p1.c1 + * in module m1 can not access p2.c2 in module m2. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build DiffCL_CheckRead + * @run main/othervm -Xbootclasspath/a:. DiffCL_CheckRead + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// ClassLoader2 --> defines m2 --> packages p2 +// defines m3 --> packages p3 +// +// m1 can not read m2 +// package p2 in m2 is exported to m1 +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2. +// Access denied since m1 can not read m2. +// +public class DiffCL_CheckRead { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publicly defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m3 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m3") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: p2 is exported to m1 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m1") + .build(); + + // Define module: m3 + // Can read: java.base, m2 + // Packages: p3 + // Packages exported: none + ModuleDescriptor descriptor_m3 = + new ModuleDescriptor.Builder("m3") + .requires("java.base") + .requires("m2") + .conceals("p3") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2, descriptor_m3); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + map.put("m3", MyDiffClassLoader.loader2); + + // Create Layer that contains m1, m2 and m3 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("m3") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is exported to m1 but m2 is not readable from m1)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("cannot access")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + DiffCL_CheckRead test = new DiffCL_CheckRead(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpQualOther.java b/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpQualOther.java new file mode 100644 index 00000000000..c815a85fac7 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpQualOther.java @@ -0,0 +1,140 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can read module m2, but package p2 in m2 + * is exported specifically to module m3, then class p1.c1 in m1 can not + * access p2.c2 in m2. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build DiffCL_ExpQualOther + * @run main/othervm -Xbootclasspath/a:. DiffCL_ExpQualOther + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// ClassLoader2 --> defines m2 --> packages p2 +// defines m3 --> packages p3 +// +// m1 can read m2 +// package p2 in m2 is exported to m3 +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2 +// Access denied since although m1 can read m2, p2 is exported only to m3. +// +public class DiffCL_ExpQualOther { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2, m3 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .requires("m3") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base, m3 + // Packages: p2 + // Packages exported: p2 is exported to m3 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m3") + .build(); + + // Define module: m3 + // Can read: java.base, m2 + // Packages: p3 + // Packages exported: none + ModuleDescriptor descriptor_m3 = + new ModuleDescriptor.Builder("m3") + .requires("java.base") + .requires("m2") + .conceals("p3") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2, descriptor_m3); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + map.put("m3", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("m3") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is exported to m3 not to m1)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + DiffCL_ExpQualOther test = new DiffCL_ExpQualOther(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpQualToM1.java b/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpQualToM1.java new file mode 100644 index 00000000000..c1c38fd61dc --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpQualToM1.java @@ -0,0 +1,120 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p1.c1 defined in m1 tries to access p2.c2 defined in m2. + * Access allowed since m1 can read m2 and package p2 is exported to m1. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build DiffCL_ExpQualToM1 + * @run main/othervm -Xbootclasspath/a:. DiffCL_ExpQualToM1 + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// ClassLoader2 --> defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is exported to m1 +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2 +// Access allowed since m1 can read m2 and package p2 is exported to m1. +// +public class DiffCL_ExpQualToM1 { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: p1 + // Packages exported: p1 is exported to unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: package p2 is exported to m1 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, an IAE should not be thrown since p2 is exported qualifiedly to m1"); + } + } + + public static void main(String args[]) throws Throwable { + DiffCL_ExpQualToM1 test = new DiffCL_ExpQualToM1(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpUnqual.java b/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpUnqual.java new file mode 100644 index 00000000000..a8668333d73 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/DiffCL_ExpUnqual.java @@ -0,0 +1,121 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can read module m2, and package p2 in m2 is + * exported unqualifiedly, then class p1.c1 in m1 can read p2.c2 in m2. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build DiffCL_ExpUnqual + * @run main/othervm -Xbootclasspath/a:. DiffCL_ExpUnqual + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// ClassLoader2 --> defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is exported to m1 +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2 +// Access allowed since m1 can read m2 and package p2 is exported +// unqualifiedly. +// +public class DiffCL_ExpUnqual { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: package p2 is exported to m1 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, an IAE should not be thrown since p2 is exported qualifiedly to m1"); + } + } + + public static void main(String args[]) throws Throwable { + DiffCL_ExpUnqual test = new DiffCL_ExpUnqual(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/DiffCL_PkgNotExp.java b/hotspot/test/runtime/modules/AccessCheck/DiffCL_PkgNotExp.java new file mode 100644 index 00000000000..d4f3d62d4ec --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/DiffCL_PkgNotExp.java @@ -0,0 +1,125 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can read module m2, but package p2 in m2 is not + * exported, then class p1.c1 in m1 can not read p2.c2 in m2. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build DiffCL_PkgNotExp + * @run main/othervm -Xbootclasspath/a:. DiffCL_PkgNotExp + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// ClassLoader2 --> defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is not exported +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2 +// Access denied since p2 is not exported. +// +public class DiffCL_PkgNotExp { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: none + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .conceals("p2") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is not exported)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + DiffCL_PkgNotExp test = new DiffCL_PkgNotExp(); + test.createLayerOnBoot(); + } +} + diff --git a/hotspot/test/runtime/modules/AccessCheck/DiffCL_Umod.java b/hotspot/test/runtime/modules/AccessCheck/DiffCL_Umod.java new file mode 100644 index 00000000000..6df74bfa689 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/DiffCL_Umod.java @@ -0,0 +1,230 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p1.c1 defined in m1 tries to access p2.c2 defined in unnamed module. + * @library /testlibrary /test/lib + * @modules java.base/jdk.internal.module + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @compile p1/c1ReadEdgeDiffLoader.java + * @compile p1/c1Loose.java + * @build DiffCL_Umod + * @run main/othervm -Xbootclasspath/a:. DiffCL_Umod + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.lang.reflect.Layer; +import java.lang.reflect.Module; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// package p1 in m1 is exported unqualifiedly +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in +// in unnamed module. +// +// Three access attempts occur in this test: +// 1. The first access is not allowed because a strict module +// cannot read an unnamed module. +// 2. In this scenario a strict module establishes readability +// to the particular unnamed module it is trying to access. +// Access is allowed. +// 3. Module m1 in the test_looseModuleLayer() method +// is transitioned to a loose module, access +// to all unnamed modules is allowed. +// +public class DiffCL_Umod { + + // Create Layers over the boot layer to test different + // accessing scenarios of a named module to an unnamed module. + + // Module m1 is a strict module and has not established + // readability to an unnamed module that p2.c2 is defined in. + public void test_strictModuleLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + MyDiffClassLoader.loader1 = new MyDiffClassLoader(); + MyDiffClassLoader.loader2 = new MyDiffClassLoader(); + + // map module m1 to class loader. + // class c2 will be loaded in an unnamed module/loader2 + // to achieve differing class loaders. + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + + // Attempt access + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Test Failed, strict module m1 should not be able " + + "to access public type p2.c2 defined in unnamed module"); + } catch (IllegalAccessError e) { + } +} + + // Module m1 is a strict module and has established + // readability to an unnamed module that p2.c2 is defined in. + public void test_strictModuleUnnamedReadableLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + MyDiffClassLoader.loader1 = new MyDiffClassLoader(); + MyDiffClassLoader.loader2 = new MyDiffClassLoader(); + + // map module m1 to class loader. + // class c2 will be loaded in an unnamed module/loader2 + // to achieve differing class loaders. + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1ReadEdgeDiffLoader + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1ReadEdgeDiffLoader"); + + try { + // Read edge between m1 and the unnamed module that loads p2.c2 is established in + // c1ReadEdgeDiffLoader's ctor before attempting access. + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, module m1 has established readability to p2/c2 loader's " + + "unnamed module, access should be allowed: " + e.getMessage()); + } + } + + // Module m1 is a loose module and thus can read all unnamed modules. + public void test_looseModuleLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + MyDiffClassLoader.loader1 = new MyDiffClassLoader(); + MyDiffClassLoader.loader2 = new MyDiffClassLoader(); + + // map module m1 to class loader. + // class c2 will be loaded in an unnamed module/loader2 + // to achieve differing class loaders. + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1Loose + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1Loose"); + + // change m1 to be a loose module + Module m1 = layer.findModule("m1").get(); + jdk.internal.module.Modules.addReads(m1, null); + + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, loose module m1 should be able to access " + + "public type p2.c2 defined in unnamed module: " + e.getMessage()); + } + } + + public static void main(String args[]) throws Throwable { + DiffCL_Umod test = new DiffCL_Umod(); + test.test_strictModuleLayer(); // access denied + test.test_strictModuleUnnamedReadableLayer(); // access allowed + test.test_looseModuleLayer(); // access allowed + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/DiffCL_UmodUpkg.java b/hotspot/test/runtime/modules/AccessCheck/DiffCL_UmodUpkg.java new file mode 100644 index 00000000000..5cd133db695 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/DiffCL_UmodUpkg.java @@ -0,0 +1,173 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p3.c3 defined in module m1 tries to access c4 defined in an unnamed package + * and an unnamed module. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile c4.java + * @compile p3/c3.jcod + * @compile p3/c3ReadEdgeDiffLoader.jcod + * @build DiffCL_UmodUpkg + * @run main/othervm -Xbootclasspath/a:. DiffCL_UmodUpkg + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p3 +// package p3 in m1 is exported unqualifiedly +// +// class p3.c3 defined in m1 tries to access c4 defined in +// in unnamed module. +// +// Two access attempts occur in this test: +// 1. The first access is not allowed because a strict module +// cannot read an unnamed module. +// 2. In this scenario a strict module establishes readability +// to the particular unnamed module it is trying to access. +// Access is allowed. +// +public class DiffCL_UmodUpkg { + + // Create Layers over the boot layer to test different + // accessing scenarios of a named module to an unnamed module. + + // Module m1 is a strict module and has not established + // readability to an unnamed module that c4 is defined in. + public void test_strictModuleLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p3 + // Packages exported: p3 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p3") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + MyDiffClassLoader.loader1 = new MyDiffClassLoader(); + MyDiffClassLoader.loader2 = new MyDiffClassLoader(); + + // map module m1 to class loader. + // class c2 will be loaded in an unnamed module/loader2 + // to achieve differing class loaders. + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p3.c3 + Class p3_c3_class = MyDiffClassLoader.loader1.loadClass("p3.c3"); + + // Attempt access + try { + p3_c3_class.newInstance(); + throw new RuntimeException("Test Failed, strict module m1 should not be able to access " + + "public type c4 defined in unnamed module"); + } catch (IllegalAccessError e) { + } +} + + // Module m1 is a strict module and has established + // readability to an unnamed module that c4 is defined in. + public void test_strictModuleUnnamedReadableLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p3 + // Packages exported: p3 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p3") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + MyDiffClassLoader.loader1 = new MyDiffClassLoader(); + MyDiffClassLoader.loader2 = new MyDiffClassLoader(); + + // map module m1 to class loader. + // class c2 will be loaded in an unnamed module/loader2 + // to achieve differing class loaders. + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p3.c3ReadEdgeDiffLoader + Class p3_c3_class = MyDiffClassLoader.loader1.loadClass("p3.c3ReadEdgeDiffLoader"); + + try { + // Read edge between m1 and the unnamed module that loads c4 is established in + // C3ReadEdgeDiffLoader's ctor before attempting access. + p3_c3_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, module m1 has established readability to " + + "c4 loader's unnamed module, access should be allowed: " + e.getMessage()); + } + } + + public static void main(String args[]) throws Throwable { + DiffCL_UmodUpkg test = new DiffCL_UmodUpkg(); + test.test_strictModuleLayer(); // access denied + test.test_strictModuleUnnamedReadableLayer(); // access allowed + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/ExpQualOther.java b/hotspot/test/runtime/modules/AccessCheck/ExpQualOther.java new file mode 100644 index 00000000000..4645d096746 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/ExpQualOther.java @@ -0,0 +1,140 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can read module m2, but package p2 in m2 + * is exported specifically to module m3, then class p1.c1 in m1 can not + * access p2.c2 in m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build ExpQualOther + * @run main/othervm -Xbootclasspath/a:. ExpQualOther + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// defines m2 --> packages p2 +// defines m3 --> packages p3 +// +// m1 can read m2 +// package p2 in m2 is exported to m3 +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2 +// Access denied since although m1 can read m2, p2 is exported only to m3. +// +public class ExpQualOther { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2, m3 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .requires("m3") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: p2 is exported to m3 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m3") + .build(); + + // Define module: m3 + // Can read: java.base, m2 + // Packages: p3 + // Packages exported: none + ModuleDescriptor descriptor_m3 = + new ModuleDescriptor.Builder("m3") + .requires("java.base") + .requires("m2") + .conceals("p3") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2, descriptor_m3); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + map.put("m3", MySameClassLoader.loader1); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m3") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is exported to m3 not to m1)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + ExpQualOther test = new ExpQualOther(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/ExpQualToM1.java b/hotspot/test/runtime/modules/AccessCheck/ExpQualToM1.java new file mode 100644 index 00000000000..fa0b0944ff1 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/ExpQualToM1.java @@ -0,0 +1,110 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can read module m2, AND package p2 in m2 is + * exported qualifiedly to m1, then class p1.c1 in m1 can read p2.c2 in m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build ExpQualToM1 + * @run main/othervm -Xbootclasspath/a:. ExpQualToM1 + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +public class ExpQualToM1 { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: p2 is exported qualifiedly to m1 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to the same class loader for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, an IAE should not be thrown since p2 is exported qualifiedly to m1"); + } + } + + public static void main(String args[]) throws Throwable { + ExpQualToM1 test = new ExpQualToM1(); + test.createLayerOnBoot(); + } + +} diff --git a/hotspot/test/runtime/modules/AccessCheck/ExpUnqual.java b/hotspot/test/runtime/modules/AccessCheck/ExpUnqual.java new file mode 100644 index 00000000000..a7c74ba5084 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/ExpUnqual.java @@ -0,0 +1,109 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can read module m2, AND package p2 in module2 is + * exported unqualifiedly, then class p1.c1 in m1 can read p2.c2 in m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build ExpUnqual + * @run main/othervm -Xbootclasspath/a:. ExpUnqual + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +public class ExpUnqual { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: p2 is exported unqualifiedly + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to the same class loader for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, an IAE should not be thrown since p2 is exported unqualifiedly."); + } + } + + public static void main(String args[]) throws Throwable { + ExpUnqual test = new ExpUnqual(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/ExportAllUnnamed.java b/hotspot/test/runtime/modules/AccessCheck/ExportAllUnnamed.java new file mode 100644 index 00000000000..cb09a5315d6 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/ExportAllUnnamed.java @@ -0,0 +1,128 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test if package p2 in module m2 is exported to all unnamed, + * then class p1.c1 in an unnamed module can read p2.c2 in module m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @compile -XaddExports:java.base/jdk.internal.module=ALL-UNNAMED ExportAllUnnamed.java + * @run main/othervm -XaddExports:java.base/jdk.internal.module=ALL-UNNAMED -Xbootclasspath/a:. ExportAllUnnamed + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.lang.reflect.Layer; +import java.lang.reflect.Module; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is exported unqualifiedly +// +// class p1.c1 defined in an unnamed module tries to access p2.c2 defined in m2 +// Access allowed, an unnamed module can read all modules and p2 in module +// m2 is exported to all unnamed modules. + +public class ExportAllUnnamed { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: p2 is exported unqualifiedly + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + Class p2_c2_class = MySameClassLoader.loader1.loadClass("p2.c2"); + Module m2 = p2_c2_class.getModule(); + + // Export m2/p2 to all unnamed modules. + jdk.internal.module.Modules.addExportsToAllUnnamed(m2, "p2"); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, unnamed module failed to access public type p2.c2 " + + "that was exported to all unnamed"); + } + } + + public static void main(String args[]) throws Throwable { + ExportAllUnnamed test = new ExportAllUnnamed(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/ModuleLibrary.java b/hotspot/test/runtime/modules/AccessCheck/ModuleLibrary.java new file mode 100644 index 00000000000..54872b9ec04 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/ModuleLibrary.java @@ -0,0 +1,82 @@ +/* + * 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. + */ + +import java.io.IOException; +import java.lang.module.ModuleReference; +import java.lang.module.ModuleFinder; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleReader; +import java.net.URI; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Optional; +import java.util.Set; +import java.util.function.Supplier; + +/** + * A container of modules that acts as a ModuleFinder for testing + * purposes. + */ + +class ModuleLibrary implements ModuleFinder { + private final Map namesToReference = new HashMap<>(); + + private ModuleLibrary() { } + + void add(ModuleDescriptor... descriptors) { + for (ModuleDescriptor descriptor: descriptors) { + String name = descriptor.name(); + if (!namesToReference.containsKey(name)) { + //modules.add(descriptor); + + URI uri = URI.create("module:/" + descriptor.name()); + + Supplier supplier = () -> { + throw new UnsupportedOperationException(); + }; + + ModuleReference mref = new ModuleReference(descriptor, uri, supplier); + + namesToReference.put(name, mref); + } + } + } + + static ModuleLibrary of(ModuleDescriptor... descriptors) { + ModuleLibrary ml = new ModuleLibrary(); + ml.add(descriptors); + return ml; + } + + @Override + public Optional find(String name) { + return Optional.ofNullable(namesToReference.get(name)); + } + + @Override + public Set findAll() { + return new HashSet<>(namesToReference.values()); + } +} + diff --git a/hotspot/test/runtime/modules/AccessCheck/PkgNotExp.java b/hotspot/test/runtime/modules/AccessCheck/PkgNotExp.java new file mode 100644 index 00000000000..b8a56173988 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/PkgNotExp.java @@ -0,0 +1,123 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if module m1 can read module m2, but package p2 in m2 is not + * exported, then class p1.c1 in m1 can not read p2.c2 in m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build PkgNotExp + * @run main/othervm -Xbootclasspath/a:. PkgNotExp + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is not exported +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2 +// Access denied since p2 is not exported. +// +public class PkgNotExp { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .exports("p1") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: none + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .conceals("p2") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to the same class loader for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + + // Create Layer that contains m1 and m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is not exported)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + PkgNotExp test = new PkgNotExp(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/Umod.java b/hotspot/test/runtime/modules/AccessCheck/Umod.java new file mode 100644 index 00000000000..639891b4339 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/Umod.java @@ -0,0 +1,221 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p1.c1 defined in m1 tries to access p2.c2 defined in unnamed module. + * @library /testlibrary /test/lib + * @modules java.base/jdk.internal.module + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @compile p1/c1ReadEdge.java + * @compile p1/c1Loose.java + * @build Umod + * @run main/othervm -Xbootclasspath/a:. Umod + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.lang.reflect.Module; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p1 +// package p1 in m1 is exported unqualifiedly +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in +// in unnamed module. +// +// Three access attempts occur in this test: +// 1. The first access is not allowed because a strict module +// cannot read an unnamed module. +// 2. In this scenario a strict module establishes readability +// to the particular unnamed module it is trying to access. +// Access is allowed. +// 3. Module m1 in the test_looseModuleLayer() method +// is transitioned to a loose module, access +// to all unnamed modules is allowed. +// +public class Umod { + + // Create Layers over the boot layer to test different + // accessing scenarios of a named module to an unnamed module. + + // Module m1 is a strict module and has not established + // readability to an unnamed module that p2.c2 is defined in. + public void test_strictModuleLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map module m1 to class loader. + // class c2 will be loaded in an unnamed module/loader. + MySameClassLoader loader = new MySameClassLoader(); + Map map = new HashMap<>(); + map.put("m1", loader); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == loader); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = loader.loadClass("p1.c1"); + + // Attempt access + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Test Failed, strict module m1, type p1.c1, should not be able " + + "to access public type p2.c2 defined in unnamed module"); + } catch (IllegalAccessError e) { + } + } + + // Module m1 is a strict module and has established + // readability to an unnamed module that p2.c2 is defined in. + public void test_strictModuleUnnamedReadableLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + MySameClassLoader loader = new MySameClassLoader(); + // map module m1 to class loader. + // class c2 will be loaded in an unnamed module/loader. + Map map = new HashMap<>(); + map.put("m1", loader); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == loader); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1ReadEdge + Class p1_c1_class = loader.loadClass("p1.c1ReadEdge"); + + try { + // Read edge between m1 and the unnamed module that loads p2.c2 is established in + // c1ReadEdge's ctor before attempting access. + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, strict module m1, type p1.c1ReadEdge, should be able to acccess public type " + + "p2.c2 defined in unnamed module: " + e.getMessage()); + } +} + + // Module m1 is a loose module and thus can read all unnamed modules. + public void test_looseModuleLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p1 + // Packages exported: p1 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + MySameClassLoader loader = new MySameClassLoader(); + // map module m1 to class loader. + // class c2 will be loaded in an unnamed module/loader. + Map map = new HashMap<>(); + map.put("m1", loader); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == loader); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1Loose + Class p1_c1_class = loader.loadClass("p1.c1Loose"); + + // change m1 to be a loose module + Module m1 = layer.findModule("m1").get(); + jdk.internal.module.Modules.addReads(m1, null); + + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, strict module m1, type p1.c1Loose, should be able to acccess public type " + + "p2.c2 defined in unnamed module: " + e.getMessage()); + } + } + + public static void main(String args[]) throws Throwable { + Umod test = new Umod(); + test.test_strictModuleLayer(); // access denied + test.test_strictModuleUnnamedReadableLayer(); // access allowed + test.test_looseModuleLayer(); // access allowed + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_ExpQualOther.java b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_ExpQualOther.java new file mode 100644 index 00000000000..34ea50f6c1e --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_ExpQualOther.java @@ -0,0 +1,127 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p1.c1 defined in an unnamed module tries to access p2.c2 defined in m2. + * Access is denied, since an unnamed module can read all modules but p2 in module + * m2 is exported specifically to module m1, not to all modules. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build UmodDiffCL_ExpQualOther + * @run main/othervm -Xbootclasspath/a:. UmodDiffCL_ExpQualOther + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// ClassLoader2 --> defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is not exported +// +// class p1.c1 defined in an unnamed module tries to access p2.c2 defined in m2 +// Access denied, an unnamed module can read all modules but p2 in module +// m2 is exported specifically to module m1 not to all modules. +// +public class UmodDiffCL_ExpQualOther { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: none + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + // NOTE: module m1 does not define a package named p1. + // p1 will be loaded in an unnamed module. + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is exported to m1, not unqualifiedly"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + UmodDiffCL_ExpQualOther test = new UmodDiffCL_ExpQualOther(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_ExpUnqual.java b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_ExpUnqual.java new file mode 100644 index 00000000000..52d253b352c --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_ExpUnqual.java @@ -0,0 +1,122 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p1.c1 defined in an unnamed module tries to access p2.c2 defined in m2. + * Access allowed, an unnamed module can read all modules and p2 in module m2 + * which is exported unqualifiedly. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build UmodDiffCL_ExpUnqual + * @run main/othervm -Xbootclasspath/a:. UmodDiffCL_ExpUnqual + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// ClassLoader2 --> defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is exported unqualifiedly. +// +// class p1.c1 defined in an unnamed module tries to access p2.c2 defined in m2 +// Access allowed, an unnamed module can read all modules and p2 in module +// m2 which is exported unqualifiedly. +// +public class UmodDiffCL_ExpUnqual { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: none + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // NOTE: module m1 does not define a package named p1. + // p1 will be loaded in an unnamed module. + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, p1.c1 defined in unnamed module can access p2.c2 in module m2"); + } + } + + public static void main(String args[]) throws Throwable { + UmodDiffCL_ExpUnqual test = new UmodDiffCL_ExpUnqual(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_PkgNotExp.java b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_PkgNotExp.java new file mode 100644 index 00000000000..ebe3e5d004e --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_PkgNotExp.java @@ -0,0 +1,126 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p1.c1 defined in unnamed module tries to access p2.c2 defined in m2. + * Access is denied since even though unnamed module can read all modules, p2 + * in module m2 is not exported at all. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p1/c1.java + * @build UmodDiffCL_PkgNotExp + * @run main/othervm -Xbootclasspath/a:. UmodDiffCL_PkgNotExp + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// ClassLoader2 --> defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is not exported +// +// class p1.c1 defined in unnamed module tries to access p2.c2 defined in m2 +// Access denied since even though unnamed module can read all modules, p2 +// in module m2 is not exported at all. +// +public class UmodDiffCL_PkgNotExp { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: none + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .conceals("p2") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + // NOTE: module m1 does not define a package named p1. + // p1 will be loaded in an unnamed module. + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is not exported to an unnamed module)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + UmodDiffCL_PkgNotExp test = new UmodDiffCL_PkgNotExp(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_Umod.java b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_Umod.java new file mode 100644 index 00000000000..6fd267961b1 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_Umod.java @@ -0,0 +1,56 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p1.c1 defined in an unnamed module tries to access p2.c2 + * defined in an unnamed module. Access allowed since unnamed module + * can read unnamed module even when class p1.c1 is loaded by + * a different loader than p2.c2. + * @compile myloaders/MyDiffClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build UmodDiffCL_Umod + * @run main/othervm -Xbootclasspath/a:. UmodDiffCL_Umod + */ + +import myloaders.MyDiffClassLoader; + +// class p1.c1 defined in an unnamed module tries to access p2.c2 defined in +// in an unnamed module. +// Access allowed since unnamed module can read unnamed module even when +// class p1.c1 is loaded by a different loader than p2.c2 +// and all packages in an unnamed module are exported unqualifiedly. +public class UmodDiffCL_Umod { + + public static void main(String args[]) throws Throwable { + Class p1_c1_class = MyDiffClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, unnamed module can access unnamed module"); + } + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_UmodUpkg.java b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_UmodUpkg.java new file mode 100644 index 00000000000..909ab0ffb29 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodDiffCL_UmodUpkg.java @@ -0,0 +1,58 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p3.c3 defined in a named package in an unnamed module tries to access c4 + * defined in an unnamed package in an unnamed module. Access allowed since + * any class in an unnamed module can read an unnamed module. + * @compile myloaders/MyDiffClassLoader.java + * @compile c4.java + * @compile p3/c3.jcod + * @build UmodDiffCL_UmodUpkg + * @run main/othervm -Xbootclasspath/a:. UmodDiffCL_UmodUpkg + */ + +import myloaders.MyDiffClassLoader; + +public class UmodDiffCL_UmodUpkg { + + public void testAccess() throws Throwable { + + Class p3_c3_class = MyDiffClassLoader.loader1.loadClass("p3.c3"); + try { + p3_c3_class.newInstance(); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + throw new RuntimeException("Test Failed, public type c3 defined in an unnamed module " + + "should be able to access public type c4 defined in an unnamed module"); + } + } + + public static void main(String args[]) throws Throwable { + UmodDiffCL_UmodUpkg test = new UmodDiffCL_UmodUpkg(); + test.testAccess(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodUPkg.java b/hotspot/test/runtime/modules/AccessCheck/UmodUPkg.java new file mode 100644 index 00000000000..97f781c8c7a --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodUPkg.java @@ -0,0 +1,167 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p3.c3 defined in module m1 tries to access c4 defined in unnamed module. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile c4.java + * @compile p3/c3.jcod + * @compile p3/c3ReadEdge.jcod + * @build UmodUPkg + * @run main/othervm -Xbootclasspath/a:. UmodUPkg + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.lang.reflect.Layer; +import java.lang.reflect.Module; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> packages p3 +// package p3 in m1 is exported unqualifiedly +// +// class p3.c3 defined in m1 tries to access c4 defined in +// in unnamed module. +// +// Two access attempts occur in this test: +// 1. The first access is not allowed because a strict module +// cannot read an unnamed module. +// 2. In this scenario a strict module establishes readability +// to the particular unnamed module it is trying to access. +// Access is allowed. +// +public class UmodUPkg { + + // Create Layers over the boot layer to test different + // accessing scenarios of a named module to an unnamed module. + + // Module m1 is a strict module and has not established + // readability to an unnamed module that c4 is defined in. + public void test_strictModuleLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p3 + // Packages exported: p3 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p3") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map module m1 to class loader. + // class c4 will be loaded in an unnamed module/loader. + MySameClassLoader loader = new MySameClassLoader(); + Map map = new HashMap<>(); + map.put("m1", loader); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == loader); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p3.c3 + Class p3_c3_class = loader.loadClass("p3.c3"); + + // Attempt access + try { + p3_c3_class.newInstance(); + throw new RuntimeException("Test Failed, strict module m1, type p3.c3, should not be able to access " + + "public type c4 defined in unnamed module"); + } catch (IllegalAccessError e) { + } + } + + // Module m1 is a strict module and has established + // readability to an unnamed module that c4 is defined in. + public void test_strictModuleUnnamedReadableLayer() throws Throwable { + + // Define module: m1 + // Can read: java.base + // Packages: p3 + // Packages exported: p3 is exported unqualifiedly + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .exports("p3") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + MySameClassLoader loader = new MySameClassLoader(); + // map module m1 to class loader. + // class c4 will be loaded in an unnamed module/loader. + Map map = new HashMap<>(); + map.put("m1", loader); + + // Create Layer that contains m1 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == loader); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p3.c3ReadEdge + Class p3_c3_class = loader.loadClass("p3.c3ReadEdge"); + + try { + // Read edge between m1 and the unnamed module that loads c4 is established in + // c3ReadEdge's ctor before attempting access. + p3_c3_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, module m1, type p3.c3ReadEdge, has established readability to " + + "c4 loader's unnamed module, access should be allowed: " + e.getMessage()); + } + } + + public static void main(String args[]) throws Throwable { + UmodUPkg test = new UmodUPkg(); + test.test_strictModuleLayer(); // access denied + test.test_strictModuleUnnamedReadableLayer(); // access allowed + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_ExpQualOther.java b/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_ExpQualOther.java new file mode 100644 index 00000000000..299a41f4003 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_ExpQualOther.java @@ -0,0 +1,125 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class c5 defined in an unnamed module tries to access p6.c6 defined in m2. + * Access is denied, since an unnamed module can read all modules but p6 in module + * m2 is exported specifically to module m1, not to all modules. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p6/c6.java + * @compile c5.java + * @build UmodUpkgDiffCL_ExpQualOther + * @run main/othervm -Xbootclasspath/a:. UmodUpkgDiffCL_ExpQualOther + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// ClassLoader2 --> defines m2 --> packages p6 +// +// m1 can read m2 +// package p6 in m2 is not exported +// +// class c5 defined in an unnamed module tries to access p6.c6 defined in m2 +// Access denied, an unnamed module can read all modules but p6 in module +// m2 is exported specifically to module m1 not to all modules. +// +public class UmodUpkgDiffCL_ExpQualOther { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p6 + // Packages exported: p6 exported to m1 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p6", "m1") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class c5 + Class c5_class = MyDiffClassLoader.loader1.loadClass("c5"); + try { + c5_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p6 in m2 is exported to m1, not unqualifiedly"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + UmodUpkgDiffCL_ExpQualOther test = new UmodUpkgDiffCL_ExpQualOther(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_NotExp.java b/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_NotExp.java new file mode 100644 index 00000000000..9210c295ac7 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_NotExp.java @@ -0,0 +1,127 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class c5 in an unnamed module can read module m2, but package p6 in module m2 is not exported. + * Access denied since even though unnamed module can read all modules, p6 in module m2 is not exported at all. + * @library /testlibrary /test/lib + * @compile myloaders/MyDiffClassLoader.java + * @compile p6/c6.java + * @compile c5.java + * @build UmodUpkgDiffCL_NotExp + * @run main/othervm -Xbootclasspath/a:. UmodUpkgDiffCL_NotExp + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MyDiffClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// ClassLoader2 --> defines m2 --> packages p6 +// +// m1 can read m2 +// package p6 in m2 is not exported +// +// class c5 defined in unnamed module tries to access p6.c6 defined in m2 +// Access denied since even though unnamed module can read all modules, p6 +// in module m2 is not exported at all. +// +public class UmodUpkgDiffCL_NotExp { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p6 + // Packages exported: none + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .conceals("p6") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MyDiffClassLoader.loader1); + map.put("m2", MyDiffClassLoader.loader2); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MyDiffClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MyDiffClassLoader.loader2); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class c5 + // NOTE: module m1 does not define any packages. + // c5 will be loaded in an unnamed module. + Class c5_class = MyDiffClassLoader.loader1.loadClass("c5"); + try { + c5_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p6 in m2 is not exported to " + + "an unnamed module that c5 is defined within)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + UmodUpkgDiffCL_NotExp test = new UmodUpkgDiffCL_NotExp(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_Umod.java b/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_Umod.java new file mode 100644 index 00000000000..597389546d8 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodUpkgDiffCL_Umod.java @@ -0,0 +1,57 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test public type c5 defined in an unnamed package and unnamed module can + * access public type p6.c6 defined in an unnamed module. + * @compile myloaders/MyDiffClassLoader.java + * @compile p6/c6.java + * @compile c5.java + * @build UmodUpkgDiffCL_Umod + * @run main/othervm -Xbootclasspath/a:. UmodUpkgDiffCL_Umod + */ + +import myloaders.MyDiffClassLoader; + +public class UmodUpkgDiffCL_Umod { + + public void testAccess() throws Throwable { + + Class c5_class = MyDiffClassLoader.loader1.loadClass("c5"); + try { + c5_class.newInstance(); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + throw new RuntimeException("Test Failed, public type c5 defined in an unnamed package and unnamed " + + "module should be able to access public type p6.c6 defined in an unnamed module"); + } + } + + public static void main(String args[]) throws Throwable { + UmodUpkgDiffCL_Umod test = new UmodUpkgDiffCL_Umod(); + test.testAccess(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_ExpQualOther.java b/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_ExpQualOther.java new file mode 100644 index 00000000000..1084294a1e5 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_ExpQualOther.java @@ -0,0 +1,136 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if class c5 in an unnamed module can read package p6 in module m2, but package p6 in module m2 is + * exported qualifiedly to module m3, then class c5 in an unnamed module can not read p6.c6 in module m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p6/c6.java + * @compile c5.java + * @build UmodUpkg_ExpQualOther + * @run main/othervm -Xbootclasspath/a:. UmodUpkg_ExpQualOther + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// defines m2 --> packages p6 +// defines m3 --> packages p3 +// +// m1 can read m2 +// package p6 in m2 is exported to m3 +// +// class c5 defined in m1 tries to access p6.c6 defined in m2 +// Access denied since although m1 can read m2, p6 is exported only to m3. +// +public class UmodUpkg_ExpQualOther { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 (need to define m1 to establish the Layer successfully) + // Can read: java.base, m2, m3 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .requires("m3") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p6 + // Packages exported: p6 is exported to m3 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p6", "m3") + .build(); + + // Define module: m3 + // Can read: java.base + // Packages: p3 + // Packages exported: none + ModuleDescriptor descriptor_m3 = + new ModuleDescriptor.Builder("m3") + .requires("java.base") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2, descriptor_m3); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + map.put("m3", MySameClassLoader.loader1); + + // Create Layer that contains m1, m2 and m3 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m3") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class c5 + Class c5_class = MySameClassLoader.loader1.loadClass("c5"); + try { + c5_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p6 in m2 is exported to m3, not unqualifiedly to everyone)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + UmodUpkg_ExpQualOther test = new UmodUpkg_ExpQualOther(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_NotExp.java b/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_NotExp.java new file mode 100644 index 00000000000..8ee6e92921c --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_NotExp.java @@ -0,0 +1,122 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test if package p6 in module m2 is not exported, then class c5 + * in an unnamed module can not access p6.c2 in module m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p6/c6.java + * @compile c5.java + * @build UmodUpkg_NotExp + * @run main/othervm -Xbootclasspath/a:. UmodUpkg_NotExp + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// ClassLoader1 --> defines m1 --> no packages +// defines m2 --> packages p6 +// +// m1 can read m2 +// package p6 in m2 is not exported +// +// class c5 defined in an unnamed module tries to access p6.c2 defined in m2 +// Access denied since p6 is not exported. +// +public class UmodUpkg_NotExp { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p6 + // Packages exported: none + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .conceals("p6") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to the same class loader for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + + // Create Layer that contains m1 and m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class c5 + Class c5_class = MySameClassLoader.loader1.loadClass("c5"); + try { + c5_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p6 in m2 is not exported)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + UmodUpkg_NotExp test = new UmodUpkg_NotExp(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_Umod.java b/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_Umod.java new file mode 100644 index 00000000000..7a96e5c2afb --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/UmodUpkg_Umod.java @@ -0,0 +1,57 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test public type c5 defined in an unnamed package and unnamed module can + * access public type p6.c6 defined in an unnamed module. + * @compile myloaders/MySameClassLoader.java + * @compile p6/c6.java + * @compile c5.java + * @build UmodUpkg_Umod + * @run main/othervm -Xbootclasspath/a:. UmodUpkg_Umod + */ + +import myloaders.MySameClassLoader; + +public class UmodUpkg_Umod { + + public void testAccess() throws Throwable { + + Class c5_class = MySameClassLoader.loader1.loadClass("c5"); + try { + c5_class.newInstance(); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + throw new RuntimeException("Test Failed, public type c5 defined in an unnamed package and unnamed " + + "module should be able to access public type c6 defined in an unnamed module"); + } + } + + public static void main(String args[]) throws Throwable { + UmodUpkg_Umod test = new UmodUpkg_Umod(); + test.testAccess(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/Umod_ExpQualOther.java b/hotspot/test/runtime/modules/AccessCheck/Umod_ExpQualOther.java new file mode 100644 index 00000000000..fe2fb7bb076 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/Umod_ExpQualOther.java @@ -0,0 +1,136 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test that if package p2 in module m2 is exported to module m3, + * then class p1.c1 in an unnamed module can not read p2.c2 in module m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build Umod_ExpQualOther + * @run main/othervm -Xbootclasspath/a:. Umod_ExpQualOther + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// defines m2 --> packages p2 +// defines m3 --> packages p3 +// +// m1 can read m2 +// package p2 in m2 is exported to m3 +// +// class p1.c1 defined in m1 tries to access p2.c2 defined in m2 +// Access denied since although m1 can read m2, p2 is exported only to m3. +// +public class Umod_ExpQualOther { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 (need to define m1 to establish the Layer successfully) + // Can read: java.base, m2, m3 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .requires("m3") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: p2 is exported to m3 + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2", "m3") + .build(); + + // Define module: m3 + // Can read: java.base + // Packages: p3 + // Packages exported: none + ModuleDescriptor descriptor_m3 = + new ModuleDescriptor.Builder("m3") + .requires("java.base") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2, descriptor_m3); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + map.put("m3", MySameClassLoader.loader1); + + // Create Layer that contains m1, m2 and m3 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m3") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is exported to m3, not unqualifiedly to everyone)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + Umod_ExpQualOther test = new Umod_ExpQualOther(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/Umod_ExpUnqual.java b/hotspot/test/runtime/modules/AccessCheck/Umod_ExpUnqual.java new file mode 100644 index 00000000000..445713c8352 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/Umod_ExpUnqual.java @@ -0,0 +1,121 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test if package p2 in module m2 is exported unqualifiedly, + * then class p1.c1 in an unnamed module can read p2.c2 in module m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build Umod_ExpUnqual + * @run main/othervm -Xbootclasspath/a:. Umod_ExpUnqual + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// +// ClassLoader1 --> defines m1 --> no packages +// defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is exported unqualifiedly +// +// class p1.c1 defined in an unnamed module tries to access p2.c2 defined in m2 +// Access allowed, an unnamed module can read all modules and p2 in module +// m2 which is exported unqualifiedly. + +public class Umod_ExpUnqual { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: p2 is exported unqualifiedly + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .exports("p2") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to differing class loaders for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + + // Create Layer that contains m1 & m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed, an unnamed module can access public type " + + "p2.c2 since it is exported unqualifiedly"); + } + } + + public static void main(String args[]) throws Throwable { + Umod_ExpUnqual test = new Umod_ExpUnqual(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/Umod_PkgNotExp.java b/hotspot/test/runtime/modules/AccessCheck/Umod_PkgNotExp.java new file mode 100644 index 00000000000..7a346cfda1b --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/Umod_PkgNotExp.java @@ -0,0 +1,122 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary Test if package p2 in module m2 is not exported, then class p1.c1 + * in an unnamed module can not access p2.c2 in module m2. + * @library /testlibrary /test/lib + * @compile myloaders/MySameClassLoader.java + * @compile p2/c2.java + * @compile p1/c1.java + * @build Umod_PkgNotExp + * @run main/othervm -Xbootclasspath/a:. Umod_PkgNotExp + */ + +import static jdk.test.lib.Asserts.*; + +import java.lang.reflect.Layer; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; +import myloaders.MySameClassLoader; + +// ClassLoader1 --> defines m1 --> no packages +// defines m2 --> packages p2 +// +// m1 can read m2 +// package p2 in m2 is not exported +// +// class p1.c1 defined in an unnamed module tries to access p2.c2 defined in m2 +// Access denied since p2 is not exported. +// +public class Umod_PkgNotExp { + + // Create a Layer over the boot layer. + // Define modules within this layer to test access between + // publically defined classes within packages of those modules. + public void createLayerOnBoot() throws Throwable { + + // Define module: m1 + // Can read: java.base, m2 + // Packages: none + // Packages exported: none + ModuleDescriptor descriptor_m1 = + new ModuleDescriptor.Builder("m1") + .requires("java.base") + .requires("m2") + .build(); + + // Define module: m2 + // Can read: java.base + // Packages: p2 + // Packages exported: none + ModuleDescriptor descriptor_m2 = + new ModuleDescriptor.Builder("m2") + .requires("java.base") + .conceals("p2") + .build(); + + // Set up a ModuleFinder containing all modules for this layer. + ModuleFinder finder = ModuleLibrary.of(descriptor_m1, descriptor_m2); + + // Resolves "m1" + Configuration cf = Layer.boot() + .configuration() + .resolveRequires(finder, ModuleFinder.empty(), Set.of("m1")); + + // map each module to the same class loader for this test + Map map = new HashMap<>(); + map.put("m1", MySameClassLoader.loader1); + map.put("m2", MySameClassLoader.loader1); + + // Create Layer that contains m1 and m2 + Layer layer = Layer.boot().defineModules(cf, map::get); + + assertTrue(layer.findLoader("m1") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("m2") == MySameClassLoader.loader1); + assertTrue(layer.findLoader("java.base") == null); + + // now use the same loader to load class p1.c1 + Class p1_c1_class = MySameClassLoader.loader1.loadClass("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is not exported)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } + + public static void main(String args[]) throws Throwable { + Umod_PkgNotExp test = new Umod_PkgNotExp(); + test.createLayerOnBoot(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/Umod_UmodUpkg.java b/hotspot/test/runtime/modules/AccessCheck/Umod_UmodUpkg.java new file mode 100644 index 00000000000..23abc1e5a35 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/Umod_UmodUpkg.java @@ -0,0 +1,58 @@ +/* + * 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary class p3.c3 defined in an unnamed module tries to access c4 defined in an unnamed package + * and an unnamed module. + * Access allowed since any class in an unnamed module can read an unnamed module. + * @compile myloaders/MySameClassLoader.java + * @compile c4.java + * @compile p3/c3.jcod + * @build Umod_UmodUpkg + * @run main/othervm -Xbootclasspath/a:. Umod_UmodUpkg + */ + +import myloaders.MySameClassLoader; + +public class Umod_UmodUpkg { + + public void testAccess() throws Throwable { + + Class p3_c3_class = MySameClassLoader.loader1.loadClass("p3.c3"); + try { + p3_c3_class.newInstance(); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + throw new RuntimeException("Test Failed, public type c3 defined in an unnamed module should be able " + + "to access public type c4 defined in an unnamed module"); + } + } + + public static void main(String args[]) throws Throwable { + Umod_UmodUpkg test = new Umod_UmodUpkg(); + test.testAccess(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/c4.java b/hotspot/test/runtime/modules/AccessCheck/c4.java new file mode 100644 index 00000000000..f2840b6e529 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/c4.java @@ -0,0 +1,28 @@ +/* + * 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. + */ + +// Small class used by multiple hotspot/runtime/modules/AccessCheck* tests. + +public class c4 { + public void method4() { } +} diff --git a/hotspot/test/runtime/BadObjectClass/Object.java b/hotspot/test/runtime/modules/AccessCheck/c5.java similarity index 71% rename from hotspot/test/runtime/BadObjectClass/Object.java rename to hotspot/test/runtime/modules/AccessCheck/c5.java index b2ad19b6b99..1a3c25a1249 100644 --- a/hotspot/test/runtime/BadObjectClass/Object.java +++ b/hotspot/test/runtime/modules/AccessCheck/c5.java @@ -4,9 +4,7 @@ * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. + * 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 @@ -23,15 +21,13 @@ * questions. */ -package java.lang; +// Small class used by multiple hotspot/runtime/modules/AccessCheck/* tests. +import p6.c6; -/** - * This is a fake java.lang.Object class. - */ -public class Object { +public class c5 { - // Add some methods - void dummy1() { return; } - void dummy2() { return; } - void dummy3() { return; } + public c5 () { + p6.c6 c6_obj = new p6.c6(); + c6_obj.method6(); + } } diff --git a/hotspot/test/runtime/modules/AccessCheck/myloaders/MyDiffClassLoader.java b/hotspot/test/runtime/modules/AccessCheck/myloaders/MyDiffClassLoader.java new file mode 100644 index 00000000000..63627590ebc --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/myloaders/MyDiffClassLoader.java @@ -0,0 +1,72 @@ +/* + * 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 myloaders; + +import java.io.*; +import java.lang.module.ModuleReference; + +// Declare a MyDiffClassLoader class to be used to map modules to. +// This class loader will also be used to load classes within modules. +public class MyDiffClassLoader extends ClassLoader +{ + public static MyDiffClassLoader loader1 = new MyDiffClassLoader(); + public static MyDiffClassLoader loader2 = new MyDiffClassLoader(); + + public Class loadClass(String name) throws ClassNotFoundException { + if (!name.equals("p1.c1") && + !name.equals("p1.c1ReadEdgeDiffLoader") && + !name.equals("p1.c1Loose") && + !name.equals("p2.c2") && + !name.equals("p3.c3") && + !name.equals("p3.c3ReadEdgeDiffLoader") && + !name.equals("c4") && + !name.equals("c5") && + !name.equals("p6.c6")) { + return super.loadClass(name); + } + if ((name.equals("p2.c2") || name.equals("c4") || name.equals("p6.c6")) && + (this == MyDiffClassLoader.loader1)) { + return MyDiffClassLoader.loader2.loadClass(name); + } + + byte[] data = getClassData(name); + return defineClass(name, data, 0, data.length); + } + byte[] getClassData(String name) { + try { + String TempName = name.replaceAll("\\.", "/"); + String currentDir = System.getProperty("test.classes"); + String filename = currentDir + File.separator + TempName + ".class"; + FileInputStream fis = new FileInputStream(filename); + byte[] b = new byte[5000]; + int cnt = fis.read(b, 0, 5000); + byte[] c = new byte[cnt]; + for (int i=0; i System ClassLoader's unnamed module + Module m1 = c1ReadEdgeDiffLoader.class.getModule(); + ClassLoader system_loader = ClassLoader.getSystemClassLoader(); + Module unnamed_module1 = system_loader.getUnnamedModule(); + m1.addReads(unnamed_module1); + + // Step #2: read edge m1 -> MyDiffClassLoader.loader2's unnamed module + ClassLoader loader2 = MyDiffClassLoader.loader2; + Module unnamed_module2 = loader2.getUnnamedModule(); + m1.addReads(unnamed_module2); + + // Attempt access - access should succeed since m1 can read + // MyDiffClassLoader.loader2's unnamed module + p2.c2 c2_obj = new p2.c2(); + c2_obj.method2(); + } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/p2/c2.java b/hotspot/test/runtime/modules/AccessCheck/p2/c2.java new file mode 100644 index 00000000000..3d21e327553 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/p2/c2.java @@ -0,0 +1,30 @@ +/* + * 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. + */ + +// Small class used by multiple hotspot/runtime/modules/AccessCheck/* tests. + +package p2; + +public class c2 { + public void method2() { } +} diff --git a/hotspot/test/runtime/modules/AccessCheck/p3/c3.jcod b/hotspot/test/runtime/modules/AccessCheck/p3/c3.jcod new file mode 100644 index 00000000000..10c97a71445 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/p3/c3.jcod @@ -0,0 +1,108 @@ +/* + * 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. + */ + +/* Small class used by multiple hotspot/runtime/modules/AccessCheck* tests. + * package p3; + * + * public class c3 { + * + * public c3 () { + * c4 c4_obj = new c4(); + * c4_obj.method4(); + * } + * } + */ + +class p3/c3 { + 0xCAFEBABE; + 0; // minor version + 52; // version + [19] { // Constant Pool + ; // first element is empty + Method #6 #13; // #1 at 0x0A + class #14; // #2 at 0x0F + Method #2 #13; // #3 at 0x12 + Method #2 #15; // #4 at 0x17 + class #16; // #5 at 0x1C + class #17; // #6 at 0x1F + Utf8 ""; // #7 at 0x22 + Utf8 "()V"; // #8 at 0x2B + Utf8 "Code"; // #9 at 0x31 + Utf8 "LineNumberTable"; // #10 at 0x38 + Utf8 "SourceFile"; // #11 at 0x4A + Utf8 "c3.java"; // #12 at 0x57 + NameAndType #7 #8; // #13 at 0x61 + Utf8 "c4"; // #14 at 0x66 + NameAndType #18 #8; // #15 at 0x6E + Utf8 "p3/c3"; // #16 at 0x73 + Utf8 "java/lang/Object"; // #17 at 0x7B + Utf8 "method4"; // #18 at 0x8E + } // Constant Pool + + 0x0021; // access + #5;// this_cpx + #6;// super_cpx + + [0] { // Interfaces + } // Interfaces + + [0] { // fields + } // fields + + [1] { // methods + { // Member at 0xA4 + 0x0001; // access + #7; // name_cpx + #8; // sig_cpx + [1] { // Attributes + Attr(#9, 53) { // Code at 0xAC + 2; // max_stack + 2; // max_locals + Bytes[17]{ + 0x2AB70001BB000259; + 0xB700034C2BB60004; + 0xB1; + }; + [0] { // Traps + } // end Traps + [1] { // Attributes + Attr(#10, 18) { // LineNumberTable at 0xCF + [4] { // LineNumberTable + 0 4; // at 0xDB + 4 5; // at 0xDF + 12 6; // at 0xE3 + 16 7; // at 0xE7 + } + } // end LineNumberTable + } // Attributes + } // end Code + } // Attributes + } // Member + } // methods + + [1] { // Attributes + Attr(#11, 2) { // SourceFile at 0xE9 + #12; + } // end SourceFile + } // Attributes +} // end class p3/c3 diff --git a/hotspot/test/runtime/modules/AccessCheck/p3/c3ReadEdge.jcod b/hotspot/test/runtime/modules/AccessCheck/p3/c3ReadEdge.jcod new file mode 100644 index 00000000000..50df6180233 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/p3/c3ReadEdge.jcod @@ -0,0 +1,142 @@ +/* + * 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 p3; + * import java.lang.reflect.*; + * public class c3ReadEdge { + * public c3ReadEdge() { + * // Establish read edge from module m1, where c3ReadEdge is defined, + * // to the unnamed module, where c4 will be defined. + * Module m1 = c3ReadEdge.class.getModule(); + * ClassLoader loader = c3ReadEdge.class.getClassLoader(); + * Module unnamed_module = loader.getUnnamedModule(); + * m1.addReads(unnamed_module); + * // Attempt access - access should succeed + * c4 c4_obj = new c4(); + * c4_obj.method4(); + * } + * } + */ + +class p3/c3ReadEdge { + 0xCAFEBABE; + 0; // minor version + 52; // version + [40] { // Constant Pool + ; // first element is empty + Method #10 #17; // #1 at 0x0A + class #18; // #2 at 0x0F + Method #19 #20; // #3 at 0x12 + Method #19 #21; // #4 at 0x17 + Method #22 #23; // #5 at 0x1C + Method #24 #25; // #6 at 0x21 + class #26; // #7 at 0x26 + Method #7 #17; // #8 at 0x29 + Method #7 #27; // #9 at 0x2E + class #28; // #10 at 0x33 + Utf8 ""; // #11 at 0x36 + Utf8 "()V"; // #12 at 0x3F + Utf8 "Code"; // #13 at 0x45 + Utf8 "LineNumberTable"; // #14 at 0x4C + Utf8 "SourceFile"; // #15 at 0x5E + Utf8 "c3ReadEdge.java"; // #16 at 0x6B + NameAndType #11 #12; // #17 at 0x7D + Utf8 "p3/c3ReadEdge"; // #18 at 0x82 + class #29; // #19 at 0x92 + NameAndType #30 #31; // #20 at 0x95 + NameAndType #32 #33; // #21 at 0x9A + class #34; // #22 at 0x9F + NameAndType #35 #31; // #23 at 0xA2 + class #36; // #24 at 0xA7 + NameAndType #37 #38; // #25 at 0xAA + Utf8 "c4"; // #26 at 0xAF + NameAndType #39 #12; // #27 at 0xB7 + Utf8 "java/lang/Object"; // #28 at 0xBC + Utf8 "java/lang/Class"; // #29 at 0xCF + Utf8 "getModule"; // #30 at 0xE1 + Utf8 "()Ljava/lang/reflect/Module;"; // #31 at 0xED + Utf8 "getClassLoader"; // #32 at 0x010C + Utf8 "()Ljava/lang/ClassLoader;"; // #33 at 0x011D + Utf8 "java/lang/ClassLoader"; // #34 at 0x0139 + Utf8 "getUnnamedModule"; // #35 at 0x0151 + Utf8 "java/lang/reflect/Module"; // #36 at 0x0164 + Utf8 "addReads"; // #37 at 0x017F + Utf8 "(Ljava/lang/reflect/Module;)Ljava/lang/reflect/Module;"; // #38 at 0x018A + Utf8 "method4"; // #39 at 0x01C3 + } // Constant Pool + + 0x0021; // access + #2;// this_cpx + #10;// super_cpx + + [0] { // Interfaces + } // Interfaces + + [0] { // fields + } // fields + + [1] { // methods + { // Member at 0x01D9 + 0x0001; // access + #11; // name_cpx + #12; // sig_cpx + [1] { // Attributes + Attr(#13, 94) { // Code at 0x01E1 + 2; // max_stack + 5; // max_locals + Bytes[42]{ + 0x2AB700011202B600; + 0x034C1202B600044D; + 0x2CB600054E2B2DB6; + 0x000657BB000759B7; + 0x00083A041904B600; + 0x09B1; + }; + [0] { // Traps + } // end Traps + [1] { // Attributes + Attr(#14, 34) { // LineNumberTable at 0x021D + [8] { // LineNumberTable + 0 32; // at 0x0229 + 4 38; // at 0x022D + 10 39; // at 0x0231 + 16 40; // at 0x0235 + 21 41; // at 0x0239 + 27 44; // at 0x023D + 36 45; // at 0x0241 + 41 46; // at 0x0245 + } + } // end LineNumberTable + } // Attributes + } // end Code + } // Attributes + } // Member + } // methods + + [1] { // Attributes + Attr(#15, 2) { // SourceFile at 0x0247 + #16; + } // end SourceFile + } // Attributes +} // end class p3/c3ReadEdge diff --git a/hotspot/test/runtime/modules/AccessCheck/p3/c3ReadEdgeDiffLoader.jcod b/hotspot/test/runtime/modules/AccessCheck/p3/c3ReadEdgeDiffLoader.jcod new file mode 100644 index 00000000000..103f7eea2c7 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/p3/c3ReadEdgeDiffLoader.jcod @@ -0,0 +1,175 @@ +/* + * 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 p3; + * import java.lang.reflect.*; + * import myloaders.MyDiffClassLoader; + * + * public class c3ReadEdgeDiffLoader { + * public c3ReadEdgeDiffLoader() { + * // The goal is to establish a read edge between module m1 + * // which is the module where p3.c3ReadEdgeDiffLoader is defined, + * // and the unnamed module that defines c4. This must be + * // done in 2 steps: + * // + * // Step #1: Establish a read edge between m1, where c3ReadEdgeDiffLoader + * // is defined, and the System ClassLoader's unnamed module, + * // where MyDiffClassLoader is defined. This read edge + * // is needed before we can obtain MyDiffClassLoader.loader2's unnamed module. + * // + * // Step #2: Establish a read edge between m1, where c3ReadEdgeDiffLoader + * // is defined, and the MyDiffClassLoader.loader2's unnamed module, + * // where c4 will be defined. + * + * // Step #1: read edge m1 -> System ClassLoader's unnamed module + * Module m1 = c3ReadEdgeDiffLoader.class.getModule(); + * ClassLoader system_loader = ClassLoader.getSystemClassLoader(); + * Module unnamed_module1 = system_loader.getUnnamedModule(); + * m1.addReads(unnamed_module1); + * + * // Step #2: read edge m1 -> MyDiffClassLoader.loader2's unnamed module + * ClassLoader loader2 = MyDiffClassLoader.loader2; + * Module unnamed_module2 = loader2.getUnnamedModule(); + * m1.addReads(unnamed_module2); + * + * // Attempt access - should succeed since m1 can read + * // MyDiffClassLoader.loader2's unnamed module + * c4 c4_obj = new c4(); + * c4_obj.method4(); + * } + * } + */ + +class p3/c3ReadEdgeDiffLoader { + 0xCAFEBABE; + 0; // minor version + 52; // version + [46] { // Constant Pool + ; // first element is empty + Method #11 #18; // #1 at 0x0A + class #19; // #2 at 0x0F + Method #20 #21; // #3 at 0x12 + Method #22 #23; // #4 at 0x17 + Method #22 #24; // #5 at 0x1C + Method #25 #26; // #6 at 0x21 + Field #27 #28; // #7 at 0x26 + class #29; // #8 at 0x2B + Method #8 #18; // #9 at 0x2E + Method #8 #30; // #10 at 0x33 + class #31; // #11 at 0x38 + Utf8 ""; // #12 at 0x3B + Utf8 "()V"; // #13 at 0x44 + Utf8 "Code"; // #14 at 0x4A + Utf8 "LineNumberTable"; // #15 at 0x51 + Utf8 "SourceFile"; // #16 at 0x63 + Utf8 "c3ReadEdgeDiffLoader.java"; // #17 at 0x70 + NameAndType #12 #13; // #18 at 0x8C + Utf8 "p3/c3ReadEdgeDiffLoader"; // #19 at 0x91 + class #32; // #20 at 0xAB + NameAndType #33 #34; // #21 at 0xAE + class #35; // #22 at 0xB3 + NameAndType #36 #37; // #23 at 0xB6 + NameAndType #38 #34; // #24 at 0xBB + class #39; // #25 at 0xC0 + NameAndType #40 #41; // #26 at 0xC3 + class #42; // #27 at 0xC8 + NameAndType #43 #44; // #28 at 0xCB + Utf8 "c4"; // #29 at 0xD0 + NameAndType #45 #13; // #30 at 0xD8 + Utf8 "java/lang/Object"; // #31 at 0xDD + Utf8 "java/lang/Class"; // #32 at 0xF0 + Utf8 "getModule"; // #33 at 0x0102 + Utf8 "()Ljava/lang/reflect/Module;"; // #34 at 0x010E + Utf8 "java/lang/ClassLoader"; // #35 at 0x012D + Utf8 "getSystemClassLoader"; // #36 at 0x0145 + Utf8 "()Ljava/lang/ClassLoader;"; // #37 at 0x015C + Utf8 "getUnnamedModule"; // #38 at 0x0178 + Utf8 "java/lang/reflect/Module"; // #39 at 0x018B + Utf8 "addReads"; // #40 at 0x01A6 + Utf8 "(Ljava/lang/reflect/Module;)Ljava/lang/reflect/Module;"; // #41 at 0x01B1 + Utf8 "myloaders/MyDiffClassLoader"; // #42 at 0x01EA + Utf8 "loader2"; // #43 at 0x0208 + Utf8 "Lmyloaders/MyDiffClassLoader;"; // #44 at 0x0212 + Utf8 "method4"; // #45 at 0x0232 + } // Constant Pool + + 0x0021; // access + #2;// this_cpx + #11;// super_cpx + + [0] { // Interfaces + } // Interfaces + + [0] { // fields + } // fields + + [1] { // methods + { // Member at 0x0248 + 0x0001; // access + #12; // name_cpx + #13; // sig_cpx + [1] { // Attributes + Attr(#14, 123) { // Code at 0x0250 + 2; // max_stack + 7; // max_locals + Bytes[59]{ + 0x2AB700011202B600; + 0x034CB800044D2CB6; + 0x00054E2B2DB60006; + 0x57B200073A041904; + 0xB600053A052B1905; + 0xB6000657BB000859; + 0xB700093A061906B6; + 0x000AB1; + }; + [0] { // Traps + } // end Traps + [1] { // Attributes + Attr(#15, 46) { // LineNumberTable at 0x029D + [11] { // LineNumberTable + 0 33; // at 0x02A9 + 4 34; // at 0x02AD + 10 36; // at 0x02B1 + 14 37; // at 0x02B5 + 19 39; // at 0x02B9 + 25 41; // at 0x02BD + 30 42; // at 0x02C1 + 37 46; // at 0x02C5 + 44 49; // at 0x02C9 + 53 50; // at 0x02CD + 58 51; // at 0x02D1 + } + } // end LineNumberTable + } // Attributes + } // end Code + } // Attributes + } // Member + } // methods + + [1] { // Attributes + Attr(#16, 2) { // SourceFile at 0x02D3 + #17; + } // end SourceFile + } // Attributes +} // end class p3/c3ReadEdgeDiffLoader diff --git a/hotspot/test/runtime/modules/AccessCheck/p6/c6.java b/hotspot/test/runtime/modules/AccessCheck/p6/c6.java new file mode 100644 index 00000000000..37c11615750 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/p6/c6.java @@ -0,0 +1,30 @@ +/* + * 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. + */ + +// Small class used by multiple hotspot/runtime/modules/AccessCheck/* tests. + +package p6; + +public class c6 { + public void method6() { } +} diff --git a/hotspot/test/runtime/modules/AccessCheckAllUnnamed.java b/hotspot/test/runtime/modules/AccessCheckAllUnnamed.java new file mode 100644 index 00000000000..a97f7ccf8d1 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheckAllUnnamed.java @@ -0,0 +1,117 @@ +/* + * 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. + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p1/c1.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI AccessCheckAllUnnamed + */ + +public class AccessCheckAllUnnamed { + + // Test a series of error conditions for API JVM_AddModuleExportsToAllUnnamed + // and then test that a class in the unnamed module can access a package in a + // named module that has been exported to all unnamed modules. + public static void main(String args[]) throws Throwable { + Object m1, m2; + + // Get the java.lang.reflect.Module object for module java.base. + Class jlObject = Class.forName("java.lang.Object"); + Object jlObject_jlrM = jlObject.getModule(); + assertNotNull(jlObject_jlrM, "jlrModule object of java.lang.Object should not be null"); + + // Get the class loader for AccessCheckWorks and assume it's also used to + // load class p2.c2. + ClassLoader this_cldr = AccessCheckAllUnnamed.class.getClassLoader(); + + // Define a module for p3. + m1 = ModuleHelper.ModuleObject("module1", this_cldr, new String[] { "p3" }); + assertNotNull(m1, "Module should not be null"); + ModuleHelper.DefineModule(m1, "9.0", "m1/there", new String[] { "p3" }); + ModuleHelper.AddReadsModule(m1, jlObject_jlrM); + + // Define a module for p2. + m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + ModuleHelper.AddReadsModule(m2, jlObject_jlrM); + + try { + ModuleHelper.AddModuleExportsToAllUnnamed((Module)null, "p2"); + throw new RuntimeException("Failed to get the expected NPE for null module"); + } catch(NullPointerException e) { + // Expected + } + + try { + ModuleHelper.AddModuleExportsToAllUnnamed(m2, null); + throw new RuntimeException("Failed to get the expected NPE for null package"); + } catch(NullPointerException e) { + // Expected + } + + try { + ModuleHelper.AddModuleExportsToAllUnnamed(this_cldr, "p2"); + throw new RuntimeException("Failed to get the expected IAE for bad module"); + } catch(IllegalArgumentException e) { + // Expected + } + + try { + ModuleHelper.AddModuleExportsToAllUnnamed(m2, "p3"); + throw new RuntimeException("Failed to get the expected IAE for package in other module"); + } catch(IllegalArgumentException e) { + // Expected + } + + try { + ModuleHelper.AddModuleExportsToAllUnnamed(m2, "p4"); + throw new RuntimeException("Failed to get the expected IAE for package not in module"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Export package p2 in m2 to allUnnamed. + ModuleHelper.AddModuleExportsToAllUnnamed(m2, "p2"); + + // p1.c1's ctor tries to call a method in p2.c2. This should succeed because + // p1 is in an unnamed module and p2.c2 is exported to all unnamed modules. + Class p1_c1_class = Class.forName("p1.c1"); + try { + Object c1_obj = p1_c1_class.newInstance(); + } catch (IllegalAccessError f) { + throw new RuntimeException( + "Class in unnamed module could not access package p2 exported to all unnamed modules"); + } + } +} + diff --git a/hotspot/test/runtime/modules/AccessCheckExp.java b/hotspot/test/runtime/modules/AccessCheckExp.java new file mode 100644 index 00000000000..304af7f9ee8 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheckExp.java @@ -0,0 +1,86 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p1/c1.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI AccessCheckExp + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +public class AccessCheckExp { + + // Test that if module1 can read module2, but package p2 in module2 is not + // exported then class p1.c1 in module1 can not read p2.c2 in module2. + public static void main(String args[]) throws Throwable { + Object m1, m2; + + // Get the java.lang.reflect.Module object for module java.base. + Class jlObject = Class.forName("java.lang.Object"); + Object jlObject_jlrM = jlObject.getModule(); + assertNotNull(jlObject_jlrM, "jlrModule object of java.lang.Object should not be null"); + + // Get the class loader for AccessCheckExp and assume it's also used to + // load classes p1.c1 and p2.c2. + ClassLoader this_cldr = AccessCheckExp.class.getClassLoader(); + + // Define a module for p1. + m1 = ModuleHelper.ModuleObject("module1", this_cldr, new String[] { "p1" }); + assertNotNull(m1, "Module should not be null"); + ModuleHelper.DefineModule(m1, "9.0", "m1/here", new String[] { "p1" }); + ModuleHelper.AddReadsModule(m1, jlObject_jlrM); + + // Define a module for p2. + m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + ModuleHelper.AddReadsModule(m2, jlObject_jlrM); + + // Make package p1 in m1 visible to everyone. + ModuleHelper.AddModuleExportsToAll(m1, "p1"); + + // p1.c1's ctor tries to call a method in p2.c2, but p2.c2 is not + // exported. So should get IllegalAccessError. + ModuleHelper.AddReadsModule(m1, m2); + + Class p1_c1_class = Class.forName("p1.c1"); + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is not exported"); + } catch (IllegalAccessError f) { + System.out.println(f.getMessage()); + if (!f.getMessage().contains("does not export")) { + throw new RuntimeException("Wrong message: " + f.getMessage()); + } + } + } +} + diff --git a/hotspot/test/runtime/modules/AccessCheckJavaBase.java b/hotspot/test/runtime/modules/AccessCheckJavaBase.java new file mode 100644 index 00000000000..764f9fb2be4 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheckJavaBase.java @@ -0,0 +1,58 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI AccessCheckJavaBase + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +public class AccessCheckJavaBase { + + // Test that a class defined to module2 always can read java.base. + public static void main(String args[]) throws Throwable { + // Get the class loader for AccessCheckJavaBase and assume it's also used to + // load class p2.c2. + ClassLoader this_cldr = AccessCheckJavaBase.class.getClassLoader(); + + // Define a module for p2. + Object m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + + // p2.c2 can read its superclass java.lang.Object defined within java.base + try { + Class p2_c2_class = Class.forName("p2.c2"); + } catch (IllegalAccessError e) { + throw new RuntimeException("Test Failed" + e.getMessage()); + } + } +} diff --git a/hotspot/test/runtime/modules/AccessCheckRead.java b/hotspot/test/runtime/modules/AccessCheckRead.java new file mode 100644 index 00000000000..8158582e866 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheckRead.java @@ -0,0 +1,86 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p1/c1.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI AccessCheckRead + */ + +import jdk.test.lib.*; +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +public class AccessCheckRead { + + // Test that a class in a package in module1 cannot access a class in + // a package n module2 if module1 cannot read module2. + public static void main(String args[]) throws Throwable { + Object m1, m2; + + // Get the java.lang.reflect.Module object for module java.base. + Class jlObject = Class.forName("java.lang.Object"); + Object jlObject_jlrM = jlObject.getModule(); + assertNotNull(jlObject_jlrM, "jlrModule object of java.lang.Object should not be null"); + + // Get the class loader for AccessCheckRead and assume it's also used to + // load classes p1.c1 and p2.c2. + ClassLoader this_cldr = AccessCheckRead.class.getClassLoader(); + + // Define a module for p1. + m1 = ModuleHelper.ModuleObject("module1", this_cldr, new String[] { "p1" }); + assertNotNull(m1, "Module should not be null"); + ModuleHelper.DefineModule(m1, "9.0", "m1/here", new String[] { "p1" }); + ModuleHelper.AddReadsModule(m1, jlObject_jlrM); + + // Define a module for p2. + m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + ModuleHelper.AddReadsModule(m2, jlObject_jlrM); + + // Make package p1 in m1 visible to everyone. + ModuleHelper.AddModuleExportsToAll(m1, "p1"); + + Class p1_c1_class = Class.forName("p1.c1"); + + // p1.c1's ctor tries to call a method in p2.c2, but p1's module + // cannot read p2's module. So should get IllegalAccessError. + try { + p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (m1 can't read m2)"); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not read") || + e.getMessage().contains("strict")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } +} diff --git a/hotspot/test/runtime/modules/AccessCheckSuper.java b/hotspot/test/runtime/modules/AccessCheckSuper.java new file mode 100644 index 00000000000..bd2065b5ff0 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheckSuper.java @@ -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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p3/c3.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI AccessCheckSuper + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +public class AccessCheckSuper { + + // Test that when a class cannot access its super class the message + // contains both "superclass" text and module text. + public static void main(String args[]) throws Throwable { + + // Get the class loader for AccessCheckSuper and assume it's also used to + // load class p2.c2 and class p3.c3. + ClassLoader this_cldr = AccessCheckSuper.class.getClassLoader(); + + // Define a module for p2. + Object m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + + // Define a module for p3. + Object m3 = ModuleHelper.ModuleObject("module3", this_cldr, new String[] { "p3" }); + assertNotNull(m3, "Module should not be null"); + ModuleHelper.DefineModule(m3, "9.0", "m3/there", new String[] { "p3" }); + + // Since a readability edge has not been established between module2 + // and module3, p3.c3 cannot read its superclass p2.c2. + try { + Class p3_c3_class = Class.forName("p3.c3"); + throw new RuntimeException("Failed to get IAE (can't read superclass)"); + } catch (IllegalAccessError e) { + if (!e.getMessage().contains("superclass access check failed") || + !e.getMessage().contains("does not read")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } +} diff --git a/hotspot/test/runtime/modules/AccessCheckUnnamed.java b/hotspot/test/runtime/modules/AccessCheckUnnamed.java new file mode 100644 index 00000000000..0451489be56 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheckUnnamed.java @@ -0,0 +1,75 @@ +/* + * 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. + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p1/c1.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI AccessCheckUnnamed + */ + +public class AccessCheckUnnamed { + + // Test that a class in the unnamed module can not access a package in a + // named module that has not been unqualifiedly exported. + public static void main(String args[]) throws Throwable { + Object m1, m2; + + // Get the java.lang.reflect.Module object for module java.base. + Class jlObject = Class.forName("java.lang.Object"); + Object jlObject_jlrM = jlObject.getModule(); + assertNotNull(jlObject_jlrM, "jlrModule object of java.lang.Object should not be null"); + + // Get the class loader for AccessCheckWorks and assume it's also used to + // load class p2.c2. + ClassLoader this_cldr = AccessCheckUnnamed.class.getClassLoader(); + + // Define a module for p2. + m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + ModuleHelper.AddReadsModule(m2, jlObject_jlrM); + + // p1.c1's ctor tries to call a method in p2.c2. This should fail because + // p1 is in the unnamed module and p2.c2 is not unqualifiedly exported. + Class p1_c1_class = Class.forName("p1.c1"); + try { + Object c1_obj = p1_c1_class.newInstance(); + throw new RuntimeException("Failed to get IAE (p2 in m2 is not exported to unnamed module)"); + } catch (IllegalAccessError f) { + System.out.println(f.getMessage()); + if (!f.getMessage().contains("does not export p2 to unnamed module")) { + throw new RuntimeException("Wrong message: " + f.getMessage()); + } + } + } +} + diff --git a/hotspot/test/runtime/modules/AccessCheckWorks.java b/hotspot/test/runtime/modules/AccessCheckWorks.java new file mode 100644 index 00000000000..3a13b04a59b --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheckWorks.java @@ -0,0 +1,79 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p1/c1.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI AccessCheckWorks + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +public class AccessCheckWorks { + + // Check that a class in a package in module1 can successfully access a + // class in module2 when module1 can read module2 and the class's package + // has been exported. + public static void main(String args[]) throws Throwable { + Object m1, m2; + + // Get the java.lang.reflect.Module object for module java.base. + Class jlObject = Class.forName("java.lang.Object"); + Object jlObject_jlrM = jlObject.getModule(); + assertNotNull(jlObject_jlrM, "jlrModule object of java.lang.Object should not be null"); + + // Get the class loader for AccessCheckWorks and assume it's also used to + // load classes p1.c1 and p2.c2. + ClassLoader this_cldr = AccessCheckWorks.class.getClassLoader(); + + // Define a module for p1. + m1 = ModuleHelper.ModuleObject("module1", this_cldr, new String[] { "p1" }); + assertNotNull(m1, "Module should not be null"); + ModuleHelper.DefineModule(m1, "9.0", "m1/here", new String[] { "p1" }); + ModuleHelper.AddReadsModule(m1, jlObject_jlrM); + + // Define a module for p2. + m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + ModuleHelper.AddReadsModule(m2, jlObject_jlrM); + + // Make package p1 in m1 visible to everyone. + ModuleHelper.AddModuleExportsToAll(m1, "p1"); + + // p1.c1's ctor tries to call a method in p2.c2. This should work because + // p1's module can read p2's module and p2 is exported to p1's module. + ModuleHelper.AddReadsModule(m1, m2); + ModuleHelper.AddModuleExports(m2, "p2", m1); + Class p1_c1_class = Class.forName("p1.c1"); + p1_c1_class.newInstance(); + } +} + diff --git a/hotspot/test/runtime/modules/CCE_module_msg.java b/hotspot/test/runtime/modules/CCE_module_msg.java new file mode 100644 index 00000000000..54ece088b29 --- /dev/null +++ b/hotspot/test/runtime/modules/CCE_module_msg.java @@ -0,0 +1,58 @@ +/* + * 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. + */ + +/** + * @test + * @run main/othervm CCE_module_msg + */ + +// Test that the message in a runtime ClassCastException contains module info. +public class CCE_module_msg { + + public static void main(String[] args) { + invalidCastTest(); + } + + public static void invalidCastTest() { + java.lang.Object instance = new java.lang.Object(); + int left = 23; + int right = 42; + try { + for (int i = 0; i < 1; i += 1) { + left = ((Derived) instance).method(left, right); + } + throw new RuntimeException("ClassCastException wasn't thrown, test failed."); + } catch (ClassCastException cce) { + System.out.println(cce.getMessage()); + if (!cce.getMessage().contains("java.lang.Object (in module: java.base) cannot be cast")) { + throw new RuntimeException("Wrong message: " + cce.getMessage()); + } + } + } +} + +class Derived extends java.lang.Object { + public int method(int left, int right) { + return right; + } +} diff --git a/hotspot/test/runtime/modules/ExportTwice.java b/hotspot/test/runtime/modules/ExportTwice.java new file mode 100644 index 00000000000..b33b55ef787 --- /dev/null +++ b/hotspot/test/runtime/modules/ExportTwice.java @@ -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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p1/c1.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI ExportTwice + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +public class ExportTwice { + + // Check that a package can not be exported unqualifiedly, and then exported + // to a specific package. + // Also, check that a package can be exported to a specific package and then + // exported unqualifiedly. + public static void main(String args[]) throws Throwable { + Object m1, m2, m3; + + // Get the java.lang.reflect.Module object for module java.base. + Class jlObject = Class.forName("java.lang.Object"); + Object jlObject_jlrM = jlObject.getModule(); + assertNotNull(jlObject_jlrM, "jlrModule object of java.lang.Object should not be null"); + + // Get the class loader for ExportTwice and assume it's also used to + // load classes p1.c1 and p2.c2. + ClassLoader this_cldr = ExportTwice.class.getClassLoader(); + + // Define a module for p1. + m1 = ModuleHelper.ModuleObject("module1", this_cldr, new String[] { "p1" }); + assertNotNull(m1, "Module should not be null"); + ModuleHelper.DefineModule(m1, "9.0", "m1/here", new String[] { "p1" }); + ModuleHelper.AddReadsModule(m1, jlObject_jlrM); + + // Define a module for p2. + m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + ModuleHelper.AddReadsModule(m2, jlObject_jlrM); + + // Define a module for p3. + m3 = ModuleHelper.ModuleObject("module3", this_cldr, new String[] { "p3" }); + assertNotNull(m3, "Module should not be null"); + ModuleHelper.DefineModule(m3, "9.0", "m3/there", new String[] { "p3" }); + ModuleHelper.AddReadsModule(m3, jlObject_jlrM); + + // Make package p1 in m1 visible to everyone. + ModuleHelper.AddModuleExportsToAll(m1, "p1"); + + // Try to export p1 only to m2 after it was exported unqualifiedly. It + // should silently succeed. + ModuleHelper.AddModuleExports(m1, "p1", m2); + + // Export p2 to m3 then export it again unqualifiedly. + ModuleHelper.AddModuleExports(m2, "p2", m3); + ModuleHelper.AddModuleExportsToAll(m2, "p2"); + + // p1.c1's ctor tries to call a method in p2.c2. This should work because + // p1's module can read p2's module and p2 is now exported unqualifiedly. + ModuleHelper.AddReadsModule(m1, m2); + Class p1_c1_class = Class.forName("p1.c1"); + p1_c1_class.newInstance(); + } +} diff --git a/hotspot/test/runtime/modules/JVMAddModuleExportToAllUnnamed.java b/hotspot/test/runtime/modules/JVMAddModuleExportToAllUnnamed.java new file mode 100644 index 00000000000..b7b34b7411f --- /dev/null +++ b/hotspot/test/runtime/modules/JVMAddModuleExportToAllUnnamed.java @@ -0,0 +1,76 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p1/c1.java + * @build sun.hotspot.WhiteBox + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMAddModuleExportToAllUnnamed + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +public class JVMAddModuleExportToAllUnnamed { + + // Check that a class in a package in module1 cannot access a class + // that is in the unnamed module if the accessing package is strict. + public static void main(String args[]) throws Throwable { + Object m1; + + // Get the java.lang.reflect.Module object for module java.base. + Class jlObject = Class.forName("java.lang.Object"); + Object jlObject_jlrM = jlObject.getModule(); + assertNotNull(jlObject_jlrM, "jlrModule object of java.lang.Object should not be null"); + + // Get the class loader for JVMAddModuleExportToAllUnnamed and assume it's also used to + // load class p1.c1. + ClassLoader this_cldr = JVMAddModuleExportToAllUnnamed.class.getClassLoader(); + + // Define a module for p1. + m1 = ModuleHelper.ModuleObject("module1", this_cldr, new String[] { "p1" }); + assertNotNull(m1, "Module should not be null"); + ModuleHelper.DefineModule(m1, "9.0", "m1/here", new String[] { "p1" }); + ModuleHelper.AddReadsModule(m1, jlObject_jlrM); + + // Make package p1 in m1 visible to everyone. + ModuleHelper.AddModuleExportsToAll(m1, "p1"); + + // p1.c1's ctor tries to call a method in p2.c2. This should not work + // because p2 is in the unnamed module and p1.c1 is strict. + try { + Class p1_c1_class = Class.forName("p1.c1"); + p1_c1_class.newInstance(); + } catch (IllegalAccessError e) { + System.out.println(e.getMessage()); + if (!e.getMessage().contains("does not read unnamed module")) { + throw new RuntimeException("Wrong message: " + e.getMessage()); + } + } + } +} diff --git a/hotspot/test/runtime/modules/JVMAddModuleExports.java b/hotspot/test/runtime/modules/JVMAddModuleExports.java new file mode 100644 index 00000000000..330dd923411 --- /dev/null +++ b/hotspot/test/runtime/modules/JVMAddModuleExports.java @@ -0,0 +1,145 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMAddModuleExports + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +public class JVMAddModuleExports { + + public static void main(String args[]) throws Throwable { + MyClassLoader from_cl = new MyClassLoader(); + MyClassLoader to_cl = new MyClassLoader(); + Module from_module, to_module; + + from_module = ModuleHelper.ModuleObject("from_module", from_cl, new String[] { "mypackage", "this/package" }); + assertNotNull(from_module, "Module should not be null"); + ModuleHelper.DefineModule(from_module, "9.0", "from_module/here", new String[] { "mypackage", "this/package" }); + to_module = ModuleHelper.ModuleObject("to_module", to_cl, new String[] { "yourpackage", "that/package" }); + assertNotNull(to_module, "Module should not be null"); + ModuleHelper.DefineModule(to_module, "9.0", "to_module/here", new String[] { "yourpackage", "that/package" }); + + // Null from_module argument, expect an NPE + try { + ModuleHelper.AddModuleExports((Module)null, "mypackage", to_module); + throw new RuntimeException("Failed to get the expected NPE for null from_module"); + } catch(NullPointerException e) { + // Expected + } + + // Null to_module argument, expect an NPE + try { + ModuleHelper.AddModuleExports(from_module, "mypackage", (Module)null); + throw new RuntimeException("Failed to get the expected NPE for null to_module"); + } catch(NullPointerException e) { + // Expected + } + + // Bad from_module argument, expect an IAE + try { + ModuleHelper.AddModuleExports(to_cl, "mypackage", to_module); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Null package argument, expect an NPE + try { + ModuleHelper.AddModuleExports(from_module, null, to_module); + throw new RuntimeException("Failed to get the expected NPE"); + } catch(NullPointerException e) { + // Expected + } + + // Bad to_module argument, expect an IAE + try { + ModuleHelper.AddModuleExports(from_module, "mypackage", from_cl); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Exporting a package to the same module + ModuleHelper.AddModuleExports(from_module, "mypackage", from_module); + + // Export a package that does not exist to to_module + try { + ModuleHelper.AddModuleExports(from_module, "notmypackage", to_module); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Export a package, that is not in from_module, to to_module + try { + ModuleHelper.AddModuleExports(from_module, "yourpackage", to_module); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Export a package, that does not exist, to from_module + try { + ModuleHelper.AddModuleExports(from_module, "notmypackage", from_module); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Export a package, that is not in from_module, to from_module + try { + ModuleHelper.AddModuleExports(from_module, "that/package", from_module); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Export the same package twice to the same module + ModuleHelper.AddModuleExports(from_module, "this/package", to_module); + ModuleHelper.AddModuleExports(from_module, "this/package", to_module); + + // Export a package, using '.' instead of '/' + try { + ModuleHelper.AddModuleExports(from_module, "this.package", to_module); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Export a package to the unnamed module and then to a specific module. + // The qualified export should be ignored. + ModuleHelper.AddModuleExportsToAll(to_module, "that/package"); + ModuleHelper.AddModuleExports(to_module, "that/package", from_module); + } + + static class MyClassLoader extends ClassLoader { } +} diff --git a/hotspot/test/runtime/modules/JVMAddModuleExportsToAll.java b/hotspot/test/runtime/modules/JVMAddModuleExportsToAll.java new file mode 100644 index 00000000000..f41d401c10d --- /dev/null +++ b/hotspot/test/runtime/modules/JVMAddModuleExportsToAll.java @@ -0,0 +1,122 @@ +/* + * 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. + */ + +import java.lang.reflect.Module; +import static jdk.test.lib.Asserts.*; + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @compile p1/c1.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMAddModuleExportsToAll + */ + +public class JVMAddModuleExportsToAll { + + // Test a series of error conditions for API JVM_AddModuleExportsToAll() + // and then test that a class in the unnamed module can access a package in + // a named module that has been exported unqualifiedly. + public static void main(String args[]) throws Throwable { + Object m1, m2, m3; + + // Get the java.lang.reflect.Module object for module java.base. + Class jlObject = Class.forName("java.lang.Object"); + Object jlObject_jlrM = jlObject.getModule(); + assertNotNull(jlObject_jlrM, "jlrModule object of java.lang.Object should not be null"); + + // Get the class loader for JVMAddModuleExportsToAll and assume it's also used to + // load class p2.c2. + ClassLoader this_cldr = JVMAddModuleExportsToAll.class.getClassLoader(); + + // Define a module for p3. + m1 = ModuleHelper.ModuleObject("module1", this_cldr, new String[] { "p3" }); + assertNotNull(m1, "Module should not be null"); + ModuleHelper.DefineModule(m1, "9.0", "m1/there", new String[] { "p3" }); + ModuleHelper.AddReadsModule(m1, jlObject_jlrM); + + // Define a module for p2. + m2 = ModuleHelper.ModuleObject("module2", this_cldr, new String[] { "p2" }); + assertNotNull(m2, "Module should not be null"); + ModuleHelper.DefineModule(m2, "9.0", "m2/there", new String[] { "p2" }); + ModuleHelper.AddReadsModule(m2, jlObject_jlrM); + + try { + ModuleHelper.AddModuleExportsToAll((Module)null, "p2"); + throw new RuntimeException("Failed to get the expected NPE for null module"); + } catch(NullPointerException e) { + // Expected + } + + try { + ModuleHelper.AddModuleExportsToAll(m2, null); + throw new RuntimeException("Failed to get the expected NPE for null package"); + } catch(NullPointerException e) { + // Expected + } + + try { // Expect IAE when passing a ClassLoader object instead of a java.lang.reflect.Module object. + ModuleHelper.AddModuleExportsToAll(this_cldr, "p2"); + throw new RuntimeException("Failed to get the expected IAE for bad module"); + } catch(IllegalArgumentException e) { + // Expected + } + + try { + ModuleHelper.AddModuleExportsToAll(m2, "p3"); + throw new RuntimeException("Failed to get the expected IAE for package that is in another module"); + } catch(IllegalArgumentException e) { + // Expected + } + + try { + ModuleHelper.AddModuleExportsToAll(m2, "p4"); + throw new RuntimeException("Failed to get the expected IAE for package not in any module"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Export package p2 in m2 unqualifiedly. Then, do a qualified export + // of p2 in m2 to m3. This should not affect the unqualified export. + m3 = ModuleHelper.ModuleObject("module3", this_cldr, new String[] { "p4" }); + assertNotNull(m3, "Module m3 should not be null"); + ModuleHelper.DefineModule(m3, "9.0", "m3/there", new String[] { "p4" }); + ModuleHelper.AddModuleExportsToAll(m2, "p2"); + ModuleHelper.AddModuleExports(m2, "p2", m3); + + // p1.c1's ctor tries to call a method in p2.c2. This should succeed because + // p1 is in an unnamed module and p2.c2 is exported unqualifiedly. + Class p1_c1_class = Class.forName("p1.c1"); + try { + Object c1_obj = p1_c1_class.newInstance(); + } catch (IllegalAccessError f) { + throw new RuntimeException( + "Class in unnamed module could not access package p2 exported unqualifieldly"); + } + } +} + diff --git a/hotspot/test/runtime/modules/JVMAddModulePackage.java b/hotspot/test/runtime/modules/JVMAddModulePackage.java new file mode 100644 index 00000000000..430c67ca6fd --- /dev/null +++ b/hotspot/test/runtime/modules/JVMAddModulePackage.java @@ -0,0 +1,129 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMAddModulePackage + */ + +import static jdk.test.lib.Asserts.*; + +public class JVMAddModulePackage { + + public static void main(String args[]) throws Throwable { + MyClassLoader cl1 = new MyClassLoader(); + MyClassLoader cl3 = new MyClassLoader(); + Object module1, module2, module3; + boolean result; + + module1 = ModuleHelper.ModuleObject("module1", cl1, new String[] { "mypackage" }); + assertNotNull(module1, "Module should not be null"); + ModuleHelper.DefineModule(module1, "9.0", "module1/here", new String[] { "mypackage" }); + module2 = ModuleHelper.ModuleObject("module2", cl1, new String[] { "yourpackage" }); + assertNotNull(module2, "Module should not be null"); + ModuleHelper.DefineModule(module2, "9.0", "module2/here", new String[] { "yourpackage" }); + module3 = ModuleHelper.ModuleObject("module3", cl3, new String[] { "package/num3" }); + assertNotNull(module3, "Module should not be null"); + ModuleHelper.DefineModule(module3, "9.0", "module3/here", new String[] { "package/num3" }); + + // Simple call + ModuleHelper.AddModulePackage(module1, "new_package"); + + // Add a package and export it + ModuleHelper.AddModulePackage(module1, "package/num3"); + ModuleHelper.AddModuleExportsToAll(module1, "package/num3"); + + // Null module argument, expect an NPE + try { + ModuleHelper.AddModulePackage(null, "new_package"); + throw new RuntimeException("Failed to get the expected NPE"); + } catch(NullPointerException e) { + // Expected + } + + // Bad module argument, expect an IAE + try { + ModuleHelper.AddModulePackage(cl1, "new_package"); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Null package argument, expect an NPE + try { + ModuleHelper.AddModulePackage(module1, null); + throw new RuntimeException("Failed to get the expected NPE"); + } catch(NullPointerException e) { + // Expected + } + + // Existing package, expect an IAE + try { + ModuleHelper.AddModulePackage(module1, "yourpackage"); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Invalid package name, expect an IAE + try { + ModuleHelper.AddModulePackage(module1, "your.package"); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Invalid package name, expect an IAE + try { + ModuleHelper.AddModulePackage(module1, ";your/package"); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Invalid package name, expect an IAE + try { + ModuleHelper.AddModulePackage(module1, "7[743"); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Empty package name, expect an IAE + try { + ModuleHelper.AddModulePackage(module1, ""); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + } + + static class MyClassLoader extends ClassLoader { } +} + diff --git a/hotspot/test/runtime/modules/JVMAddReadsModule.java b/hotspot/test/runtime/modules/JVMAddReadsModule.java new file mode 100644 index 00000000000..78f1d3bf62e --- /dev/null +++ b/hotspot/test/runtime/modules/JVMAddReadsModule.java @@ -0,0 +1,83 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMAddReadsModule + */ + +import static jdk.test.lib.Asserts.*; + +public class JVMAddReadsModule { + + public static void main(String args[]) throws Throwable { + MyClassLoader from_cl = new MyClassLoader(); + MyClassLoader to_cl = new MyClassLoader(); + Object from_module, to_module; + + from_module = ModuleHelper.ModuleObject("from_module", from_cl, new String[] { "mypackage" }); + assertNotNull(from_module, "Module should not be null"); + ModuleHelper.DefineModule(from_module, "9.0", "from_module/here", new String[] { "mypackage" }); + + to_module = ModuleHelper.ModuleObject("to_module", to_cl, new String[] { "yourpackage" }); + assertNotNull(to_module, "Module should not be null"); + ModuleHelper.DefineModule(to_module, "9.0", "to_module/here", new String[] { "yourpackage" }); + + // Null from_module argument, expect an NPE + try { + ModuleHelper.AddReadsModule(null, to_module); + throw new RuntimeException("Failed to get the expected NPE"); + } catch(NullPointerException e) { + // Expected + } + + // Null to_module argument, do not expect an NPE + try { + ModuleHelper.AddReadsModule(from_module, null); + } catch(NullPointerException e) { + throw new RuntimeException("Unexpected NPE was thrown"); + } + + // Null from_module and to_module arguments, expect an NPE + try { + ModuleHelper.AddReadsModule(null, null); + throw new RuntimeException("Failed to get the expected NPE"); + } catch(NullPointerException e) { + // Expected + } + + // Both modules are the same, should not throw an exception + ModuleHelper.AddReadsModule(from_module, from_module); + + // Duplicate calls, should not throw an exception + ModuleHelper.AddReadsModule(from_module, to_module); + ModuleHelper.AddReadsModule(from_module, to_module); + } + + static class MyClassLoader extends ClassLoader { } +} diff --git a/hotspot/test/runtime/modules/JVMCanReadModule.java b/hotspot/test/runtime/modules/JVMCanReadModule.java new file mode 100644 index 00000000000..ca131df42ec --- /dev/null +++ b/hotspot/test/runtime/modules/JVMCanReadModule.java @@ -0,0 +1,96 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMCanReadModule + */ + +import static jdk.test.lib.Asserts.*; + +public class JVMCanReadModule { + + public static void main(String args[]) throws Throwable { + MyClassLoader asking_cl = new MyClassLoader(); + MyClassLoader target_cl = new MyClassLoader(); + Object asking_module, target_module; + boolean result; + + asking_module = ModuleHelper.ModuleObject("asking_module", asking_cl, new String[] { "mypackage" }); + assertNotNull(asking_module, "Module should not be null"); + ModuleHelper.DefineModule(asking_module, "9.0", "asking_module/here", new String[] { "mypackage" }); + target_module = ModuleHelper.ModuleObject("target_module", target_cl, new String[] { "yourpackage" }); + assertNotNull(target_module, "Module should not be null"); + ModuleHelper.DefineModule(target_module, "9.0", "target_module/here", new String[] { "yourpackage" }); + + // Set up relationship + ModuleHelper.AddReadsModule(asking_module, target_module); + + // Null asking_module argument, expect an NPE + try { + result = ModuleHelper.CanReadModule(null, target_module); + throw new RuntimeException("Failed to get the expected NPE"); + } catch(NullPointerException e) { + // Expected + } + + // Bad asking_module argument, expect an IAE + try { + result = ModuleHelper.CanReadModule(asking_cl, target_module); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Bad target_module argument, expect an IAE + try { + result = ModuleHelper.CanReadModule(asking_module, asking_cl); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Verify strict modules can not read the unnamed module + result = ModuleHelper.CanReadModule(target_module, null); + assertFalse(result, "target_module can not read unnamed module"); + + // Verify asking_module can read itself + result = ModuleHelper.CanReadModule(asking_module, asking_module); + assertTrue(result, "asking_module can read itself"); + + // Verify asking_module can read target_module + result = ModuleHelper.CanReadModule(asking_module, target_module); + assertTrue(result, "asking_module can read target_module"); + + // Verify target_module cannot read asking_module + result = ModuleHelper.CanReadModule(target_module, asking_module); + assertTrue(!result, "target_module cannot read asking_module"); + } + + static class MyClassLoader extends ClassLoader { } +} diff --git a/hotspot/test/runtime/modules/JVMDefineModule.java b/hotspot/test/runtime/modules/JVMDefineModule.java new file mode 100644 index 00000000000..77d3c0f3940 --- /dev/null +++ b/hotspot/test/runtime/modules/JVMDefineModule.java @@ -0,0 +1,262 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMDefineModule + */ + +import static jdk.test.lib.Asserts.*; + +public class JVMDefineModule { + + public static void main(String args[]) throws Throwable { + MyClassLoader cl = new MyClassLoader(); + Object m; + + // NULL classloader argument, expect success + m = ModuleHelper.ModuleObject("mymodule", null, new String[] { "mypackage" }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, "9.0", "mymodule/here", new String[] { "mypackage" }); + +/* Invalid test, won't compile. + // Invalid classloader argument, expect an IAE + try { + m = ModuleHelper.ModuleObject("mymodule1", new Object(), new String[] { "mypackage1" }); + ModuleHelper.DefineModule(m, "9.0", "mymodule/here", new String[] { "mypackage1" }); + throw new RuntimeException("Failed to get expected IAE for bad loader"); + } catch(IllegalArgumentException e) { + // Expected + } +*/ + + // NULL package argument, should not throw an exception + m = ModuleHelper.ModuleObject("mymodule2", cl, new String[] { "nullpkg" }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, "9.0", "mymodule2/here", null); + + // Null module argument, expect an NPE + try { + ModuleHelper.DefineModule(null, "9.0", "mymodule/here", new String[] { "mypackage1" }); + throw new RuntimeException("Failed to get expected NPE for null module"); + } catch(NullPointerException e) { + if (!e.getMessage().contains("Null module object")) { + throw new RuntimeException("Failed to get expected IAE message for null module: " + e.getMessage()); + } + // Expected + } + + // Invalid module argument, expect an IAE + try { + ModuleHelper.DefineModule(new Object(), "9.0", "mymodule/here", new String[] { "mypackage1" }); + throw new RuntimeException("Failed to get expected IAE or NPE for bad module"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("module is not a subclass")) { + throw new RuntimeException("Failed to get expected IAE message for bad module: " + e.getMessage()); + } + } + + // NULL module name, expect an IAE or NPE + try { + m = ModuleHelper.ModuleObject(null, cl, new String[] { "mypackage2" }); + ModuleHelper.DefineModule(m, "9.0", "mymodule/here", new String[] { "mypackage2" }); + throw new RuntimeException("Failed to get expected NPE for NULL module"); + } catch(IllegalArgumentException e) { + // Expected + } catch(NullPointerException e) { + // Expected + } + + // module name is java.base, expect an IAE + m = ModuleHelper.ModuleObject("java.base", cl, new String[] { "mypackage3" }); + try { + ModuleHelper.DefineModule(m, "9.0", "mymodule/here", new String[] { "mypackage3" }); + throw new RuntimeException("Failed to get expected IAE for java.base, not defined with boot class loader"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("Class loader must be the boot class loader")) { + throw new RuntimeException("Failed to get expected IAE message for java.base: " + e.getMessage()); + } + } + + // Duplicates in package list, expect an IAE + m = ModuleHelper.ModuleObject("module.x", cl, new String[] { "mypackage4", "mypackage5" }); + try { + ModuleHelper.DefineModule(m, "9.0", "mymodule/here", new String[] { "mypackage4", "mypackage5", "mypackage4" }); + throw new RuntimeException("Failed to get IAE for duplicate packages"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("Duplicate package name")) { + throw new RuntimeException("Failed to get expected IAE message for duplicate package: " + e.getMessage()); + } + } + + // Empty entry in package list, expect an IAE + m = ModuleHelper.ModuleObject("module.y", cl, new String[] { "mypackageX", "mypackageY" }); + try { + ModuleHelper.DefineModule(m, "9.0", "mymodule/here", new String[] { "mypackageX", "", "mypackageY" }); + throw new RuntimeException("Failed to get IAE for empty package"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("Invalid package name")) { + throw new RuntimeException("Failed to get expected IAE message empty package entry: " + e.getMessage()); + } + } + + // Duplicate module name, expect an IAE + m = ModuleHelper.ModuleObject("module.name", cl, new String[] { "mypackage6" }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage6" }); + try { + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage6a" }); + throw new RuntimeException("Failed to get IAE for duplicate module"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("Module module.name is already defined")) { + throw new RuntimeException("Failed to get expected IAE message for duplicate module: " + e.getMessage()); + } + } + + // Package is already defined for class loader, expect an IAE + m = ModuleHelper.ModuleObject("dupl.pkg.module", cl, new String[] { "mypackage6b" }); + try { + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage6" }); + throw new RuntimeException("Failed to get IAE for existing package"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("Package mypackage6 for module dupl.pkg.module already exists for class loader")) { + throw new RuntimeException("Failed to get expected IAE message for duplicate package: " + e.getMessage()); + } + } + + // Empty module name, expect an IAE + try { + m = ModuleHelper.ModuleObject("", cl, new String[] { "mypackage8" }); + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage8" }); + throw new RuntimeException("Failed to get expected IAE for empty module name"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Bad module name, expect an IAE + try { + m = ModuleHelper.ModuleObject("bad;name", cl, new String[] { "mypackage9" }); + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage9" }); + throw new RuntimeException("Failed to get expected IAE for bad;name"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Bad module name, expect an IAE + try { + m = ModuleHelper.ModuleObject(".leadingdot", cl, new String[] { "mypackage9a" }); + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage9a" }); + throw new RuntimeException("Failed to get expected IAE for .leadingdot"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Bad module name, expect an IAE + try { + m = ModuleHelper.ModuleObject("trailingdot.", cl, new String[] { "mypackage9b" }); + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage9b" }); + throw new RuntimeException("Failed to get expected IAE for trailingdot."); + } catch(IllegalArgumentException e) { + // Expected + } + + // Bad module name, expect an IAE + m = ModuleHelper.ModuleObject("consecutive..dots", cl, new String[] { "mypackage9c" }); + try { + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage9c" }); + throw new RuntimeException("Failed to get expected IAE for consecutive..dots"); + } catch(IllegalArgumentException e) { + // Expected + } + + // module name with multiple dots, should be okay + m = ModuleHelper.ModuleObject("more.than.one.dat", cl, new String[] { "mypackage9d" }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "mypackage9d" }); + + // Zero length package list, should be okay + m = ModuleHelper.ModuleObject("zero.packages", cl, new String[] { }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { }); + + // Invalid package name, expect an IAE + m = ModuleHelper.ModuleObject("module5", cl, new String[] { "your.package" }); + try { + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "your.package" }); + throw new RuntimeException("Failed to get expected IAE for your.package"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("Invalid package name")) { + throw new RuntimeException("Failed to get expected IAE message for bad package name: " + e.getMessage()); + } + } + + // Invalid package name, expect an IAE + m = ModuleHelper.ModuleObject("module6", cl, new String[] { "foo" }); // Name irrelevant + try { + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { ";your/package" }); + throw new RuntimeException("Failed to get expected IAE for ;your.package"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("Invalid package name")) { + throw new RuntimeException("Failed to get expected IAE message for bad package name: " + e.getMessage()); + } + } + + // Invalid package name, expect an IAE + m = ModuleHelper.ModuleObject("module7", cl, new String[] { "foo" }); // Name irrelevant + try { + ModuleHelper.DefineModule(m, "9.0", "module.name/here", new String[] { "7[743" }); + throw new RuntimeException("Failed to get expected IAE for package 7[743"); + } catch(IllegalArgumentException e) { + if (!e.getMessage().contains("Invalid package name")) { + throw new RuntimeException("Failed to get expected IAE message for bad package name: " + e.getMessage()); + } + } + + // module version that is null, should be okay + m = ModuleHelper.ModuleObject("module8", cl, new String[] { "a_package_8" }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, null, "module8/here", new String[] { "a_package_8" }); + + // module version that is "", should be okay + m = ModuleHelper.ModuleObject("module9", cl, new String[] { "a_package_9" }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, "", "module9/here", new String[] { "a_package_9" }); + + // module location that is null, should be okay + m = ModuleHelper.ModuleObject("module10", cl, new String[] { "a_package_10" }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, "9.0", null, new String[] { "a_package_10" }); + + // module location that is "", should be okay + m = ModuleHelper.ModuleObject("module11", cl, new String[] { "a_package_11" }); + assertNotNull(m, "Module should not be null"); + ModuleHelper.DefineModule(m, "9.0", "", new String[] { "a_package_11" }); + } + + static class MyClassLoader extends ClassLoader { } +} diff --git a/hotspot/test/runtime/modules/JVMGetModuleByPkgName.java b/hotspot/test/runtime/modules/JVMGetModuleByPkgName.java new file mode 100644 index 00000000000..25065e2812c --- /dev/null +++ b/hotspot/test/runtime/modules/JVMGetModuleByPkgName.java @@ -0,0 +1,90 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @compile p2/c2.java + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMGetModuleByPkgName + */ + +import static jdk.test.lib.Asserts.*; +import java.lang.ClassLoader; +import java.lang.reflect.Module; + +public class JVMGetModuleByPkgName { + + public static void main(String args[]) throws Throwable { + + Module javaBase = ModuleHelper.GetModuleByPackageName(null, "java/lang"); + if (!javaBase.getName().equals("java.base")) { + throw new RuntimeException( + "Failed to get module java.base for package java/lang"); + } + + if (ModuleHelper.GetModuleByPackageName(null, "bad.package.name") != null) { + throw new RuntimeException("Failed to get null for bad.package.name"); + } + + ClassLoader systemLoader = ClassLoader.getSystemClassLoader(); + if (ModuleHelper.GetModuleByPackageName(systemLoader, "java/lang") != null) { + throw new RuntimeException( + "Failed to get null for systemClassLoader and java/lang"); + } + + try { + ModuleHelper.GetModuleByPackageName(systemLoader, null); + throw new RuntimeException( + "Failed to throw NullPointerException for null package name"); + } catch(NullPointerException e) { + // Expected + } + + Module unnamedModule = ModuleHelper.GetModuleByPackageName(systemLoader, ""); + if (unnamedModule.isNamed()) { + throw new RuntimeException( + "Unexpected named module returned for unnamed package"); + } + + p2.c2 obj = new p2.c2(); + unnamedModule = ModuleHelper.GetModuleByPackageName(systemLoader, "p2"); + if (unnamedModule.isNamed()) { + throw new RuntimeException( + "Unexpected named module returned for package p2 in unnamed module"); + } + + MyClassLoader cl1 = new MyClassLoader(); + Module module1 = (Module)ModuleHelper.ModuleObject("module1", cl1, new String[] { "mypackage" }); + assertNotNull(module1, "Module should not be null"); + ModuleHelper.DefineModule(module1, "9.0", "module1/here", new String[] { "mypackage" }); + if (ModuleHelper.GetModuleByPackageName(cl1, "mypackage") != module1) { + throw new RuntimeException("Wrong module returned for cl1 mypackage"); + } + } + + static class MyClassLoader extends ClassLoader { } +} diff --git a/hotspot/test/runtime/modules/JVMIsExportedToModule.java b/hotspot/test/runtime/modules/JVMIsExportedToModule.java new file mode 100644 index 00000000000..0e35b7d8d1c --- /dev/null +++ b/hotspot/test/runtime/modules/JVMIsExportedToModule.java @@ -0,0 +1,123 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary /test/lib /compiler/whitebox .. + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI JVMIsExportedToModule + */ + +import static jdk.test.lib.Asserts.*; + +public class JVMIsExportedToModule { + + public static void main(String args[]) throws Throwable { + MyClassLoader from_cl = new MyClassLoader(); + MyClassLoader to_cl = new MyClassLoader(); + Object from_module, to_module; + boolean result; + + from_module = ModuleHelper.ModuleObject("from_module", from_cl, new String[] { "mypackage", "this/package" }); + assertNotNull(from_module, "Module from_module should not be null"); + ModuleHelper.DefineModule(from_module, "9.0", "from_module/here", new String[] { "mypackage", "this/package" }); + to_module = ModuleHelper.ModuleObject("to_module", to_cl, new String[] { "yourpackage", "that/package" }); + assertNotNull(to_module, "Module to_module should not be null"); + ModuleHelper.DefineModule(to_module, "9.0", "to_module/here", new String[] { "yourpackage", "that/package" }); + + Object unnamed_module = JVMIsExportedToModule.class.getModule(); + assertNotNull(unnamed_module, "Module unnamed_module should not be null"); + + // Null from_module argument, expect an NPE + try { + result = ModuleHelper.IsExportedToModule(null, "mypackage", to_module); + throw new RuntimeException("Failed to get the expected NPE for null from_module"); + } catch(NullPointerException e) { + // Expected + } + + // Null to_module argument, expect an NPE + try { + result = ModuleHelper.IsExportedToModule(from_module, "mypackage", null); + throw new RuntimeException("Failed to get expected NPE for null to_module"); + } catch(NullPointerException e) { + // Expected + } + + // Null package argument, expect an NPE + try { + result = ModuleHelper.IsExportedToModule(from_module, null, to_module); + throw new RuntimeException("Failed to get the expected NPE for null package"); + } catch(NullPointerException e) { + // Expected + } + + // Bad from_module argument, expect an IAE + try { + result = ModuleHelper.IsExportedToModule(to_cl, "mypackage", to_module); + throw new RuntimeException("Failed to get the expected IAE for bad from_module"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Bad to_module argument, expect an IAE + try { + result = ModuleHelper.IsExportedToModule(from_module, "mypackage", from_cl); + throw new RuntimeException("Failed to get the expected IAE"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Check that package is exported to its own module + result = ModuleHelper.IsExportedToModule(from_module, "mypackage", from_module); + assertTrue(result, "Package is always exported to itself"); + + // Package is not in to_module, expect an IAE + try { + result = ModuleHelper.IsExportedToModule(from_module, "yourpackage", from_cl); + throw new RuntimeException("Failed to get the expected IAE for package not in to_module"); + } catch(IllegalArgumentException e) { + // Expected + } + + // Package is accessible when exported to unnamed module + ModuleHelper.AddModuleExportsToAll(from_module, "mypackage"); + result = ModuleHelper.IsExportedToModule(from_module, "mypackage", to_module); + assertTrue(result, "Package exported to unnamed module is visible to named module"); + + result = ModuleHelper.IsExportedToModule(from_module, "mypackage", unnamed_module); + assertTrue(result, "Package exported to unnamed module is visible to unnamed module"); + + // Package is accessible only to named module when exported to named module + ModuleHelper.AddModuleExports(from_module, "this/package", to_module); + result = ModuleHelper.IsExportedToModule(from_module, "this/package", to_module); + assertTrue(result, "Package exported to named module is visible to named module"); + result = ModuleHelper.IsExportedToModule(from_module, "this/package", unnamed_module); + assertTrue(!result, "Package exported to named module is not visible to unnamed module"); + } + + static class MyClassLoader extends ClassLoader { } +} diff --git a/hotspot/test/runtime/modules/LoadUnloadModuleStress.java b/hotspot/test/runtime/modules/LoadUnloadModuleStress.java new file mode 100644 index 00000000000..53d45c9a892 --- /dev/null +++ b/hotspot/test/runtime/modules/LoadUnloadModuleStress.java @@ -0,0 +1,70 @@ +/* + * 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. + */ + +/* + * @test + * @summary Ensure module information is cleaned when owning class loader unloads + * @library /testlibrary /test/lib /compiler/whitebox .. + * @build sun.hotspot.WhiteBox + * @compile/module=java.base java/lang/reflect/ModuleHelper.java + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xmx64m -Xmx64m LoadUnloadModuleStress 15000 + */ + +import java.lang.ref.WeakReference; + +import static jdk.test.lib.Asserts.*; + +public class LoadUnloadModuleStress { + private static long timeout; + private static long timeStamp; + + public static byte[] garbage; + public static volatile WeakReference clweak; + + public static Object createModule() throws Throwable { + MyClassLoader cl = new MyClassLoader(); + Object module = ModuleHelper.ModuleObject("mymodule", cl, new String [] {"PackageA"}); + assertNotNull(module); + ModuleHelper.DefineModule(module, "9.0", "mymodule", new String[] { "PackageA" }); + clweak = new WeakReference<>(cl); + return module; + } + + public static void main(String args[]) throws Throwable { + timeout = Long.valueOf(args[0]); + timeStamp = System.currentTimeMillis(); + + while(System.currentTimeMillis() - timeStamp < timeout) { + WeakReference modweak = new WeakReference<>(createModule()); + + while(clweak.get() != null) { + garbage = new byte[8192]; + System.gc(); + } + assertNull(modweak.get()); + } + } + static class MyClassLoader extends ClassLoader { } +} diff --git a/hotspot/test/runtime/modules/ModuleHelper.java b/hotspot/test/runtime/modules/ModuleHelper.java new file mode 100644 index 00000000000..2761b48ff4a --- /dev/null +++ b/hotspot/test/runtime/modules/ModuleHelper.java @@ -0,0 +1,104 @@ +/* + * 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. + */ + +import java.net.URI; +import java.lang.module.ModuleDescriptor; +import java.lang.reflect.Module; +import java.util.Collections; +import java.util.HashSet; +import java.util.Set; + +import sun.hotspot.WhiteBox; + +public class ModuleHelper { + + public static void DefineModule(Object module, String version, String location, + String[] pkgs) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + wb.DefineModule(module, version, location, pkgs); + } + + public static void AddModuleExports(Object from, String pkg, Object to) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + wb.AddModuleExports(from, pkg, to); + java.lang.reflect.ModuleHelper.addExportsNoSync((Module)from, pkg, (Module)to); + } + + public static void AddReadsModule(Object from, Object to) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + wb.AddReadsModule(from, to); + java.lang.reflect.ModuleHelper.addReadsNoSync((Module)from, (Module)to); + } + + public static void AddModulePackage(Object m, String pkg) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + wb.AddModulePackage(m, pkg); + java.lang.reflect.ModuleHelper.addPackageNoSync((Module)m, pkg); + } + + public static Module GetModuleByPackageName(Object ldr, String pkg) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + return (Module)wb.GetModuleByPackageName(ldr, pkg); + } + + public static void AddModuleExportsToAllUnnamed(Object m, String pkg) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + wb.AddModuleExportsToAllUnnamed(m, pkg); + //java.lang.reflect.ModuleHelper.addExportsToAllUnnamedNoSync((Module)m, pkg); + } + + public static void AddModuleExportsToAll(Object m, String pkg) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + wb.AddModuleExportsToAll(m, pkg); + java.lang.reflect.ModuleHelper.addExportsNoSync((Module)m, pkg, (Module)null); + } + + public static boolean CanReadModule(Object from, Object to) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + return wb.CanReadModule(from, to); + } + + public static boolean IsExportedToModule(Object from, String pkg, + Object to) throws Throwable { + WhiteBox wb = WhiteBox.getWhiteBox(); + return wb.IsExportedToModule(from, pkg, to); + } + + public static Module ModuleObject(String name, ClassLoader loader, String[] pkgs) throws Throwable { + Set pkg_set = new HashSet<>(); + if (pkgs != null) { + for (String pkg: pkgs) { + pkg_set.add(pkg.replace('/', '.')); + } + } else { + pkg_set = Collections.emptySet(); + } + + ModuleDescriptor descriptor = + new ModuleDescriptor.Builder(name).conceals(pkg_set).build(); + URI uri = URI.create("module:/" + name); + + return java.lang.reflect.ModuleHelper.newModule(loader, descriptor); + } + +} diff --git a/hotspot/test/runtime/modules/Visibility/XbootcpNoVisibility.java b/hotspot/test/runtime/modules/Visibility/XbootcpNoVisibility.java new file mode 100644 index 00000000000..3dd36caaa03 --- /dev/null +++ b/hotspot/test/runtime/modules/Visibility/XbootcpNoVisibility.java @@ -0,0 +1,77 @@ +/* + * 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. + */ + +/* + * @test + * @summary Ensure that a class defined within a java.base package can not + * be located via -Xbootclasspath/a + * @library /testlibrary + * @modules java.base/sun.misc + * java.management + * @run main/othervm XbootcpNoVisibility + */ + +import jdk.test.lib.*; + +public class XbootcpNoVisibility { + public static void main(String args[]) throws Exception { + + String Vis3_B_src = + "package sun.util;" + + "public class Vis3_B { public void m() { System.out.println(\"In B's m()\"); } }"; + + ClassFileInstaller.writeClassToDisk("sun/util/Vis3_B", + InMemoryJavaCompiler.compile("sun.util.Vis3_B", Vis3_B_src), System.getProperty("test.classes")); + + String Vis3_A_src = + "import sun.util.*;" + + "public class Vis3_A {" + + " public static void main(String args[]) throws Exception {" + + // Try loading a class within a named package in a module which has been defined + // to the boot loader. In this situation, the class should only be attempted + // to be loaded from the boot loader's module path which consists of: + // [-Xpatch]; exploded build | "modules" jimage + // + // Since the class is located on the boot loader's append path via + // -Xbootclasspath/a specification, it should not be found. + " try {" + + " sun.util.Vis3_B b = new sun.util.Vis3_B();" + + " } catch (NoClassDefFoundError e) {" + + " System.out.println(\"XbootcpNoVisibility PASSED - " + + "test should throw exception\\n\");" + + " return;" + + " }" + + " throw new RuntimeException(\"XbootcpNoVisibility FAILED - " + + "test should have thrown exception\");" + + " }" + + "}"; + + ClassFileInstaller.writeClassToDisk("Vis3_A", + InMemoryJavaCompiler.compile("Vis3_A", Vis3_A_src), System.getProperty("test.classes")); + + new OutputAnalyzer(ProcessTools.createJavaProcessBuilder( + "-Xbootclasspath/a:.", + "Vis3_A") + .start()).shouldContain("XbootcpNoVisibility PASSED"); + } +} diff --git a/hotspot/test/runtime/modules/Visibility/XbootcpVisibility.java b/hotspot/test/runtime/modules/Visibility/XbootcpVisibility.java new file mode 100644 index 00000000000..5eb8882e870 --- /dev/null +++ b/hotspot/test/runtime/modules/Visibility/XbootcpVisibility.java @@ -0,0 +1,109 @@ +/* + * 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. + */ + +/* + * @test + * @summary Ensure that a package whose module has not been defined to the boot loader + * is correctly located with -Xbootclasspath/a + * @requires !(os.family == "windows") + * @library /testlibrary + * @modules java.base/sun.misc + * java.management + * @run main/othervm XbootcpVisibility + */ + +import java.io.File; +import java.nio.file.Files; +import java.nio.file.Paths; + +import jdk.test.lib.*; + +public class XbootcpVisibility { + + public static void main(String[] args) throws Throwable { + + String Vis1_B_src = + "package p2;" + + "public class Vis1_B { public void m() { System.out.println(\"In B's m()\"); } }"; + + ClassFileInstaller.writeClassToDisk("p2/Vis1_B", + InMemoryJavaCompiler.compile("p2.Vis1_B", Vis1_B_src), System.getProperty("test.classes")); + ClassFileInstaller.writeClassToDisk("p2/Vis1_B", "mods1"); + + String Vis1_C_src = + "package p2;" + + "public class Vis1_C { public void m() { System.out.println(\"In C's m()\"); } }"; + + ClassFileInstaller.writeClassToDisk("p2/Vis1_C", + InMemoryJavaCompiler.compile("p2.Vis1_C", Vis1_C_src), System.getProperty("test.classes")); + ClassFileInstaller.writeClassToDisk("p2/Vis1_C", "mods1"); + + String Vis1_A_src = + "public class Vis1_A {" + + " public static void main(String args[]) throws Exception {" + + // Try loading a class within a named package + // in the unnamed module. + // Ensure the class can still be loaded successfully by the + // boot loader since it is located on -Xbootclasspath/a. + " try {" + + " p2.Vis1_B b = new p2.Vis1_B();" + + " if (b.getClass().getClassLoader() != null) {" + + " throw new RuntimeException(\"XbootcpVisibility FAILED - class B " + + "should be loaded by boot class loader\\n\");" + + " }" + + " b.m();" + + // Now that the package p2 has been recorded in the + // unnamed module within the boot loader's PackageEntryTable, + // ensure that a different class within the same package + // can be located on -Xbootclasspath/a as well. + " p2.Vis1_C c = new p2.Vis1_C();" + + " if (c.getClass().getClassLoader() != null) {" + + " throw new RuntimeException(\"XbootcpVisibility FAILED - class C " + + "should be loaded by boot class loader\\n\");" + + " }" + + " c.m();" + + " } catch (Exception e) {" + + " System.out.println(e);" + + " throw new RuntimeException(\"XbootcpVisibility FAILED - " + + "test should not throw exception\\n\");" + + " }" + + " System.out.println(\"XbootcpVisibility PASSED\\n\");" + + " }" + + "}"; + + ClassFileInstaller.writeClassToDisk("Vis1_A", + InMemoryJavaCompiler.compile("Vis1_A", Vis1_A_src), System.getProperty("test.classes")); + + // Make sure the classes are actually being loaded from mods1 + Files.delete(Paths.get(System.getProperty("test.classes") + File.separator + + "p2" + File.separator + "Vis1_B.class")); + Files.delete(Paths.get(System.getProperty("test.classes") + File.separator + + "p2" + File.separator + "Vis1_C.class")); + + new OutputAnalyzer(ProcessTools.createJavaProcessBuilder( + "-Xbootclasspath/a:nonexistent.jar", + "-Xbootclasspath/a:mods1", + "Vis1_A") + .start()).shouldHaveExitValue(0); + } +} diff --git a/hotspot/test/runtime/modules/Visibility/XpatchVisibility.java b/hotspot/test/runtime/modules/Visibility/XpatchVisibility.java new file mode 100644 index 00000000000..1fe53d19b74 --- /dev/null +++ b/hotspot/test/runtime/modules/Visibility/XpatchVisibility.java @@ -0,0 +1,91 @@ +/* + * 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. + */ + +/* + * @test + * @summary Ensure that a newly introduced java.base package placed within the -Xpatch directory + * is considered part of the boot loader's visibility boundary + * @requires !(os.family == "windows") + * @library /testlibrary + * @modules java.base/sun.misc + * java.management + * @run main/othervm XpatchVisibility + */ + +import java.io.File; +import java.nio.file.Files; +import java.nio.file.Paths; + +import jdk.test.lib.*; + +public class XpatchVisibility { + + public static void main(String[] args) throws Throwable { + + String Vis2_B_src = + "package p2;" + + "public class Vis2_B {" + + " public void m() {" + + " System.out.println(\"In B's m()\");" + + " }" + + "}"; + + String Vis2_A_src = + "import p2.*;" + + "public class Vis2_A {" + + " public static void main(String args[]) throws Exception {" + + // Try loading a class within a newly introduced java.base + // package. Make sure the class can be found via -Xpatch. + " try {" + + " p2.Vis2_B b = new p2.Vis2_B();" + + " if (b.getClass().getClassLoader() != null) {" + + " throw new RuntimeException(\"XpatchVisibility FAILED - class B " + + "should be loaded by boot class loader\\n\");" + + " }" + + " b.m();" + + " } catch (Throwable e) {" + + " throw new RuntimeException(\"XpatchVisibility FAILED - test " + + "should not throw an error or exception\\n\");" + + " }" + + " System.out.println(\"XpatchVisibility PASSED\\n\");" + + " }" + + "}"; + + ClassFileInstaller.writeClassToDisk("p2/Vis2_B", + InMemoryJavaCompiler.compile("p2.Vis2_B", Vis2_B_src), System.getProperty("test.classes")); + ClassFileInstaller.writeClassToDisk("p2/Vis2_B", "mods2/java.base"); + + ClassFileInstaller.writeClassToDisk("Vis2_A", + InMemoryJavaCompiler.compile("Vis2_A", Vis2_A_src), System.getProperty("test.classes")); + + // Make sure the classes are actually being loaded from mods2 + Files.delete(Paths.get(System.getProperty("test.classes") + File.separator + + "p2" + File.separator + "Vis2_B.class")); + + new OutputAnalyzer(ProcessTools.createJavaProcessBuilder( + "-Xpatch:mods2", + "-XaddExports:java.base/p2=ALL-UNNAMED", + "Vis2_A") + .start()).shouldHaveExitValue(0); + } +} diff --git a/hotspot/test/runtime/modules/Xpatch/Xpatch2Dirs.java b/hotspot/test/runtime/modules/Xpatch/Xpatch2Dirs.java new file mode 100644 index 00000000000..014476cbd6e --- /dev/null +++ b/hotspot/test/runtime/modules/Xpatch/Xpatch2Dirs.java @@ -0,0 +1,68 @@ +/* + * 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. + */ + +/* + * @test + * @summary Make sure -Xpatch works with multiple directories. + * @library /testlibrary + * @compile Xpatch2DirsMain.java + * @run main Xpatch2Dirs + */ + +import jdk.test.lib.*; +import java.io.File; + +public class Xpatch2Dirs { + + public static void main(String[] args) throws Exception { + String source1 = "package javax.naming.spi; " + + "public class NamingManager { " + + " static { " + + " System.out.println(\"I pass one!\"); " + + " } " + + "}"; + String source2 = "package java.beans; " + + "public class Encoder { " + + " static { " + + " System.out.println(\"I pass two!\"); " + + " } " + + "}"; + + ClassFileInstaller.writeClassToDisk("javax/naming/spi/NamingManager", + InMemoryJavaCompiler.compile("javax.naming.spi.NamingManager", source1, "-Xmodule:java.naming"), + "mods/java.naming"); + + ClassFileInstaller.writeClassToDisk("java/beans/Encoder", + InMemoryJavaCompiler.compile("java.beans.Encoder", source2, "-Xmodule:java.desktop"), + "mods2/java.desktop"); + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-Xpatch:mods" + File.pathSeparator + "mods2", + "Xpatch2DirsMain", "javax.naming.spi.NamingManager", "java.beans.Encoder"); + + OutputAnalyzer oa = new OutputAnalyzer(pb.start()); + oa.shouldContain("I pass one!"); + oa.shouldContain("I pass two!"); + oa.shouldHaveExitValue(0); + } +} diff --git a/hotspot/test/runtime/modules/Xpatch/Xpatch2DirsMain.java b/hotspot/test/runtime/modules/Xpatch/Xpatch2DirsMain.java new file mode 100644 index 00000000000..9691156030b --- /dev/null +++ b/hotspot/test/runtime/modules/Xpatch/Xpatch2DirsMain.java @@ -0,0 +1,31 @@ +/* + * 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. + */ + +// This loads the class affected by the -Xpatch option. For the test to pass +// it must load both classes from the -Xpatch directory, not the jimage file. +public class Xpatch2DirsMain { + public static void main(String[] args) throws Exception { + Class.forName(args[0]); + Class.forName(args[1]); + } +} diff --git a/hotspot/test/runtime/modules/Xpatch/XpatchMain.java b/hotspot/test/runtime/modules/Xpatch/XpatchMain.java new file mode 100644 index 00000000000..5509cbc7621 --- /dev/null +++ b/hotspot/test/runtime/modules/Xpatch/XpatchMain.java @@ -0,0 +1,30 @@ +/* + * 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. + */ + +// This loads the class affected by the -Xpatch option. For the test to pass +// it must load the class from the -Xpatch directory, not the jimage file. +public class XpatchMain { + public static void main(String[] args) throws Exception { + Class.forName(args[0]); + } +} diff --git a/hotspot/test/runtime/modules/Xpatch/XpatchTest.java b/hotspot/test/runtime/modules/Xpatch/XpatchTest.java new file mode 100644 index 00000000000..ddc873feca5 --- /dev/null +++ b/hotspot/test/runtime/modules/Xpatch/XpatchTest.java @@ -0,0 +1,56 @@ +/* + * 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. + */ + +/* + * @test + * @bug 8130399 + * @summary Make sure -Xpatch works for modules besides java.base. + * @library /testlibrary + * @compile XpatchMain.java + * @run main XpatchTest + */ + +import jdk.test.lib.*; + +public class XpatchTest { + + public static void main(String[] args) throws Exception { + String source = "package javax.naming.spi; " + + "public class NamingManager { " + + " static { " + + " System.out.println(\"I pass!\"); " + + " } " + + "}"; + + ClassFileInstaller.writeClassToDisk("javax/naming/spi/NamingManager", + InMemoryJavaCompiler.compile("javax.naming.spi.NamingManager", source, "-Xmodule:java.naming"), + "mods/java.naming"); + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-Xpatch:mods", + "XpatchMain", "javax.naming.spi.NamingManager"); + + new OutputAnalyzer(pb.start()) + .shouldContain("I pass!") + .shouldHaveExitValue(0); + } +} diff --git a/hotspot/test/runtime/modules/Xpatch/XpatchTraceCL.java b/hotspot/test/runtime/modules/Xpatch/XpatchTraceCL.java new file mode 100644 index 00000000000..02edebf60c0 --- /dev/null +++ b/hotspot/test/runtime/modules/Xpatch/XpatchTraceCL.java @@ -0,0 +1,83 @@ +/* + * 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. + */ + +/* + * @test + * @bug 8069469 + * @summary Make sure -XX:+TraceClassLoading works properly with "modules" jimage, + -Xpatch, and with -Xbootclasspath/a + * @library /testlibrary + * @compile XpatchMain.java + * @run main XpatchTraceCL + */ + +import java.io.File; +import jdk.test.lib.*; + +public class XpatchTraceCL { + + public static void main(String[] args) throws Exception { + String source = "package javax.naming.spi; " + + "public class NamingManager { " + + " static { " + + " System.out.println(\"I pass!\"); " + + " } " + + "}"; + + // Test -XX:+TraceClassLoading output for -Xpatch + ClassFileInstaller.writeClassToDisk("javax/naming/spi/NamingManager", + InMemoryJavaCompiler.compile("javax.naming.spi.NamingManager", source, "-Xmodule:java.naming"), + "mods/java.naming"); + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-Xpatch:mods", + "-XX:+TraceClassLoading", "XpatchMain", "javax.naming.spi.NamingManager"); + + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + // "modules" jimage case. + output.shouldContain("[classload] java.lang.Thread source: jrt:/java.base"); + // -Xpatch case. + output.shouldContain("[classload] javax.naming.spi.NamingManager source: mods" + + File.separator + "java.naming"); + // -cp case. + output.shouldContain("[classload] XpatchMain source: file"); + + // Test -XX:+TraceClassLoading output for -Xbootclasspath/a + source = "package XpatchTraceCL_pkg; " + + "public class ItIsI { " + + " static { " + + " System.out.println(\"I also pass!\"); " + + " } " + + "}"; + + ClassFileInstaller.writeClassToDisk("XpatchTraceCL_pkg/ItIsI", + InMemoryJavaCompiler.compile("XpatchTraceCL_pkg.ItIsI", source), + "xbcp"); + + pb = ProcessTools.createJavaProcessBuilder("-Xbootclasspath/a:xbcp", + "-XX:+TraceClassLoading", "XpatchMain", "XpatchTraceCL_pkg.ItIsI"); + output = new OutputAnalyzer(pb.start()); + // -Xbootclasspath/a case. + output.shouldContain("[classload] XpatchTraceCL_pkg.ItIsI source: xbcp"); + output.shouldHaveExitValue(0); + } +} diff --git a/hotspot/test/runtime/modules/XpatchCDS.java b/hotspot/test/runtime/modules/XpatchCDS.java new file mode 100644 index 00000000000..4c80beca9fb --- /dev/null +++ b/hotspot/test/runtime/modules/XpatchCDS.java @@ -0,0 +1,62 @@ +/* + * 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. + */ + +/* + * @test + * @library /testlibrary + * @modules java.base/sun.misc + * @run main XpatchCDS + */ + +import java.io.File; +import jdk.test.lib.*; + +public class XpatchCDS { + + public static void main(String args[]) throws Throwable { + System.out.println("Test that -Xpatch and -Xshare:dump are incompatibable"); + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-Xpatch:.", "-Xshare:dump"); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("Cannot use the following option when dumping the shared archive: -Xpatch"); + + System.out.println("Test that -Xpatch and -Xshare:on are incompatibable"); + String filename = "Xpatch.jsa"; + pb = ProcessTools.createJavaProcessBuilder( + "-XX:+UnlockDiagnosticVMOptions", + "-XX:SharedArchiveFile=" + filename, + "-Xshare:dump"); + output = new OutputAnalyzer(pb.start()); + output.shouldContain("ro space:"); // Make sure archive got created. + + pb = ProcessTools.createJavaProcessBuilder( + "-XX:+UnlockDiagnosticVMOptions", + "-XX:SharedArchiveFile=" + filename, + "-Xshare:on", + "-Xpatch:.", + "-version"); + output = new OutputAnalyzer(pb.start()); + output.shouldContain("The shared archive file cannot be used with -Xpatch"); + + output.shouldHaveExitValue(1); + } +} diff --git a/hotspot/test/runtime/modules/acc_module.jcod b/hotspot/test/runtime/modules/acc_module.jcod new file mode 100644 index 00000000000..573c7dde2a6 --- /dev/null +++ b/hotspot/test/runtime/modules/acc_module.jcod @@ -0,0 +1,137 @@ +/* + * 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. + */ + +/* + * This class consists of the following java code, but has an illegal class + * access_flags value of 0x8000, that should be ignored by the JVM. + * + * public class acc_module { + * public static void main(String[] args) { + * System.out.println("hello"); + * } + * } + */ +class acc_module { + 0xCAFEBABE; + 0; // minor version + 52; // version + [29] { // Constant Pool + ; // first element is empty + Method #6 #15; // #1 at 0x0A + Field #16 #17; // #2 at 0x0F + String #18; // #3 at 0x14 + Method #19 #20; // #4 at 0x17 + class #21; // #5 at 0x1C + class #22; // #6 at 0x1F + Utf8 ""; // #7 at 0x22 + Utf8 "()V"; // #8 at 0x2B + Utf8 "Code"; // #9 at 0x31 + Utf8 "LineNumberTable"; // #10 at 0x38 + Utf8 "main"; // #11 at 0x4A + Utf8 "([Ljava/lang/String;)V"; // #12 at 0x51 + Utf8 "SourceFile"; // #13 at 0x6A + Utf8 "acc_module.java"; // #14 at 0x77 + NameAndType #7 #8; // #15 at 0x89 + class #23; // #16 at 0x8E + NameAndType #24 #25; // #17 at 0x91 + Utf8 "hello"; // #18 at 0x96 + class #26; // #19 at 0x9E + NameAndType #27 #28; // #20 at 0xA1 + Utf8 "acc_module"; // #21 at 0xA6 + Utf8 "java/lang/Object"; // #22 at 0xB3 + Utf8 "java/lang/System"; // #23 at 0xC6 + Utf8 "out"; // #24 at 0xD9 + Utf8 "Ljava/io/PrintStream;"; // #25 at 0xDF + Utf8 "java/io/PrintStream"; // #26 at 0xF7 + Utf8 "println"; // #27 at 0x010D + Utf8 "(Ljava/lang/String;)V"; // #28 at 0x0117 + } // Constant Pool + + 0x8000; // access + #5;// this_cpx + #6;// super_cpx + + [0] { // Interfaces + } // Interfaces + + [0] { // fields + } // fields + + [2] { // methods + { // Member at 0x013B + 0x0001; // access + #7; // name_cpx + #8; // sig_cpx + [1] { // Attributes + Attr(#9, 29) { // Code at 0x0143 + 1; // max_stack + 1; // max_locals + Bytes[5]{ + 0x2AB70001B1; + }; + [0] { // Traps + } // end Traps + [1] { // Attributes + Attr(#10, 6) { // LineNumberTable at 0x015A + [1] { // LineNumberTable + 0 1; // at 0x0166 + } + } // end LineNumberTable + } // Attributes + } // end Code + } // Attributes + } // Member + ; + { // Member at 0x0166 + 0x0009; // access + #11; // name_cpx + #12; // sig_cpx + [1] { // Attributes + Attr(#9, 37) { // Code at 0x016E + 2; // max_stack + 1; // max_locals + Bytes[9]{ + 0xB200021203B60004; + 0xB1; + }; + [0] { // Traps + } // end Traps + [1] { // Attributes + Attr(#10, 10) { // LineNumberTable at 0x0189 + [2] { // LineNumberTable + 0 3; // at 0x0195 + 8 4; // at 0x0199 + } + } // end LineNumberTable + } // Attributes + } // end Code + } // Attributes + } // Member + } // methods + + [1] { // Attributes + Attr(#13, 2) { // SourceFile at 0x019B + #14; + } // end SourceFile + } // Attributes +} // end class acc_module diff --git a/hotspot/test/runtime/modules/getModuleJNI/GetModule.java b/hotspot/test/runtime/modules/getModuleJNI/GetModule.java new file mode 100644 index 00000000000..bc9883c6a64 --- /dev/null +++ b/hotspot/test/runtime/modules/getModuleJNI/GetModule.java @@ -0,0 +1,167 @@ +/* + * 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. + */ + +/* @test + * @summary test JNI_GetModule() API + * @run main/native GetModule + */ + +import java.lang.reflect.Module; +import java.lang.management.LockInfo; +public class GetModule { + + static { + System.loadLibrary("GetModule"); + } + + static native Object callGetModule(java.lang.Class clazz); + static native void callAddModuleReads(java.lang.reflect.Module from_module, + java.lang.reflect.Module source_module); + static native boolean callCanReadModule(java.lang.reflect.Module asking_module, + java.lang.reflect.Module source_module); + + public static void main(String[] args) { + Module module; + + // Module for array of primitives, should be "java.base" + int[] int_array = {1, 2, 3}; + Module javaBaseModule; + try { + javaBaseModule = (Module)callGetModule(int_array.getClass()); + if (!javaBaseModule.getName().equals("java.base")) { + throw new RuntimeException("Unexpected module name for array of primitives: " + + javaBaseModule.getName()); + } + } catch(Throwable e) { + throw new RuntimeException("Unexpected exception for [I: " + e.toString()); + } + + // Module for java.lang.String + java.lang.String str = "abc"; + try { + module = (Module)callGetModule(str.getClass()); + if (!module.getName().equals("java.base")) { + throw new RuntimeException("Unexpected module name for class String: " + + module.getName()); + } + } catch(Throwable e) { + throw new RuntimeException("Unexpected exception for String: " + e.toString()); + } + + // Module for java.lang.management.LockInfo + try { + LockInfo li = new LockInfo("java.lang.Class", 57); + module = (Module)callGetModule(li.getClass()); + if (!module.getName().equals("java.management")) { + throw new RuntimeException("Unexpected module name for class LockInfo: " + + module.getName()); + } + } catch(Throwable e) { + throw new RuntimeException("Unexpected exception for LockInfo: " + e.toString()); + } + + // Unnamed module. + try { + module = (Module)callGetModule(MyClassLoader.class); + if (module == null || module.getName() != null) { + throw new RuntimeException("Bad module for unnamed module"); + } + } catch(Throwable e) { + throw new RuntimeException("Unexpected exception for unnamed module: " + e.toString()); + } + + try { + module = (Module)callGetModule(null); + throw new RuntimeException("Failed to get expected NullPointerException"); + } catch(NullPointerException e) { + // Expected + } + + + // Tests for JNI_AddModuleReads() // + + Module javaScriptingModule = javax.script.Bindings.class.getModule(); + if (javaScriptingModule == null) { + throw new RuntimeException("Failed to get java.scripting module"); + } + Module javaLoggingModule = java.util.logging.Level.class.getModule(); + if (javaLoggingModule == null) { + throw new RuntimeException("Failed to get java.logging module"); + } + + if (callCanReadModule(javaLoggingModule, javaScriptingModule)) { + throw new RuntimeException( + "Expected FALSE because javaLoggingModule cannot read javaScriptingModule"); + } + + callAddModuleReads(javaLoggingModule, javaScriptingModule); + callAddModuleReads(javaScriptingModule, GetModule.class.getModule()); // unnamed module + + try { + callAddModuleReads(null, javaLoggingModule); + throw new RuntimeException( + "Expected NullPointerException for bad from_module not thrown"); + } catch(NullPointerException e) { + // expected + } + + try { + callAddModuleReads(javaLoggingModule, null); + throw new RuntimeException( + "Expected NullPointerException for bad source_module not thrown"); + } catch(NullPointerException e) { + // expected + } + + + // Tests for JNI_CanReadModule() // + + if (!callCanReadModule(javaLoggingModule, javaScriptingModule)) { + throw new RuntimeException( + "Expected TRUE because javaLoggingModule can read javaScriptingModule"); + } + + if (callCanReadModule(javaBaseModule, javaScriptingModule)) { + throw new RuntimeException( + "Expected FALSE because javaBaseModule cannnot read javaScriptingModule"); + } + + try { + callCanReadModule(javaLoggingModule, null); + throw new RuntimeException( + "Expected NullPointerException for bad sourceModule not thrown"); + } catch(NullPointerException e) { + // expected + } + + try { + callCanReadModule(null, javaScriptingModule); + throw new RuntimeException( + "Expected NullPointerException for bad asking_module not thrown"); + } catch(NullPointerException e) { + // expected + } + } + + static class MyClassLoader extends ClassLoader { } +} diff --git a/hotspot/test/runtime/modules/getModuleJNI/libGetModule.c b/hotspot/test/runtime/modules/getModuleJNI/libGetModule.c new file mode 100644 index 00000000000..a7b1b6d2dc0 --- /dev/null +++ b/hotspot/test/runtime/modules/getModuleJNI/libGetModule.c @@ -0,0 +1,42 @@ +/* + * 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. + */ + +#include + +JNIEXPORT jobject JNICALL +Java_GetModule_callGetModule(JNIEnv *env, jclass unused, jclass clazz) { + jobject res = (jobject)((*env)->GetModule(env, clazz)); + return res; +} + +JNIEXPORT void JNICALL +Java_GetModule_callAddModuleReads(JNIEnv *env, jclass unused, jobject from_module, jobject source_module) { + (*env)->AddModuleReads(env, from_module, source_module); +} + +JNIEXPORT jboolean JNICALL +Java_GetModule_callCanReadModule(JNIEnv *env, jclass unused, jobject asking_module, jobject source_module) { + jboolean res = (*env)->CanReadModule(env, asking_module, source_module); + return res; +} + diff --git a/hotspot/test/runtime/modules/java.base/java/lang/reflect/ModuleHelper.java b/hotspot/test/runtime/modules/java.base/java/lang/reflect/ModuleHelper.java new file mode 100644 index 00000000000..ca7977aee8d --- /dev/null +++ b/hotspot/test/runtime/modules/java.base/java/lang/reflect/ModuleHelper.java @@ -0,0 +1,69 @@ +/* + * 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 java.lang.reflect; + +import java.lang.module.ModuleDescriptor; + +/** + * A helper class intended to be injected into java.lang.reflect using the + * java -Xpatch option. The helper class provides access to package private + * methods in java.lang.reflect.Module. + */ + +public final class ModuleHelper { + + private ModuleHelper() { } + + /** + * Creates a named module but without defining the module to the VM. + */ + public static Module newModule(ClassLoader loader, ModuleDescriptor descriptor) { + return new Module(loader, descriptor); + } + + /** + * Updates module {@code from} to that it reads module {@code to} without + * notifying the VM. + */ + public static void addReadsNoSync(Module from, Module to) { + from.implAddReadsNoSync(to); + } + + /** + * Updates module {@code from} so that it exports package {@code pkg} + * to module {@code to} but without notifying the VM. If {@code to} is + * {@code null} then the package is exported unconditionally. + */ + public static void addExportsNoSync(Module from, String pkg, Module to) { + from.implAddExportsNoSync(pkg, to); + } + + /** + * Adds a package to a module without notifying the VM. + */ + public static void addPackageNoSync(Module m, String pkg) { + m.implAddPackageNoSync(pkg); + } + +} diff --git a/hotspot/test/runtime/modules/p1/c1.java b/hotspot/test/runtime/modules/p1/c1.java new file mode 100644 index 00000000000..7fbe4f9dc1d --- /dev/null +++ b/hotspot/test/runtime/modules/p1/c1.java @@ -0,0 +1,35 @@ +/* + * 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. + */ + +// Small class used by multiple hotspot/runtime/modules/AccessCheck* tests. +package p1; + +import p2.c2; + +public class c1 { + + public c1 () { + p2.c2 c2_obj = new p2.c2(); + c2_obj.method2(); + } +} diff --git a/hotspot/test/runtime/modules/p2/c2.java b/hotspot/test/runtime/modules/p2/c2.java new file mode 100644 index 00000000000..74adc645a74 --- /dev/null +++ b/hotspot/test/runtime/modules/p2/c2.java @@ -0,0 +1,29 @@ +/* + * 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. + */ + +// Small class used by multiple hotspot/runtime/modules/AccessCheck* tests. +package p2; + +public class c2 { + public void method2() { } +} diff --git a/hotspot/test/runtime/modules/p3/c3.java b/hotspot/test/runtime/modules/p3/c3.java new file mode 100644 index 00000000000..1e8130c49ad --- /dev/null +++ b/hotspot/test/runtime/modules/p3/c3.java @@ -0,0 +1,29 @@ +/* + * 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. + */ + +// Small class used by multiple hotspot/runtime/modules/AccessCheck* tests. +package p3; + +public class c3 extends p2.c2 { + public void method3() { } +} diff --git a/hotspot/test/serviceability/sa/jmap-hprof/JMapHProfLargeHeapTest.java b/hotspot/test/serviceability/sa/jmap-hprof/JMapHProfLargeHeapTest.java index 33b59c9af07..4e2abad9078 100644 --- a/hotspot/test/serviceability/sa/jmap-hprof/JMapHProfLargeHeapTest.java +++ b/hotspot/test/serviceability/sa/jmap-hprof/JMapHProfLargeHeapTest.java @@ -95,7 +95,7 @@ public class JMapHProfLargeHeapTest { String expectedFormat) throws Exception, IOException, InterruptedException, FileNotFoundException { ProcessBuilder procBuilder = ProcessTools.createJavaProcessBuilder( - vmArgs, "JMapHProfLargeHeapProc", String.valueOf(heapSize)); + "-XaddExports:java.management/sun.management=ALL-UNNAMED", vmArgs, "JMapHProfLargeHeapProc", String.valueOf(heapSize)); procBuilder.redirectError(ProcessBuilder.Redirect.INHERIT); Process largeHeapProc = procBuilder.start(); diff --git a/hotspot/test/testlibrary/ClassFileInstaller.java b/hotspot/test/testlibrary/ClassFileInstaller.java index 4e042dacde4..2c1541d0457 100644 --- a/hotspot/test/testlibrary/ClassFileInstaller.java +++ b/hotspot/test/testlibrary/ClassFileInstaller.java @@ -23,6 +23,7 @@ import java.io.FileNotFoundException; import java.io.InputStream; +import java.io.ByteArrayInputStream; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; @@ -38,22 +39,47 @@ public class ClassFileInstaller { */ public static void main(String... args) throws Exception { for (String arg : args) { - ClassLoader cl = ClassFileInstaller.class.getClassLoader(); - - // Convert dotted class name to a path to a class file - String pathName = arg.replace('.', '/').concat(".class"); - InputStream is = cl.getResourceAsStream(pathName); - if (is == null) { - throw new FileNotFoundException(pathName); - } - - // Create the class file's package directory - Path p = Paths.get(pathName); - if (pathName.contains("/")) { - Files.createDirectories(p.getParent()); - } - // Create the class file - Files.copy(is, p, StandardCopyOption.REPLACE_EXISTING); + writeClassToDisk(arg); } } + + public static void writeClassToDisk(String className) throws Exception { + writeClassToDisk(className, ""); + } + + public static void writeClassToDisk(String className, String prependPath) throws Exception { + ClassLoader cl = ClassFileInstaller.class.getClassLoader(); + + // Convert dotted class name to a path to a class file + String pathName = className.replace('.', '/').concat(".class"); + InputStream is = cl.getResourceAsStream(pathName); + if (prependPath.length() > 0) { + pathName = prependPath + "/" + pathName; + } + writeToDisk(pathName, is); + } + + public static void writeClassToDisk(String className, byte[] bytecode) throws Exception { + writeClassToDisk(className, bytecode, ""); + } + + public static void writeClassToDisk(String className, byte[] bytecode, String prependPath) throws Exception { + // Convert dotted class name to a path to a class file + String pathName = className.replace('.', '/').concat(".class"); + if (prependPath.length() > 0) { + pathName = prependPath + "/" + pathName; + } + writeToDisk(pathName, new ByteArrayInputStream(bytecode)); + } + + + private static void writeToDisk(String pathName, InputStream is) throws Exception { + // Create the class file's package directory + Path p = Paths.get(pathName); + if (pathName.contains("/")) { + Files.createDirectories(p.getParent()); + } + // Create the class file + Files.copy(is, p, StandardCopyOption.REPLACE_EXISTING); + } } diff --git a/hotspot/test/testlibrary/jdk/test/lib/InMemoryJavaCompiler.java b/hotspot/test/testlibrary/jdk/test/lib/InMemoryJavaCompiler.java index 80439f898b6..8384b28ca5d 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/InMemoryJavaCompiler.java +++ b/hotspot/test/testlibrary/jdk/test/lib/InMemoryJavaCompiler.java @@ -128,12 +128,13 @@ public class InMemoryJavaCompiler { * * @param className The name of the class * @param sourceCode The source code for the class with name {@code className} + * @param options additional command line options * @throws RuntimeException if the compilation did not succeed * @return The resulting byte code from the compilation */ - public static byte[] compile(String className, CharSequence sourceCode) { + public static byte[] compile(String className, CharSequence sourceCode, String... options) { MemoryJavaFileObject file = new MemoryJavaFileObject(className, sourceCode); - CompilationTask task = getCompilationTask(file); + CompilationTask task = getCompilationTask(file, options); if(!task.call()) { throw new RuntimeException("Could not compile " + className + " with source code " + sourceCode); @@ -146,7 +147,7 @@ public class InMemoryJavaCompiler { return ToolProvider.getSystemJavaCompiler(); } - private static CompilationTask getCompilationTask(MemoryJavaFileObject file) { - return getCompiler().getTask(null, new FileManagerWrapper(file), null, null, null, Arrays.asList(file)); + private static CompilationTask getCompilationTask(MemoryJavaFileObject file, String... options) { + return getCompiler().getTask(null, new FileManagerWrapper(file), null, Arrays.asList(options), null, Arrays.asList(file)); } } diff --git a/hotspot/test/testlibrary/jdk/test/lib/cli/CommandLineOptionTest.java b/hotspot/test/testlibrary/jdk/test/lib/cli/CommandLineOptionTest.java index e1bb112ceb3..5fd3326c173 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/cli/CommandLineOptionTest.java +++ b/hotspot/test/testlibrary/jdk/test/lib/cli/CommandLineOptionTest.java @@ -101,6 +101,7 @@ public abstract class CommandLineOptionTest { throws Throwable { List finalOptions = new ArrayList<>(); if (addTestVMOptions) { + Collections.addAll(finalOptions, ProcessTools.getVmInputArgs()); Collections.addAll(finalOptions, Utils.getTestJavaOpts()); } Collections.addAll(finalOptions, options); diff --git a/hotspot/test/testlibrary/whitebox/sun/hotspot/WhiteBox.java b/hotspot/test/testlibrary/whitebox/sun/hotspot/WhiteBox.java new file mode 100644 index 00000000000..367e46965c0 --- /dev/null +++ b/hotspot/test/testlibrary/whitebox/sun/hotspot/WhiteBox.java @@ -0,0 +1,264 @@ +/* + * Copyright (c) 2012, 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 sun.hotspot; + +import java.lang.reflect.Executable; +import java.util.Arrays; +import java.util.List; +import java.util.function.Function; +import java.util.stream.Stream; +import java.security.BasicPermission; + +import sun.hotspot.parser.DiagnosticCommand; + +public class WhiteBox { + + @SuppressWarnings("serial") + public static class WhiteBoxPermission extends BasicPermission { + public WhiteBoxPermission(String s) { + super(s); + } + } + + private WhiteBox() {} + private static final WhiteBox instance = new WhiteBox(); + private static native void registerNatives(); + + /** + * Returns the singleton WhiteBox instance. + * + * The returned WhiteBox object should be carefully guarded + * by the caller, since it can be used to read and write data + * at arbitrary memory addresses. It must never be passed to + * untrusted code. + */ + public synchronized static WhiteBox getWhiteBox() { + SecurityManager sm = System.getSecurityManager(); + if (sm != null) { + sm.checkPermission(new WhiteBoxPermission("getInstance")); + } + return instance; + } + + static { + registerNatives(); + } + + // Get the maximum heap size supporting COOPs + public native long getCompressedOopsMaxHeapSize(); + // Arguments + public native void printHeapSizes(); + + // Memory + public native long getObjectAddress(Object o); + public native int getHeapOopSize(); + public native int getVMPageSize(); + public native boolean isObjectInOldGen(Object o); + public native long getObjectSize(Object o); + + // Runtime + // Make sure class name is in the correct format + public boolean isClassAlive(String name) { + return isClassAlive0(name.replace('.', '/')); + } + private native boolean isClassAlive0(String name); + + // JVMTI + public native void addToBootstrapClassLoaderSearch(String segment); + public native void addToSystemClassLoaderSearch(String segment); + + // G1 + public native boolean g1InConcurrentMark(); + public native boolean g1IsHumongous(Object o); + public native long g1NumFreeRegions(); + public native int g1RegionSize(); + public native Object[] parseCommandLine(String commandline, char delim, DiagnosticCommand[] args); + + // NMT + public native long NMTMalloc(long size); + public native void NMTFree(long mem); + public native long NMTReserveMemory(long size); + public native void NMTCommitMemory(long addr, long size); + public native void NMTUncommitMemory(long addr, long size); + public native void NMTReleaseMemory(long addr, long size); + public native long NMTMallocWithPseudoStack(long size, int index); + public native boolean NMTIsDetailSupported(); + public native boolean NMTChangeTrackingLevel(); + public native int NMTGetHashSize(); + + // Compiler + public native void deoptimizeAll(); + public boolean isMethodCompiled(Executable method) { + return isMethodCompiled(method, false /*not osr*/); + } + public native boolean isMethodCompiled(Executable method, boolean isOsr); + public boolean isMethodCompilable(Executable method) { + return isMethodCompilable(method, -1 /*any*/); + } + public boolean isMethodCompilable(Executable method, int compLevel) { + return isMethodCompilable(method, compLevel, false /*not osr*/); + } + public native boolean isMethodCompilable(Executable method, int compLevel, boolean isOsr); + public native boolean isMethodQueuedForCompilation(Executable method); + public int deoptimizeMethod(Executable method) { + return deoptimizeMethod(method, false /*not osr*/); + } + public native int deoptimizeMethod(Executable method, boolean isOsr); + public void makeMethodNotCompilable(Executable method) { + makeMethodNotCompilable(method, -1 /*any*/); + } + public void makeMethodNotCompilable(Executable method, int compLevel) { + makeMethodNotCompilable(method, compLevel, false /*not osr*/); + } + public native void makeMethodNotCompilable(Executable method, int compLevel, boolean isOsr); + public int getMethodCompilationLevel(Executable method) { + return getMethodCompilationLevel(method, false /*not ost*/); + } + public native int getMethodCompilationLevel(Executable method, boolean isOsr); + public native boolean testSetDontInlineMethod(Executable method, boolean value); + public int getCompileQueuesSize() { + return getCompileQueueSize(-1 /*any*/); + } + public native int getCompileQueueSize(int compLevel); + public native boolean testSetForceInlineMethod(Executable method, boolean value); + public boolean enqueueMethodForCompilation(Executable method, int compLevel) { + return enqueueMethodForCompilation(method, compLevel, -1 /*InvocationEntryBci*/); + } + public native boolean enqueueMethodForCompilation(Executable method, int compLevel, int entry_bci); + public native void clearMethodState(Executable method); + public native void lockCompilation(); + public native void unlockCompilation(); + public native int getMethodEntryBci(Executable method); + public native Object[] getNMethod(Executable method, boolean isOsr); + public native long allocateCodeBlob(int size, int type); + public long allocateCodeBlob(long size, int type) { + int intSize = (int) size; + if ((long) intSize != size || size < 0) { + throw new IllegalArgumentException( + "size argument has illegal value " + size); + } + return allocateCodeBlob( intSize, type); + } + public native void freeCodeBlob(long addr); + public void forceNMethodSweep() { + try { + forceNMethodSweep0().join(); + } catch (InterruptedException e) { + Thread.currentThread().interrupt(); + } + } + public native Thread forceNMethodSweep0(); + public native Object[] getCodeHeapEntries(int type); + public native int getCompilationActivityMode(); + public native Object[] getCodeBlob(long addr); + + // Intered strings + public native boolean isInStringTable(String str); + + // Memory + public native void readReservedMemory(); + public native long allocateMetaspace(ClassLoader classLoader, long size); + public native void freeMetaspace(ClassLoader classLoader, long addr, long size); + public native long incMetaspaceCapacityUntilGC(long increment); + public native long metaspaceCapacityUntilGC(); + + // Force Young GC + public native void youngGC(); + + // Force Full GC + public native void fullGC(); + + // Method tries to start concurrent mark cycle. + // It returns false if CM Thread is always in concurrent cycle. + public native boolean g1StartConcMarkCycle(); + + // Tests on ReservedSpace/VirtualSpace classes + public native int stressVirtualSpaceResize(long reservedSpaceSize, long magnitude, long iterations); + public native void runMemoryUnitTests(); + public native void readFromNoaccessArea(); + public native long getThreadStackSize(); + public native long getThreadRemainingStackSize(); + + // CPU features + public native String getCPUFeatures(); + + // Native extensions + public native long getHeapUsageForContext(int context); + public native long getHeapRegionCountForContext(int context); + public native int getContextForObject(Object obj); + public native void printRegionInfo(int context); + + // VM flags + public native boolean isConstantVMFlag(String name); + public native boolean isLockedVMFlag(String name); + public native void setBooleanVMFlag(String name, boolean value); + public native void setIntxVMFlag(String name, long value); + public native void setUintxVMFlag(String name, long value); + public native void setUint64VMFlag(String name, long value); + public native void setSizeTVMFlag(String name, long value); + public native void setStringVMFlag(String name, String value); + public native void setDoubleVMFlag(String name, double value); + public native Boolean getBooleanVMFlag(String name); + public native Long getIntxVMFlag(String name); + public native Long getUintxVMFlag(String name); + public native Long getUint64VMFlag(String name); + public native Long getSizeTVMFlag(String name); + public native String getStringVMFlag(String name); + public native Double getDoubleVMFlag(String name); + private final List> flagsGetters = Arrays.asList( + this::getBooleanVMFlag, this::getIntxVMFlag, this::getUintxVMFlag, + this::getUint64VMFlag, this::getSizeTVMFlag, this::getStringVMFlag, + this::getDoubleVMFlag); + + public Object getVMFlag(String name) { + return flagsGetters.stream() + .map(f -> f.apply(name)) + .filter(x -> x != null) + .findAny() + .orElse(null); + } + + // Jigsaw + public native Object DefineModule(String name, Object loader, Object[] packages); + public native void AddModuleExports(Object from_module, String pkg, Object to_module); + public native void AddReadsModule(Object from_module, Object to_module); + public native boolean CanReadModule(Object asking_module, Object target_module); + public native boolean IsExportedToModule(Object from_module, String pkg, Object to_module); + public native Object GetModule(Class clazz); + public native void AddModulePackage(Object module, String pkg); + + // Image File + public native boolean readImageFile(String imagefile); + + public native int getOffsetForName0(String name); + public int getOffsetForName(String name) throws Exception { + int offset = getOffsetForName0(name); + if (offset == -1) { + throw new RuntimeException(name + " not found"); + } + return offset; + } + +}