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 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/corba/.hgtags b/corba/.hgtags index 5c2b765d93d..9c059fc2de3 100644 --- a/corba/.hgtags +++ b/corba/.hgtags @@ -352,3 +352,4 @@ e385e95e6101711d5c63e7b1a827e99b6ec7a1cc jdk-9+104 49202432b69445164a42be7cbdf74ed5fce98157 jdk-9+107 84f2862a25eb3232ff36c376b4e2bf2a83dfced3 jdk-9+108 b75afa17aefe480c23c616a6a2497063312f7189 jdk-9+109 +9666775734fb6028ee86df9972626b3667b6a318 jdk-9+110 diff --git a/corba/make/gensrc/Gensrc-java.corba.gmk b/corba/make/gensrc/Gensrc-java.corba.gmk index 608267e0df1..2d186902777 100644 --- a/corba/make/gensrc/Gensrc-java.corba.gmk +++ b/corba/make/gensrc/Gensrc-java.corba.gmk @@ -47,7 +47,7 @@ $(eval $(call SetupJavaCompilation,BUILD_IDLJ, \ BIN := $(BUILDTOOLS_OUTPUTDIR)/idlj_classes, \ COPY := .prp, \ INCLUDES := com/sun/tools/corba/se/idl, \ - EXCLUDE_FILES := ResourceBundleUtil.java)) + EXCLUDE_FILES := ResourceBundleUtil.java module-info.java)) # Force the language to english for predictable source code generation. TOOL_IDLJ_CMD := $(JAVA) -cp $(BUILDTOOLS_OUTPUTDIR)/idlj_classes \ diff --git a/corba/src/java.corba/share/classes/module-info.java b/corba/src/java.corba/share/classes/module-info.java new file mode 100644 index 00000000000..23da4b785a9 --- /dev/null +++ b/corba/src/java.corba/share/classes/module-info.java @@ -0,0 +1,77 @@ +/* + * 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 java.corba { + requires public java.desktop; + requires public java.rmi; + requires java.logging; + requires java.naming; + requires java.transaction; + + exports javax.activity; + exports javax.rmi; + exports javax.rmi.CORBA; + exports org.omg.CORBA; + exports org.omg.CORBA.DynAnyPackage; + exports org.omg.CORBA.ORBPackage; + exports org.omg.CORBA.TypeCodePackage; + exports org.omg.CORBA.portable; + exports org.omg.CORBA_2_3; + exports org.omg.CORBA_2_3.portable; + exports org.omg.CosNaming; + exports org.omg.CosNaming.NamingContextExtPackage; + exports org.omg.CosNaming.NamingContextPackage; + exports org.omg.Dynamic; + exports org.omg.DynamicAny; + exports org.omg.DynamicAny.DynAnyFactoryPackage; + exports org.omg.DynamicAny.DynAnyPackage; + exports org.omg.IOP; + exports org.omg.IOP.CodecFactoryPackage; + exports org.omg.IOP.CodecPackage; + exports org.omg.Messaging; + exports org.omg.PortableInterceptor; + exports org.omg.PortableInterceptor.ORBInitInfoPackage; + exports org.omg.PortableServer; + exports org.omg.PortableServer.CurrentPackage; + exports org.omg.PortableServer.POAManagerPackage; + exports org.omg.PortableServer.POAPackage; + exports org.omg.PortableServer.ServantLocatorPackage; + exports org.omg.PortableServer.portable; + exports org.omg.SendingContext; + exports org.omg.stub.java.rmi; + exports com.sun.corba.se.impl.util to + jdk.rmic; + exports com.sun.jndi.cosnaming to + java.naming; + exports com.sun.jndi.url.corbaname to + java.naming; + exports com.sun.jndi.url.iiop to + java.naming; + exports com.sun.jndi.url.iiopname to + java.naming; + + provides javax.naming.spi.InitialContextFactory + with com.sun.jndi.cosnaming.CNCtxFactory; +} 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 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/runtime/modules/AccModuleTest.java b/hotspot/test/runtime/modules/AccModuleTest.java new file mode 100644 index 00000000000..f1f8ab0bc60 --- /dev/null +++ b/hotspot/test/runtime/modules/AccModuleTest.java @@ -0,0 +1,42 @@ +/* + * 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 + * @compile acc_module.jcod + * @build AccModuleTest + * @run main AccModuleTest + */ + +import java.io.File; +import jdk.test.lib.*; + +public class AccModuleTest { + + 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/modules/AccessCheck/c5.java b/hotspot/test/runtime/modules/AccessCheck/c5.java new file mode 100644 index 00000000000..1a3c25a1249 --- /dev/null +++ b/hotspot/test/runtime/modules/AccessCheck/c5.java @@ -0,0 +1,33 @@ +/* + * 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. +import p6.c6; + +public class c5 { + + 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; + } + +} diff --git a/jdk/.hgtags b/jdk/.hgtags index 8d4f94f1f2c..49cf4dd5ea8 100644 --- a/jdk/.hgtags +++ b/jdk/.hgtags @@ -352,3 +352,4 @@ eee1ced1d8e78293f2a004af818ca474387dbebf jdk-9+103 8701b2bb1d2e1b9abc2a9be0933993c7150a9dbe jdk-9+107 42794e648cfe9fd67461dcbe8b7594241a84bcff jdk-9+108 1c7bad0798900fe58f4db01ae7ffdc84f5baee8c jdk-9+109 +9417e1bcded6af5532c3b26235437ab227758877 jdk-9+110 diff --git a/jdk/make/CompileInterimRmic.gmk b/jdk/make/CompileInterimRmic.gmk index f31db8dfd04..ce60127e6d1 100644 --- a/jdk/make/CompileInterimRmic.gmk +++ b/jdk/make/CompileInterimRmic.gmk @@ -44,8 +44,9 @@ RMIC_PKGS := \ $(eval $(call SetupJavaCompilation,BUILD_INTERIM_RMIC, \ SETUP := GENERATE_OLDBYTECODE, \ SRC := $(JDK_TOPDIR)/src/jdk.rmic/share/classes, \ + EXCLUDE_FILES := module-info.java, \ INCLUDES := $(RMIC_PKGS), \ - BIN := $(BUILDTOOLS_OUTPUTDIR)/interim_rmic_classes, \ + BIN := $(BUILDTOOLS_OUTPUTDIR)/override_modules/jdk.rmic, \ COPY := .properties)) ########################################################################################## diff --git a/jdk/make/CompileTools.gmk b/jdk/make/CompileTools.gmk index 27aaffaafdb..46dab28f146 100644 --- a/jdk/make/CompileTools.gmk +++ b/jdk/make/CompileTools.gmk @@ -32,42 +32,12 @@ include SetupJavaCompilers.gmk ################################################################################ -JIMAGE_PKGS := \ - jdk/internal/jimage \ - jdk/internal/jrtfs \ - # - -$(eval $(call SetupJavaCompilation,BUILD_INTERIM_JIMAGE, \ - SETUP := GENERATE_OLDBYTECODE, \ - SRC := $(JDK_TOPDIR)/src/java.base/share/classes, \ - INCLUDES := $(JIMAGE_PKGS), \ - BIN := $(BUILDTOOLS_OUTPUTDIR)/interim_jimage_classes)) - -TARGETS += $(BUILD_INTERIM_JIMAGE) - -# 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 := $(BUILDTOOLS_OUTPUTDIR)/interim_jimage_classes, \ - FILES := META-INF/services/java.nio.file.spi.FileSystemProvider)) - -TARGETS += $(COPY_JIMAGE_SERVICE_PROVIDER) - -################################################################################ - $(eval $(call SetupJavaCompilation,BUILD_TOOLS_JDK, \ SETUP := GENERATE_OLDBYTECODE, \ - ADD_JAVAC_FLAGS := -Xbootclasspath/p:$(call PathList, \ - $(BUILDTOOLS_OUTPUTDIR)/interim_jimage_classes \ - $(BUILDTOOLS_OUTPUTDIR)/interim_cldrconverter_classes), \ SRC := $(JDK_TOPDIR)/make/src/classes $(BUILDTOOLS_OUTPUTDIR)/interim_cldrconverter_classes, \ - BIN := $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes, \ - COPY := boot.modules ext.modules)) - -$(BUILD_TOOLS_JDK): $(BUILD_INTERIM_JIMAGE) $(COPY_JIMAGE_SERVICE_PROVIDER) + EXCLUDES := build/tools/deps \ + build/tools/jigsaw, \ + BIN := $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes)) TARGETS += $(BUILD_TOOLS_JDK) diff --git a/make/CheckModules.gmk b/jdk/make/GenerateModuleSummary.gmk similarity index 67% rename from make/CheckModules.gmk rename to jdk/make/GenerateModuleSummary.gmk index 4872c912e49..285de739088 100644 --- a/make/CheckModules.gmk +++ b/jdk/make/GenerateModuleSummary.gmk @@ -28,19 +28,20 @@ default: all include $(SPEC) include MakeBase.gmk +include ModuleTools.gmk -JDEPS_MODULES_XML := $(JDK_OUTPUTDIR)/modules/jdk.jdeps/com/sun/tools/jdeps/resources/jdeps-modules.xml +GENGRAPHS_DIR := $(IMAGES_OUTPUTDIR)/gengraphs +TOOLS_MODULE_SRCDIR := $(JDK_TOPDIR)/make/src/classes/build/tools/jigsaw -# -# 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 +$(GENGRAPHS_DIR)/jdk.dot: $(BUILD_JIGSAW_TOOLS) + $(MKDIR) -p $(@D) + $(TOOL_GENGRAPHS) $(GENGRAPHS_DIR) -all: checkdeps +$(GENGRAPHS_DIR)/technology-summary.html: $(TOOLS_MODULE_SRCDIR)/technology-summary.html + $(install-file) -.PHONY: all +$(GENGRAPHS_DIR)/module-summary.html: $(BUILD_JIGSAW_TOOLS) $(GENGRAPHS_DIR)/technology-summary.html + $(MKDIR) -p $(@D) + $(TOOL_MODULESUMMARY) -o $@ -mp $(IMAGES_OUTPUTDIR)/jmods + +all: $(GENGRAPHS_DIR)/jdk.dot $(GENGRAPHS_DIR)/module-summary.html diff --git a/jdk/make/gendata/Gendata-jdk.jdeps.gmk b/jdk/make/ModuleTools.gmk similarity index 57% rename from jdk/make/gendata/Gendata-jdk.jdeps.gmk rename to jdk/make/ModuleTools.gmk index 3616796eed4..2943b9c8d21 100644 --- a/jdk/make/gendata/Gendata-jdk.jdeps.gmk +++ b/jdk/make/ModuleTools.gmk @@ -1,5 +1,5 @@ # -# Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # # This code is free software; you can redistribute it and/or modify it @@ -23,20 +23,25 @@ # questions. # -include GendataCommon.gmk +include $(SPEC) +include MakeBase.gmk +include JavaCompilation.gmk +include SetupJavaCompilers.gmk -$(eval $(call IncludeCustomExtension, jdk, gendata/Gendata-jdk.jdeps.gmk)) +TOOLS_CLASSES_DIR := $(BUILDTOOLS_OUTPUTDIR)/tools_jigsaw_classes -JDEPS_MODULES_XML := $(JDK_OUTPUTDIR)/modules/jdk.jdeps/com/sun/tools/jdeps/resources/jdeps-modules.xml -MODULES_XML += $(TOPDIR)/modules.xml +$(eval $(call SetupJavaCompilation,BUILD_JIGSAW_TOOLS, \ + SETUP := GENERATE_USINGJDKBYTECODE, \ + SRC := $(JDK_TOPDIR)/make/src/classes, \ + INCLUDES := build/tools/deps \ + build/tools/jigsaw, \ + BIN := $(TOOLS_CLASSES_DIR), \ + ADD_JAVAC_FLAGS := -XaddExports:jdk.jdeps/com.sun.tools.classfile=ALL-UNNAMED )) -# -# Generate modules.xml for jdeps to use -# It augments $(TOPDIR)/modules.xml to include module membership -# -$(JDEPS_MODULES_XML): $(BUILD_TOOLS_JDK) $(MODULES_XML) - $(MKDIR) -p $(@D) - $(RM) $@ - $(TOOL_GENMODULESXML) -o $@ -mp $(JDK_OUTPUTDIR)/modules $(MODULES_XML) -TARGETS += $(JDEPS_MODULES_XML) +TOOL_GENGRAPHS := $(BUILD_JAVA) -esa -ea -cp $(TOOLS_CLASSES_DIR) \ + build.tools.jigsaw.GenGraphs + +TOOL_MODULESUMMARY := $(BUILD_JAVA) -esa -ea -cp $(TOOLS_CLASSES_DIR) \ + -XaddExports:jdk.jdeps/com.sun.tools.classfile=ALL-UNNAMED \ + build.tools.jigsaw.ModuleSummary diff --git a/jdk/make/Tools.gmk b/jdk/make/Tools.gmk index a087e3c83ae..1ae471d72b3 100644 --- a/jdk/make/Tools.gmk +++ b/jdk/make/Tools.gmk @@ -41,7 +41,12 @@ BUILD_TOOLS_JDK := $(call SetupJavaCompilationCompileTarget, \ TOOL_ADDJSUM = $(JAVA_SMALL) -cp $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes \ build.tools.addjsum.AddJsum +ifeq ($(BOOT_JDK_MODULAR), true) + COMPILEFONTCONFIG_ADD_EXPORTS := -XaddExports:java.desktop/sun.awt=ALL-UNNAMED +endif + TOOL_COMPILEFONTCONFIG = $(JAVA_SMALL) -cp $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes \ + $(COMPILEFONTCONFIG_ADD_EXPORTS) \ build.tools.compilefontconfig.CompileFontConfig TOOL_COMPILEPROPERTIES = $(JAVA_SMALL) -cp $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes \ @@ -106,12 +111,18 @@ TOOL_OSX_TOBIN = $(JAVA_SMALL) -Djava.awt.headless=true -cp $(BUILDTOOLS_OUTPUTD TOOL_CLDRCONVERTER = $(JAVA_SMALL) -cp $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes \ build.tools.cldrconverter.CLDRConverter -TOOL_GENMODULESXML = $(JAVA_SMALL) -Xbootclasspath/p:$(INTERIM_LANGTOOLS_JAR) \ - -cp $(call PathList, $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes $(JDK_OUTPUTDIR)) \ +TOOL_GENMODULESXML = $(JAVA_SMALL) $(INTERIM_LANGTOOLS_BOOTCLASSPATH) \ + -cp $(call PathList, $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes) \ build.tools.module.GenJdepsModulesXml -TOOL_IMAGEBUILDER = $(JAVA_SMALL) -Xbootclasspath/p:$(BUILDTOOLS_OUTPUTDIR)/interim_jimage_classes \ - -cp $(call PathList, $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes $(JDK_OUTPUTDIR)) \ - build.tools.module.ImageBuilder +TOOL_GENMODULEINFOSOURCE = $(JAVA_SMALL) $(INTERIM_LANGTOOLS_BOOTCLASSPATH) \ + -cp $(call PathList, $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes) \ + build.tools.module.GenModuleInfoSource + +TOOL_GENCLASSLOADERMAP = $(JAVA_SMALL) $(INTERIM_LANGTOOLS_BOOTCLASSPATH) \ + -cp $(call PathList, $(BUILDTOOLS_OUTPUTDIR)/jdk_tools_classes) \ + build.tools.module.GenModuleLoaderMap + +########################################################################################## endif # _TOOLS_GMK diff --git a/jdk/make/copy/Copy-java.base.gmk b/jdk/make/copy/Copy-java.base.gmk index ed8e42ae272..04988de5f85 100644 --- a/jdk/make/copy/Copy-java.base.gmk +++ b/jdk/make/copy/Copy-java.base.gmk @@ -172,6 +172,11 @@ ifeq ($(OPENJDK_TARGET_OS), windows) POLICY_SRC_LIST += $(JDK_TOPDIR)/src/java.base/$(OPENJDK_TARGET_OS)/conf/security/java.policy endif +# Allow imported modules to modify the java.policy +ifneq ($(IMPORT_MODULES_CONF), ) + POLICY_SRC_LIST += $(wildcard $(IMPORT_MODULES_CONF)/java.base/security/java.policy.extra) +endif + POLICY_SRC_LIST += $(POLICY_SRC) $(POLICY_DST): $(POLICY_SRC_LIST) diff --git a/jdk/make/data/jdwp/jdwp.spec b/jdk/make/data/jdwp/jdwp.spec index 8d05bebcc02..a144c896f92 100644 --- a/jdk/make/data/jdwp/jdwp.spec +++ b/jdk/make/data/jdwp/jdwp.spec @@ -1,5 +1,5 @@ /* - * Copyright (c) 1998, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1998, 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 @@ -569,6 +569,21 @@ JDWP "Java(tm) Debug Wire Protocol" (Error VM_DEAD) ) ) + (Command AllModules=22 + "Returns all modules in the target VM." + "

Since JDWP version 9." + (Out + ) + (Reply + (Repeat modules "The number of the modules that follow." + (moduleID module "One of the modules.") + ) + ) + (ErrorSet + (Error NOT_IMPLEMENTED) + (Error VM_DEAD) + ) + ) ) (CommandSet ReferenceType=2 @@ -1029,6 +1044,22 @@ JDWP "Java(tm) Debug Wire Protocol" (Error VM_DEAD) ) ) + (Command Module=19 + "Returns the module that this reference type belongs to." + "

Since JDWP version 9." + (Out + (referenceType refType "The reference type.") + ) + (Reply + (moduleID module "The module this reference type belongs to.") + ) + (ErrorSet + (Error INVALID_CLASS "refType is not the ID of a reference type.") + (Error INVALID_OBJECT "refType is not a known ID.") + (Error NOT_IMPLEMENTED) + (Error VM_DEAD) + ) + ) ) (CommandSet ClassType=3 (Command Superclass=1 @@ -2647,6 +2678,54 @@ JDWP "Java(tm) Debug Wire Protocol" ) ) ) +(CommandSet ModuleReference=18 + (Command Name=1 + "Returns the name of this module." + "

Since JDWP version 9." + (Out + (moduleID module "This module.") + ) + (Reply + (string name "The module's name.") + ) + (ErrorSet + (Error INVALID_MODULE) + (Error NOT_IMPLEMENTED) + (Error VM_DEAD) + ) + ) + (Command ClassLoader=2 + "Returns the class loader of this module." + "

Since JDWP version 9." + (Out + (moduleID module "This module.") + ) + (Reply + (classLoaderObject classLoader "The module's class loader.") + ) + (ErrorSet + (Error INVALID_MODULE) + (Error NOT_IMPLEMENTED) + (Error VM_DEAD) + ) + ) + (Command CanRead=3 + "Returns true if this module can read the source module; false otherwise." + "

Since JDWP version 9." + (Out + (moduleID module "This module.") + (moduleID sourceModule "The source module.") + ) + (Reply + (boolean canRead "true if this module can read the source module; false otherwise.") + ) + (ErrorSet + (Error INVALID_MODULE "This module or sourceModule is not the ID of a module.") + (Error NOT_IMPLEMENTED) + (Error VM_DEAD) + ) + ) +) (CommandSet Event=64 (Command Composite=100 "Several events may occur at a given time in the target VM. " @@ -3054,6 +3133,7 @@ JDWP "Java(tm) Debug Wire Protocol" (Constant INVALID_SLOT =35 "Invalid slot.") (Constant DUPLICATE =40 "Item already set.") (Constant NOT_FOUND =41 "Desired element not found.") + (Constant INVALID_MODULE =42 "Invalid module.") (Constant INVALID_MONITOR =50 "Invalid monitor.") (Constant NOT_MONITOR_OWNER =51 "This thread doesn't own the monitor.") (Constant INTERRUPT =52 "The call has been interrupted before completion.") diff --git a/jdk/make/gendata/GendataBreakIterator.gmk b/jdk/make/gendata/GendataBreakIterator.gmk index 2ca835c23d3..1aa4becefac 100644 --- a/jdk/make/gendata/GendataBreakIterator.gmk +++ b/jdk/make/gendata/GendataBreakIterator.gmk @@ -33,8 +33,6 @@ # They are used at JDK build phase in order to create $(BIFILES) which # are used on runtime instead. # -TEXT_SRCDIR := $(JDK_TOPDIR)/src/java.base/share/classes \ - $(JDK_TOPDIR)/src/jdk.localedata/share/classes TEXT_PKG := sun/text/resources TEXT_PKG_LD := $(TEXT_PKG)/ext TEXT_SOURCES := $(TEXT_PKG)/BreakIteratorRules.java \ @@ -46,11 +44,35 @@ TEXT_SOURCES := $(TEXT_PKG)/BreakIteratorRules.java \ BREAK_ITERATOR_CLASSES := $(BUILDTOOLS_OUTPUTDIR)/break_iterator_classes # These two files should be moved out to a build tool! -$(eval $(call SetupJavaCompilation,BUILD_BREAKITERATOR, \ +$(eval $(call SetupJavaCompilation,BUILD_BREAKITERATOR_BASE, \ SETUP := GENERATE_OLDBYTECODE, \ - SRC := $(TEXT_SRCDIR), \ - INCLUDE_FILES := $(TEXT_SOURCES), \ - BIN := $(BREAK_ITERATOR_CLASSES))) + SRC := $(JDK_TOPDIR)/src/java.base/share/classes, \ + INCLUDE_FILES := \ + $(TEXT_PKG)/BreakIteratorRules.java \ + $(TEXT_PKG)/BreakIteratorInfo.java, \ + BIN := $(BREAK_ITERATOR_CLASSES)/java.base)) + +$(eval $(call SetupJavaCompilation,BUILD_BREAKITERATOR_LD, \ + SETUP := GENERATE_OLDBYTECODE, \ + SRC := $(JDK_TOPDIR)/src/jdk.localedata/share/classes, \ + INCLUDES := $(TEXT_PKG_LD), \ + INCLUDE_FILES := \ + $(TEXT_PKG_LD)/BreakIteratorRules_th.java \ + $(TEXT_PKG_LD)/BreakIteratorInfo_th.java, \ + BIN := $(BREAK_ITERATOR_CLASSES)/jdk.localedata)) + +ifeq ($(BOOT_JDK_MODULAR), true) + BREAK_ITERATOR_BOOTCLASSPATH := -Xpatch:$(BREAK_ITERATOR_CLASSES) \ + -XaddExports:$(subst $(SPACE),$(COMMA),$(strip \ + java.base/sun.text=ALL-UNNAMED \ + java.base/sun.text.resources=ALL-UNNAMED \ + jdk.localedata/sun.text.resources.ext=ALL-UNNAMED \ + )) +else + BREAK_ITERATOR_BOOTCLASSPATH := -Xbootclasspath/p:$(call PathList, \ + $(BREAK_ITERATOR_CLASSES)/java.base \ + $(BREAK_ITERATOR_CLASSES)/jdk.localedata) +endif # Generate data resource files. # input @@ -67,8 +89,9 @@ BIFILES_TH := $(LD_DATA_PKG_DIR)/WordBreakIteratorData_th \ $(LD_DATA_PKG_DIR)/LineBreakIteratorData_th $(BIFILES): $(BASE_DATA_PKG_DIR)/_the.bifiles -$(BASE_DATA_PKG_DIR)/_the.bifiles: JAVA_FLAGS += -Xbootclasspath/p:$(BREAK_ITERATOR_CLASSES) -$(BASE_DATA_PKG_DIR)/_the.bifiles: $(BUILD_TOOLS) $(UNICODEDATA) $(BUILD_BREAKITERATOR) +$(BASE_DATA_PKG_DIR)/_the.bifiles: JAVA_FLAGS += $(BREAK_ITERATOR_BOOTCLASSPATH) +$(BASE_DATA_PKG_DIR)/_the.bifiles: $(BUILD_TOOLS) $(UNICODEDATA) \ + $(BUILD_BREAKITERATOR_BASE) $(BUILD_BREAKITERATOR_LD) $(call LogInfo, Generating BreakIteratorData) $(call MakeDir, $(@D)) $(RM) $(BIFILES) @@ -78,8 +101,9 @@ $(BASE_DATA_PKG_DIR)/_the.bifiles: $(BUILD_TOOLS) $(UNICODEDATA) $(BUILD_BREAKIT $(TOUCH) $@ $(BIFILES_TH): $(LD_DATA_PKG_DIR)/_the.bifiles_th -$(LD_DATA_PKG_DIR)/_the.bifiles_th: JAVA_FLAGS += -Xbootclasspath/p:$(BREAK_ITERATOR_CLASSES) -$(LD_DATA_PKG_DIR)/_the.bifiles_th: $(BUILD_TOOLS) $(UNICODEDATA) $(BUILD_BREAKITERATOR) +$(LD_DATA_PKG_DIR)/_the.bifiles_th: JAVA_FLAGS += $(BREAK_ITERATOR_BOOTCLASSPATH) +$(LD_DATA_PKG_DIR)/_the.bifiles_th: $(BUILD_TOOLS) $(UNICODEDATA) \ + $(BUILD_BREAKITERATOR_BASE) $(BUILD_BREAKITERATOR_LD) $(call LogInfo, Generating BreakIteratorData_th) $(RM) $(BIFILES_TH) $(TOOL_GENERATEBREAKITERATORDATA) \ diff --git a/jdk/make/gensrc/Gensrc-java.base.gmk b/jdk/make/gensrc/Gensrc-java.base.gmk index e176d88ea7e..1a188a9a786 100644 --- a/jdk/make/gensrc/Gensrc-java.base.gmk +++ b/jdk/make/gensrc/Gensrc-java.base.gmk @@ -33,6 +33,7 @@ include GensrcCharsetMapping.gmk include GensrcCharsetCoder.gmk include GensrcBuffer.gmk include GensrcExceptions.gmk +include GensrcModuleLoaderMap.gmk ################################################################################ diff --git a/jdk/make/gensrc/Gensrc-jdk.jdi.gmk b/jdk/make/gensrc/Gensrc-jdk.jdi.gmk index e634c12b6bb..720b685ef4b 100644 --- a/jdk/make/gensrc/Gensrc-jdk.jdi.gmk +++ b/jdk/make/gensrc/Gensrc-jdk.jdi.gmk @@ -61,30 +61,6 @@ GENSRC_JDK_JDI += $(GENSRC_JDWP) ################################################################################ -define process-provider - $(call MakeDir, $(@D)) - $(CAT) $^ | $(SED) -e "s/^#\[$(OPENJDK_TARGET_OS)\]//" > $@ -endef - -# Filter com.sun.jdi.connect.Connector -$(SUPPORT_OUTPUTDIR)/gensrc/jdk.jdi/META-INF/services/com.sun.jdi.connect.Connector: \ - $(JDK_TOPDIR)/src/jdk.jdi/share/classes/META-INF/services/com.sun.jdi.connect.Connector \ - $(HOTSPOT_TOPDIR)/src/jdk.hotspot.agent/share/classes/META-INF/services/com.sun.jdi.connect.Connector - $(process-provider) - -# Copy the same service file into jdk.hotspot.agent so that they are kept the same. -$(SUPPORT_OUTPUTDIR)/gensrc/jdk.hotspot.agent/META-INF/services/com.sun.jdi.connect.Connector: \ - $(SUPPORT_OUTPUTDIR)/gensrc/jdk.jdi/META-INF/services/com.sun.jdi.connect.Connector - $(install-file) - -# Some platforms don't have the serviceability agent -ifeq ($(INCLUDE_SA), true) - GENSRC_JDK_JDI += $(SUPPORT_OUTPUTDIR)/gensrc/jdk.jdi/META-INF/services/com.sun.jdi.connect.Connector \ - $(SUPPORT_OUTPUTDIR)/gensrc/jdk.hotspot.agent/META-INF/services/com.sun.jdi.connect.Connector -endif - -################################################################################ - include GensrcProperties.gmk $(eval $(call SetupCompileProperties, COMPILE_PROPERTIES, \ diff --git a/jdk/make/gensrc/Gensrc-jdk.dev.gmk b/jdk/make/gensrc/Gensrc-jdk.jlink.gmk similarity index 71% rename from jdk/make/gensrc/Gensrc-jdk.dev.gmk rename to jdk/make/gensrc/Gensrc-jdk.jlink.gmk index 4316c485883..c841345c737 100644 --- a/jdk/make/gensrc/Gensrc-jdk.dev.gmk +++ b/jdk/make/gensrc/Gensrc-jdk.jlink.gmk @@ -29,12 +29,23 @@ include GensrcCommon.gmk include GensrcProperties.gmk -$(eval $(call SetupCompileProperties, COMPILE_PROPERTIES, \ - SRC_DIRS := $(JDK_TOPDIR)/src/jdk.dev/share/classes/jdk/tools/jimage/resources, \ +$(eval $(call SetupCompileProperties, JLINK_PROPERTIES, \ + SRC_DIRS := $(JDK_TOPDIR)/src/jdk.jlink/share/classes/jdk/tools/jlink/resources, \ CLASS := ListResourceBundle, \ )) -TARGETS += $(COMPILE_PROPERTIES) +$(eval $(call SetupCompileProperties, JMOD_PROPERTIES, \ + SRC_DIRS := $(JDK_TOPDIR)/src/jdk.jlink/share/classes/jdk/tools/jmod/resources, \ + CLASS := ListResourceBundle, \ +)) + +$(eval $(call SetupCompileProperties, JIMAGE_PROPERTIES, \ + SRC_DIRS := $(JDK_TOPDIR)/src/jdk.jlink/share/classes/jdk/tools/jimage/resources, \ + CLASS := ListResourceBundle, \ +)) + + +TARGETS += $(JLINK_PROPERTIES) $(JMOD_PROPERTIES) $(JIMAGE_PROPERTIES) ################################################################################ diff --git a/jdk/make/gensrc/Gensrc-jdk.jvmstat.gmk b/jdk/make/gensrc/Gensrc-jdk.jvmstat.gmk deleted file mode 100644 index 11e54a6f635..00000000000 --- a/jdk/make/gensrc/Gensrc-jdk.jvmstat.gmk +++ /dev/null @@ -1,55 +0,0 @@ -# -# Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. -# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. -# -# This code is free software; you can redistribute it and/or modify it -# under the terms of the GNU General Public License version 2 only, as -# published by the Free Software Foundation. Oracle designates this -# particular file as subject to the "Classpath" exception as provided -# by Oracle in the LICENSE file that accompanied this code. -# -# This code is distributed in the hope that it will be useful, but WITHOUT -# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or -# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License -# version 2 for more details (a copy is included in the LICENSE file that -# accompanied this code). -# -# You should have received a copy of the GNU General Public License version -# 2 along with this work; if not, write to the Free Software Foundation, -# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. -# -# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA -# or visit www.oracle.com if you need additional information or have any -# questions. -# - -include GensrcCommon.gmk - -################################################################################ - -define merge-providers - $(MKDIR) -p $(@D) - $(CAT) $^ > $@ -endef - -PROVIDER_FILE := META-INF/services/sun.jvmstat.monitor.MonitoredHostService - -# Merge the local and remote sevice providers into jdk.jvmstat/META-INF/services -$(SUPPORT_OUTPUTDIR)/gensrc/jdk.jvmstat/$(PROVIDER_FILE): \ - $(JDK_TOPDIR)/src/jdk.jvmstat/share/classes/$(PROVIDER_FILE) \ - $(JDK_TOPDIR)/src/jdk.jvmstat.rmi/share/classes/$(PROVIDER_FILE) - $(merge-providers) - -# Copy the same service file into jdk.jvmstat.rmi so that they are kept the same. -$(SUPPORT_OUTPUTDIR)/gensrc/jdk.jvmstat.rmi/$(PROVIDER_FILE): \ - $(SUPPORT_OUTPUTDIR)/gensrc/jdk.jvmstat/$(PROVIDER_FILE) - $(install-file) - -################################################################################ - -jdk.jvmstat: $(SUPPORT_OUTPUTDIR)/gensrc/jdk.jvmstat/$(PROVIDER_FILE) \ - $(SUPPORT_OUTPUTDIR)/gensrc/jdk.jvmstat.rmi/$(PROVIDER_FILE) - -all: jdk.jvmstat - -.PHONY: all \ No newline at end of file diff --git a/jdk/make/gensrc/GensrcLocaleData.gmk b/jdk/make/gensrc/GensrcLocaleData.gmk index 602a4f4752f..0acc8f9591f 100644 --- a/jdk/make/gensrc/GensrcLocaleData.gmk +++ b/jdk/make/gensrc/GensrcLocaleData.gmk @@ -143,19 +143,6 @@ $(SUPPORT_OUTPUTDIR)/gensrc/jdk.localedata/sun/util/resources/provider/NonBaseLo GENSRC_BASELOCALEDATA := $(SUPPORT_OUTPUTDIR)/gensrc/java.base/sun/util/locale/provider/BaseLocaleDataMetaInfo.java GENSRC_LOCALEDATA := $(SUPPORT_OUTPUTDIR)/gensrc/jdk.localedata/sun/util/resources/provider/NonBaseLocaleDataMetaInfo.java -################################################################################ - -GENSRC_CRBC_DST := $(SUPPORT_OUTPUTDIR)/gensrc/java.base/sun/util/CoreResourceBundleControl.java -GENSRC_CRBC_CMD := $(JDK_TOPDIR)/make/scripts/localelist.sh - -JRE_NONEXIST_LOCALES := en en_US de_DE es_ES fr_FR it_IT ja_JP ko_KR sv_SE zh - -$(GENSRC_CRBC_DST): $(JDK_TOPDIR)/src/java.base/share/classes/sun/util/CoreResourceBundleControl-XLocales.java.template \ - $(GENSRC_CRBC_CMD) - $(MKDIR) -p $(@D) - NAWK="$(NAWK)" SED="$(SED)" $(SH) $(GENSRC_CRBC_CMD) "$(JRE_NONEXIST_LOCALES)" $< $@ - -GENSRC_BASELOCALEDATA += $(GENSRC_CRBC_DST) GENSRC_JAVA_BASE += $(GENSRC_BASELOCALEDATA) GENSRC_JDK_LOCALEDATA += $(GENSRC_LOCALEDATA) diff --git a/jdk/make/gensrc/GensrcModuleLoaderMap.gmk b/jdk/make/gensrc/GensrcModuleLoaderMap.gmk new file mode 100644 index 00000000000..7e8fb339f5d --- /dev/null +++ b/jdk/make/gensrc/GensrcModuleLoaderMap.gmk @@ -0,0 +1,164 @@ +# +# 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. +# + +include Modules.gmk + +BOOT_MODULES := +UPGRADEABLE_MDOULES := +AGGREGATOR_MDOULES := +OTHER_PLATFORM_MODULES := + +# Hook to include the corresponding custom file, if present. +$(eval $(call IncludeCustomExtension, jdk, gensrc/GensrcModuleLoaderMap.gmk)) + +BOOT_MODULES += \ + java.base \ + java.datatransfer \ + java.desktop \ + java.httpclient \ + java.instrument \ + java.logging \ + java.management \ + java.naming \ + java.prefs \ + java.rmi \ + java.security.jgss \ + java.security.sasl \ + java.sql \ + java.xml \ + java.xml.crypto \ + jdk.httpserver \ + jdk.management \ + jdk.sctp \ + jdk.security.auth \ + jdk.security.jgss \ + # + +# to be deprivileged +BOOT_MODULES += \ + java.compiler \ + java.scripting \ + java.sql.rowset \ + java.smartcardio \ + jdk.charsets \ + jdk.naming.rmi \ + # + +UPGRADEABLE_MODULES += \ + java.activation \ + java.annotations.common \ + java.corba \ + java.transaction \ + java.xml.bind \ + java.xml.ws \ + # + +AGGREGATOR_MODULES += \ + java.compact1 \ + java.compact2 \ + java.compact3 \ + java.se \ + java.se.ee \ + # + +OTHER_PLATFORM_MODULES += \ + jdk.accessibility \ + jdk.crypto.ec \ + jdk.crypto.pkcs11 \ + jdk.dynalink \ + jdk.jsobject \ + jdk.xml.dom \ + jdk.localedata \ + jdk.naming.dns \ + jdk.scripting.nashorn \ + jdk.zipfs \ + # + +ifeq ($(OPENJDK_TARGET_OS), macsox) + BOOT_MODULES += jdk.deploy.osx +endif +ifeq ($(OPENJDK_TARGET_OS), windows) + OTHER_PLATFORM_MODULES += jdk.crypto.mscapi +endif +ifeq ($(OPENJDK_TARGET_OS), solaris) + OTHER_PLATFORM_MODULES += jdk.crypto.ucrypto +endif + +# Param 1 - Name of module +define ReadImportMetaData + ifneq ($$(wildcard $(IMPORT_MODULES_MAKE)/$$(strip $1)/build.properties), ) + classloader := + include $(IMPORT_MODULES_MAKE)/$$(strip $1)/build.properties + ifeq ($$(classloader), boot) + BOOT_MODULES += $1 + else ifeq ($$(classloader), ext) + OTHER_PLATFORM_MODULES += $1 + endif + endif +endef + +IMPORTED_MODULES := $(call FindImportedModules) +$(foreach m, $(IMPORTED_MODULES), $(eval $(call ReadImportMetaData, $m))) + + +# 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 +$(strip \ + $(subst $(COMMA)$(COMMA),$(COMMA),$(subst $(SPACE),$(COMMA),$(strip $1))) \ +) +endef +BOOT_MODULES_LIST := $(call SubstComma, $(BOOT_MODULES)) +PLATFORM_MODULES_LIST := $(call SubstComma, $(UPGRADEABLE_MODULES) $(AGGREGATOR_MODULES) $(OTHER_PLATFORM_MODULES)) + +VARDEPS_VALUE := $(BOOT_MODULES_LIST) $(PLATFORM_MODULES_LIST) +VARDEPS_FILE := $(call DependOnVariable, VARDEPS_VALUE) + +############################################################################ + +$(SUPPORT_OUTPUTDIR)/gensrc/java.base/jdk/internal/module/ModuleLoaderMap.java: \ + $(JDK_TOPDIR)/src/java.base/share/classes/jdk/internal/module/ModuleLoaderMap.java \ + $(VARDEPS_FILE) $(BUILD_TOOLS_JDK) + $(MKDIR) -p $(@D) + $(RM) $@ $@.tmp + $(TOOL_GENCLASSLOADERMAP) -boot $(BOOT_MODULES_LIST) \ + -platform $(PLATFORM_MODULES_LIST) -o $@.tmp $< + $(MV) $@.tmp $@ + +GENSRC_JAVA_BASE += $(SUPPORT_OUTPUTDIR)/gensrc/java.base/jdk/internal/module/ModuleLoaderMap.java + +$(SUPPORT_OUTPUTDIR)/gensrc/java.base/jdk/internal/vm/cds/resources/ModuleLoaderMap.dat: \ + $(JDK_TOPDIR)/src/java.base/share/classes/jdk/internal/vm/cds/resources/ModuleLoaderMap.dat \ + $(VARDEPS_FILE) $(BUILD_TOOLS_JDK) + $(MKDIR) -p $(@D) + $(RM) $@ $@.tmp + $(TOOL_GENCLASSLOADERMAP) -boot $(BOOT_MODULES_LIST) \ + -platform $(PLATFORM_MODULES_LIST) -o $@.tmp $< + $(MV) $@.tmp $@ + +GENSRC_JAVA_BASE += $(SUPPORT_OUTPUTDIR)/gensrc/java.base/jdk/internal/vm/cds/resources/ModuleLoaderMap.dat + +################################################################################ diff --git a/jdk/make/launcher/Launcher-java.desktop.gmk b/jdk/make/launcher/Launcher-java.desktop.gmk index 2bb3c4923f2..c9f730f4d71 100644 --- a/jdk/make/launcher/Launcher-java.desktop.gmk +++ b/jdk/make/launcher/Launcher-java.desktop.gmk @@ -31,6 +31,7 @@ $(eval $(call IncludeCustomExtension, jdk, launcher/Launcher-java.desktop.gmk)) ifndef BUILD_HEADLESS_ONLY $(eval $(call SetupBuildLauncher, appletviewer, \ MAIN_CLASS := sun.applet.Main, \ + JAVA_ARGS := -addmods ALL-SYSTEM, \ LIBS_unix := $(X_LIBS), \ )) endif diff --git a/jdk/make/launcher/Launcher-jdk.dev.gmk b/jdk/make/launcher/Launcher-jdk.jlink.gmk similarity index 69% rename from jdk/make/launcher/Launcher-jdk.dev.gmk rename to jdk/make/launcher/Launcher-jdk.jlink.gmk index d874ae9372c..acf8228b6c4 100644 --- a/jdk/make/launcher/Launcher-jdk.dev.gmk +++ b/jdk/make/launcher/Launcher-jdk.jlink.gmk @@ -1,5 +1,5 @@ # -# Copyright (c) 2011, 2015, Oracle and/or its affiliates. All rights reserved. +# 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 @@ -27,4 +27,19 @@ include LauncherCommon.gmk $(eval $(call SetupBuildLauncher, jimage,\ MAIN_CLASS := jdk.tools.jimage.Main, \ + CFLAGS := -DENABLE_ARG_FILES, \ +)) + +$(eval $(call SetupBuildLauncher, jlink,\ + MAIN_CLASS := jdk.tools.jlink.internal.Main, \ + CFLAGS := -DENABLE_ARG_FILES \ + -DEXPAND_CLASSPATH_WILDCARDS \ + -DNEVER_ACT_AS_SERVER_CLASS_MACHINE, \ +)) + +$(eval $(call SetupBuildLauncher, jmod,\ + MAIN_CLASS := jdk.tools.jmod.Main, \ + CFLAGS := -DENABLE_ARG_FILES \ + -DEXPAND_CLASSPATH_WILDCARDS \ + -DNEVER_ACT_AS_SERVER_CLASS_MACHINE, \ )) diff --git a/jdk/make/launcher/Launcher-jdk.pack200.gmk b/jdk/make/launcher/Launcher-jdk.pack200.gmk index 818828a2929..af050554a79 100644 --- a/jdk/make/launcher/Launcher-jdk.pack200.gmk +++ b/jdk/make/launcher/Launcher-jdk.pack200.gmk @@ -26,6 +26,7 @@ include LauncherCommon.gmk $(eval $(call SetupBuildLauncher, pack200, \ + MAIN_MODULE := java.base, \ MAIN_CLASS := com.sun.java.util.jar.pack.Driver, \ )) diff --git a/jdk/make/launcher/Launcher-jdk.rmic.gmk b/jdk/make/launcher/Launcher-jdk.rmic.gmk index d60c3d9b60b..b0a8b6dc81a 100644 --- a/jdk/make/launcher/Launcher-jdk.rmic.gmk +++ b/jdk/make/launcher/Launcher-jdk.rmic.gmk @@ -26,6 +26,6 @@ include LauncherCommon.gmk $(eval $(call SetupBuildLauncher, rmic, \ - MAIN_CLASS := sun.rmi.rmic.Main, \ + MAIN_CLASS := jdk.rmi.rmic.Main, \ CFLAGS := -DEXPAND_CLASSPATH_WILDCARDS, \ )) diff --git a/jdk/make/launcher/LauncherCommon.gmk b/jdk/make/launcher/LauncherCommon.gmk index d35b8b71348..1bef88f8a42 100644 --- a/jdk/make/launcher/LauncherCommon.gmk +++ b/jdk/make/launcher/LauncherCommon.gmk @@ -63,6 +63,8 @@ JAVA_MANIFEST := $(JDK_TOPDIR)/src/java.base/windows/native/launcher/java.manife # used as the name of the executable. # # Remaining parameters are named arguments. These include: +# MAIN_MODULE The module of the main class to launch if different from the +# current module # MAIN_CLASS The Java main class to launch # JAVA_ARGS Processed into a -DJAVA_ARGS C flag # CFLAGS Additional CFLAGS @@ -97,9 +99,13 @@ define SetupBuildLauncherBody $1_JAVA_ARGS += -ms8m endif + ifeq ($$($1_MAIN_MODULE), ) + $1_MAIN_MODULE := $(MODULE) + endif + ifneq ($$($1_JAVA_ARGS), ) $1_JAVA_ARGS_STR := '{ $$(strip $$(foreach a, \ - $$(addprefix -J, $$($1_JAVA_ARGS)) $$($1_MAIN_CLASS), "$$a"$(COMMA) )) }' + $$(addprefix -J, $$($1_JAVA_ARGS)) -m $$($1_MAIN_MODULE)/$$($1_MAIN_CLASS), "$$a"$(COMMA) )) }' $1_CFLAGS += -DJAVA_ARGS=$$($1_JAVA_ARGS_STR) endif diff --git a/jdk/make/lib/CoreLibraries.gmk b/jdk/make/lib/CoreLibraries.gmk index 9dcf06be11a..f114c9b7d33 100644 --- a/jdk/make/lib/CoreLibraries.gmk +++ b/jdk/make/lib/CoreLibraries.gmk @@ -146,6 +146,7 @@ $(eval $(call SetupNativeCompilation,BUILD_LIBJAVA, \ $(LIBJAVA_CFLAGS), \ System.c_CFLAGS := $(VERSION_CFLAGS), \ jdk_util.c_CFLAGS := $(VERSION_CFLAGS), \ + DISABLED_WARNINGS_gcc := unused-result, \ DISABLED_WARNINGS_solstudio := E_STATEMENT_NOT_REACHED, \ MAPFILE := $(LIBJAVA_MAPFILE), \ LDFLAGS := $(LDFLAGS_JDKLIB) \ diff --git a/jdk/make/lib/Lib-java.instrument.gmk b/jdk/make/lib/Lib-java.instrument.gmk index 44ba88dad40..64f25d0eb82 100644 --- a/jdk/make/lib/Lib-java.instrument.gmk +++ b/jdk/make/lib/Lib-java.instrument.gmk @@ -68,14 +68,14 @@ $(eval $(call SetupNativeCompilation,BUILD_LIBINSTRUMENT, \ LDFLAGS_macosx := -Wl$(COMMA)-all_load, \ LDFLAGS_aix := -L$(SUPPORT_OUTPUTDIR)/native/java.base, \ LIBS := $(JDKLIB_LIBS), \ - LIBS_unix := -ljava $(LIBZ), \ + LIBS_unix := -ljava -ljvm $(LIBZ), \ LIBS_linux := -ljli $(LIBDL), \ LIBS_solaris := -ljli $(LIBDL), \ LIBS_aix := -liconv -ljli_static $(LIBDL), \ LIBS_macosx := -liconv -framework Cocoa -framework Security \ -framework ApplicationServices \ $(SUPPORT_OUTPUTDIR)/native/java.base/libjli_static.a, \ - LIBS_windows := $(WIN_JAVA_LIB) advapi32.lib \ + LIBS_windows := jvm.lib $(WIN_JAVA_LIB) advapi32.lib \ $(SUPPORT_OUTPUTDIR)/native/java.base/jli_static.lib, \ VERSIONINFO_RESOURCE := $(GLOBAL_VERSION_INFO_RESOURCE), \ RC_FLAGS := $(RC_FLAGS) \ diff --git a/jdk/make/mapfiles/libjava/mapfile-vers b/jdk/make/mapfiles/libjava/mapfile-vers index ac21dd24890..82fd583e48e 100644 --- a/jdk/make/mapfiles/libjava/mapfile-vers +++ b/jdk/make/mapfiles/libjava/mapfile-vers @@ -137,7 +137,6 @@ SUNWprivate_1.1 { Java_java_lang_ClassLoader_registerNatives; Java_java_lang_Double_longBitsToDouble; Java_java_lang_Double_doubleToRawLongBits; - Java_java_lang_reflect_Proxy_defineClass0; Java_java_lang_Float_intBitsToFloat; Java_java_lang_Float_floatToRawIntBits; Java_java_lang_StackFrameInfo_fillInStackFrames; @@ -165,8 +164,6 @@ SUNWprivate_1.1 { Java_java_lang_StrictMath_expm1; Java_java_lang_Object_getClass; Java_java_lang_Object_registerNatives; - Java_java_lang_Package_getSystemPackage0; - Java_java_lang_Package_getSystemPackages0; Java_java_lang_ProcessEnvironment_environ; Java_java_lang_ProcessHandleImpl_destroy0; Java_java_lang_ProcessHandleImpl_getCurrentPid0; @@ -204,11 +201,11 @@ SUNWprivate_1.1 { Java_java_lang_reflect_Executable_getTypeAnnotationBytes0; Java_java_lang_reflect_Field_getTypeAnnotationBytes0; Java_java_lang_Runtime_freeMemory; - Java_java_lang_Runtime_maxMemory; + Java_java_lang_Runtime_maxMemory; Java_java_lang_Runtime_gc; Java_java_lang_Runtime_runFinalization0; Java_java_lang_Runtime_totalMemory; - Java_java_lang_Runtime_availableProcessors; + Java_java_lang_Runtime_availableProcessors; Java_java_lang_SecurityManager_classDepth; Java_java_lang_SecurityManager_classLoaderDepth0; Java_java_lang_SecurityManager_currentClassLoader0; @@ -274,6 +271,18 @@ SUNWprivate_1.1 { Java_jdk_internal_misc_VM_getgid; Java_jdk_internal_misc_VM_getegid; Java_jdk_internal_misc_VM_initialize; + + Java_java_lang_reflect_Module_defineModule0; + Java_java_lang_reflect_Module_addReads0; + Java_java_lang_reflect_Module_addExports0; + Java_java_lang_reflect_Module_addExportsToAll0; + Java_java_lang_reflect_Module_addExportsToAllUnnamed0; + Java_java_lang_reflect_Module_addPackage0; + + Java_jdk_internal_loader_BootLoader_getSystemPackageLocation; + Java_jdk_internal_loader_BootLoader_getSystemPackageNames; + Java_jdk_internal_loader_BootLoader_setBootLoaderUnnamedModule0; + Java_sun_misc_VMSupport_initAgentProperties; Java_sun_misc_VMSupport_getVMTemporaryDirectory; diff --git a/jdk/make/mapfiles/libjimage/mapfile-vers b/jdk/make/mapfiles/libjimage/mapfile-vers index 68e01c657d8..48eac21b53a 100644 --- a/jdk/make/mapfiles/libjimage/mapfile-vers +++ b/jdk/make/mapfiles/libjimage/mapfile-vers @@ -27,23 +27,7 @@ SUNWprivate_1.1 { global: - JNI_OnLoad; - Java_jdk_internal_jimage_ImageNativeSubstrate_openImage; - Java_jdk_internal_jimage_ImageNativeSubstrate_closeImage; - Java_jdk_internal_jimage_ImageNativeSubstrate_getIndexAddress; - Java_jdk_internal_jimage_ImageNativeSubstrate_getDataAddress; - Java_jdk_internal_jimage_ImageNativeSubstrate_read; - Java_jdk_internal_jimage_ImageNativeSubstrate_readCompressed; - Java_jdk_internal_jimage_ImageNativeSubstrate_getStringBytes; - Java_jdk_internal_jimage_ImageNativeSubstrate_getAttributes; - Java_jdk_internal_jimage_ImageNativeSubstrate_findAttributes; - Java_jdk_internal_jimage_ImageNativeSubstrate_attributeOffsets; - Java_jdk_internal_jimage_ImageNativeSubstrate_JIMAGE_1Open; - Java_jdk_internal_jimage_ImageNativeSubstrate_JIMAGE_1Close; - Java_jdk_internal_jimage_ImageNativeSubstrate_JIMAGE_1FindResource; - Java_jdk_internal_jimage_ImageNativeSubstrate_JIMAGE_1GetResource; - Java_jdk_internal_jimage_ImageNativeSubstrate_JIMAGE_1PackageToModule; - Java_jdk_internal_jimage_ImageNativeSubstrate_JIMAGE_1Resources; + Java_jdk_internal_jimage_NativeImageBuffer_getNativeMap; JIMAGE_Open; JIMAGE_Close; JIMAGE_PackageToModule; diff --git a/jdk/make/rmic/Rmic-java.management.gmk b/jdk/make/rmic/Rmic-java.management.gmk index 9eca430ed63..0452d5cbba5 100644 --- a/jdk/make/rmic/Rmic-java.management.gmk +++ b/jdk/make/rmic/Rmic-java.management.gmk @@ -40,7 +40,7 @@ JMX_RMI_CLASSES := javax.management.remote.rmi.RMIConnectionImpl \ $(eval $(call SetupRMICompilation,RMI_GEN, \ CLASSES := $(JMX_RMI_CLASSES), \ CLASSES_DIR := $(CLASSES_DIR)/java.management, \ - STUB_CLASSES_DIR := $(RMIC_GENSRC_DIR), \ + STUB_CLASSES_DIR := $(RMIC_GENSRC_DIR)/java.management, \ RUN_V12 := true, \ KEEP_GENERATED := true, \ )) @@ -50,7 +50,7 @@ $(RMIC_GENSRC_DIR)/_classes.moved: $(RMI_GEN) $(eval classfiles := $(shell $(FIND) $(RMIC_GENSRC_DIR) -name "*.class")) $(foreach src, $(classfiles), \ $(eval target := $(patsubst $(RMIC_GENSRC_DIR)/%, \ - $(STUB_CLASSES_DIR)/java.management/%, $(src))) \ + $(STUB_CLASSES_DIR)/%, $(src))) \ $(MKDIR) -p $(dir $(target)) ; \ $(MV) $(src) $(target) $(NEWLINE)) $(TOUCH) $@ diff --git a/jdk/make/rmic/RmicCommon.gmk b/jdk/make/rmic/RmicCommon.gmk index 8c6affb1015..45fd1de853f 100644 --- a/jdk/make/rmic/RmicCommon.gmk +++ b/jdk/make/rmic/RmicCommon.gmk @@ -31,10 +31,7 @@ include RMICompilation.gmk ########################################################################################## -BTRMIC_CP := $(call PathList, \ - $(BUILDTOOLS_OUTPUTDIR)/interim_rmic_classes $(INTERIM_LANGTOOLS_JAR)) -BTRMIC_ARGS := -cp $(BTRMIC_CP) -RMIC := $(JAVA) $(BTRMIC_ARGS) sun.rmi.rmic.Main +RMIC := $(JAVA) $(INTERIM_OVERRIDE_MODULES_ARGS) sun.rmi.rmic.Main CLASSES_DIR := $(JDK_OUTPUTDIR)/modules # NOTE: If the smart javac dependency management is reintroduced, these classes risk diff --git a/jdk/make/scripts/localelist.sh b/jdk/make/scripts/localelist.sh deleted file mode 100644 index 96fa02b0c9f..00000000000 --- a/jdk/make/scripts/localelist.sh +++ /dev/null @@ -1,106 +0,0 @@ -#!/bin/sh - -# -# Copyright (c) 2005, 2012, Oracle and/or its affiliates. All rights reserved. -# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. -# -# This code is free software; you can redistribute it and/or modify it -# under the terms of the GNU General Public License version 2 only, as -# published by the Free Software Foundation. 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 script is to generate the supported locale list string and replace the -# #LOCALE_LIST# in /src/share/classes/sun/util/CoreResourceBundleControl.java. -# -# NAWK & SED is passed in as environment variables. -# -LOCALE_LIST=$1 -INUT_FILE=$2 -OUTPUT_FILE=$3 - -LOCALES=`(for I in $LOCALE_LIST; do echo $I;done) | sort | uniq` -JAVA_LOCALES= - -toJavaLocale() -{ - NewLocale=`echo $1 | $NAWK ' - BEGIN { - # The following values have to be consistent with java.util.Locale. - javalocales["en"] = "ENGLISH"; - javalocales["fr"] = "FRENCH"; - javalocales["de"] = "GERMAN"; - javalocales["it"] = "ITALIAN"; - javalocales["ja"] = "JAPANESE"; - javalocales["ko"] = "KOREAN"; - javalocales["zh"] = "CHINESE"; - javalocales["zh_CN"] = "SIMPLIFIED_CHINESE"; - javalocales["zh_TW"] = "TRADITIONAL_CHINESE"; - javalocales["fr_FR"] = "FRANCE"; - javalocales["de_DE"] = "GERMANY"; - javalocales["it_IT"] = "ITALY"; - javalocales["ja_JP"] = "JAPAN"; - javalocales["ko_KR"] = "KOREA"; - javalocales["en_GB"] = "UK"; - javalocales["en_US"] = "US"; - javalocales["en_CA"] = "CANADA"; - javalocales["fr_CA"] = "CANADA_FRENCH"; - } - { - if ($0 in javalocales) { - print " Locale." javalocales[$0]; - } else { - split($0, a, "_"); - if (a[3] != "") { - print " new Locale(\"" a[1] "\", \"" a[2] "\", \"" a[3] "\")"; - } else if (a[2] != "") { - print " new Locale(\"" a[1] "\", \"" a[2] "\")"; - } else { - print " new Locale(\"" a[1] "\")"; - } - } - }'` - - JAVA_LOCALES=$JAVA_LOCALES$NewLocale -} - -# count the number of locales -counter=0 -for i in $LOCALES -do - counter=`expr $counter + 1` -done - -index=0 -for locale in $LOCALES -do - index=`expr $index + 1`; - if [ $index != $counter ] - then - toJavaLocale $locale - JAVA_LOCALES=$JAVA_LOCALES"," - else - toJavaLocale $locale - fi -done - -# replace the #LOCALE_LIST# in the precompiled CoreResourceBundleControl.java file. - -$SED -e "s@^#warn .*@// -- This file was mechanically generated: Do not edit! -- //@" \ - -e "s/#LOCALE_LIST#/$JAVA_LOCALES/g" $2 > $3 diff --git a/jdk/make/src/classes/build/tools/cldrconverter/ResourceBundleGenerator.java b/jdk/make/src/classes/build/tools/cldrconverter/ResourceBundleGenerator.java index 500cf2b10e4..b291758402f 100644 --- a/jdk/make/src/classes/build/tools/cldrconverter/ResourceBundleGenerator.java +++ b/jdk/make/src/classes/build/tools/cldrconverter/ResourceBundleGenerator.java @@ -343,9 +343,7 @@ class ResourceBundleGenerator implements BundleGenerator { if (!all && CLDRConverter.isBaseModule ^ isBaseLocale(id)) { continue; } - if (sb.length() > 0) { - sb.append(' '); - } + sb.append(' '); sb.append(id); } } diff --git a/jdk/make/src/classes/build/tools/generatebreakiteratordata/GenerateBreakIteratorData.java b/jdk/make/src/classes/build/tools/generatebreakiteratordata/GenerateBreakIteratorData.java index c22fa683c80..f0821e8a366 100644 --- a/jdk/make/src/classes/build/tools/generatebreakiteratordata/GenerateBreakIteratorData.java +++ b/jdk/make/src/classes/build/tools/generatebreakiteratordata/GenerateBreakIteratorData.java @@ -62,25 +62,37 @@ public class GenerateBreakIteratorData { CharacterCategory.makeCategoryMap(unicodeData); /* Generate files */ - generateFiles(); + try { + generateFiles(); + } catch (Exception e) { + e.printStackTrace(); + System.exit(1); + } + } + + private static String localizedBundleName(String pkg, String clazz) { + if (language.length() > 0) { + return pkg + ".ext." + clazz + '_' + language; + } else { + return pkg + '.' + clazz; + } } /** * Generate data files whose names are included in * sun.text.resources.BreakIteratorInfo+ */ - private static void generateFiles() { + private static void generateFiles() throws Exception { String[] classNames; ResourceBundle rules, info; - String pkgName = "sun.text.resources" + (language.length() > 0 ? ".ext" : ""); + info = (ResourceBundle) Class.forName( + localizedBundleName("sun.text.resources", "BreakIteratorInfo")).newInstance(); - info = ResourceBundle.getBundle(pkgName + ".BreakIteratorInfo", - new Locale(language, country, valiant)); classNames = info.getStringArray("BreakIteratorClasses"); - rules = ResourceBundle.getBundle(pkgName + ".BreakIteratorRules", - new Locale(language, country, valiant)); + rules = (ResourceBundle) Class.forName( + localizedBundleName("sun.text.resources", "BreakIteratorRules")).newInstance(); if (info.containsKey("CharacterData")) { generateDataFile(info.getString("CharacterData"), diff --git a/jdk/make/src/classes/build/tools/jdwpgen/ModuleTypeNode.java b/jdk/make/src/classes/build/tools/jdwpgen/ModuleTypeNode.java new file mode 100644 index 00000000000..af128520cfd --- /dev/null +++ b/jdk/make/src/classes/build/tools/jdwpgen/ModuleTypeNode.java @@ -0,0 +1,56 @@ +/* + * 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. + */ + +package build.tools.jdwpgen; + +import java.util.*; +import java.io.*; + +class ModuleTypeNode extends AbstractSimpleTypeNode { + + String docType() { + return "moduleID"; + } + + String javaType() { + return "ModuleReferenceImpl"; + } + + String debugValue(String label) { + return "(" + label + "==null?\"NULL\":\"ref=\"+" + label + ".ref())"; + } + + public void genJavaWrite(PrintWriter writer, int depth, + String writeLabel) { + genJavaDebugWrite(writer, depth, writeLabel, + debugValue(writeLabel)); + indent(writer, depth); + writer.println("ps.writeModuleRef(" + writeLabel + ".ref());"); + } + + String javaRead() { + return "ps.readModule()"; + } +} diff --git a/jdk/make/src/classes/build/tools/jdwpgen/Parse.java b/jdk/make/src/classes/build/tools/jdwpgen/Parse.java index 5b172e5dae5..4ac854b09c8 100644 --- a/jdk/make/src/classes/build/tools/jdwpgen/Parse.java +++ b/jdk/make/src/classes/build/tools/jdwpgen/Parse.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1998, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1998, 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 @@ -80,6 +80,7 @@ class Parse { kindMap.put("field", new FieldTypeNode()); kindMap.put("frame", new FrameTypeNode()); kindMap.put("string", new StringTypeNode()); + kindMap.put("moduleID", new ModuleTypeNode()); kindMap.put("value", new ValueTypeNode()); kindMap.put("byte", new ByteTypeNode()); kindMap.put("location", new LocationTypeNode()); diff --git a/jdk/make/src/classes/build/tools/jigsaw/GenGraphs.java b/jdk/make/src/classes/build/tools/jigsaw/GenGraphs.java new file mode 100644 index 00000000000..da774751e1c --- /dev/null +++ b/jdk/make/src/classes/build/tools/jigsaw/GenGraphs.java @@ -0,0 +1,231 @@ +/* + * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package build.tools.jigsaw; + +import java.io.IOException; +import java.io.PrintStream; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.lang.module.ModuleReference; +import java.lang.module.ResolvedModule; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; +import java.util.TreeSet; +import java.util.function.Function; +import java.util.stream.Collectors; +import static java.lang.module.ModuleDescriptor.Requires.Modifier.PUBLIC; + +/** + * Generate the DOT file for a module graph for each module in the JDK + * after transitive reduction. + */ +public class GenGraphs { + + public static void main(String[] args) throws Exception { + + if (args.length != 1) { + System.err.println("ERROR: specify the output directory"); + System.exit(1); + } + Path dir = Paths.get(args[0]); + Files.createDirectories(dir); + + ModuleFinder finder = ModuleFinder.ofSystem(); + + Set javaSEModules + = new TreeSet<>(finder.findAll().stream() + .map(ModuleReference::descriptor) + .filter(m -> (m.name().startsWith("java.") && + !m.name().equals("java.smartcardio"))) + .collect(Collectors.toSet())); + Set jdkModules + = new TreeSet<>(finder.findAll().stream() + .map(ModuleReference::descriptor) + .filter(m -> !javaSEModules.contains(m)) + .collect(Collectors.toSet())); + + GenGraphs genGraphs = new GenGraphs(javaSEModules, jdkModules); + Set mods = new HashSet<>(); + for (ModuleReference mref: finder.findAll()) { + ModuleDescriptor descriptor = mref.descriptor(); + String name = descriptor.name(); + mods.add(name); + Configuration cf = Configuration.empty() + .resolveRequires(finder, + ModuleFinder.empty(), + Set.of(name)); + genGraphs.genDotFile(dir, name, cf); + } + + Configuration cf = Configuration.empty() + .resolveRequires(finder, + ModuleFinder.empty(), + mods); + genGraphs.genDotFile(dir, "jdk", cf); + + } + + private final Set javaGroup; + private final Set jdkGroup; + + GenGraphs(Set javaGroup, Set jdkGroup) { + this.javaGroup = Collections.unmodifiableSet(javaGroup); + this.jdkGroup = Collections.unmodifiableSet(jdkGroup); + } + + private static final String ORANGE = "#e76f00"; + private static final String BLUE = "#437291"; + private static final String GRAY = "#dddddd"; + + private static final String REEXPORTS = ""; + private static final String REQUIRES = "style=\"dashed\""; + private static final String REQUIRES_BASE = "color=\"" + GRAY + "\""; + + private static final Map weights = new HashMap<>(); + + private static void weight(String s, String t, int w) { + weights.put(s + ":" + t, w); + } + + private static int weightOf(String s, String t) { + int w = weights.getOrDefault(s + ":" + t, 1); + if (w != 1) + return w; + if (s.startsWith("java.") && t.startsWith("java.")) + return 10; + return 1; + } + + static { + int h = 1000; + weight("java.se", "java.compact3", h * 10); + weight("jdk.compact3", "java.compact3", h * 10); + weight("java.compact3", "java.compact2", h * 10); + weight("java.compact2", "java.compact1", h * 10); + weight("java.compact1", "java.logging", h * 10); + weight("java.logging", "java.base", h * 10); + } + + private void genDotFile(Path dir, String name, Configuration cf) throws IOException { + try (PrintStream out + = new PrintStream(Files.newOutputStream(dir.resolve(name + ".dot")))) { + + Map nameToModule = cf.modules().stream() + .map(ResolvedModule::reference) + .map(ModuleReference::descriptor) + .collect(Collectors.toMap(ModuleDescriptor::name, Function.identity())); + + Set descriptors = new TreeSet<>(nameToModule.values()); + + out.format("digraph \"%s\" {%n", name); + out.format("size=\"25,25\";"); + out.format("nodesep=.5;%n"); + out.format("ranksep=1.5;%n"); + out.format("pencolor=transparent;%n"); + out.format("node [shape=plaintext, fontname=\"DejaVuSans\", fontsize=36, margin=\".2,.2\"];%n"); + out.format("edge [penwidth=4, color=\"#999999\", arrowhead=open, arrowsize=2];%n"); + + out.format("subgraph %sse {%n", name.equals("jdk") ? "cluster_" : ""); + descriptors.stream() + .filter(javaGroup::contains) + .map(ModuleDescriptor::name) + .forEach(mn -> out.format(" \"%s\" [fontcolor=\"%s\", group=%s];%n", + mn, ORANGE, "java")); + out.format("}%n"); + descriptors.stream() + .filter(jdkGroup::contains) + .map(ModuleDescriptor::name) + .forEach(mn -> out.format(" \"%s\" [fontcolor=\"%s\", group=%s];%n", + mn, BLUE, "jdk")); + + // transitive reduction + Graph graph = gengraph(cf); + descriptors.forEach(md -> { + String mn = md.name(); + Set requiresPublic = md.requires().stream() + .filter(d -> d.modifiers().contains(PUBLIC)) + .map(d -> d.name()) + .collect(Collectors.toSet()); + + graph.adjacentNodes(mn).forEach(dn -> { + String attr = dn.equals("java.base") ? REQUIRES_BASE + : (requiresPublic.contains(dn) ? REEXPORTS : REQUIRES); + int w = weightOf(mn, dn); + if (w > 1) + attr += "weight=" + w; + out.format(" \"%s\" -> \"%s\" [%s];%n", mn, dn, attr); + }); + }); + + out.println("}"); + } + } + + /** + * Returns a Graph of the given Configuration after transitive reduction. + * + * Transitive reduction of requires public edge and requires edge have + * to be applied separately to prevent the requires public edges + * (e.g. U -> V) from being reduced by a path (U -> X -> Y -> V) + * in which V would not be re-exported from U. + */ + private Graph gengraph(Configuration cf) { + Graph.Builder builder = new Graph.Builder<>(); + for (ResolvedModule resolvedModule : cf.modules()) { + String mn = resolvedModule.reference().descriptor().name(); + builder.addNode(mn); + resolvedModule.reads().stream() + .map(ResolvedModule::name) + .forEach(target -> builder.addEdge(mn, target)); + } + Graph rpg = requiresPublicGraph(cf); + return builder.build().reduce(rpg); + } + + /** + * Returns a Graph containing only requires public edges + * with transitive reduction. + */ + private Graph requiresPublicGraph(Configuration cf) { + Graph.Builder builder = new Graph.Builder<>(); + for (ResolvedModule resolvedModule : cf.modules()) { + ModuleDescriptor descriptor = resolvedModule.reference().descriptor(); + String mn = descriptor.name(); + descriptor.requires().stream() + .filter(d -> d.modifiers().contains(PUBLIC)) + .map(d -> d.name()) + .forEach(d -> builder.addEdge(mn, d)); + } + return builder.build().reduce(); + } +} diff --git a/jdk/make/src/classes/build/tools/jigsaw/Graph.java b/jdk/make/src/classes/build/tools/jigsaw/Graph.java new file mode 100644 index 00000000000..a835da1ad93 --- /dev/null +++ b/jdk/make/src/classes/build/tools/jigsaw/Graph.java @@ -0,0 +1,171 @@ +/* + * 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. + */ + +package build.tools.jigsaw; + +import java.io.PrintStream; +import java.util.Deque; +import java.util.HashMap; +import java.util.HashSet; +import java.util.LinkedList; +import java.util.Map; +import java.util.Set; + +public class Graph { + private static boolean traceOn = Boolean.getBoolean("build.tools.module.trace"); + private final Set nodes; + private final Map> edges; + private Graph(Set nodes, Map> edges) { + this.nodes = nodes; + this.edges = edges; + } + + public Set nodes() { + return nodes; + } + + public Map> edges() { + return edges; + } + + public Set adjacentNodes(T u) { + return edges.get(u); + } + + /** + * Returns a new Graph after transitive reduction + */ + public Graph reduce() { + Graph.Builder builder = new Builder<>(); + nodes.stream() + .forEach(u -> { + builder.addNode(u); + edges.get(u).stream() + .filter(v -> !pathExists(u, v, false)) + .forEach(v -> builder.addEdge(u, v)); + }); + return builder.build(); + } + + /** + * Returns a new Graph after transitive reduction. All edges in + * the given g takes precedence over this graph. + * + * @throw IllegalArgumentException g must be a subgraph this graph + */ + public Graph reduce(Graph g) { + boolean subgraph = nodes.containsAll(g.nodes) && g.edges.keySet().stream() + .allMatch(u -> adjacentNodes(u).containsAll(g.adjacentNodes(u))); + if (!subgraph) { + throw new IllegalArgumentException("the given argument is not a subgraph of this graph"); + } + + Graph.Builder builder = new Builder<>(); + nodes.stream() + .forEach(u -> { + builder.addNode(u); + // filter the edge if there exists a path from u to v in the given g + // or there exists another path from u to v in this graph + edges.get(u).stream() + .filter(v -> !g.pathExists(u, v) && !pathExists(u, v, false)) + .forEach(v -> builder.addEdge(u, v)); + }); + + // add the overlapped edges from this graph and the given g + g.edges().keySet().stream() + .forEach(u -> g.adjacentNodes(u).stream() + .filter(v -> isAdjacent(u, v)) + .forEach(v -> builder.addEdge(u, v))); + return builder.build(); + } + + private boolean isAdjacent(T u, T v) { + return edges.containsKey(u) && edges.get(u).contains(v); + } + + private boolean pathExists(T u, T v) { + return pathExists(u, v, true); + } + + /** + * Returns true if there exists a path from u to v in this graph. + * If includeAdjacent is false, it returns true if there exists + * another path from u to v of distance > 1 + */ + private boolean pathExists(T u, T v, boolean includeAdjacent) { + if (!nodes.contains(u) || !nodes.contains(v)) { + return false; + } + if (includeAdjacent && isAdjacent(u, v)) { + return true; + } + Deque stack = new LinkedList<>(); + Set visited = new HashSet<>(); + stack.push(u); + while (!stack.isEmpty()) { + T node = stack.pop(); + if (node.equals(v)) { + if (traceOn) { + System.out.format("Edge %s -> %s removed%n", u, v); + } + return true; + } + if (!visited.contains(node)) { + visited.add(node); + edges.get(node).stream() + .filter(e -> includeAdjacent || !node.equals(u) || !e.equals(v)) + .forEach(e -> stack.push(e)); + } + } + assert !visited.contains(v); + return false; + } + + void printGraph(PrintStream out) { + nodes.stream() + .forEach(u -> adjacentNodes(u).stream() + .forEach(v -> out.format("%s -> %s%n", u, v))); + } + + public static class Builder { + final Set nodes = new HashSet<>(); + final Map> edges = new HashMap<>(); + public void addNode(T node) { + if (nodes.contains(node)) { + return; + } + nodes.add(node); + edges.computeIfAbsent(node, _e -> new HashSet<>()); + } + public void addEdge(T u, T v) { + addNode(u); + addNode(v); + edges.get(u).add(v); + } + public Graph build() { + return new Graph<>(nodes, edges); + } + } +} diff --git a/jdk/make/src/classes/build/tools/jigsaw/ModuleSummary.java b/jdk/make/src/classes/build/tools/jigsaw/ModuleSummary.java new file mode 100644 index 00000000000..c1b15d39219 --- /dev/null +++ b/jdk/make/src/classes/build/tools/jigsaw/ModuleSummary.java @@ -0,0 +1,766 @@ +/* + * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package build.tools.jigsaw; + +import java.io.IOException; +import java.io.PrintStream; +import java.lang.module.Configuration; +import java.lang.module.ModuleDescriptor; +import java.lang.module.ModuleFinder; +import java.lang.module.ModuleReference; +import java.lang.module.ResolvedModule; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.Arrays; +import java.util.Comparator; +import java.util.Date; +import java.util.Enumeration; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; +import java.util.stream.Collectors; +import java.util.zip.ZipEntry; +import java.util.zip.ZipFile; +import static java.lang.module.ModuleDescriptor.*; +import static build.tools.jigsaw.ModuleSummary.HtmlDocument.Selector.*; +import static build.tools.jigsaw.ModuleSummary.HtmlDocument.Division.*; + +public class ModuleSummary { + private static final String USAGE = "Usage: ModuleSummary -mp

-o [-root mn]*"; + + public static void main(String[] args) throws Exception { + int i=0; + Path modpath = null; + Path outfile = null; + Set roots = new HashSet<>(); + while (i < args.length && args[i].startsWith("-")) { + String arg = args[i++]; + switch (arg) { + case "-mp": + modpath = Paths.get(args[i++]); + break; + case "-o": + outfile = Paths.get(args[i++]); + break; + case "-root": + roots.add(args[i++]); + default: + System.err.println(USAGE); + System.exit(-1); + } + } + if (outfile == null || modpath == null) { + System.err.println(USAGE); + System.exit(1); + } + Path dir = outfile.getParent() != null ? outfile.getParent() : Paths.get("."); + Files.createDirectories(dir); + + Map modules = new HashMap<>(); + Set mrefs = ModuleFinder.ofSystem().findAll(); + for (ModuleReference mref : mrefs) { + String mn = mref.descriptor().name(); + Path jmod = modpath.resolve(mn + ".jmod"); + modules.put(mn, new ModuleSummary(mref, jmod)); + } + + if (roots.isEmpty()) { + roots.addAll(modules.keySet()); + } + genReport(outfile, modules, roots, "JDK Module Summary"); + } + + static void genReport(Path outfile, Map modules, Set roots, String title) + throws IOException + { + Configuration cf = resolve(roots); + try (PrintStream out = new PrintStream(Files.newOutputStream(outfile))) { + HtmlDocument doc = new HtmlDocument(title, modules); + Set descriptors = cf.modules().stream() + .map(ResolvedModule::reference) + .map(ModuleReference::descriptor) + .collect(Collectors.toSet()); + doc.writeTo(out, descriptors); + } + } + + private final String name; + private final ModuleDescriptor descriptor; + private final JmodInfo jmodInfo; + ModuleSummary(ModuleReference mref, Path jmod) throws IOException { + this.name = mref.descriptor().name(); + this.descriptor = mref.descriptor(); + this.jmodInfo = new JmodInfo(jmod); + } + + String name() { + return name; + } + + long uncompressedSize() { + return jmodInfo.size; + } + + long jmodFileSize() { + return jmodInfo.filesize; // estimated compressed size + } + + ModuleDescriptor descriptor() { + return descriptor; + } + + int numClasses() { + return jmodInfo.classCount; + } + + long classBytes() { + return jmodInfo.classBytes; + } + + int numResources() { + return jmodInfo.resourceCount; + } + + long resourceBytes() { + return jmodInfo.resourceBytes; + } + + int numConfigs() { + return jmodInfo.configCount; + } + long configBytes() { + return jmodInfo.configBytes; + } + int numCommands() { + return jmodInfo.nativeCmds.size(); + } + + long commandBytes() { + return jmodInfo.nativeCmds.values().stream() + .mapToLong(l -> l.longValue()).sum() - jmodInfo.debugInfoCmdBytes; + } + int numCommandsDebug() { + return jmodInfo.debugInfoCmdCount; + } + long commandDebugBytes() { + return jmodInfo.debugInfoCmdBytes; + } + int numNativeLibraries() { + return jmodInfo.nativeLibs.size(); + } + + long nativeLibrariesBytes() { + return jmodInfo.nativeLibs.values().stream() + .mapToLong(l -> l.longValue()).sum() - jmodInfo.debugInfoLibBytes; + } + int numNativeLibrariesDebug() { + return jmodInfo.debugInfoLibCount; + } + + long nativeLibrariesDebugBytes() { + return jmodInfo.debugInfoLibBytes; + } + + Map commands() { + return jmodInfo.nativeCmds; + } + + Map nativeLibs() { + return jmodInfo.nativeLibs; + } + + Map configFiles() { + return jmodInfo.configFiles; + } + + + static class JmodInfo { + final long size; + final long filesize; + final int classCount; + final long classBytes; + final int resourceCount; + final long resourceBytes; + final int configCount; + final long configBytes; + final int debugInfoLibCount; + final long debugInfoLibBytes; + final int debugInfoCmdCount; + final long debugInfoCmdBytes; + final Map configFiles = new HashMap<>(); + final Map nativeCmds = new HashMap<>(); + final Map nativeLibs = new HashMap<>(); + + JmodInfo(Path jmod) throws IOException { + long total = 0; + long cBytes = 0, rBytes = 0, cfBytes = 0, dizLibBytes = 0, dizCmdBytes = 0; + int cCount = 0, rCount = 0, cfCount = 0, dizLibCount = 0, dizCmdCount = 0; + try (ZipFile zf = new ZipFile(jmod.toFile())) { + for (Enumeration e = zf.entries(); e.hasMoreElements(); ) { + ZipEntry ze = e.nextElement(); + String fn = ze.getName(); + int pos = fn.indexOf('/'); + String dir = fn.substring(0, pos); + String filename = fn.substring(fn.lastIndexOf('/') + 1); + // name shown in the column + String name = filename; + + long len = ze.getSize(); + total += len; + switch (dir) { + case NATIVE_LIBS: + nativeLibs.put(name, len); + if (filename.endsWith(".diz")) { + dizLibCount++; + dizLibBytes += len; + } + break; + case NATIVE_CMDS: + nativeCmds.put(name, len); + if (filename.endsWith(".diz")) { + dizCmdCount++; + dizCmdBytes += len; + } + break; + case CLASSES: + if (filename.endsWith(".class")) { + cCount++; + cBytes += len; + } else { + rCount++; + rBytes += len; + } + break; + case CONFIG: + configFiles.put(name, len); + cfCount++; + cfBytes += len; + break; + default: + break; + } + } + this.filesize = jmod.toFile().length(); + this.classCount = cCount; + this.classBytes = cBytes; + this.resourceCount = rCount; + this.resourceBytes = rBytes; + this.configCount = cfCount; + this.configBytes = cfBytes; + this.size = total; + this.debugInfoLibCount = dizLibCount; + this.debugInfoLibBytes = dizLibBytes; + this.debugInfoCmdCount = dizCmdCount; + this.debugInfoCmdBytes = dizCmdBytes; + } + } + + static final String NATIVE_LIBS = "native"; + static final String NATIVE_CMDS = "bin"; + static final String CLASSES = "classes"; + static final String CONFIG = "conf"; + + static final String MODULE_ID = "module/id"; + static final String MODULE_MAIN_CLASS = "module/main-class"; + } + + static Configuration resolve(Set roots) { + return Configuration.empty() + .resolveRequires(ModuleFinder.ofSystem(), + ModuleFinder.empty(), + roots); + } + + static class HtmlDocument { + final String title; + final Map modules; + boolean requiresPublicNote = false; + boolean aggregatorNote = false; + boolean totalBytesNote = false; + HtmlDocument(String title, Map modules) { + this.title = title; + this.modules = modules; + } + + void writeTo(PrintStream out, Set selectedModules) { + out.format("%n"); + out.format("%s%n", title); + // stylesheet + Arrays.stream(HtmlDocument.STYLES).forEach(out::println); + out.format("%n"); + + // body begins + out.format("%n"); + + // title and date + out.println(DOCTITLE.toString(title)); + out.println(VERSION.toString(String.format("%tc", new Date()))); + + // total modules and sizes + long totalBytes = selectedModules.stream() + .map(ModuleDescriptor::name) + .map(modules::get) + .mapToLong(ModuleSummary::uncompressedSize) + .sum(); + String[] sections = new String[] { + String.format("%s: %d", "Total modules", selectedModules.size()), + String.format("%s: %,d bytes (%s %s)", "Total size", + totalBytes, + System.getProperty("os.name"), + System.getProperty("os.arch")) + }; + out.println(SECTION.toString(sections)); + + // write table and header + out.println(String.format("", MODULES)); + out.println(header("Module", "Requires", "Exports", + "Services", "Commands/Native Libraries/Configs")); + + // write contents - one row per module + selectedModules.stream() + .sorted(Comparator.comparing(ModuleDescriptor::name)) + .map(m -> modules.get(m.name())) + .map(ModuleTableRow::new) + .forEach(table -> table.writeTo(out)); + + out.format("
"); // end table + out.format(""); + out.println(""); + } + + String header(String... columns) { + StringBuilder sb = new StringBuilder(); + sb.append(""); + Arrays.stream(columns) + .forEach(cn -> sb.append(" ").append(cn).append("").append("\n")); + sb.append(""); + return sb.toString(); + } + + static enum Selector { + MODULES("modules"), + MODULE("module"), + MODULE_DEF("code name def"), + AGGREGATOR("code name def agg"), + REQUIRES("code"), + REQUIRES_PUBLIC("code reexp"), + BR("br"), + CODE("code"), + NUMBER("number"),; + final String name; + Selector(String name) { + this.name = name; + } + @Override + public String toString() { + return name; + } + } + + static enum Division { + DOCTITLE("doctitle"), + VERSION("versions"), + SECTION("section"); + final String name; + + Division(String name) { + this.name = name; + } + + public String toString(String... lines) { + String value = Arrays.stream(lines).collect(Collectors.joining("
\n")); + return "
" + value + "
"; + } + } + + class ModuleTableRow { + private final ModuleSummary ms; + private final Set deps; + private final int maxRows; + private final boolean aggregator; + ModuleTableRow(ModuleSummary ms) { + this.ms = ms; + Configuration cf = resolve(Set.of(ms.name())); + this.deps = cf.modules().stream() + .map(ResolvedModule::reference) + .map(ModuleReference::descriptor) + .collect(Collectors.toSet()); + int count = (ms.numClasses() > 0 ? 1 : 0) + + (ms.numResources() > 0 ? 1 : 0) + + (ms.numConfigs() > 0 ? 1 : 0) + + (ms.numNativeLibraries() > 0 ? 1 : 0) + + (ms.numNativeLibrariesDebug() > 0 ? 1 : 0) + + (ms.numCommands() > 0 ? 1 : 0) + + (ms.numCommandsDebug() > 0 ? 1 : 0); + this.aggregator = ms.numClasses() == 1 && count == 1; // only module-info.class + + // 5 fixed rows (name + 2 transitive count/size + 2 blank rows) + this.maxRows = 5 + count + (aggregator && !aggregatorNote ? 2 : 0); + } + + public void writeTo(PrintStream out) { + out.println(String.format("", ms.name(), MODULE)); + out.println(moduleColumn()); + out.println(requiresColumn()); + out.println(exportsColumn()); + out.println(servicesColumn()); + out.println(otherSectionColumn()); + out.println(""); + out.println(""); + } + + public String moduleColumn() { + // module name + StringBuilder sb = new StringBuilder(" "); + sb.append(""); + sb.append(String.format("", MODULE)).append("\n"); + sb.append(moduleName(ms.name())); + sb.append(blankRow()); + // metadata + sb.append(toTableRow("class", "classes", ms.numClasses(), ms.classBytes())); + sb.append(toTableRow("resource", "resources", ms.numResources(), ms.resourceBytes())); + sb.append(toTableRow("config", "configs", ms.numConfigs(), ms.configBytes())); + sb.append(toTableRow("native library", "native libraries", + ms.numNativeLibraries(), ms.nativeLibrariesBytes())); + sb.append(toTableRow("native library debug", "native libraries debug", + ms.numNativeLibrariesDebug(), ms.nativeLibrariesDebugBytes())); + sb.append(toTableRow("command", "commands", ms.numCommands(), ms.commandBytes())); + sb.append(toTableRow("command debug", "commands debug", + ms.numCommandsDebug(), ms.commandDebugBytes())); + sb.append(blankRow()); + + // transitive dependencies + long reqBytes = deps.stream() + .filter(d -> !d.name().equals(ms.name())) + .mapToLong(d -> modules.get(d.name()).uncompressedSize()) + .sum(); + long reqJmodFileSize = deps.stream() + .mapToLong(d -> modules.get(d.name()).jmodFileSize()) + .sum(); + // size + if (totalBytesNote) { + sb.append(toTableRow("Total bytes", ms.uncompressedSize())); + sb.append(toTableRow("Total bytes of dependencies", reqBytes)); + } else { + // print footnote + sb.append(toTableRow("Total bytes1", ms.uncompressedSize())); + sb.append(toTableRow("Total bytes of dependencies2", reqBytes)); + } + String files = deps.size() == 1 ? "file" : "files"; + sb.append(toTableRow(String.format("Total jmod bytes (%d %s)", deps.size(), files), reqJmodFileSize)); + + if (aggregator && !aggregatorNote) { + aggregatorNote = true; + sb.append(blankRow()); + sb.append(toTableRow("* aggregator is a module with module-info.class only", BR)); + } + if (!totalBytesNote) { + totalBytesNote = true; + sb.append(blankRow()); + sb.append(toTableRow("1sum of all files including debug files", BR)); + sb.append(toTableRow("2sum of direct and indirect dependencies", BR)); + } + sb.append("
").append(""); + return sb.toString(); + } + + private String moduleName(String mn) { + if (aggregator) { + StringBuilder sb = new StringBuilder(); + sb.append(String.format("", AGGREGATOR)) + .append(mn) + .append("").append("  "); + if (!aggregatorNote) { + sb.append("(aggregator*)"); + } else { + sb.append("(aggregator)"); + } + sb.append(""); + return sb.toString(); + } else { + return toTableRow(mn, MODULE_DEF); + } + } + + public String requiresColumn() { + StringBuilder sb = new StringBuilder(); + sb.append(String.format("")); + boolean footnote = requiresPublicNote; + ms.descriptor().requires().stream() + .sorted(Comparator.comparing(Requires::name)) + .forEach(r -> { + boolean requiresPublic = r.modifiers().contains(Requires.Modifier.PUBLIC); + Selector sel = requiresPublic ? REQUIRES_PUBLIC : REQUIRES; + String req = String.format("%s", + sel, r.name(), r.name()); + if (!requiresPublicNote && requiresPublic) { + requiresPublicNote = true; + req += "*"; + } + sb.append(req).append("\n").append("
"); + }); + + if (!ms.name().equals("java.base")) { + int directDeps = ms.descriptor().requires().size(); + int indirectDeps = deps.size()-directDeps-1; + for (int i=directDeps; i< (maxRows-1); i++) { + sb.append("
"); + } + sb.append("
"); + sb.append("+").append(indirectDeps).append(" transitive dependencies"); + } + if (footnote != requiresPublicNote) { + sb.append("

").append("* bold denotes requires public"); + } + sb.append(""); + return sb.toString(); + } + + public String exportsColumn() { + StringBuilder sb = new StringBuilder(); + sb.append(String.format(" ", CODE)); + ms.descriptor().exports().stream() + .sorted(Comparator.comparing(Exports::source)) + .filter(e -> !e.isQualified()) + .forEach(e -> sb.append(e.source()).append("
").append("\n")); + sb.append(""); + return sb.toString(); + } + + public String servicesColumn() { + StringBuilder sb = new StringBuilder(); + sb.append(String.format(" ", CODE)); + ms.descriptor().uses().stream() + .sorted() + .forEach(s -> sb.append("uses ").append(s).append("
").append("\n")); + ms.descriptor().provides().entrySet().stream() + .sorted(Map.Entry.comparingByKey()) + .flatMap(e -> e.getValue().providers().stream() + .map(p -> String.format("provides %s
    with %s", + e.getKey(), p))) + .forEach(p -> sb.append(p).append("
").append("\n")); + sb.append(""); + return sb.toString(); + } + + public String otherSectionColumn() { + StringBuilder sb = new StringBuilder(); + sb.append(""); + sb.append(String.format("", MODULE)).append("\n"); + // commands + if (ms.numCommands() > 0) { + sb.append(toTableRow("bin/", CODE)); + ms.commands().entrySet().stream() + .sorted(Map.Entry.comparingByKey()) + .forEach(e -> sb.append(toTableRow(e.getKey(), e.getValue(), CODE))); + sb.append(blankRow()); + } + + // native libraries + if (ms.numNativeLibraries() > 0) { + sb.append(toTableRow("lib/", CODE)); + ms.nativeLibs().entrySet().stream() + .sorted(Map.Entry.comparingByKey()) + .forEach(e -> sb.append(toTableRow(e.getKey(), e.getValue(), CODE))); + sb.append(blankRow()); + } + + // config files + if (ms.numConfigs() > 0) { + sb.append(toTableRow("conf/", CODE)); + ms.configFiles().entrySet().stream() + .sorted(Map.Entry.comparingByKey()) + .forEach(e -> sb.append(toTableRow(e.getKey(), e.getValue(), CODE))); + } + // totals + sb.append("
").append(""); + return sb.toString(); + } + + private String blankRow() { + return toTableRow(" ", BR); + } + + private String toTableRow(String col, Selector selector) { + TableDataBuilder builder = new TableDataBuilder(); + builder.colspan(selector, 2, col); + return builder.build(); + } + + private String toTableRow(String col1, long col2) { + return toTableRow(col1, col2, BR); + } + + private String toTableRow(String col1, long col2, Selector selector) { + TableDataBuilder builder = new TableDataBuilder(); + builder.data(selector, col1); + builder.data(col2); + return builder.build(); + + } + + private String toTableRow(String singular, String plural, int count, long bytes) { + if (count == 0) { + return ""; + } + TableDataBuilder builder = new TableDataBuilder(); + if (count == 1) { + builder.data(count + " " + singular); + } else { + builder.data(count + " " + plural); + } + builder.data(bytes); + return builder.build(); + } + + class TableDataBuilder { + private final StringBuilder sb; + TableDataBuilder() { + this.sb = new StringBuilder(""); + } + TableDataBuilder data(String s) { + data(BR, s); + return this; + } + TableDataBuilder data(long num) { + data(NUMBER, String.format("%,d", num)); + return this; + } + TableDataBuilder colspan(Selector selector, int columns, String data) { + sb.append(""); + sb.append(""); + sb.append(data).append(""); + return this; + } + + TableDataBuilder data(Selector selector, String data) { + sb.append(""); + sb.append(data).append(""); + return this; + } + String build() { + sb.append(""); + return sb.toString(); + } + } + } + + private static final String[] STYLES = new String[]{ + "", + "", + }; + } +} diff --git a/jdk/make/src/classes/build/tools/jigsaw/technology-summary.html b/jdk/make/src/classes/build/tools/jigsaw/technology-summary.html new file mode 100644 index 00000000000..8414f3521ec --- /dev/null +++ b/jdk/make/src/classes/build/tools/jigsaw/technology-summary.html @@ -0,0 +1,627 @@ + + +JDK Technology Summary + + + +

JCP Technologies in the Modular JDK

+ +

Last updated 2015-03-06 (Added java.datatransfer. Assumes JNLP is modularized, and StAX joins the Java SE Platform.)

+ +

JDK Module Summary | Technologies in the Java SE Documentation

+ + + + + + + + +
Legend
JCP technology in the Java SE Platform only -- in java.base
JCP technology in the Java SE Platform only -- not in java.base
JCP technology in the Java SE Platform and the Java EE Platform
JCP technology in the Java SE Platform based on non-JCP standards
JCP technology in neither the Java SE or EE Platforms
+ +

An upgradeable module contains JCP technology that is in the Java SE Platform but is not exclusive to the Java SE Platform, i.e., the green and pink technologies. Most upgradeable modules are defined by loaders other than the bootstrap.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TechnologyOriginal JSROriginal TargetModuleEvolved ByHistoryProfile/SELoaderUpg?
JMX3Java SEjava.managementUJSR for Java SE3bootNo
Print Service6Java SEjava.desktopUJSR for Java SESEbootNo
Preferences10Java SEjava.prefsUJSR for Java SE3bootNo
Image I/O15Java SEjava.desktopUJSR for Java SESEbootNo
SASL28Java SEjava.security.saslUJSR for Java SE3bootNo
Logging47Java SEjava.loggingUJSR for Java SE1bootNo
NIO51Java SEjava.baseUJSR for Java SE1bootNo
JNLP56Java SEjava.jnlpOriginal JSRN/AbootNo
Beans Persistence57Java SEjava.desktopUJSR for Java SESEbootNo
GSS72Java SEjava.security.jgssUJSR for Java SE3bootNo
XML Digital Signature105Java SEjava.xml.cryptoUJSR for Java SE3bootNo
JDBC Rowset114Java SEjava.sql.rowsetOriginal JSRCo-evolved with JDBC3bootNo
JMX Remote160Java SEjava.managementUJSR for Java SE3bootNo
Profiling (Agent)163Java SEjava.instrumentUJSR for Java SE3bootNo
Profiling (JMX)163Java SEjava.managementUJSR for Java SE3bootNo
Concurrency Utilities166Java SEjava.baseUJSR for Java SE1bootNo
Annotations175Java SEjava.baseUJSR for Java SE1bootNo
StAX173Java SEjava.xmlOriginal JSRFormerly a Standalone Technology2bootNo
Annotations (Language Model)175Java SEjava.compilerUJSR for Java SE3bootNo
Compiler199Java SEjava.compilerOriginal JSRFormerly a Standalone Technology3bootNo
Pack200200Java SEjava.baseUJSR for Java SE1bootNo
NIO.2203Java SEjava.baseUJSR for Java SE1bootNo
JAXP206Java SEjava.xmlUJSR for Java SEFormerly a Standalone Technology2bootNo
JDBC221Java SEjava.sqlOriginal JSRCo-evolved with JDBC Rowset2bootNo
Scripting223Java SEjava.scriptingOriginal JSRFormerly a Standalone Technology1bootNo
Smart Card I/O268Java SEjava.smartcardioOriginal JSRFormerly a Standalone Technology (unlisted)N/AbootNo
Annotation Processing269Java SEjava.compilerOriginal JSRFormerly a Standalone Technology3bootNo
InvokeDynamic292Java SEjava.baseUJSR for Java SE1bootNo
Type Annotations308Java SEjava.baseUJSR for Java SE1bootNo
Type Annotations (Language Model)308Java SEjava.compilerUJSR for Java SE3bootNo
Date and Time310Java SEjava.baseUJSR for Java SE1bootNo
Streams335Java SEjava.baseUJSR for Java SE1bootNo
Collections, Math, I18N, I/O, Net, Reflection------java.baseUJSR for Java SE1bootNo
JCA, JAAS, JSSE------java.baseUJSR for Java SE1bootNo
Applet, AWT, Swing, Java 2D, Beans, A11Y, Sound------java.desktopUJSR for Java SESEbootNo
Data Transfer------java.datatransferUJSR for Java SESEbootNo
JNDI------java.namingUJSR for Java SE3bootNo
RMI------java.rmiUJSR for Java SE2bootNo
JAF925---java.activationOriginal JSRFormerly a Standalone Technology (unlisted)SEextYes
RMI-IIOP, IDL(OMG)---java.corbaUJSR for Java SEFormerly an Endorsed StandardSEextYes
DOM, SAX(W3C)---java.xmlUJSR for Java SEFormerly an Endorsed Standard2bootNo
SAAJ67Java SEjava.xml.wsOriginal JSRFormerly a Standalone Technology (f.k.a. JAXM)SEextYes
Web Services Metadata181Java EEjava.xml.wsOriginal JSRFormerly a Standalone Technology (unlisted)SEextYes
JAXB222Java SEjava.xml.bindOriginal JSRFormerly a Standalone TechnologySEextYes
JAXWS224Java SEjava.xml.wsOriginal JSRFormerly a Standalone TechnologySEextYes
Common Annotations250Java SE,EEjava.annotations.commonOriginal JSRFormerly a Standalone Technology (unlisted)SEextYes
JTA (non-XA)907---java.transactionOriginal JSRFormerly a Standalone Technology (unlisted)SEextYes
JTA (XA)907---java.sqlOriginal JSRFormerly a Standalone Technology (unlisted)2bootNo
+ + diff --git a/jdk/make/src/classes/build/tools/module/GenJdepsModulesXml.java b/jdk/make/src/classes/build/tools/module/GenJdepsModulesXml.java deleted file mode 100644 index 17a63e79546..00000000000 --- a/jdk/make/src/classes/build/tools/module/GenJdepsModulesXml.java +++ /dev/null @@ -1,130 +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. - */ - -package build.tools.module; - -import java.io.File; -import java.io.IOException; -import java.io.UncheckedIOException; -import java.nio.file.Files; -import java.nio.file.NoSuchFileException; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.nio.file.attribute.BasicFileAttributes; -import java.util.HashSet; -import java.util.Set; -import java.util.stream.Collectors; - -/** - * GenJdepsModulesXml augments the input modules.xml file(s) - * to include the module membership from the given path to - * the JDK exploded image. The output file is used by jdeps - * to analyze dependencies and enforce module boundaries. - * - * The input modules.xml file defines the modular structure of - * the JDK as described in JEP 200: The Modular JDK - * (http://openjdk.java.net/jeps/200). - * - * $ java build.tools.module.GenJdepsModulesXml \ - * -o com/sun/tools/jdeps/resources/modules.xml \ - * -mp $OUTPUTDIR/modules \ - * top/modules.xml - */ -public final class GenJdepsModulesXml { - private final static String USAGE = - "Usage: GenJdepsModulesXml -o -mp build/modules path-to-modules-xml"; - - public static void main(String[] args) throws Exception { - Path outfile = null; - Path modulepath = null; - int i = 0; - while (i < args.length) { - String arg = args[i]; - if (arg.equals("-o")) { - outfile = Paths.get(args[i+1]); - i = i+2; - } else if (arg.equals("-mp")) { - modulepath = Paths.get(args[i+1]); - i = i+2; - if (!Files.isDirectory(modulepath)) { - System.err.println(modulepath + " is not a directory"); - System.exit(1); - } - } else { - break; - } - } - if (outfile == null || modulepath == null || i >= args.length) { - System.err.println(USAGE); - System.exit(-1); - } - - GenJdepsModulesXml gentool = new GenJdepsModulesXml(modulepath); - Set modules = new HashSet<>(); - for (; i < args.length; i++) { - Path p = Paths.get(args[i]); - modules.addAll(ModulesXmlReader.readModules(p) - .stream() - .map(gentool::buildIncludes) - .collect(Collectors.toSet())); - } - - Files.createDirectories(outfile.getParent()); - ModulesXmlWriter.writeModules(modules, outfile); - } - - final Path modulepath; - public GenJdepsModulesXml(Path modulepath) { - this.modulepath = modulepath; - } - - private static String packageName(Path p) { - return packageName(p.toString().replace(File.separatorChar, '/')); - } - private static String packageName(String name) { - int i = name.lastIndexOf('/'); - return (i > 0) ? name.substring(0, i).replace('/', '.') : ""; - } - - private static boolean includes(String name) { - return name.endsWith(".class"); - } - - public Module buildIncludes(Module module) { - Module.Builder mb = new Module.Builder(module); - Path mclasses = modulepath.resolve(module.name()); - try { - Files.find(mclasses, Integer.MAX_VALUE, (Path p, BasicFileAttributes attr) - -> includes(p.getFileName().toString())) - .map(p -> packageName(mclasses.relativize(p))) - .forEach(mb::include); - } catch (NoSuchFileException e) { - // aggregate module may not have class - } catch (IOException ioe) { - throw new UncheckedIOException(ioe); - } - return mb.build(); - } -} diff --git a/jdk/make/src/classes/build/tools/module/GenModuleInfoSource.java b/jdk/make/src/classes/build/tools/module/GenModuleInfoSource.java new file mode 100644 index 00000000000..255e5ef64ca --- /dev/null +++ b/jdk/make/src/classes/build/tools/module/GenModuleInfoSource.java @@ -0,0 +1,134 @@ +/* + * 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. + */ +package build.tools.module; + +import java.io.BufferedWriter; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; + +/** + * A build tool to extend the module-info.java in the source tree + * for platform-specific exports, uses, and provides and write + * to the specified output file. + * + * GenModulesList build tool currently generates the modules.list from + * the module-info.java from the source tree that will be used for + * the make target and dependences. + * + * The build currently invokes gensrc-$MODULE.gmk after modules.list + * is generated. Hence, platform-specific requires is not supported. + */ +public class GenModuleInfoSource { + private final static String USAGE = + "Usage: GenModuleInfoSource [option] -o \n" + + "Options are:\n" + + " -exports \n" + + " -exports /\n" + + " -uses \n" + + " -provides /\n"; + + public static void main(String... args) throws Exception { + Path outfile = null; + Path moduleInfoJava = null; + Map> options = new HashMap<>(); + + // validate input arguments + for (int i = 0; i < args.length; i++){ + String option = args[i]; + if (option.startsWith("-")) { + String arg = args[++i]; + if (option.equals("-exports") || + option.equals("-uses") || + option.equals("-provides")) { + options.computeIfAbsent(option, _k -> new HashSet<>()).add(arg); + } else if (option.equals("-o")) { + outfile = Paths.get(arg); + } else { + throw new IllegalArgumentException("invalid option: " + option); + } + } else if (moduleInfoJava != null) { + throw new IllegalArgumentException("more than one module-info.java"); + } else { + moduleInfoJava = Paths.get(option); + if (Files.notExists(moduleInfoJava)) { + throw new IllegalArgumentException(option + " not exist"); + } + } + } + + if (moduleInfoJava == null || outfile == null) { + System.err.println(USAGE); + System.exit(-1); + } + // read module-info.java + Module.Builder builder = ModuleInfoReader.builder(moduleInfoJava); + augment(builder, options); + + // generate new module-info.java + Module module = builder.build(); + Path parent = outfile.getParent(); + if (parent != null) + Files.createDirectories(parent); + + try (BufferedWriter writer = Files.newBufferedWriter(outfile)) { + writer.write(module.toString()); + } + } + + private static void augment(Module.Builder builder, Map> options) { + for (String opt : options.keySet()) { + if (opt.equals("-exports")) { + for (String arg : options.get(opt)) { + int index = arg.indexOf('/'); + if (index > 0) { + String pn = arg.substring(0, index); + String mn = arg.substring(index + 1, arg.length()); + builder.exportTo(pn, mn); + } else { + builder.export(arg); + } + } + } else if (opt.equals("-uses")) { + options.get(opt).stream() + .forEach(builder::use); + } else if (opt.equals("-provides")) { + for (String arg : options.get(opt)) { + int index = arg.indexOf('/'); + if (index <= 0) { + throw new IllegalArgumentException("invalid -provide argument: " + arg); + } + String service = arg.substring(0, index); + String impl = arg.substring(index + 1, arg.length()); + builder.provide(service, impl); + } + } + } + } +} diff --git a/jdk/make/src/classes/build/tools/module/GenModuleLoaderMap.java b/jdk/make/src/classes/build/tools/module/GenModuleLoaderMap.java new file mode 100644 index 00000000000..6719277fac1 --- /dev/null +++ b/jdk/make/src/classes/build/tools/module/GenModuleLoaderMap.java @@ -0,0 +1,116 @@ +/* + * 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. + */ + +package build.tools.module; + +import java.io.BufferedReader; +import java.io.BufferedWriter; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.io.PrintWriter; +import java.nio.charset.StandardCharsets; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.Arrays; +import java.util.Set; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +public class GenModuleLoaderMap { + private static final String USAGE = + "GenModuleLoaderMap -o -boot m1[,m2]* -platform m3[,m4]* "; + + public static void main(String... args) throws Exception { + // default set of boot modules and ext modules + Stream bootModules = Stream.empty(); + Stream platformModules = Stream.empty(); + Path outfile = null; + Path source = null; + for (int i=0; i < args.length; i++) { + String option = args[i]; + if (option.startsWith("-")) { + String arg = args[++i]; + if (option.equals("-boot")) { + String[] mns = arg.split(","); + bootModules = Stream.concat(bootModules, Arrays.stream(mns)); + } else if (option.equals("-platform")) { + String[] mns = arg.split(","); + platformModules = Stream.concat(platformModules, Arrays.stream(mns)); + } else if (option.equals("-o")) { + outfile = Paths.get(arg); + } else { + throw new IllegalArgumentException("invalid option: " + option); + } + } else { + source = Paths.get(option); + } + } + + if (outfile == null) { + throw new IllegalArgumentException("-o must be specified"); + } + if (Files.notExists(source)) { + throw new IllegalArgumentException(source + " not exist"); + } + + boolean needsQuotes = outfile.toString().contains(".java.tmp"); + + try (BufferedWriter bw = Files.newBufferedWriter(outfile, StandardCharsets.UTF_8); + PrintWriter writer = new PrintWriter(bw)) { + for (String line : Files.readAllLines(source)) { + if (line.contains("@@BOOT_MODULE_NAMES@@")) { + line = patch(line, "@@BOOT_MODULE_NAMES@@", bootModules, needsQuotes); + } else if (line.contains("@@PLATFORM_MODULE_NAMES@@")) { + line = patch(line, "@@PLATFORM_MODULE_NAMES@@", platformModules, needsQuotes); + } + writer.println(line); + } + } + } + + private static String patch(String s, String tag, Stream stream, boolean needsQuotes) { + String mns = null; + if (needsQuotes) { + mns = stream.sorted() + .collect(Collectors.joining("\",\n \"")); + } else { + mns = stream.sorted() + .collect(Collectors.joining("\n")); + } + return s.replace(tag, mns); + } + + /** + * Reads the contents of the given modules file. + */ + private static Set readModuleSet(String name) throws IOException { + try (InputStream is = GenModuleLoaderMap.class.getResourceAsStream(name); + BufferedReader reader = new BufferedReader(new InputStreamReader(is))) { + return reader.lines().collect(Collectors.toSet()); + } + } +} diff --git a/jdk/make/src/classes/build/tools/module/GenModulesList.java b/jdk/make/src/classes/build/tools/module/GenModulesList.java deleted file mode 100644 index 66a9f8c76ed..00000000000 --- a/jdk/make/src/classes/build/tools/module/GenModulesList.java +++ /dev/null @@ -1,158 +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. - */ - -package build.tools.module; - -import java.io.PrintWriter; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.*; -import java.util.stream.Collectors; - -/** - * $ java build.tools.module.GenModulesList \ - * -o modules.list \ - * top/modules.xml ... - */ -public final class GenModulesList { - private final static String USAGE = - "Usage: GenModulesList -o path-to-modules-xml"; - - private Set modules = new HashSet<>(); - private HashMap nameToModule = new HashMap<>(); - - public static void main(String[] args) throws Exception { - GenModulesList gen = new GenModulesList(); - gen.run(args); - } - - void run(String[] args) throws Exception { - Path outfile = null; - int i = 0; - while (i < args.length) { - String arg = args[i]; - if (arg.equals("-o")) { - outfile = Paths.get(args[i+1]); - i = i+2; - } else { - break; - } - } - if (outfile == null || i >= args.length) { - System.err.println(USAGE); - System.exit(-1); - } - - for (; i < args.length; i++) { - Path p = Paths.get(args[i]); - modules.addAll(ModulesXmlReader.readModules(p)); - } - - modules.stream() - .forEach(m -> nameToModule.put(m.name(), m)); - - Path parent = outfile.getParent(); - if (parent != null) - Files.createDirectories(parent); - - Iterable sortedModules = (new TopoSorter(modules)).result(); - try (PrintWriter writer = new PrintWriter(outfile.toFile())) { - for (Module m : sortedModules) { - if (isNotAggregator(m)) { - String deps = getModuleDependences(m).stream() - .filter(GenModulesList::isNotAggregator) - .map(Module::name) - .collect(Collectors.joining(" ")); - writer.format("%s: %s%n", m.name(), deps); - } - } - } - } - - private Module nameToModule(String name) { - return nameToModule.get(name); - } - - private Set getModuleDependences(Module m) { - return m.requires().stream() - .map(d -> d.name()) - .map(this::nameToModule) - .collect(Collectors.toSet()); - } - - static boolean isNotAggregator(Module m) { - return isNotAggregator(m.name()); - } - - static boolean isNotAggregator(String name) { - return AGGREGATORS.contains(name) ? false : true; - } - - static final List AGGREGATORS = Arrays.asList(new String[] { - "java.se", "java.compact1", "java.compact2", "java.compact3"}); - - class TopoSorter { - final Deque result = new LinkedList<>(); - final Deque nodes = new LinkedList<>(); - - TopoSorter(Collection nodes) { - nodes.stream() - .forEach(m -> this.nodes.add(m)); - - sort(); - } - - public Iterable result() { - return result; - } - - private void sort() { - Deque visited = new LinkedList<>(); - Deque done = new LinkedList<>(); - Module node; - while ((node = nodes.poll()) != null) { - if (!visited.contains(node)) { - visit(node, visited, done); - } - } - } - - private void visit(Module m, Deque visited, Deque done) { - if (visited.contains(m)) { - if (!done.contains(m)) { - throw new IllegalArgumentException("Cyclic detected: " + - m + " " + getModuleDependences(m)); - } - return; - } - visited.add(m); - getModuleDependences(m).stream() - .forEach(x -> visit(x, visited, done)); - done.add(m); - result.addLast(m); - } - } -} diff --git a/jdk/make/src/classes/build/tools/module/ImageBuilder.java b/jdk/make/src/classes/build/tools/module/ImageBuilder.java deleted file mode 100644 index b87d59306c2..00000000000 --- a/jdk/make/src/classes/build/tools/module/ImageBuilder.java +++ /dev/null @@ -1,499 +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. - */ - -package build.tools.module; - -import jdk.internal.jimage.Archive; - -import java.io.BufferedReader; -import java.io.File; -import java.io.IOException; -import java.io.InputStream; -import java.io.InputStreamReader; -import java.io.PrintWriter; -import java.nio.ByteOrder; -import java.nio.file.Files; -import java.nio.file.InvalidPathException; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.ArrayList; -import java.util.Collection; -import java.util.HashMap; -import java.util.HashSet; -import java.util.LinkedList; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.Set; -import java.util.stream.Collectors; -import jdk.internal.jimage.ImageFileCreator; - -/** - * A tool for building a runtime image. - * - * java build.tools.module.ImageBuilder --output top/modules.xml,... - * Possible options are: - * --cmds Location of native commands - * --configs Location of config files - * --help Print this usage message - * --classes Location of module classes files - * --libs Location of native libraries - * --mods Comma separated list of module names - * --output Location of the output path - * --endian Byte order of the target runtime; {little,big} - */ -class ImageBuilder { - static class BadArgs extends Exception { - private static final long serialVersionUID = 0L; - BadArgs(String format, Object... args) { - super(String.format(format, args)); - this.format = format; - this.args = args; - } - BadArgs showUsage(boolean b) { - showUsage = b; - return this; - } - final String format; - final Object[] args; - boolean showUsage; - } - - static class Option { - - interface Processing { - - void process(ImageBuilder task, String opt, String arg) throws BadArgs; - } - - final boolean hasArg; - final String[] aliases; - final String description; - final Processing processing; - - Option(boolean hasArg, String description, Processing processing, - String... aliases) { - this.hasArg = hasArg; - this.description = description; - this.processing = processing; - this.aliases = aliases; - } - boolean isHidden() { - return false; - } - boolean matches(String opt) { - for (String a : aliases) { - if (a.equals(opt)) { - return true; - } else if (opt.startsWith("--") && hasArg && opt.startsWith(a + "=")) { - return true; - } - } - return false; - } - boolean ignoreRest() { - return false; - } - void process(ImageBuilder task, String opt, String arg) throws BadArgs { - processing.process(task, opt, arg); - } - String description() { - return description; - } - } - - private static Path CWD = Paths.get(""); - - private static List splitPath(String arg, String separator) - throws BadArgs - { - List paths = new ArrayList<>(); - for (String p: arg.split(separator)) { - if (p.length() > 0) { - try { - Path path = CWD.resolve(p); - if (Files.notExists(path)) - throw new BadArgs("path not found: %s", path); - paths.add(path); - } catch (InvalidPathException x) { - throw new BadArgs("path not valid: %s", p); - } - } - } - return paths; - } - - static Option[] recognizedOptions = { - new Option(true, "Location of native commands", (task, opt, arg) -> { - task.options.cmds = splitPath(arg, File.pathSeparator); - }, "--cmds"), - new Option(true, "Location of config files", (task, opt, arg) -> { - task.options.configs = splitPath(arg, File.pathSeparator); - }, "--configs"), - new Option(false, "Print this usage message", (task, opt, arg) -> { - task.options.help = true; - }, "--help"), - new Option(true, "Location of module classes files", (task, opt, arg) -> { - task.options.classes = splitPath(arg, File.pathSeparator); - }, "--classes"), - new Option(true, "Location of native libraries", (task, opt, arg) -> { - task.options.libs = splitPath(arg, File.pathSeparator); - }, "--libs"), - new Option(true, "Comma separated list of module names", - (task, opt, arg) -> { - for (String mn : arg.split(",")) { - if (mn.isEmpty()) { - throw new BadArgs("Module not found", mn); - } - task.options.mods.add(mn); - } - }, "--mods"), - new Option(true, "Location of the output path", (task, opt, arg) -> { - Path path = Paths.get(arg); - task.options.output = path; - }, "--output"), - new Option(true, "Byte order of the target runtime; {little,big}", - (task, opt, arg) -> { - if (arg.equals("little")) { - task.options.endian = ByteOrder.LITTLE_ENDIAN; - } else if (arg.equals("big")) { - task.options.endian = ByteOrder.BIG_ENDIAN; - } else { - throw new BadArgs("Unknown byte order " + arg); - } - }, "--endian") - }; - - private final Options options = new Options(); - - private PrintWriter log; - void setLog(PrintWriter out) { - log = out; - } - - Set moduleGraph = new java.util.HashSet<>(); - - /** Module list files */ - private static final String BOOT_MODULES = "boot.modules"; - private static final String EXT_MODULES = "ext.modules"; - - /** - * Result codes. - */ - static final int EXIT_OK = 0, // Completed with no errors. - EXIT_ERROR = 1, // Completed but reported errors. - EXIT_CMDERR = 2, // Bad command-line arguments - EXIT_SYSERR = 3, // System error or resource exhaustion. - EXIT_ABNORMAL = 4; // terminated abnormally - - - static class Options { - boolean help; - List classes; - List cmds; - List configs; - List libs; - Set mods = new HashSet<>(); - Path output; - ByteOrder endian = ByteOrder.nativeOrder(); // default, if not specified - } - - public static void main(String[] args) throws Exception { - ImageBuilder builder = new ImageBuilder(); - int rc = builder.run(args); - System.exit(rc); - } - - int run(String[] args) { - if (log == null) - log = new PrintWriter(System.out); - - try { - handleOptions(args); - if (options.help) { - showHelp(); - return EXIT_OK; - } - - if (options.classes == null) - throw new BadArgs("--classes must be specified").showUsage(true); - - Path output = options.output; - if (output == null) - throw new BadArgs("--output must be specified").showUsage(true); - Files.createDirectories(output); - if (Files.list(output).findFirst().isPresent()) - throw new BadArgs("dir not empty", output); - - if (options.mods.isEmpty()) - throw new BadArgs("--mods must be specified").showUsage(true); - - if (moduleGraph.isEmpty()) - throw new BadArgs("modules.xml must be specified").showUsage(true); - - if (options.cmds == null || options.cmds.isEmpty()) - warning("--commands is not set"); - if (options.libs == null || options.libs.isEmpty()) - warning("--libs is not set"); - //if (options.configs == null || options.configs.isEmpty()) - // warning("--configs is not set"); - - // additional option combination validation - - boolean ok = run(); - return ok ? EXIT_OK : EXIT_ERROR; - } catch (BadArgs e) { - reportError(e.format, e.args); - if (e.showUsage) - log.println(USAGE_SUMMARY); - return EXIT_CMDERR; - } catch (Exception x) { - x.printStackTrace(); - return EXIT_ABNORMAL; - } finally { - log.flush(); - } - } - - private boolean run() throws IOException { - createImage(); - return true; - } - - class SimpleResolver { - private final Set initialMods; - private final Map nameToModule = new HashMap<>(); - - SimpleResolver(Set mods, Set graph) { - graph.stream() - .forEach(m -> nameToModule.put(m.name(), m)); - initialMods = mods.stream() - .map(this::nameToModule) - .collect(Collectors.toSet()); - } - - /** Returns the transitive closure, in topological order */ - List resolve() { - List result = new LinkedList<>(); - Set visited = new HashSet<>(); - Set done = new HashSet<>(); - for (Module m : initialMods) { - if (!visited.contains(m)) - visit(m, visited, result, done); - } - return result.stream() - .map(m -> m.name()) - .collect(Collectors.toList()); - } - - private void visit(Module m, Set visited, - List result, Set done) { - if (visited.contains(m)) { - if (!done.contains(m)) - throw new IllegalArgumentException("Cyclic detected: " + - m + " " + getModuleDependences(m)); - return; - } - visited.add(m); - getModuleDependences(m).stream() - .forEach(d -> visit(d, visited, result, done)); - done.add(m); - result.add(m); - } - - private Module nameToModule(String name) { - Module m = nameToModule.get(name); - if (m == null) - throw new RuntimeException("No module definition for " + name); - return m; - } - - private Set getModuleDependences(Module m) { - return m.requires().stream() - .map(d -> d.name()) - .map(this::nameToModule) - .collect(Collectors.toSet()); - } - } - - private List resolve(Set mods ) { - return (new SimpleResolver(mods, moduleGraph)).resolve(); - } - - private void createImage() throws IOException { - Collection modules = resolve(options.mods); - log.print(modules.stream().collect(Collectors.joining(" "))); - ImageFileHelper imageHelper = new ImageFileHelper(modules); - imageHelper.createModularImage(options.output); - - // jspawnhelper, might be in lib or lib/ARCH - Path jspawnhelper = Paths.get("jspawnhelper"); - Path lib = options.output.resolve("lib"); - Optional helper = Files.walk(lib, 2) - .filter(f -> f.getFileName().equals(jspawnhelper)) - .findFirst(); - if (helper.isPresent()) - helper.get().toFile().setExecutable(true, false); - } - - private class ImageFileHelper { - final Collection modules; - final Set bootModules; - final Set extModules; - final Set appModules; - - ImageFileHelper(Collection modules) throws IOException { - this.modules = modules; - this.bootModules = modulesFor(BOOT_MODULES).stream() - .filter(modules::contains) - .collect(Collectors.toSet()); - this.extModules = modulesFor(EXT_MODULES).stream() - .filter(modules::contains) - .collect(Collectors.toSet()); - this.appModules = modules.stream() - .filter(m -> m.length() != 0 && - !bootModules.contains(m) && - !extModules.contains(m)) - .collect(Collectors.toSet()); - } - - void createModularImage(Path output) throws IOException { - Set bootArchives = bootModules.stream() - .map(this::toModuleArchive) - .collect(Collectors.toSet()); - Set extArchives = extModules.stream() - .map(this::toModuleArchive) - .collect(Collectors.toSet()); - Set appArchives = appModules.stream() - .map(this::toModuleArchive) - .collect(Collectors.toSet()); - ImageFileCreator.create(output, "bootmodules", bootArchives, options.endian); - ImageFileCreator.create(output, "extmodules", extArchives, options.endian); - ImageFileCreator.create(output, "appmodules", appArchives, options.endian); - } - - ModuleArchive toModuleArchive(String mn) { - return new ModuleArchive(mn, - moduleToPath(mn, options.classes, false/*true*/), - moduleToPath(mn, options.cmds, false), - moduleToPath(mn, options.libs, false), - moduleToPath(mn, options.configs, false)); - } - - private Path moduleToPath(String name, List paths, boolean expect) { - Set foundPaths = new HashSet<>(); - if (paths != null) { - for (Path p : paths) { - Path rp = p.resolve(name); - if (Files.exists(rp)) - foundPaths.add(rp); - } - } - if (foundPaths.size() > 1) - throw new RuntimeException("Found more that one path for " + name); - if (expect && foundPaths.size() != 1) - throw new RuntimeException("Expected to find classes path for " + name); - return foundPaths.size() == 0 ? null : foundPaths.iterator().next(); - } - - private List modulesFor(String name) throws IOException { - try (InputStream is = ImageBuilder.class.getResourceAsStream(name); - BufferedReader reader = new BufferedReader(new InputStreamReader(is))) { - return reader.lines().collect(Collectors.toList()); - } - } - } - - public void handleOptions(String[] args) throws BadArgs { - // process options - for (int i=0; i < args.length; i++) { - if (args[i].charAt(0) == '-') { - String name = args[i]; - Option option = getOption(name); - String param = null; - if (option.hasArg) { - if (name.startsWith("--") && name.indexOf('=') > 0) { - param = name.substring(name.indexOf('=') + 1, name.length()); - } else if (i + 1 < args.length) { - param = args[++i]; - } - if (param == null || param.isEmpty() || param.charAt(0) == '-') { - throw new BadArgs("Missing arg for %n", name).showUsage(true); - } - } - option.process(this, name, param); - if (option.ignoreRest()) { - i = args.length; - } - } else { - // process rest of the input arguments - Path p = Paths.get(args[i]); - try { - moduleGraph.addAll(ModulesXmlReader.readModules(p) - .stream() - .collect(Collectors.toSet())); - } catch (Exception e) { - throw new RuntimeException(e); - } - } - } - } - - private Option getOption(String name) throws BadArgs { - for (Option o : recognizedOptions) { - if (o.matches(name)) { - return o; - } - } - throw new BadArgs("Unknown option %s", name).showUsage(true); - } - - private void reportError(String format, Object... args) { - log.format("Error: " + format + "%n", args); - } - - private void warning(String format, Object... args) { - log.format("Warning: " + format + "%n", args); - } - - private static final String USAGE = - "ImageBuilder --output path-to-modules-xml\n"; - - private static final String USAGE_SUMMARY = - USAGE + "Use --help for a list of possible options."; - - private void showHelp() { - log.format(USAGE); - log.format("Possible options are:%n"); - for (Option o : recognizedOptions) { - String name = o.aliases[0].substring(1); // there must always be at least one name - name = name.charAt(0) == '-' ? name.substring(1) : name; - if (o.isHidden() || name.equals("h")) - continue; - - log.format(" --%s\t\t\t%s%n", name, o.description()); - } - } -} diff --git a/jdk/make/src/classes/build/tools/module/Module.java b/jdk/make/src/classes/build/tools/module/Module.java index 8264f3c7c76..e553d92ace8 100644 --- a/jdk/make/src/classes/build/tools/module/Module.java +++ b/jdk/make/src/classes/build/tools/module/Module.java @@ -25,10 +25,17 @@ package build.tools.module; -import java.util.*; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Objects; +import java.util.Set; +import java.util.stream.Collectors; +import java.util.stream.Stream; public class Module { - static class Dependence { + public static class Dependence implements Comparable { final String name; final boolean reexport; Dependence(String name) { @@ -43,6 +50,10 @@ public class Module { return name; } + public boolean reexport(){ + return reexport; + } + @Override public int hashCode() { int hash = 5; @@ -55,20 +66,35 @@ public class Module { Dependence d = (Dependence)o; return this.name.equals(d.name) && this.reexport == d.reexport; } + + @Override + public int compareTo(Dependence o) { + int rc = this.name.compareTo(o.name); + return rc != 0 ? rc : Boolean.compare(this.reexport, o.reexport); + } + + @Override + public String toString() { + return String.format("requires %s%s;", + reexport ? "public " : "", name); + } } private final String moduleName; private final Set requires; private final Map> exports; - private final Set packages; + private final Set uses; + private final Map> provides; private Module(String name, - Set requires, - Map> exports, - Set packages) { + Set requires, + Map> exports, + Set uses, + Map> provides) { this.moduleName = name; this.requires = Collections.unmodifiableSet(requires); this.exports = Collections.unmodifiableMap(exports); - this.packages = Collections.unmodifiableSet(packages); + this.uses = Collections.unmodifiableSet(uses); + this.provides = Collections.unmodifiableMap(provides); } public String name() { @@ -83,8 +109,12 @@ public class Module { return exports; } - public Set packages() { - return packages; + public Set uses() { + return uses; + } + + public Map> provides() { + return provides; } @Override @@ -95,8 +125,7 @@ public class Module { Module that = (Module) ob; return (moduleName.equals(that.moduleName) && requires.equals(that.requires) - && exports.equals(that.exports) - && packages.equals(that.packages)); + && exports.equals(that.exports)); } @Override @@ -104,43 +133,55 @@ public class Module { int hc = moduleName.hashCode(); hc = hc * 43 + requires.hashCode(); hc = hc * 43 + exports.hashCode(); - hc = hc * 43 + packages.hashCode(); return hc; } @Override public String toString() { StringBuilder sb = new StringBuilder(); - sb.append("module ").append(moduleName).append(" {").append("\n"); - requires.stream().sorted().forEach(d -> - sb.append(String.format(" requires %s%s%n", d.reexport ? "public " : "", d.name))); - exports.entrySet().stream().filter(e -> e.getValue().isEmpty()) + sb.append(String.format("module %s {%n", moduleName)); + requires.stream() + .sorted() + .map(d -> String.format(" requires %s%s;%n", d.reexport ? "public " : "", d.name)) + .forEach(sb::append); + exports.entrySet().stream() + .filter(e -> e.getValue().isEmpty()) .sorted(Map.Entry.comparingByKey()) - .forEach(e -> sb.append(String.format(" exports %s%n", e.getKey()))); - exports.entrySet().stream().filter(e -> !e.getValue().isEmpty()) + .map(e -> String.format(" exports %s;%n", e.getKey())) + .forEach(sb::append); + exports.entrySet().stream() + .filter(e -> !e.getValue().isEmpty()) .sorted(Map.Entry.comparingByKey()) - .forEach(e -> sb.append(String.format(" exports %s to %s%n", e.getKey(), e.getValue()))); - packages.stream().sorted().forEach(pn -> sb.append(String.format(" includes %s%n", pn))); - sb.append("}"); + .map(e -> String.format(" exports %s to%n%s;%n", e.getKey(), + e.getValue().stream().sorted() + .map(mn -> String.format(" %s", mn)) + .collect(Collectors.joining(",\n")))) + .forEach(sb::append); + uses.stream().sorted() + .map(s -> String.format(" uses %s;%n", s)) + .forEach(sb::append); + provides.entrySet().stream() + .sorted(Map.Entry.comparingByKey()) + .flatMap(e -> e.getValue().stream().sorted() + .map(impl -> String.format(" provides %s with %s;%n", e.getKey(), impl))) + .forEach(sb::append); + sb.append("}").append("\n"); return sb.toString(); } + /** + * Module Builder + */ static class Builder { private String name; - private final Set requires = new HashSet<>(); - private final Map> exports = new HashMap<>(); - private final Set packages = new HashSet<>(); + final Set requires = new HashSet<>(); + final Map> exports = new HashMap<>(); + final Set uses = new HashSet<>(); + final Map> provides = new HashMap<>(); public Builder() { } - public Builder(Module module) { - name = module.name(); - requires.addAll(module.requires()); - exports.putAll(module.exports()); - packages.addAll(module.packages()); - } - public Builder name(String n) { name = n; return this; @@ -151,28 +192,89 @@ public class Module { return this; } - public Builder include(String p) { - packages.add(p); - return this; + public Builder export(String p) { + Objects.requireNonNull(p); + if (exports.containsKey(p)) { + throw new RuntimeException(name + " already exports " + p + + " " + exports.get(p)); + } + return exportTo(p, Collections.emptySet()); } - public Builder export(String p) { - return exportTo(p, Collections.emptySet()); + public Builder exportTo(String p, String mn) { + Objects.requireNonNull(p); + Objects.requireNonNull(mn); + Set ms = exports.get(p); + if (ms != null && ms.isEmpty()) { + throw new RuntimeException(name + " already has unqualified exports " + p); + } + exports.computeIfAbsent(p, _k -> new HashSet<>()).add(mn); + return this; } public Builder exportTo(String p, Set ms) { Objects.requireNonNull(p); Objects.requireNonNull(ms); if (exports.containsKey(p)) { - throw new RuntimeException(name + " already exports " + p); + throw new RuntimeException(name + " already exports " + p + + " " + exports.get(p)); } exports.put(p, new HashSet<>(ms)); return this; } + public Builder use(String cn) { + uses.add(cn); + return this; + } + + public Builder provide(String s, String impl) { + provides.computeIfAbsent(s, _k -> new HashSet<>()).add(impl); + return this; + } + + public Builder merge(Module m1, Module m2) { + if (!m1.name().equals(m2.name())) { + throw new IllegalArgumentException(m1.name() + " != " + m2.name()); + } + name = m1.name(); + // ## reexports + requires.addAll(m1.requires()); + requires.addAll(m2.requires()); + Stream.concat(m1.exports().keySet().stream(), m2.exports().keySet().stream()) + .distinct() + .forEach(pn -> { + Set s1 = m2.exports().get(pn); + Set s2 = m2.exports().get(pn); + if (s1 == null || s2 == null) { + exportTo(pn, s1 != null ? s1 : s2); + } else if (s1.isEmpty() || s2.isEmpty()) { + // unqualified exports + export(pn); + } else { + exportTo(pn, Stream.concat(s1.stream(), s2.stream()) + .collect(Collectors.toSet())); + } + }); + uses.addAll(m1.uses()); + uses.addAll(m2.uses()); + m1.provides().keySet().stream() + .forEach(s -> m1.provides().get(s).stream() + .forEach(impl -> provide(s, impl))); + m2.provides().keySet().stream() + .forEach(s -> m2.provides().get(s).stream() + .forEach(impl -> provide(s, impl))); + return this; + } + public Module build() { - Module m = new Module(name, requires, exports, packages); + Module m = new Module(name, requires, exports, uses, provides); return m; } + + @Override + public String toString() { + return name != null ? name : "Unknown"; + } } } diff --git a/jdk/make/src/classes/build/tools/module/ModuleArchive.java b/jdk/make/src/classes/build/tools/module/ModuleArchive.java deleted file mode 100644 index c4be7b504a9..00000000000 --- a/jdk/make/src/classes/build/tools/module/ModuleArchive.java +++ /dev/null @@ -1,184 +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. - */ - -package build.tools.module; - -import jdk.internal.jimage.Archive; - -import java.io.IOException; -import java.io.InputStream; -import java.io.UncheckedIOException; -import java.nio.file.Files; -import java.nio.file.Path; -import java.util.ArrayList; -import java.util.List; -import java.util.stream.Collectors; -import java.util.stream.Stream; -import jdk.internal.jimage.Archive.Entry.EntryType; - -/** - * An Archive backed by an exploded representation on disk. - */ -public class ModuleArchive implements Archive { - private final Path classes; - private final Path cmds; - private final Path libs; - private final Path configs; - private final String moduleName; - - private final List opened = new ArrayList<>(); - - public ModuleArchive(String moduleName, Path classes, Path cmds, - Path libs, Path configs) { - this.moduleName = moduleName; - this.classes = classes; - this.cmds = cmds; - this.libs = libs; - this.configs = configs; - } - - @Override - public String moduleName() { - return moduleName; - } - - @Override - public void open() throws IOException { - // NOOP - } - - @Override - public void close() throws IOException { - IOException e = null; - for (InputStream stream : opened) { - try { - stream.close(); - } catch (IOException ex) { - if (e == null) { - e = ex; - } else { - e.addSuppressed(ex); - } - } - } - if (e != null) { - throw e; - } - } - - @Override - public Stream entries() { - List entries = new ArrayList<>(); - try { - /* - * This code should be revisited to avoid buffering of the entries. - * 1) Do we really need sorting classes? This force buffering of entries. - * libs, cmds and configs are not sorted. - * 2) I/O streams should be concatenated instead of buffering into - * entries list. - * 3) Close I/O streams in a close handler. - */ - if (classes != null) { - try (Stream stream = Files.walk(classes)) { - entries.addAll(stream - .filter(p -> !Files.isDirectory(p) - && !classes.relativize(p).toString().startsWith("_the.") - && !classes.relativize(p).toString().endsWith(".bc") - && !classes.relativize(p).toString().equals("javac_state")) - .sorted() - .map(p -> toEntry(p, classes, EntryType.CLASS_OR_RESOURCE)) - .collect(Collectors.toList())); - } - } - if (cmds != null) { - try (Stream stream = Files.walk(cmds)) { - entries.addAll(stream - .filter(p -> !Files.isDirectory(p)) - .map(p -> toEntry(p, cmds, EntryType.NATIVE_CMD)) - .collect(Collectors.toList())); - } - } - if (libs != null) { - try (Stream stream = Files.walk(libs)) { - entries.addAll(stream - .filter(p -> !Files.isDirectory(p)) - .map(p -> toEntry(p, libs, EntryType.NATIVE_LIB)) - .collect(Collectors.toList())); - } - } - if (configs != null) { - try (Stream stream = Files.walk(configs)) { - entries.addAll(stream - .filter(p -> !Files.isDirectory(p)) - .map(p -> toEntry(p, configs, EntryType.CONFIG)) - .collect(Collectors.toList())); - } - } - } catch (IOException ioe) { - throw new UncheckedIOException(ioe); - } - return entries.stream(); - } - - private class FileEntry extends Entry { - private final boolean isDirectory; - private final long size; - private final Path entryPath; - FileEntry(Path entryPath, String path, EntryType type, - boolean isDirectory, long size) { - super(ModuleArchive.this, path, path, type); - this.entryPath = entryPath; - this.isDirectory = isDirectory; - this.size = size; - } - - public boolean isDirectory() { - return isDirectory; - } - - @Override - public long size() { - return size; - } - - @Override - public InputStream stream() throws IOException { - InputStream stream = Files.newInputStream(entryPath); - opened.add(stream); - return stream; - } - } - - private Entry toEntry(Path entryPath, Path basePath, EntryType section) { - try { - String path = basePath.relativize(entryPath).toString().replace('\\', '/'); - return new FileEntry(entryPath, path, section, - false, Files.size(entryPath)); - } catch (IOException e) { - throw new UncheckedIOException(e); - } - } -} - diff --git a/jdk/make/src/classes/build/tools/module/ModuleInfoReader.java b/jdk/make/src/classes/build/tools/module/ModuleInfoReader.java new file mode 100644 index 00000000000..04ff42b39ba --- /dev/null +++ b/jdk/make/src/classes/build/tools/module/ModuleInfoReader.java @@ -0,0 +1,357 @@ +/* + * 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. + */ +package build.tools.module; + +import java.io.IOException; +import java.nio.file.Files; +import java.nio.file.Path; +import java.util.List; +import java.util.function.Supplier; +import java.util.regex.Pattern; +import java.util.stream.Stream; + +import build.tools.module.Module.Builder; + +/** + * Source reader of module-info.java + */ +public class ModuleInfoReader { + private final Path sourcefile; + private final Builder builder; + private ModuleInfoReader(Path file) { + this.sourcefile = file; + this.builder = new Builder(); + } + + public static Builder builder(Path file) throws IOException { + ModuleInfoReader reader = new ModuleInfoReader(file); + reader.readFile(); + return reader.builder; + } + + /** + * Reads the source file. + */ + void readFile() throws IOException { + List lines = Files.readAllLines(sourcefile); + boolean done = false; + int lineNumber = 0; + boolean inBlockComment = false; + boolean inRequires = false; + boolean reexports = false; + boolean inProvides = false; + boolean inWith = false; + String serviceIntf = null; + String providerClass = null; + boolean inUses = false; + boolean inExports = false; + boolean inExportsTo = false; + String qualifiedExports = null; + Counter counter = new Counter(); + + for (String line : lines) { + lineNumber++; + if (inBlockComment) { + int c = line.indexOf("*/"); + if (c >= 0) { + line = line.substring(c + 2, line.length()); + inBlockComment = false; + } else { + // skip lines until end of comment block + continue; + } + } + inBlockComment = beginBlockComment(line); + + line = trimComment(line).trim(); + // ignore empty lines + if (line.length() == 0) { + continue; + } + String values; + if (inRequires || inExports | inUses | (inWith && providerClass == null)) { + values = line; + } else { + String[] s = line.split("\\s+"); + String keyword = s[0].trim(); + int nextIndex = keyword.length(); + switch (keyword) { + case "module": + if (s.length != 3 || !s[2].trim().equals("{")) { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + ", is malformed"); + } + builder.name(s[1].trim()); + continue; // next line + case "requires": + inRequires = true; + counter.numRequires++; + if (s.length >= 2) { + String ss = s[1].trim(); + if (ss.equals("public")) { + nextIndex = line.indexOf(ss) + ss.length(); + reexports = true; + } + } + break; + case "exports": + inExports = true; + inExportsTo = false; + counter.numExports++; + qualifiedExports = null; + if (s.length >= 3) { + qualifiedExports = s[1].trim(); + nextIndex = line.indexOf(qualifiedExports, nextIndex) + + qualifiedExports.length(); + if (s[2].trim().equals("to")) { + inExportsTo = true; + nextIndex = line.indexOf("to", nextIndex) + "to".length(); + } else { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + ", is malformed: " + s[2]); + } + } + break; + case "to": + if (!inExports || qualifiedExports == null) { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + ", is malformed"); + } + inExportsTo = true; + break; + case "uses": + inUses = true; + counter.numUses++; + break; + case "provides": + inProvides = true; + inWith = false; + counter.numProvides++; + serviceIntf = null; + providerClass = null; + if (s.length >= 2) { + serviceIntf = s[1].trim(); + nextIndex = line.indexOf(serviceIntf) + serviceIntf.length(); + } + if (s.length >= 3) { + if (s[2].trim().equals("with")) { + inWith = true; + nextIndex = line.indexOf("with") + "with".length(); + } else { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + ", is malformed: " + s[2]); + } + } + break; + case "with": + if (!inProvides || serviceIntf == null) { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + ", is malformed"); + } + inWith = true; + nextIndex = line.indexOf("with") + "with".length(); + break; + case "}": + counter.validate(builder); + done = true; + continue; // next line + default: + throw new RuntimeException(sourcefile + ", \"" + + keyword + "\" on line " + + lineNumber + ", is not recognized"); + } + values = line.substring(nextIndex, line.length()).trim(); + } + + int len = values.length(); + if (len == 0) { + continue; // next line + } + char lastchar = values.charAt(len - 1); + if (lastchar != ',' && lastchar != ';') { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + ", is malformed:" + + " ',' or ';' is missing."); + } + + values = values.substring(0, len - 1).trim(); + // parse the values specified for a keyword specified + for (String s : values.split(",")) { + s = s.trim(); + if (s.length() > 0) { + if (inRequires) { + if (builder.requires.contains(s)) { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + " duplicated requires: \"" + s + "\""); + } + builder.require(s, reexports); + } else if (inExports) { + if (!inExportsTo && qualifiedExports == null) { + builder.export(s); + } else { + builder.exportTo(qualifiedExports, s); + } + } else if (inUses) { + builder.use(s); + } else if (inProvides) { + if (!inWith) { + serviceIntf = s; + } else { + providerClass = s; + builder.provide(serviceIntf, providerClass); + } + } + } + } + if (lastchar == ';') { + inRequires = false; + reexports = false; + inExports = false; + inExportsTo = false; + inProvides = false; + inWith = false; + inUses = false; + } + } + + if (inBlockComment) { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + ", missing \"*/\" to end a block comment"); + } + if (!done) { + throw new RuntimeException(sourcefile + ", line " + + lineNumber + ", missing \"}\" to end module definition" + + " for \"" + builder + "\""); + } + return; + } + + // the naming convention for the module names without dashes + private static final Pattern CLASS_NAME_PATTERN = Pattern.compile("[\\w\\.\\*_$/]+"); + private static boolean beginBlockComment(String line) { + int pos = 0; + while (pos >= 0 && pos < line.length()) { + int c = line.indexOf("/*", pos); + if (c < 0) { + return false; + } + + if (c > 0 && !Character.isWhitespace(line.charAt(c - 1))) { + return false; + } + + int c1 = line.indexOf("//", pos); + if (c1 >= 0 && c1 < c) { + return false; + } + + int c2 = line.indexOf("*/", c + 2); + if (c2 < 0) { + return true; + } + pos = c + 2; + } + return false; + } + private static String trimComment(String line) { + StringBuilder sb = new StringBuilder(); + + int pos = 0; + while (pos >= 0 && pos < line.length()) { + int c1 = line.indexOf("//", pos); + if (c1 > 0 && !Character.isWhitespace(line.charAt(c1 - 1))) { + // not a comment + c1 = -1; + } + + int c2 = line.indexOf("/*", pos); + if (c2 > 0 && !Character.isWhitespace(line.charAt(c2 - 1))) { + // not a comment + c2 = -1; + } + + int c = line.length(); + int n = line.length(); + if (c1 >= 0 || c2 >= 0) { + if (c1 >= 0) { + c = c1; + } + if (c2 >= 0 && c2 < c) { + c = c2; + } + int c3 = line.indexOf("*/", c2 + 2); + if (c == c2 && c3 > c2) { + n = c3 + 2; + } + } + if (c > 0) { + if (sb.length() > 0) { + // add a whitespace if multiple comments on one line + sb.append(" "); + } + sb.append(line.substring(pos, c)); + } + pos = n; + } + return sb.toString(); + } + + + static class Counter { + int numRequires; + int numExports; + int numUses; + int numProvides; + + void validate(Builder builder) { + assertEquals("requires", numRequires, builder.requires.size(), + () -> builder.requires.stream() + .map(Module.Dependence::toString)); + assertEquals("exports", numExports, builder.exports.size(), + () -> builder.exports.entrySet().stream() + .map(e -> "exports " + e.getKey() + " to " + e.getValue())); + assertEquals("uses", numUses, builder.uses.size(), + () -> builder.uses.stream()); + assertEquals("provides", numProvides, + (int)builder.provides.values().stream() + .flatMap(s -> s.stream()) + .count(), + () -> builder.provides.entrySet().stream() + .map(e -> "provides " + e.getKey() + " with " + e.getValue())); + } + + private static void assertEquals(String msg, int expected, int got, + Supplier> supplier) { + if (expected != got){ + System.err.println("ERROR: mismatched " + msg + + " expected: " + expected + " got: " + got ); + supplier.get().sorted() + .forEach(System.err::println); + throw new AssertionError("mismatched " + msg + + " expected: " + expected + " got: " + got + " "); + } + } + } +} diff --git a/jdk/make/src/classes/build/tools/module/boot.modules b/jdk/make/src/classes/build/tools/module/boot.modules deleted file mode 100644 index 0bf1c55fc5d..00000000000 --- a/jdk/make/src/classes/build/tools/module/boot.modules +++ /dev/null @@ -1,36 +0,0 @@ -java.base -java.compiler -java.datatransfer -java.desktop -java.httpclient -java.instrument -java.logging -java.management -java.naming -java.prefs -java.rmi -java.scripting -java.security.jgss -java.security.sasl -java.smartcardio -java.sql -java.sql.rowset -java.xml -java.xml.crypto -jdk.charsets -jdk.deploy -jdk.httpserver -jdk.jfr -jdk.jsobject -jdk.net -jdk.vm.cds -jdk.vm.ci -jdk.management -jdk.management.cmm -jdk.management.jfr -jdk.management.resource -jdk.naming.rmi -jdk.sctp -jdk.security.auth -jdk.security.jgss -jdk.snmp diff --git a/jdk/make/src/classes/build/tools/module/ext.modules b/jdk/make/src/classes/build/tools/module/ext.modules deleted file mode 100644 index d266d40847f..00000000000 --- a/jdk/make/src/classes/build/tools/module/ext.modules +++ /dev/null @@ -1,17 +0,0 @@ -java.activation -java.annotations.common -java.corba -java.transaction -java.xml.bind -java.xml.ws -jdk.accessibility -jdk.crypto.ec -jdk.crypto.mscapi -jdk.crypto.pkcs11 -jdk.crypto.ucrypto -jdk.dynalink -jdk.localedata -jdk.naming.dns -jdk.scripting.nashorn -jdk.xml.dom -jdk.zipfs diff --git a/jdk/src/demo/share/java2d/J2DBench/src/j2dbench/ResultSet.java b/jdk/src/demo/share/java2d/J2DBench/src/j2dbench/ResultSet.java index f84f8835901..018a9d579e5 100644 --- a/jdk/src/demo/share/java2d/J2DBench/src/j2dbench/ResultSet.java +++ b/jdk/src/demo/share/java2d/J2DBench/src/j2dbench/ResultSet.java @@ -84,7 +84,6 @@ public class ResultSet { "java.util.prefs.PreferencesFactory", "sun.java2d.fontpath", "sun.boot.library.path", - "sun.boot.class.path", }; /* diff --git a/jdk/src/java.base/macosx/classes/module-info.java.extra b/jdk/src/java.base/macosx/classes/module-info.java.extra new file mode 100644 index 00000000000..6815c121dbb --- /dev/null +++ b/jdk/src/java.base/macosx/classes/module-info.java.extra @@ -0,0 +1,27 @@ +/* + * 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. + */ + +exports jdk.internal.loader to java.desktop; +provides java.security.Provider with apple.security.AppleProvider; diff --git a/jdk/src/java.base/share/classes/com/sun/java/util/jar/pack/intrinsic.properties b/jdk/src/java.base/share/classes/com/sun/java/util/jar/pack/intrinsic.properties index f29fee3216c..be1687076cc 100644 --- a/jdk/src/java.base/share/classes/com/sun/java/util/jar/pack/intrinsic.properties +++ b/jdk/src/java.base/share/classes/com/sun/java/util/jar/pack/intrinsic.properties @@ -6,7 +6,7 @@ # but may be overridden here, if necessary. # The makefile for this directory must copy this file -# into the target class hierarchy so it will get into rt.jar. +# into the target class hierarchy so it will get into runtime image. # JCOV attributes pack.code.attribute.CoverageTable = NH[PHHII] @@ -14,6 +14,15 @@ pack.code.attribute.CharacterRangeTable = NH[PHPOHIIH] pack.class.attribute.SourceID = RUH pack.class.attribute.CompilationID = RUH +# Module attributes, supported by the tool and not JSR-200 +pack.class.attribute.Module = NH[RUHFH]NH[RUHNH[RUH]]NH[RCH]NH[RCHRCH] +pack.class.attribute.ConcealedPackages = NH[RUH] +pack.class.attribute.Version = RUH +pack.class.attribute.MainClass = RUH +pack.class.attribute.TargetPlatform = RUHRUHRUH +pack.class.attribute.Hashes = RUHNH[RUHRUH] + + # Note: Zero-length ("marker") attributes do not need to be specified here. # They are automatically defined to have an empty layout. #pack.class.attribute.Deprecated = diff --git a/jdk/src/java.base/share/classes/java/io/ObjectInputStream.java b/jdk/src/java.base/share/classes/java/io/ObjectInputStream.java index ee25d91c6cd..73741eb4bcb 100644 --- a/jdk/src/java.base/share/classes/java/io/ObjectInputStream.java +++ b/jdk/src/java.base/share/classes/java/io/ObjectInputStream.java @@ -713,9 +713,11 @@ public class ObjectInputStream classObjs[i] = cl; } try { - return Proxy.getProxyClass( + @SuppressWarnings("deprecation") + Class proxyClass = Proxy.getProxyClass( hasNonPublicInterface ? nonPublicLoader : latestLoader, classObjs); + return proxyClass; } catch (IllegalArgumentException e) { throw new ClassNotFoundException(null, e); } diff --git a/jdk/src/java.base/share/classes/java/lang/Class.java b/jdk/src/java.base/share/classes/java/lang/Class.java index af75faf371a..fa6971dabfa 100644 --- a/jdk/src/java.base/share/classes/java/lang/Class.java +++ b/jdk/src/java.base/share/classes/java/lang/Class.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1994, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1994, 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,6 +25,8 @@ package java.lang; +import java.lang.annotation.Annotation; +import java.lang.module.ModuleReader; import java.lang.reflect.AnnotatedElement; import java.lang.reflect.Array; import java.lang.reflect.GenericArrayType; @@ -33,15 +35,19 @@ import java.lang.reflect.Member; import java.lang.reflect.Field; import java.lang.reflect.Executable; import java.lang.reflect.Method; +import java.lang.reflect.Module; import java.lang.reflect.Constructor; import java.lang.reflect.Modifier; import java.lang.reflect.Type; import java.lang.reflect.TypeVariable; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.AnnotatedType; +import java.lang.reflect.Proxy; import java.lang.ref.SoftReference; +import java.io.IOException; import java.io.InputStream; import java.io.ObjectStreamField; +import java.net.URL; import java.security.AccessController; import java.security.PrivilegedAction; import java.util.ArrayList; @@ -55,9 +61,11 @@ import java.util.Map; import java.util.HashMap; import java.util.Objects; import java.util.StringJoiner; +import jdk.internal.HotSpotIntrinsicCandidate; +import jdk.internal.loader.BootLoader; +import jdk.internal.loader.BuiltinClassLoader; import jdk.internal.misc.Unsafe; import jdk.internal.misc.VM; -import jdk.internal.HotSpotIntrinsicCandidate; import sun.reflect.CallerSensitive; import sun.reflect.ConstantPool; import sun.reflect.Reflection; @@ -69,8 +77,6 @@ import sun.reflect.generics.repository.MethodRepository; import sun.reflect.generics.repository.ConstructorRepository; import sun.reflect.generics.scope.ClassScope; import sun.security.util.SecurityConstants; -import java.lang.annotation.Annotation; -import java.lang.reflect.Proxy; import sun.reflect.annotation.*; import sun.reflect.misc.ReflectUtil; @@ -378,6 +384,86 @@ public final class Class implements java.io.Serializable, Class caller) throws ClassNotFoundException; + + /** + * Returns the {@code Class} with the given + * binary name in the given module. + * + *

This method attempts to locate, load, and link the class or interface. + * It does not run the class initializer. If the class is not found, this + * method returns {@code null}.

+ * + *

If the class loader of the given module defines other modules and + * the given name is a class defined in a different module, this method + * returns {@code null} after the class is loaded.

+ * + *

This method does not check whether the requested class is + * accessible to its caller.

+ * + * @apiNote + * This method returns {@code null} on failure rather than + * throwing a {@link ClassNotFoundException}, as is done by + * the {@link #forName(String, boolean, ClassLoader)} method. + * The security check is a stack-based permission check if the caller + * loads a class in another module. + * + * @param module A module + * @param name The binary name + * of the class + * @return {@code Class} object of the given name defined in the given module; + * {@code null} if not found. + * + * @throws NullPointerException if the given module or name is {@code null} + * + * @throws LinkageError if the linkage fails + * + * @throws SecurityException + *
    + *
  • if the caller is not the specified module and + * {@code RuntimePermission("getClassLoader")} permission is denied; or
  • + *
  • access to the module content is denied. For example, + * permission check will be performed when a class loader calls + * {@link ModuleReader#open(String)} to read the bytes of a class file + * in a module.
  • + *
+ * + * @since 9 + */ + @CallerSensitive + public static Class forName(Module module, String name) { + Objects.requireNonNull(module); + Objects.requireNonNull(name); + + Class caller = Reflection.getCallerClass(); + if (caller != null && caller.getModule() != module) { + // if caller is null, Class.forName is the last java frame on the stack. + // java.base has all permissions + SecurityManager sm = System.getSecurityManager(); + if (sm != null) { + sm.checkPermission(SecurityConstants.GET_CLASSLOADER_PERMISSION); + } + } + + PrivilegedAction pa = module::getClassLoader; + ClassLoader cl = AccessController.doPrivileged(pa); + if (module.isNamed() && cl != null) { + return cl.loadLocalClass(module, name); + } + + final Class c; + if (cl != null) { + c = cl.loadLocalClass(name); + } else { + c = BootLoader.loadClassOrNull(name); + } + + if (c != null && c.getModule() == module) { + return c; + } else { + return null; + } + } + /** * Creates a new instance of the class represented by this {@code Class} * object. The class is instantiated as if by a {@code new} @@ -453,13 +539,11 @@ public final class Class implements java.io.Serializable, } Constructor tmpConstructor = cachedConstructor; // Security check (same as in java.lang.reflect.Constructor) - int modifiers = tmpConstructor.getModifiers(); - if (!Reflection.quickCheckMemberAccess(this, modifiers)) { - Class caller = Reflection.getCallerClass(); - if (newInstanceCallerCache != caller) { - Reflection.ensureMemberAccess(caller, this, null, modifiers); - newInstanceCallerCache = caller; - } + Class caller = Reflection.getCallerClass(); + if (newInstanceCallerCache != caller) { + int modifiers = tmpConstructor.getModifiers(); + Reflection.ensureMemberAccess(caller, this, null, modifiers); + newInstanceCallerCache = caller; } // Run constructor try { @@ -717,6 +801,29 @@ public final class Class implements java.io.Serializable, // Package-private to allow ClassLoader access ClassLoader getClassLoader0() { return classLoader; } + /** + * Returns the module that this class or interface is a member of. + * + * If this class represents an array type then this method returns the + * {@code Module} for the element type. If this class represents a + * primitive type or void, then the {@code Module} object for the + * {@code java.base} module is returned. + * + * If this class is in an unnamed module then the {@link + * ClassLoader#getUnnamedModule() unnamed} {@code Module} of the class + * loader for this class is returned. + * + * @return the module that this class or interface is a member of + * + * @since 9 + */ + public Module getModule() { + return module; + } + + // set by VM + private transient Module module; + // Initialized in JVM not by private constructor // This field is filtered from reflection access, i.e. getDeclaredField // will throw NoSuchFieldException @@ -808,24 +915,60 @@ public final class Class implements java.io.Serializable, } /** - * Gets the package for this class. The class loader of this class is used - * to find the package. If the class was loaded by the bootstrap class - * loader the set of packages loaded from CLASSPATH is searched to find the - * package of the class. Null is returned if no package object was created - * by the class loader of this class. + * Gets the package of this class. * - *

Packages have attributes for versions and specifications only if the - * information was defined in the manifests that accompany the classes, and - * if the class loader created the package instance with the attributes - * from the manifest. + *

If this class represents an array type, a primitive type or void, + * this method returns {@code null}. * - * @return the package of the class, or null if no package - * information is available from the archive or codebase. + * @return the package of this class. */ public Package getPackage() { - return Package.getPackage(this); + if (isPrimitive() || isArray()) { + return null; + } + ClassLoader cl = getClassLoader0(); + return cl != null ? cl.definePackage(this) + : BootLoader.definePackage(this); } + /** + * Returns the fully qualified package name. + * + *

If this class is a top level class, then this method returns the fully + * qualified name of the package that the class is a member of, or the + * empty string if the class is in an unnamed package. + * + *

If this class is a member class, then this method is equivalent to + * invoking {@code getPackageName()} on the {@link #getEnclosingClass + * enclosing class}. + * + *

If this class is a {@link #isLocalClass local class} or an {@link + * #isAnonymousClass() anonymous class}, then this method is equivalent to + * invoking {@code getPackageName()} on the {@link #getDeclaringClass + * declaring class} of the {@link #getEnclosingMethod enclosing method} or + * {@link #getEnclosingConstructor enclosing constructor}. + * + *

This method returns {@code null} if this class represents an array type, + * a primitive type or void. + * + * @return the fully qualified package name + * + * @since 9 + * @jls 6.7 Fully Qualified Names + */ + public String getPackageName() { + String pn = this.packageName; + if (pn == null && !isArray() && !isPrimitive()) { + String cn = getName(); + int dot = cn.lastIndexOf('.'); + pn = (dot != -1) ? cn.substring(0, dot).intern() : ""; + this.packageName = pn; + } + return pn; + } + + // cached package name + private String packageName; /** * Returns the interfaces directly implemented by the class or interface @@ -2213,15 +2356,19 @@ public final class Class implements java.io.Serializable, } /** - * Finds a resource with a given name. The rules for searching resources + * Finds a resource with a given name. If this class is in a named {@link + * Module Module}, and the caller of this method is in the same module, + * then this method will attempt to find the resource in that module. + * Otherwise, the rules for searching resources * associated with a given class are implemented by the defining * {@linkplain ClassLoader class loader} of the class. This method * delegates to this object's class loader. If this object was loaded by * the bootstrap class loader, the method delegates to {@link * ClassLoader#getSystemResourceAsStream}. * - *

Before delegation, an absolute resource name is constructed from the - * given resource name using this algorithm: + *

Before finding a resource in the caller's module or delegation to a + * class loader, an absolute resource name is constructed from the given + * resource name using this algorithm: * *

    * @@ -2242,26 +2389,60 @@ public final class Class implements java.io.Serializable, *
* * @param name name of the desired resource - * @return A {@link java.io.InputStream} object or {@code null} if - * no resource with this name is found + * @return A {@link java.io.InputStream} object or {@code null} if + * no resource with this name is found * @throws NullPointerException If {@code name} is {@code null} * @since 1.1 */ - public InputStream getResourceAsStream(String name) { + @CallerSensitive + public InputStream getResourceAsStream(String name) { name = resolveName(name); - ClassLoader cl = getClassLoader0(); - if (cl==null) { - // A system class. - return ClassLoader.getSystemResourceAsStream(name); + + // if this Class and the caller are in the same named module + // then attempt to get an input stream to the resource in the + // module + Module module = getModule(); + if (module.isNamed()) { + Class caller = Reflection.getCallerClass(); + if (caller != null && caller.getModule() == module) { + ClassLoader cl = getClassLoader0(); + String mn = module.getName(); + try { + + // special-case built-in class loaders to avoid the + // need for a URL connection + if (cl == null) { + return BootLoader.findResourceAsStream(mn, name); + } else if (cl instanceof BuiltinClassLoader) { + return ((BuiltinClassLoader) cl).findResourceAsStream(mn, name); + } else { + URL url = cl.findResource(mn, name); + return (url != null) ? url.openStream() : null; + } + + } catch (IOException | SecurityException e) { + return null; + } + } + } + + // this Class and caller not in the same named module + ClassLoader cl = getClassLoader0(); + if (cl == null) { + return ClassLoader.getSystemResourceAsStream(name); + } else { + return cl.getResourceAsStream(name); } - return cl.getResourceAsStream(name); } /** - * Finds a resource with a given name. The rules for searching resources + * Finds a resource with a given name. If this class is in a named {@link + * Module Module}, and the caller of this method is in the same module, + * then this method will attempt to find the resource in that module. + * Otherwise, the rules for searching resources * associated with a given class are implemented by the defining * {@linkplain ClassLoader class loader} of the class. This method - * delegates to this object's class loader. If this object was loaded by + * delegates to this object's class loader. If this object was loaded by * the bootstrap class loader, the method delegates to {@link * ClassLoader#getSystemResource}. * @@ -2287,22 +2468,43 @@ public final class Class implements java.io.Serializable, * * * @param name name of the desired resource - * @return A {@link java.net.URL} object or {@code null} if no - * resource with this name is found + * @return A {@link java.net.URL} object; {@code null} if no + * resource with this name is found or the resource cannot + * be located by a URL. * @since 1.1 */ - public java.net.URL getResource(String name) { + @CallerSensitive + public URL getResource(String name) { name = resolveName(name); - ClassLoader cl = getClassLoader0(); - if (cl==null) { - // A system class. - return ClassLoader.getSystemResource(name); + + // if this Class and the caller are in the same named module + // then attempt to get URL to the resource in the module + Module module = getModule(); + if (module.isNamed()) { + Class caller = Reflection.getCallerClass(); + if (caller != null && caller.getModule() == module) { + String mn = getModule().getName(); + ClassLoader cl = getClassLoader0(); + try { + if (cl == null) { + return BootLoader.findResource(mn, name); + } else { + return cl.findResource(mn, name); + } + } catch (IOException ioe) { + return null; + } + } + } + + ClassLoader cl = getClassLoader0(); + if (cl == null) { + return ClassLoader.getSystemResource(name); + } else { + return cl.getResource(name); } - return cl.getResource(name); } - - /** protection domain returned when the internal domain is null */ private static java.security.ProtectionDomain allPermDomain; @@ -2845,15 +3047,15 @@ public final class Class implements java.io.Serializable, private void remove(int i) { if (methods[i] != null && methods[i].isDefault()) defaults--; - methods[i] = null; - } + methods[i] = null; + } private boolean matchesNameAndDescriptor(Method m1, Method m2) { return m1.getReturnType() == m2.getReturnType() && m1.getName() == m2.getName() && // name is guaranteed to be interned arrayContentsEq(m1.getParameterTypes(), m2.getParameterTypes()); - } + } void compactAndTrim() { int newPos = 0; diff --git a/jdk/src/java.base/share/classes/java/lang/ClassLoader.java b/jdk/src/java.base/share/classes/java/lang/ClassLoader.java index 1bb15800454..fa6a652067b 100644 --- a/jdk/src/java.base/share/classes/java/lang/ClassLoader.java +++ b/jdk/src/java.base/share/classes/java/lang/ClassLoader.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2013, 2015 Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2013, 2016, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -22,38 +22,44 @@ * or visit www.oracle.com if you need additional information or have any * questions. */ + package java.lang; import java.io.InputStream; import java.io.IOException; import java.io.File; import java.lang.reflect.Constructor; -import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Field; +import java.lang.reflect.Module; import java.net.URL; import java.security.AccessController; import java.security.AccessControlContext; import java.security.CodeSource; import java.security.PrivilegedAction; -import java.security.PrivilegedActionException; -import java.security.PrivilegedExceptionAction; import java.security.ProtectionDomain; import java.security.cert.Certificate; import java.util.Collections; import java.util.Enumeration; import java.util.HashMap; import java.util.HashSet; +import java.util.Hashtable; +import java.util.Map; +import java.util.Objects; import java.util.Set; import java.util.Stack; -import java.util.Map; import java.util.NoSuchElementException; import java.util.Vector; -import java.util.Hashtable; import java.util.WeakHashMap; import java.util.concurrent.ConcurrentHashMap; +import java.util.stream.Stream; import jdk.internal.perf.PerfCounter; -import sun.misc.Resource; -import sun.misc.URLClassPath; +import jdk.internal.module.ServicesCatalog; +import jdk.internal.loader.BootLoader; +import jdk.internal.loader.ClassLoaders; +import jdk.internal.misc.SharedSecrets; +import jdk.internal.misc.Unsafe; +import jdk.internal.misc.VM; import sun.reflect.CallerSensitive; import sun.reflect.Reflection; import sun.reflect.misc.ReflectUtil; @@ -90,9 +96,7 @@ import sun.security.util.SecurityConstants; * associated parent class loader. When requested to find a class or * resource, a ClassLoader instance will delegate the search for the * class or resource to its parent class loader before attempting to find the - * class or resource itself. The virtual machine's built-in class loader, - * called the "bootstrap class loader", does not itself have a parent but may - * serve as the parent of a ClassLoader instance. + * class or resource itself. * *

Class loaders that support concurrent loading of classes are known as * parallel capable class loaders and are required to register @@ -101,19 +105,39 @@ import sun.security.util.SecurityConstants; * #registerAsParallelCapable ClassLoader.registerAsParallelCapable} * method. Note that the ClassLoader class is registered as parallel * capable by default. However, its subclasses still need to register themselves - * if they are parallel capable.
+ * if they are parallel capable. * In environments in which the delegation model is not strictly * hierarchical, class loaders need to be parallel capable, otherwise class * loading can lead to deadlocks because the loader lock is held for the * duration of the class loading process (see {@link #loadClass * loadClass} methods). * - *

Normally, the Java virtual machine loads classes from the local file - * system in a platform-dependent manner. For example, on UNIX systems, the - * virtual machine loads classes from the directory defined by the - * CLASSPATH environment variable. + *

Run-time Built-in Class Loaders

* - *

However, some classes may not originate from a file; they may originate + * The Java run-time has the following built-in class loaders: + * + *

    + *
  • Bootstrap class loader. + * It is the virtual machine's built-in class loader, typically represented + * as {@code null}, and does not have a parent.
  • + *
  • {@linkplain #getPlatformClassLoader() Platform class loader}. + * All platform classes are visible to the platform class loader + * that can be used as the parent of a {@code ClassLoader} instance. + * Platform classes include Java SE platform APIs, their implementation + * classes and JDK-specific run-time classes that are defined by the + * platform class loader or its ancestors.
  • + *
  • {@linkplain #getSystemClassLoader() System class loader}. + * It is also known as application class + * loader and is distinct from the platform class loader. + * The system class loader is typically used to define classes on the + * application class path, module path, and JDK-specific tools. + * The platform class loader is a parent or an ancestor of the system class + * loader that all platform classes are visible to it.
  • + *
+ * + *

Normally, the Java virtual machine loads classes from the local file + * system in a platform-dependent manner. + * However, some classes may not originate from a file; they may originate * from other sources, such as the network, or they could be constructed by an * application. The method {@link #defineClass(String, byte[], int, int) * defineClass} converts an array of bytes into an instance of class @@ -159,8 +183,8 @@ import sun.security.util.SecurityConstants; * *

Binary names

* - *

Any class name provided as a {@link String} parameter to methods in - * ClassLoader must be a binary name as defined by + *

Any class name provided as a {@code String} parameter to methods in + * {@code ClassLoader} must be a binary name as defined by * The Java™ Language Specification. * *

Examples of valid class names include: @@ -171,9 +195,12 @@ import sun.security.util.SecurityConstants; * "java.net.URLClassLoader$3$1" * * - * {@code Class} objects for array classes are not created by {@code ClassLoader}; - * use the {@link Class#forName} method instead. + *

Any package name provided as a {@code String} parameter to methods in + * {@code ClassLoader} must be either the empty string (denoting an unnamed package) + * or a fully qualified name as defined by + * The Java™ Language Specification. * + * @jls 6.7 Fully Qualified Names * @jls 13.1 The Form of a Binary * @see #resolveClass(Class) * @since 1.0 @@ -190,6 +217,9 @@ public abstract class ClassLoader { // must be added *after* it. private final ClassLoader parent; + // the unnamed module for this ClassLoader + private final Module unnamedModule; + /** * Encapsulates the set of parallel capable loader types. */ @@ -241,7 +271,7 @@ public abstract class ClassLoader { // is parallel capable and the appropriate lock object for class loading. private final ConcurrentHashMap parallelLockMap; - // Hashtable that maps packages to certs + // Maps packages to certs private final Map package2certs; // Shared among all packages with unsigned classes @@ -265,11 +295,39 @@ public abstract class ClassLoader { classes.addElement(c); } - // The packages defined in this class loader. Each package name is mapped - // to its corresponding Package object. - private final ConcurrentHashMap packages + // The packages defined in this class loader. Each package name is + // mapped to its corresponding NamedPackage object. + // + // The value is a Package object if ClassLoader::definePackage, + // Class::getPackage, ClassLoader::getDefinePackage(s) or + // Package::getPackage(s) method is called to define it. + // Otherwise, the value is a NamedPackage object. + private final ConcurrentHashMap packages = new ConcurrentHashMap<>(); + /* + * Returns a named package for the given module. + */ + private NamedPackage getNamedPackage(String pn, Module m) { + NamedPackage p = packages.get(pn); + if (p == null) { + p = new NamedPackage(pn, m); + + NamedPackage value = packages.putIfAbsent(pn, p); + if (value != null) { + // Package object already be defined for the named package + p = value; + // if definePackage is called by this class loader to define + // a package in a named module, this will return Package + // object of the same name. Package object may contain + // unexpected information but it does not impact the runtime. + // this assertion may be helpful for troubleshooting + assert value.module() == m; + } + } + return p; + } + private static Void checkCreateClassLoader() { SecurityManager security = System.getSecurityManager(); if (security != null) { @@ -280,6 +338,9 @@ public abstract class ClassLoader { private ClassLoader(Void unused, ClassLoader parent) { this.parent = parent; + this.unnamedModule + = SharedSecrets.getJavaLangReflectModuleAccess() + .defineUnnamedModule(this); if (ParallelLoaders.isRegistered(this.getClass())) { parallelLockMap = new ConcurrentHashMap<>(); package2certs = new ConcurrentHashMap<>(); @@ -437,6 +498,74 @@ public abstract class ClassLoader { } } + /** + * Loads the class with the specified binary name + * in a module defined to this class loader. This method returns {@code null} + * if the class could not be found. + * + * @apiNote This method does not delegate to the parent class loader. + * + * @implSpec The default implementation of this method searches for classes + * in the following order: + * + *

    + *
  1. Invoke {@link #findLoadedClass(String)} to check if the class + * has already been loaded.
  2. + *
  3. Invoke the {@link #findClass(String, String)} method to find the + * class in the given module.
  4. + *
+ * + * @param module + * The module + * @param name + * The binary name of the class + * + * @return The resulting {@code Class} object in a module defined by + * this class loader, or {@code null} if the class could not be found. + */ + final Class loadLocalClass(Module module, String name) { + synchronized (getClassLoadingLock(name)) { + // First, check if the class has already been loaded + Class c = findLoadedClass(name); + if (c == null) { + c = findClass(module.getName(), name); + } + if (c != null && c.getModule() == module) { + return c; + } else { + return null; + } + } + } + + /** + * Loads the class with the specified binary name + * defined by this class loader. This method returns {@code null} + * if the class could not be found. + * + * @apiNote This method does not delegate to the parent class loader. + * + * @param name + * The binary name of the class + * + * @return The resulting {@code Class} object in a module defined by + * this class loader, or {@code null} if the class could not be found. + */ + final Class loadLocalClass(String name) { + synchronized (getClassLoadingLock(name)) { + // First, check if the class has already been loaded + Class c = findLoadedClass(name); + if (c == null) { + try { + return findClass(name); + } catch (ClassNotFoundException e) { + // ignore + } + } + return c; + } + } + /** * Returns the lock object for class loading operations. * For backward compatibility, the default implementation of this method @@ -531,6 +660,32 @@ public abstract class ClassLoader { throw new ClassNotFoundException(name); } + /** + * Finds the class with the given binary name + * in a module defined to this class loader. + * Class loader implementations that support the loading from modules + * should override this method. + * + * @apiNote This method returns {@code null} rather than throwing + * {@code ClassNotFoundException} if the class could not be found + * + * @implSpec The default implementation returns {@code null}. + * + * @param moduleName + * The module name + * @param name + * The binary name of the class + * + * @return The resulting {@code Class} object, or {@code null} + * if the class could not be found. + * + * @since 9 + */ + protected Class findClass(String moduleName, String name) { + return null; + } + + /** * Converts an array of bytes into an instance of class Class. * Before the Class can be used it must be resolved. This method @@ -580,55 +735,63 @@ public abstract class ClassLoader { } /** - * Converts an array of bytes into an instance of class Class. - * Before the Class can be used it must be resolved. + * Converts an array of bytes into an instance of class {@code Class}. + * Before the {@code Class} can be used it must be resolved. * *

This method assigns a default {@link java.security.ProtectionDomain - * ProtectionDomain} to the newly defined class. The - * ProtectionDomain is effectively granted the same set of + * ProtectionDomain} to the newly defined class. The + * {@code ProtectionDomain} is effectively granted the same set of * permissions returned when {@link * java.security.Policy#getPermissions(java.security.CodeSource) - * Policy.getPolicy().getPermissions(new CodeSource(null, null))} - * is invoked. The default domain is created on the first invocation of - * {@link #defineClass(String, byte[], int, int) defineClass}, + * Policy.getPolicy().getPermissions(new CodeSource(null, null))} + * is invoked. The default protection domain is created on the first invocation + * of {@link #defineClass(String, byte[], int, int) defineClass}, * and re-used on subsequent invocations. * - *

To assign a specific ProtectionDomain to the class, use + *

To assign a specific {@code ProtectionDomain} to the class, use * the {@link #defineClass(String, byte[], int, int, - * java.security.ProtectionDomain) defineClass} method that takes a - * ProtectionDomain as one of its arguments.

+ * java.security.ProtectionDomain) defineClass} method that takes a + * {@code ProtectionDomain} as one of its arguments.

+ * + *

+ * This method defines a package in this class loader corresponding to the + * package of the {@code Class} (if such a package has not already been defined + * in this class loader). The name of the defined package is derived from + * the binary name of the class specified by + * the byte array {@code b}. + * Other properties of the defined package are as specified by {@link Package}. * * @param name * The expected binary name of the class, or - * null if not known + * {@code null} if not known * * @param b * The bytes that make up the class data. The bytes in positions - * off through off+len-1 should have the format + * {@code off} through {@code off+len-1} should have the format * of a valid class file as defined by * The Java™ Virtual Machine Specification. * * @param off - * The start offset in b of the class data + * The start offset in {@code b} of the class data * * @param len * The length of the class data * - * @return The Class object that was created from the specified + * @return The {@code Class} object that was created from the specified * class data. * * @throws ClassFormatError * If the data did not contain a valid class * * @throws IndexOutOfBoundsException - * If either off or len is negative, or if - * off+len is greater than b.length. + * If either {@code off} or {@code len} is negative, or if + * {@code off+len} is greater than {@code b.length}. * * @throws SecurityException * If an attempt is made to add this class to a package that * contains classes that were signed by a different set of * certificates than this class (which is unsigned), or if - * name begins with "java.". + * {@code name} begins with "{@code java.}". * * @see #loadClass(String, boolean) * @see #resolveClass(Class) @@ -654,7 +817,8 @@ public abstract class ClassLoader { if (!checkName(name)) throw new NoClassDefFoundError("IllegalName: " + name); - if ((name != null) && name.startsWith("java.")) { + if ((name != null) && name.startsWith("java.") + && this != getBuiltinPlatformClassLoader()) { throw new SecurityException ("Prohibited package name: " + name.substring(0, name.lastIndexOf('.'))); @@ -663,13 +827,14 @@ public abstract class ClassLoader { pd = defaultDomain; } - if (name != null) checkCerts(name, pd.getCodeSource()); + if (name != null) { + checkCerts(name, pd.getCodeSource()); + } return pd; } - private String defineClassSourceLocation(ProtectionDomain pd) - { + private String defineClassSourceLocation(ProtectionDomain pd) { CodeSource cs = pd.getCodeSource(); String source = null; if (cs != null && cs.getLocation() != null) { @@ -678,8 +843,10 @@ public abstract class ClassLoader { return source; } - private void postDefineClass(Class c, ProtectionDomain pd) - { + private void postDefineClass(Class c, ProtectionDomain pd) { + // define a named package, if not present + getNamedPackage(c.getPackageName(), c.getModule()); + if (pd.getCodeSource() != null) { Certificate certs[] = pd.getCodeSource().getCertificates(); if (certs != null) @@ -688,69 +855,80 @@ public abstract class ClassLoader { } /** - * Converts an array of bytes into an instance of class Class, - * with an optional ProtectionDomain. If the domain is - * null, then a default domain will be assigned to the class as - * specified in the documentation for {@link #defineClass(String, byte[], - * int, int)}. Before the class can be used it must be resolved. + * Converts an array of bytes into an instance of class {@code Class}, + * with a given {@code ProtectionDomain}. + * + *

If the given {@code ProtectionDomain} is {@code null}, + * then a default protection domain will be assigned to the class as specified + * in the documentation for {@link #defineClass(String, byte[], int, int)}. + * Before the class can be used it must be resolved. * *

The first class defined in a package determines the exact set of * certificates that all subsequent classes defined in that package must * contain. The set of certificates for a class is obtained from the - * {@link java.security.CodeSource CodeSource} within the - * ProtectionDomain of the class. Any classes added to that + * {@link java.security.CodeSource CodeSource} within the + * {@code ProtectionDomain} of the class. Any classes added to that * package must contain the same set of certificates or a - * SecurityException will be thrown. Note that if - * name is null, this check is not performed. + * {@code SecurityException} will be thrown. Note that if + * {@code name} is {@code null}, this check is not performed. * You should always pass in the binary name of the * class you are defining as well as the bytes. This ensures that the * class you are defining is indeed the class you think it is. * - *

The specified name cannot begin with "java.", since - * all classes in the "java.* packages can only be defined by the - * bootstrap class loader. If name is not null, it - * must be equal to the binary name of the class - * specified by the byte array "b", otherwise a {@link - * NoClassDefFoundError NoClassDefFoundError} will be thrown.

+ *

If the specified {@code name} begins with "{@code java.}", it can + * only be defined by the {@linkplain #getPlatformClassLoader() + * platform class loader} or its ancestors; otherwise {@code SecurityException} + * will be thrown. If {@code name} is not {@code null}, it must be equal to + * the binary name of the class + * specified by the byte array {@code b}, otherwise a {@link + * NoClassDefFoundError NoClassDefFoundError} will be thrown. + * + *

This method defines a package in this class loader corresponding to the + * package of the {@code Class} (if such a package has not already been defined + * in this class loader). The name of the defined package is derived from + * the binary name of the class specified by + * the byte array {@code b}. + * Other properties of the defined package are as specified by {@link Package}. * * @param name * The expected binary name of the class, or - * null if not known + * {@code null} if not known * * @param b * The bytes that make up the class data. The bytes in positions - * off through off+len-1 should have the format + * {@code off} through {@code off+len-1} should have the format * of a valid class file as defined by * The Java™ Virtual Machine Specification. * * @param off - * The start offset in b of the class data + * The start offset in {@code b} of the class data * * @param len * The length of the class data * * @param protectionDomain - * The ProtectionDomain of the class + * The {@code ProtectionDomain} of the class * - * @return The Class object created from the data, - * and optional ProtectionDomain. + * @return The {@code Class} object created from the data, + * and {@code ProtectionDomain}. * * @throws ClassFormatError * If the data did not contain a valid class * * @throws NoClassDefFoundError - * If name is not equal to the binary - * name of the class specified by b + * If {@code name} is not {@code null} and not equal to the + * binary name of the class specified by {@code b} * * @throws IndexOutOfBoundsException - * If either off or len is negative, or if - * off+len is greater than b.length. + * If either {@code off} or {@code len} is negative, or if + * {@code off+len} is greater than {@code b.length}. * * @throws SecurityException * If an attempt is made to add this class to a package that * contains classes that were signed by a different set of - * certificates than this class, or if name begins with - * "java.". + * certificates than this class, or if {@code name} begins with + * "{@code java.}" and this class loader is not the platform + * class loader or its ancestor. */ protected final Class defineClass(String name, byte[] b, int off, int len, ProtectionDomain protectionDomain) @@ -764,15 +942,16 @@ public abstract class ClassLoader { } /** - * Converts a {@link java.nio.ByteBuffer ByteBuffer} - * into an instance of class Class, - * with an optional ProtectionDomain. If the domain is - * null, then a default domain will be assigned to the class as + * Converts a {@link java.nio.ByteBuffer ByteBuffer} into an instance + * of class {@code Class}, with the given {@code ProtectionDomain}. + * If the given {@code ProtectionDomain} is {@code null}, then a default + * protection domain will be assigned to the class as * specified in the documentation for {@link #defineClass(String, byte[], * int, int)}. Before the class can be used it must be resolved. * *

The rules about the first class defined in a package determining the - * set of certificates for the package, and the restrictions on class names + * set of certificates for the package, the restrictions on class names, + * and the defined package of the class * are identical to those specified in the documentation for {@link * #defineClass(String, byte[], int, int, ProtectionDomain)}. * @@ -803,23 +982,23 @@ public abstract class ClassLoader { * The Java™ Virtual Machine Specification. * * @param protectionDomain - * The ProtectionDomain of the class, or null. + * The {@code ProtectionDomain} of the class, or {@code null}. * - * @return The Class object created from the data, - * and optional ProtectionDomain. + * @return The {@code Class} object created from the data, + * and {@code ProtectionDomain}. * * @throws ClassFormatError * If the data did not contain a valid class. * * @throws NoClassDefFoundError - * If name is not equal to the binary - * name of the class specified by b + * If {@code name} is not {@code null} and not equal to the + * binary name of the class specified by {@code b} * * @throws SecurityException * If an attempt is made to add this class to a package that * contains classes that were signed by a different set of - * certificates than this class, or if name begins with - * "java.". + * certificates than this class, or if {@code name} begins with + * "{@code java.}". * * @see #defineClass(String, byte[], int, int, ProtectionDomain) * @@ -1002,8 +1181,7 @@ public abstract class ClassLoader { * Returns a class loaded by the bootstrap class loader; * or return null if not found. */ - private Class findBootstrapClassOrNull(String name) - { + Class findBootstrapClassOrNull(String name) { if (!checkName(name)) return null; return findBootstrapClass(name); @@ -1051,13 +1229,44 @@ public abstract class ClassLoader { } - // -- Resource -- + // -- Resources -- + + /** + * Returns a URL to a resource in a module defined to this class loader. + * Class loader implementations that support the loading from modules + * should override this method. + * + * @implSpec The default implementation returns {@code null}. + * + * @param moduleName + * The module name + * @param name + * The resource name + * + * @return A URL to the resource; {@code null} if the resource could not be + * found, a URL could not be constructed to locate the resource, + * access to the resource is denied by the security manager, or + * there isn't a module of the given name defined to the class + * loader. + * + * @throws IOException + * If I/O errors occur + * + * @see java.lang.module.ModuleReader#find(String) + * @since 9 + */ + protected URL findResource(String moduleName, String name) throws IOException { + return null; + } /** * Finds the resource with the given name. A resource is some data * (images, audio, text, etc) that can be accessed by class code in a way * that is independent of the location of the code. * + * Resources in a named module are private to that module. This method does + * not find resource in named modules. + * *

The name of a resource is a '/'-separated path name that * identifies the resource. * @@ -1084,7 +1293,7 @@ public abstract class ClassLoader { if (parent != null) { url = parent.getResource(name); } else { - url = getBootstrapResource(name); + url = BootLoader.findResource(name); } if (url == null) { url = findResource(name); @@ -1097,6 +1306,9 @@ public abstract class ClassLoader { * (images, audio, text, etc) that can be accessed by class code in a way * that is independent of the location of the code. * + * Resources in a named module are private to that module. This method does + * not find resources in named modules. + * *

The name of a resource is a /-separated path name that * identifies the resource. * @@ -1131,7 +1343,7 @@ public abstract class ClassLoader { if (parent != null) { tmp[0] = parent.getResources(name); } else { - tmp[0] = getBootstrapResources(name); + tmp[0] = BootLoader.findResources(name); } tmp[1] = findResources(name); @@ -1142,6 +1354,9 @@ public abstract class ClassLoader { * Finds the resource with the given name. Class loader implementations * should override this method to specify where to find resources. * + * Resources in a named module are private to that module. This method does + * not find resources in named modules defined to this class loader. + * * @param name * The resource name * @@ -1160,6 +1375,9 @@ public abstract class ClassLoader { * implementations should override this method to specify where to load * resources from. * + * Resources in a named module are private to that module. This method does + * not find resources in named modules defined to this class loader. + * * @param name * The resource name * @@ -1204,6 +1422,9 @@ public abstract class ClassLoader { * classes. This method locates the resource through the system class * loader (see {@link #getSystemClassLoader()}). * + * Resources in a named module are private to that module. This method does + * not find resources in named modules. + * * @param name * The resource name * @@ -1215,7 +1436,7 @@ public abstract class ClassLoader { public static URL getSystemResource(String name) { ClassLoader system = getSystemClassLoader(); if (system == null) { - return getBootstrapResource(name); + return BootLoader.findResource(name); } return system.getResource(name); } @@ -1226,6 +1447,9 @@ public abstract class ClassLoader { * {@link java.util.Enumeration Enumeration} of {@link * java.net.URL URL} objects. * + * Resources in a named module are private to that module. This method does + * not find resources in named modules. + * *

The search order is described in the documentation for {@link * #getSystemResource(String)}.

* @@ -1245,47 +1469,17 @@ public abstract class ClassLoader { { ClassLoader system = getSystemClassLoader(); if (system == null) { - return getBootstrapResources(name); + return BootLoader.findResources(name); } return system.getResources(name); } - /** - * Find resources from the VM's built-in classloader. - */ - private static URL getBootstrapResource(String name) { - URLClassPath ucp = getBootstrapClassPath(); - Resource res = ucp.getResource(name); - return res != null ? res.getURL() : null; - } - - /** - * Find resources from the VM's built-in classloader. - */ - private static Enumeration getBootstrapResources(String name) - throws IOException - { - final Enumeration e = - getBootstrapClassPath().getResources(name); - return new Enumeration<> () { - public URL nextElement() { - return e.nextElement().getURL(); - } - public boolean hasMoreElements() { - return e.hasMoreElements(); - } - }; - } - - // Returns the URLClassPath that is used for finding system resources. - static URLClassPath getBootstrapClassPath() { - return sun.misc.Launcher.getBootstrapClassPath(); - } - - /** * Returns an input stream for reading the specified resource. * + * Resources in a named module are private to that module. This method does + * not find resources in named modules. + * *

The search order is described in the documentation for {@link * #getResource(String)}.

* @@ -1311,6 +1505,9 @@ public abstract class ClassLoader { * used to load classes. This method locates the resource through the * system class loader (see {@link #getSystemClassLoader()}). * + * Resources in a named module are private to that module. This method does + * not find resources in named modules. + * * @param name * The resource name * @@ -1370,14 +1567,55 @@ public abstract class ClassLoader { return parent; } + /** + * Returns the unnamed {@code Module} for this class loader. + * + * @return The unnamed Module for this class loader + * + * @see Module#isNamed() + * @since 9 + */ + public final Module getUnnamedModule() { + return unnamedModule; + } + + /** + * Returns the platform class loader for delegation. All + * platform classes are visible to + * the platform class loader. + * + * @return The platform {@code ClassLoader}. + * + * @throws SecurityException + * If a security manager exists and the caller's class loader is + * not {@code null} and the caller's class loader is not the same + * as or an ancestor of the platform class loader, + * and the {@link SecurityManager#checkPermission(java.security.Permission) + * checkPermission} method denies {@code RuntimePermission("getClassLoader")} + * to access the platform class loader. + * + * @since 9 + */ + @CallerSensitive + public static ClassLoader getPlatformClassLoader() { + SecurityManager sm = System.getSecurityManager(); + ClassLoader loader = getBuiltinPlatformClassLoader(); + if (sm != null) { + checkClassLoaderPermission(loader, Reflection.getCallerClass()); + } + return loader; + } + /** * Returns the system class loader for delegation. This is the default * delegation parent for new ClassLoader instances, and is * typically the class loader used to start the application. * *

This method is first invoked early in the runtime's startup - * sequence, at which point it creates the system class loader and sets it - * as the context class loader of the invoking Thread. + * sequence, at which point it creates the system class loader. This + * class loader will be the context class loader for the main application + * thread (for example, the thread that invokes the {@code main} method of + * the main class). * *

The default system class loader is an implementation-dependent * instance of this class. @@ -1390,7 +1628,10 @@ public abstract class ClassLoader { * type ClassLoader which is used as the delegation parent. An * instance is then created using this constructor with the default system * class loader as the parameter. The resulting class loader is defined - * to be the system class loader. + * to be the system class loader. During construction, the class loader + * should take great care to avoid calling {@code getSystemClassLoader()}. + * If circular initialization of the system class loader is detected then + * an unspecified error or exception is thrown. * *

If a security manager is present, and the invoker's class loader is * not null and the invoker's class loader is not the same as or @@ -1403,6 +1644,11 @@ public abstract class ClassLoader { * access to the system class loader. If not, a * SecurityException will be thrown.

* + * @implNote The system property to override the system class loader is not + * examined until the VM is almost fully initialized. Code that executes + * this method during startup should take care not to cache the return + * value until the system is fully initialized. + * * @return The system ClassLoader for delegation, or * null if none * @@ -1427,45 +1673,69 @@ public abstract class ClassLoader { */ @CallerSensitive public static ClassLoader getSystemClassLoader() { - initSystemClassLoader(); - if (scl == null) { - return null; + switch (VM.initLevel()) { + case 0: + case 1: + case 2: + // the system class loader is the built-in app class loader during startup + return getBuiltinAppClassLoader(); + case 3: + throw new InternalError("getSystemClassLoader should only be called after VM booted"); + case 4: + // system fully initialized + assert VM.isBooted() && scl != null; + SecurityManager sm = System.getSecurityManager(); + if (sm != null) { + checkClassLoaderPermission(scl, Reflection.getCallerClass()); + } + return scl; + default: + throw new InternalError("should not reach here"); } - SecurityManager sm = System.getSecurityManager(); - if (sm != null) { - checkClassLoaderPermission(scl, Reflection.getCallerClass()); - } - return scl; } - private static synchronized void initSystemClassLoader() { - if (!sclSet) { - if (scl != null) - throw new IllegalStateException("recursive invocation"); - sun.misc.Launcher l = sun.misc.Launcher.getLauncher(); - if (l != null) { - Throwable oops = null; - scl = l.getClassLoader(); - try { - scl = AccessController.doPrivileged( - new SystemClassLoaderAction(scl)); - } catch (PrivilegedActionException pae) { - oops = pae.getCause(); - if (oops instanceof InvocationTargetException) { - oops = oops.getCause(); - } - } - if (oops != null) { - if (oops instanceof Error) { - throw (Error) oops; - } else { - // wrap the exception - throw new Error(oops); - } - } - } - sclSet = true; + static ClassLoader getBuiltinPlatformClassLoader() { + return ClassLoaders.platformClassLoader(); + } + + static ClassLoader getBuiltinAppClassLoader() { + return ClassLoaders.appClassLoader(); + } + + /* + * Initialize the system class loader that may be a custom class on the + * application class path or application module path. + * + * @see java.lang.System#initPhase3 + */ + static synchronized ClassLoader initSystemClassLoader() { + if (VM.initLevel() != 3) { + throw new InternalError("system class loader cannot be set at initLevel " + + VM.initLevel()); } + + // detect recursive initialization + if (scl != null) { + throw new IllegalStateException("recursive invocation"); + } + + ClassLoader builtinLoader = getBuiltinAppClassLoader(); + + // All are privileged frames. No need to call doPrivileged. + String cn = System.getProperty("java.system.class.loader"); + if (cn != null) { + try { + // custom class loader is only supported to be loaded from unnamed module + Constructor ctor = Class.forName(cn, false, builtinLoader) + .getDeclaredConstructor(ClassLoader.class); + scl = (ClassLoader) ctor.newInstance(builtinLoader); + } catch (Exception e) { + throw new Error(e); + } + } else { + scl = builtinLoader; + } + return scl; } // Returns true if the specified class loader can be found in this class @@ -1524,26 +1794,102 @@ public abstract class ClassLoader { } } - // The class loader for the system + // The system class loader // @GuardedBy("ClassLoader.class") - private static ClassLoader scl; - - // Set to true once the system class loader has been set - // @GuardedBy("ClassLoader.class") - private static boolean sclSet; - + private static volatile ClassLoader scl; // -- Package -- /** - * Defines a package by name in this ClassLoader. This allows - * class loaders to define the packages for their classes. Packages must - * be created before the class is defined, and package names must be - * unique within a class loader and cannot be redefined or changed once - * created. + * Define a Package of the given Class object. + * + * If the given class represents an array type, a primitive type or void, + * this method returns {@code null}. + * + * This method does not throw IllegalArgumentException. + */ + Package definePackage(Class c) { + if (c.isPrimitive() || c.isArray()) { + return null; + } + + return definePackage(c.getPackageName(), c.getModule()); + } + + /** + * Defines a Package of the given name and module + * + * This method does not throw IllegalArgumentException. + * + * @param name package name + * @param m module + */ + Package definePackage(String name, Module m) { + if (name.isEmpty() && m.isNamed()) { + throw new InternalError("unnamed package in " + m); + } + + // check if Package object is already defined + NamedPackage pkg = packages.get(name); + if (pkg instanceof Package) + return (Package)pkg; + + return (Package)packages.compute(name, (n, p) -> toPackage(n, p, m)); + } + + /* + * Returns a Package object for the named package + */ + private Package toPackage(String name, NamedPackage p, Module m) { + // define Package object if the named package is not yet defined + if (p == null) + return NamedPackage.toPackage(name, m); + + // otherwise, replace the NamedPackage object with Package object + if (p instanceof Package) + return (Package)p; + + return NamedPackage.toPackage(p.packageName(), p.module()); + } + + /** + * Defines a package by name in this {@code ClassLoader}. + *

+ * Package names must be unique within a class loader and + * cannot be redefined or changed once created. + *

+ * If a class loader wishes to define a package with specific properties, + * such as version information, then the class loader should call this + * {@code definePackage} method before calling {@code defineClass}. + * Otherwise, the + * {@link #defineClass(String, byte[], int, int, ProtectionDomain) defineClass} + * method will define a package in this class loader corresponding to the package + * of the newly defined class; the properties of this defined package are + * specified by {@link Package}. + * + * @apiNote + * A class loader that wishes to define a package for classes in a JAR + * typically uses the specification and implementation titles, versions, and + * vendors from the JAR's manifest. If the package is specified as + * {@linkplain java.util.jar.Attributes.Name#SEALED sealed} in the JAR's manifest, + * the {@code URL} of the JAR file is typically used as the {@code sealBase}. + * If classes of package {@code 'p'} defined by this class loader + * are loaded from multiple JARs, the {@code Package} object may contain + * different information depending on the first class of package {@code 'p'} + * defined and which JAR's manifest is read first to explicitly define + * package {@code 'p'}. + * + *

It is strongly recommended that a class loader does not call this + * method to explicitly define packages in named modules; instead, + * the package will be automatically defined when a class is {@linkplain + * #defineClass(String, byte[], int, int, ProtectionDomain) being defined}. + * If it is desirable to define {@code Package} explicitly, it should ensure + * that all packages in a named module are defined with the properties + * specified by {@link Package}. Otherwise, some {@code Package} objects + * in a named module may be for example sealed with different seal base. * * @param name - * The package name + * The package name * * @param specTitle * The specification title @@ -1564,89 +1910,155 @@ public abstract class ClassLoader { * The implementation vendor * * @param sealBase - * If not null, then this package is sealed with - * respect to the given code source {@link java.net.URL - * URL} object. Otherwise, the package is not sealed. + * If not {@code null}, then this package is sealed with + * respect to the given code source {@link java.net.URL URL} + * object. Otherwise, the package is not sealed. * - * @return The newly defined Package object + * @return The newly defined {@code Package} object + * + * @throws NullPointerException + * if {@code name} is {@code null}. * * @throws IllegalArgumentException - * If package name duplicates an existing package either in this - * class loader or one of its ancestors + * if a package of the given {@code name} is already + * defined by this class loader * * @since 1.2 + * + * @see + * The JAR File Specification: Package Versioning + * @see + * The JAR File Specification: Package Sealing */ protected Package definePackage(String name, String specTitle, String specVersion, String specVendor, String implTitle, String implVersion, String implVendor, URL sealBase) - throws IllegalArgumentException { - Package pkg = getPackage(name); - if (pkg != null) { + Objects.requireNonNull(name); + + // definePackage is not final and may be overridden by custom class loader + Package p = new Package(name, specTitle, specVersion, specVendor, + implTitle, implVersion, implVendor, + sealBase, this); + + if (packages.putIfAbsent(name, p) != null) throw new IllegalArgumentException(name); - } - pkg = new Package(name, specTitle, specVersion, specVendor, - implTitle, implVersion, implVendor, - sealBase, this); - if (packages.putIfAbsent(name, pkg) != null) { - throw new IllegalArgumentException(name); - } - return pkg; + + return p; } /** - * Returns a Package that has been defined by this class loader - * or any of its ancestors. + * Returns a {@code Package} of the given name that has been + * defined by this class loader. + * + * @param name The package name + * + * @return The {@code Package} of the given name defined by this class loader, + * or {@code null} if not found + * + * @since 9 + */ + public final Package getDefinedPackage(String name) { + NamedPackage p = packages.get(name); + if (p == null) + return null; + + return definePackage(name, p.module()); + } + + /** + * Returns all of the {@code Package}s defined by this class loader. + * The returned array has no duplicated {@code Package}s of the same name. + * + * @apiNote This method returns an array rather than a {@code Set} or {@code Stream} + * for consistency with the existing {@link #getPackages} method. + * + * @return The array of {@code Package} objects defined by this class loader; + * or an zero length array if no package has been defined by this class loader. + * + * @since 9 + */ + public final Package[] getDefinedPackages() { + return packages().toArray(Package[]::new); + } + + /** + * Finds a package by name in this class loader and its ancestors. + *

+ * If this class loader defines a {@code Package} of the given name, + * the {@code Package} is returned. Otherwise, the ancestors of + * this class loader are searched recursively (parent by parent) + * for a {@code Package} of the given name. * * @param name - * The package name + * The package name * - * @return The Package corresponding to the given name, or - * null if not found + * @return The {@code Package} corresponding to the given name defined by + * this class loader or its ancestors, or {@code null} if not found. + * + * @deprecated + * If multiple class loaders delegate to each other and define classes + * with the same package name, and one such loader relies on the lookup + * behavior of {@code getPackage} to return a {@code Package} from + * a parent loader, then the properties exposed by the {@code Package} + * may not be as expected in the rest of the program. + * For example, the {@code Package} will only expose annotations from the + * {@code package-info.class} file defined by the parent loader, even if + * annotations exist in a {@code package-info.class} file defined by + * a child loader. A more robust approach is to use the + * {@link ClassLoader#getDefinedPackage} method which returns + * a {@code Package} for the specified class loader. * * @since 1.2 */ + @Deprecated protected Package getPackage(String name) { - Package pkg = packages.get(name); + Package pkg = getDefinedPackage(name); if (pkg == null) { if (parent != null) { pkg = parent.getPackage(name); } else { - pkg = Package.getSystemPackage(name); + pkg = BootLoader.getDefinedPackage(name); } } return pkg; } /** - * Returns all of the Packages defined by this class loader and - * its ancestors. + * Returns all of the {@code Package}s defined by this class loader + * and its ancestors. The returned array may contain more than one + * {@code Package} object of the same package name, each defined by + * a different class loader in the class loader hierarchy. * - * @return The array of Package objects defined by this - * ClassLoader + * @return The array of {@code Package} objects defined by this + * class loader and its ancestors * * @since 1.2 */ protected Package[] getPackages() { - Package[] pkgs; - if (parent != null) { - pkgs = parent.getPackages(); - } else { - pkgs = Package.getSystemPackages(); + Stream pkgs = packages(); + ClassLoader ld = parent; + while (ld != null) { + pkgs = Stream.concat(ld.packages(), pkgs); + ld = ld.parent; } - - Map map = packages; - if (pkgs != null) { - map = new HashMap<>(packages); - for (Package pkg : pkgs) { - map.putIfAbsent(pkg.getName(), pkg); - } - } - return map.values().toArray(new Package[map.size()]); + return Stream.concat(BootLoader.packages(), pkgs) + .toArray(Package[]::new); } + + // package-private + + /** + * Returns a stream of Packages defined in this class loader + */ + Stream packages() { + return packages.values().stream() + .map(p -> definePackage(p.packageName(), p.module())); + } + // -- Native library access -- /** @@ -2183,29 +2595,53 @@ public abstract class ClassLoader { // Retrieves the assertion directives from the VM. private static native AssertionStatusDirectives retrieveDirectives(); -} -class SystemClassLoaderAction - implements PrivilegedExceptionAction { - private ClassLoader parent; - - SystemClassLoaderAction(ClassLoader parent) { - this.parent = parent; + /** + * Returns the ServiceCatalog for modules defined to this class loader + * or {@code null} if this class loader does not have a services catalog. + */ + ServicesCatalog getServicesCatalog() { + return servicesCatalog; } - public ClassLoader run() throws Exception { - String cls = System.getProperty("java.system.class.loader"); - if (cls == null) { - return parent; + /** + * Returns the ServiceCatalog for modules defined to this class loader, + * creating it if it doesn't already exist. + */ + ServicesCatalog createOrGetServicesCatalog() { + ServicesCatalog catalog = servicesCatalog; + if (catalog == null) { + catalog = new ServicesCatalog(); + boolean set = trySetObjectField("servicesCatalog", catalog); + if (!set) { + // beaten by someone else + catalog = servicesCatalog; + } } + return catalog; + } - Constructor ctor = Class.forName(cls, true, parent) - .getDeclaredConstructor(new Class[] { ClassLoader.class }); - ClassLoader sys = (ClassLoader) ctor.newInstance( - new Object[] { parent }); - Thread.currentThread().setContextClassLoader(sys); - return sys; + // the ServiceCatalog for modules associated with this class loader. + private volatile ServicesCatalog servicesCatalog; + + + /** + * Attempts to atomically set a volatile field in this object. Returns + * {@code true} if not beaten by another thread. Avoids the use of + * AtomicReferenceFieldUpdater in this class. + */ + private boolean trySetObjectField(String name, Object obj) { + Unsafe unsafe = Unsafe.getUnsafe(); + Class k = ClassLoader.class; + long offset; + try { + Field f = k.getDeclaredField(name); + offset = unsafe.objectFieldOffset(f); + } catch (NoSuchFieldException e) { + throw new InternalError(e); + } + return unsafe.compareAndSwapObject(this, offset, null, obj); } } diff --git a/jdk/src/java.base/share/classes/java/lang/NamedPackage.java b/jdk/src/java.base/share/classes/java/lang/NamedPackage.java new file mode 100644 index 00000000000..5d86cc7fd9c --- /dev/null +++ b/jdk/src/java.base/share/classes/java/lang/NamedPackage.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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package java.lang; + +import java.lang.module.Configuration; +import java.lang.module.ModuleReference; +import java.lang.reflect.Module; +import java.net.URI; + +/** + * A NamedPackage represents a package by name in a specific module. + * + * A class loader will automatically create NamedPackage for each + * package when a class is defined. Package object is lazily + * defined until Class::getPackage, Package::getPackage(s), or + * ClassLoader::getDefinedPackage(s) method is called. + * + * NamedPackage allows ClassLoader to keep track of the runtime + * packages with minimal footprint and avoid constructing Package + * object. + */ +class NamedPackage { + private final String name; + private final Module module; + + NamedPackage(String pn, Module module) { + if (pn.isEmpty() && module.isNamed()) { + throw new InternalError("unnamed package in " + module); + } + this.name = pn.intern(); + this.module = module; + } + + /** + * Returns the name of this package. + */ + String packageName() { + return name; + } + + /** + * Returns the module of this named package. + */ + Module module() { + return module; + } + + /** + * Returns the location of the module if this named package is in + * a named module; otherwise, returns null. + */ + URI location() { + if (module.isNamed() && module.getLayer() != null) { + Configuration cf = module.getLayer().configuration(); + ModuleReference mref + = cf.findModule(module.getName()).get().reference(); + return mref.location().orElse(null); + } + return null; + } + + /** + * Creates a Package object of the given name and module. + */ + static Package toPackage(String name, Module module) { + return new Package(name, module); + } +} diff --git a/jdk/src/java.base/share/classes/java/lang/Package.java b/jdk/src/java.base/share/classes/java/lang/Package.java index 2ab9fe3c4bf..5b80278b097 100644 --- a/jdk/src/java.base/share/classes/java/lang/Package.java +++ b/jdk/src/java.base/share/classes/java/lang/Package.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 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 @@ -25,85 +25,94 @@ package java.lang; +import java.lang.annotation.Annotation; import java.lang.reflect.AnnotatedElement; - -import java.io.File; -import java.io.FileInputStream; -import java.io.IOException; -import java.net.URL; +import java.lang.reflect.Module; import java.net.MalformedURLException; +import java.net.URI; +import java.net.URL; import java.security.AccessController; import java.security.PrivilegedAction; +import java.util.Objects; -import java.util.concurrent.ConcurrentHashMap; -import java.util.jar.JarInputStream; -import java.util.jar.Manifest; -import java.util.jar.Attributes; -import java.util.jar.Attributes.Name; -import java.util.Map; - -import sun.net.www.ParseUtil; +import jdk.internal.loader.BootLoader; import sun.reflect.CallerSensitive; import sun.reflect.Reflection; -import java.lang.annotation.Annotation; /** - * {@code Package} objects contain version information - * about the implementation and specification of a Java package. - * This versioning information is retrieved and made available - * by the {@link ClassLoader} instance that - * loaded the class(es). Typically, it is stored in the manifest that is - * distributed with the classes. + * Represents metadata about a run-time package associated with a class loader. + * Metadata includes annotations, versioning, and sealing. + *

+ * Annotations for the run-time package are read from {@code package-info.class} + * at the same code source as classes in the run-time package. + *

+ * The set of classes that make up the run-time package may implement a + * particular specification. The specification title, version, and vendor + * (indicating the owner/maintainer of the specification) can be provided + * when the {@code Package} is defined. An application can ask if the + * {@code Package} is compatible with a particular specification version + * by using the {@link #isCompatibleWith Package.isCompatibleWith(String)} + * method. In addition, information about the actual classes that make up the + * run-time package can be provided when the Package is defined. + * This information consists of an implementation title, version, and vendor + * (indicating the supplier of the classes). + *

+ * A {@code Package} may be explicitly defined with + * the {@link ClassLoader#definePackage(String, String, String, String, + * String, String, String, URL)} method. + * The caller supplies the specification and implementation titles, versions, and + * vendors. The caller also indicates whether the package is + * {@linkplain java.util.jar.Attributes.Name#SEALED sealed}. + * If a {@code Package} is not explicitly defined for a run-time package when + * a class in that run-time package is defined, then a {@code Package} is + * automatically defined by the class's defining class loader, as follows. + *

+ * A {@code Package} automatically defined for classes in a named module has + * the following properties: + *

    + *
  • The name of the package is derived from the {@linkplain Class#getName() binary names} + * of the classes. Since classes in a named module must be in a named package, + * the derived name is never empty.
  • + *
  • The package is sealed with the {@linkplain java.lang.module.ModuleReference#location() + * module location} as the code source, if known.
  • + *
  • The specification and implementation titles, versions, and vendors + * are unspecified.
  • + *
  • Any annotations on the package are read from {@code package-info.class} + * as specified above.
  • + *
+ *

+ * A {@code Package} automatically defined for classes in an unnamed module + * has the following properties: + *

    + *
  • The name of the package is either {@code ""} (for classes in an unnamed package) + * or derived from the {@linkplain Class#getName() binary names} of the classes + * (for classes in a named package).
  • + *
  • The package is not sealed.
  • + *
  • The specification and implementation titles, versions, and vendors + * are unspecified.
  • + *
  • Any annotations on the package are read from {@code package-info.class} + * as specified above.
  • + *
* - *

The set of classes that make up the package may implement a - * particular specification and if so the specification title, version number, - * and vendor strings identify that specification. - * An application can ask if the package is - * compatible with a particular version, see the {@link - * #isCompatibleWith isCompatibleWith} - * method for details. + *

+ * A {@code Package} can be obtained with the {@link Package#getPackage + * Package.getPackage(String)} and {@link ClassLoader#getDefinedPackage + * ClassLoader.getDefinedPackage(String)} methods. + * Every {@code Package} defined by a class loader can be obtained + * with the {@link Package#getPackages Package.getPackages()} and + * {@link ClassLoader#getDefinedPackages} methods. * - *

Specification version numbers use a syntax that consists of nonnegative - * decimal integers separated by periods ".", for example "2.0" or - * "1.2.3.4.5.6.7". This allows an extensible number to be used to represent - * major, minor, micro, etc. versions. The version specification is described - * by the following formal grammar: - *

- *
- *
SpecificationVersion: - *
Digits RefinedVersionopt - - *
RefinedVersion: - *
{@code .} Digits - *
{@code .} Digits RefinedVersion + * @jvms 5.3 Run-time package + * @see + * The JAR File Specification: Package Versioning + * @see + * The JAR File Specification: Package Sealing + * @see ClassLoader#definePackage(String, String, String, String, String, String, String, URL) * - *
Digits: - *
Digit - *
Digits - * - *
Digit: - *
any character for which {@link Character#isDigit} returns {@code true}, - * e.g. 0, 1, 2, ... - *
- *
- * - *

The implementation title, version, and vendor strings identify an - * implementation and are made available conveniently to enable accurate - * reporting of the packages involved when a problem occurs. The contents - * all three implementation strings are vendor specific. The - * implementation version strings have no specified syntax and should - * only be compared for equality with desired version identifiers. - * - *

Within each {@code ClassLoader} instance all classes from the same - * java package have the same Package object. The static methods allow a package - * to be found by name or the set of all packages known to the current class - * loader to be found. - * - * @see ClassLoader#definePackage * @since 1.2 */ -public class Package implements java.lang.reflect.AnnotatedElement { +public class Package extends NamedPackage implements java.lang.reflect.AnnotatedElement { /** * Return the name of this package. * @@ -112,47 +121,72 @@ public class Package implements java.lang.reflect.AnnotatedElement { * for example, {@code java.lang} */ public String getName() { - return pkgName; + return packageName(); } - /** * Return the title of the specification that this package implements. - * @return the specification title, null is returned if it is not known. + * @return the specification title, {@code null} is returned if it is not known. */ public String getSpecificationTitle() { - return specTitle; + return versionInfo.specTitle; } /** * Returns the version number of the specification * that this package implements. - * This version string must be a sequence of nonnegative decimal + * This version string must be a sequence of non-negative decimal * integers separated by "."'s and may have leading zeros. * When version strings are compared the most significant * numbers are compared. - * @return the specification version, null is returned if it is not known. + * + * + *

Specification version numbers use a syntax that consists of non-negative + * decimal integers separated by periods ".", for example "2.0" or + * "1.2.3.4.5.6.7". This allows an extensible number to be used to represent + * major, minor, micro, etc. versions. The version specification is described + * by the following formal grammar: + *

+ *
+ *
SpecificationVersion: + *
Digits RefinedVersionopt + + *
RefinedVersion: + *
{@code .} Digits + *
{@code .} Digits RefinedVersion + * + *
Digits: + *
Digit + *
Digits + * + *
Digit: + *
any character for which {@link Character#isDigit} returns {@code true}, + * e.g. 0, 1, 2, ... + *
+ *
+ * + * @return the specification version, {@code null} is returned if it is not known. */ public String getSpecificationVersion() { - return specVersion; + return versionInfo.specVersion; } /** * Return the name of the organization, vendor, * or company that owns and maintains the specification * of the classes that implement this package. - * @return the specification vendor, null is returned if it is not known. + * @return the specification vendor, {@code null} is returned if it is not known. */ public String getSpecificationVendor() { - return specVendor; + return versionInfo.specVendor; } /** * Return the title of this package. - * @return the title of the implementation, null is returned if it is not known. + * @return the title of the implementation, {@code null} is returned if it is not known. */ public String getImplementationTitle() { - return implTitle; + return versionInfo.implTitle; } /** @@ -162,19 +196,20 @@ public class Package implements java.lang.reflect.AnnotatedElement { * runtime. It may be compared for equality with other * package version strings used for this implementation * by this vendor for this package. - * @return the version of the implementation, null is returned if it is not known. + * @return the version of the implementation, {@code null} is returned if it is not known. */ public String getImplementationVersion() { - return implVersion; + return versionInfo.implVersion; } /** - * Returns the name of the organization, - * vendor or company that provided this implementation. - * @return the vendor that implemented this package.. + * Returns the vendor that implemented this package, {@code null} + * is returned if it is not known. + * @return the vendor that implemented this package, {@code null} + * is returned if it is not known. */ public String getImplementationVendor() { - return implVendor; + return versionInfo.implVendor; } /** @@ -183,17 +218,29 @@ public class Package implements java.lang.reflect.AnnotatedElement { * @return true if the package is sealed, false otherwise */ public boolean isSealed() { - return sealBase != null; + return module().isNamed() || versionInfo.sealBase != null; } /** * Returns true if this package is sealed with respect to the specified - * code source url. + * code source {@code url}. * - * @param url the code source url - * @return true if this package is sealed with respect to url + * @param url the code source URL + * @return true if this package is sealed with respect to the given {@code url} */ public boolean isSealed(URL url) { + Objects.requireNonNull(url); + + URL sealBase = null; + if (versionInfo != VersionInfo.NULL_VERSION_INFO) { + sealBase = versionInfo.sealBase; + } else { + try { + URI uri = location(); + sealBase = uri != null ? uri.toURL() : null; + } catch (MalformedURLException e) { + } + } return url.equals(sealBase); } @@ -216,17 +263,17 @@ public class Package implements java.lang.reflect.AnnotatedElement { * @return true if this package's version number is greater * than or equal to the desired version number * - * @exception NumberFormatException if the desired or current version - * is not of the correct dotted form. + * @exception NumberFormatException if the current version is not known or + * the desired or current version is not of the correct dotted form. */ public boolean isCompatibleWith(String desired) throws NumberFormatException { - if (specVersion == null || specVersion.length() < 1) { + if (versionInfo.specVersion == null || versionInfo.specVersion.length() < 1) { throw new NumberFormatException("Empty version string"); } - String [] sa = specVersion.split("\\.", -1); + String [] sa = versionInfo.specVersion.split("\\.", -1); int [] si = new int[sa.length]; for (int i = 0; i < sa.length; i++) { si[i] = Integer.parseInt(sa[i]); @@ -255,92 +302,71 @@ public class Package implements java.lang.reflect.AnnotatedElement { } /** - * Find a package by name in the callers {@code ClassLoader} instance. - * The callers {@code ClassLoader} instance is used to find the package - * instance corresponding to the named class. If the callers - * {@code ClassLoader} instance is null then the set of packages loaded - * by the system {@code ClassLoader} instance is searched to find the - * named package.

+ * Finds a package by name in the caller's class loader and its + * ancestors. + *

+ * If the caller's class loader defines a {@code Package} of the given name, + * the {@code Package} is returned. Otherwise, the ancestors of the + * caller's class loader are searched recursively (parent by parent) + * for a {@code Package} of the given name. + *

+ * Calling this method is equivalent to calling {@link ClassLoader#getPackage} + * on a {@code ClassLoader} instance which is the caller's class loader. * - * Packages have attributes for versions and specifications only if the class - * loader created the package instance with the appropriate attributes. Typically, - * those attributes are defined in the manifests that accompany the classes. + * @param name A package name, such as "{@code java.lang}". + * @return The {@code Package} of the given name defined by the caller's + * class loader or its ancestors, or {@code null} if not found. * - * @param name a package name, for example, java.lang. - * @return the package of the requested name. It may be null if no package - * information is available from the archive or codebase. + * @deprecated + * If multiple class loaders delegate to each other and define classes + * with the same package name, and one such loader relies on the lookup + * behavior of {@code getPackage} to return a {@code Package} from + * a parent loader, then the properties exposed by the {@code Package} + * may not be as expected in the rest of the program. + * For example, the {@code Package} will only expose annotations from the + * {@code package-info.class} file defined by the parent loader, even if + * annotations exist in a {@code package-info.class} file defined by + * a child loader. A more robust approach is to use the + * {@link ClassLoader#getDefinedPackage} method which returns + * a {@code Package} for the specified class loader. + * + * @see ClassLoader#getDefinedPackage */ @CallerSensitive + @Deprecated + @SuppressWarnings("deprecation") public static Package getPackage(String name) { ClassLoader l = ClassLoader.getClassLoader(Reflection.getCallerClass()); - if (l != null) { - return l.getPackage(name); - } else { - return getSystemPackage(name); - } + return l != null ? l.getPackage(name) : BootLoader.getDefinedPackage(name); } /** - * Get all the packages currently known for the caller's {@code ClassLoader} - * instance. Those packages correspond to classes loaded via or accessible by - * name to that {@code ClassLoader} instance. If the caller's - * {@code ClassLoader} instance is the bootstrap {@code ClassLoader} - * instance, which may be represented by {@code null} in some implementations, - * only packages corresponding to classes loaded by the bootstrap - * {@code ClassLoader} instance will be returned. + * Returns all of the {@code Package}s defined by the caller's class loader + * and its ancestors. The returned array may contain more than one + * {@code Package} object of the same package name, each defined by + * a different class loader in the class loader hierarchy. + *

+ * Calling this method is equivalent to calling {@link ClassLoader#getPackages} + * on a {@code ClassLoader} instance which is the caller's class loader. * - * @return a new array of packages known to the callers {@code ClassLoader} - * instance. An zero length array is returned if none are known. + * @return The array of {@code Package} objects defined by this + * class loader and its ancestors + * + * @see ClassLoader#getDefinedPackages */ @CallerSensitive public static Package[] getPackages() { - ClassLoader l = ClassLoader.getClassLoader(Reflection.getCallerClass()); - if (l != null) { - return l.getPackages(); - } else { - return getSystemPackages(); - } - } - - /** - * Get the package for the specified class. - * The class's class loader is used to find the package instance - * corresponding to the specified class. If the class loader - * is the bootstrap class loader, which may be represented by - * {@code null} in some implementations, then the set of packages - * loaded by the bootstrap class loader is searched to find the package. - *

- * Packages have attributes for versions and specifications only - * if the class loader created the package - * instance with the appropriate attributes. Typically those - * attributes are defined in the manifests that accompany - * the classes. - * - * @param c the class to get the package of. - * @return the package of the class. It may be null if no package - * information is available from the archive or codebase. */ - static Package getPackage(Class c) { - String name = c.getName(); - int i = name.lastIndexOf('.'); - if (i != -1) { - name = name.substring(0, i); - ClassLoader cl = c.getClassLoader(); - if (cl != null) { - return cl.getPackage(name); - } else { - return getSystemPackage(name); - } - } else { - return null; - } + ClassLoader cl = ClassLoader.getClassLoader(Reflection.getCallerClass()); + return cl != null ? cl.getPackages() : BootLoader.packages().toArray(Package[]::new); } /** * Return the hash code computed from the package name. * @return the hash code computed from the package name. */ + @Override public int hashCode(){ - return pkgName.hashCode(); + return packageName().hashCode(); } /** @@ -350,9 +376,10 @@ public class Package implements java.lang.reflect.AnnotatedElement { * If the package version is defined it is appended. * @return the string representation of the package. */ + @Override public String toString() { - String spec = specTitle; - String ver = specVersion; + String spec = versionInfo.specTitle; + String ver = versionInfo.specVersion; if (spec != null && spec.length() > 0) spec = ", " + spec; else @@ -361,14 +388,20 @@ public class Package implements java.lang.reflect.AnnotatedElement { ver = ", version " + ver; else ver = ""; - return "package " + pkgName + spec + ver; + return "package " + packageName() + spec + ver; } private Class getPackageInfo() { if (packageInfo == null) { - try { - packageInfo = Class.forName(pkgName + ".package-info", false, loader); - } catch (ClassNotFoundException ex) { + // find package-info.class defined by loader + String cn = packageName() + ".package-info"; + PrivilegedAction pa = module()::getClassLoader; + ClassLoader loader = AccessController.doPrivileged(pa); + Class c = loader != null ? loader.loadLocalClass(cn) + : BootLoader.loadClassOrNull(cn); + if (c != null) { + packageInfo = c; + } else { // store a proxy for the package info that has no annotations class PackageInfoProxy {} packageInfo = PackageInfoProxy.class; @@ -437,8 +470,12 @@ public class Package implements java.lang.reflect.AnnotatedElement { } /** - * Construct a package instance with the specified version - * information. + * Construct a package instance for an unnamed module + * with the specified version information. + * + * @apiNote + * This method should not be called to define a Package for named module. + * * @param name the name of the package * @param spectitle the title of the specification * @param specversion the version of the specification @@ -446,216 +483,73 @@ public class Package implements java.lang.reflect.AnnotatedElement { * @param impltitle the title of the implementation * @param implversion the version of the implementation * @param implvendor the organization that maintains the implementation + * @param sealbase code source where this Package comes from + * @param loader defining class loader */ Package(String name, String spectitle, String specversion, String specvendor, String impltitle, String implversion, String implvendor, URL sealbase, ClassLoader loader) { - pkgName = name; - implTitle = impltitle; - implVersion = implversion; - implVendor = implvendor; - specTitle = spectitle; - specVersion = specversion; - specVendor = specvendor; - sealBase = sealbase; - this.loader = loader; + super(Objects.requireNonNull(name), + loader != null ? loader.getUnnamedModule() + : BootLoader.getUnnamedModule()); + + this.versionInfo = VersionInfo.getInstance(spectitle, specversion, + specvendor, impltitle, + implversion, implvendor, + sealbase); + } + + Package(String name, Module module) { + super(name, module); + this.versionInfo = VersionInfo.NULL_VERSION_INFO; } /* - * Construct a package using the attributes from the specified manifest. - * - * @param name the package name - * @param man the optional manifest for the package - * @param url the optional code source url for the package + * Versioning information. Only for packages in unnamed modules. */ - private Package(String name, Manifest man, URL url, ClassLoader loader) { - String path = name.replace('.', '/').concat("/"); - String sealed = null; - String specTitle= null; - String specVersion= null; - String specVendor= null; - String implTitle= null; - String implVersion= null; - String implVendor= null; - URL sealBase= null; - Attributes attr = man.getAttributes(path); - if (attr != null) { - specTitle = attr.getValue(Name.SPECIFICATION_TITLE); - specVersion = attr.getValue(Name.SPECIFICATION_VERSION); - specVendor = attr.getValue(Name.SPECIFICATION_VENDOR); - implTitle = attr.getValue(Name.IMPLEMENTATION_TITLE); - implVersion = attr.getValue(Name.IMPLEMENTATION_VERSION); - implVendor = attr.getValue(Name.IMPLEMENTATION_VENDOR); - sealed = attr.getValue(Name.SEALED); - } - attr = man.getMainAttributes(); - if (attr != null) { - if (specTitle == null) { - specTitle = attr.getValue(Name.SPECIFICATION_TITLE); - } - if (specVersion == null) { - specVersion = attr.getValue(Name.SPECIFICATION_VERSION); - } - if (specVendor == null) { - specVendor = attr.getValue(Name.SPECIFICATION_VENDOR); - } - if (implTitle == null) { - implTitle = attr.getValue(Name.IMPLEMENTATION_TITLE); - } - if (implVersion == null) { - implVersion = attr.getValue(Name.IMPLEMENTATION_VERSION); - } - if (implVendor == null) { - implVendor = attr.getValue(Name.IMPLEMENTATION_VENDOR); - } - if (sealed == null) { - sealed = attr.getValue(Name.SEALED); - } - } - if ("true".equalsIgnoreCase(sealed)) { - sealBase = url; - } - pkgName = name; - this.specTitle = specTitle; - this.specVersion = specVersion; - this.specVendor = specVendor; - this.implTitle = implTitle; - this.implVersion = implVersion; - this.implVendor = implVendor; - this.sealBase = sealBase; - this.loader = loader; - } + static class VersionInfo { + static final VersionInfo NULL_VERSION_INFO + = new VersionInfo(null, null, null, null, null, null, null); - /* - * Returns the loaded system package for the specified name. - */ - static Package getSystemPackage(String name) { - Package pkg = pkgs.get(name); - if (pkg == null) { - name = name.replace('.', '/').concat("/"); - String fn = getSystemPackage0(name); - if (fn != null) { - pkg = defineSystemPackage(name, fn); + private final String specTitle; + private final String specVersion; + private final String specVendor; + private final String implTitle; + private final String implVersion; + private final String implVendor; + private final URL sealBase; + + static VersionInfo getInstance(String spectitle, String specversion, + String specvendor, String impltitle, + String implversion, String implvendor, + URL sealbase) { + if (spectitle == null && specversion == null && + specvendor == null && impltitle == null && + implvendor == null && sealbase == null) { + return NULL_VERSION_INFO; } - } - return pkg; - } - - /* - * Return an array of loaded system packages. - */ - static Package[] getSystemPackages() { - // First, update the system package map with new package names - String[] names = getSystemPackages0(); - for (String name : names) { - if (!pkgs.containsKey(name)) { - defineSystemPackage(name, getSystemPackage0(name)); - } - } - return pkgs.values().toArray(new Package[pkgs.size()]); - } - - private static Package defineSystemPackage(final String iname, - final String fn) - { - // Convert to "."-separated package name - String name = iname.substring(0, iname.length() - 1).replace('/', '.'); - // Creates a cached manifest for the file name, allowing - // only-once, lazy reads of manifest from jar files - CachedManifest cachedManifest = createCachedManifest(fn); - pkgs.putIfAbsent(name, new Package(name, cachedManifest.getManifest(), - cachedManifest.getURL(), null)); - // Ensure we only expose one Package object - return pkgs.get(name); - } - - private static CachedManifest createCachedManifest(String fn) { - if (!manifests.containsKey(fn)) { - manifests.putIfAbsent(fn, new CachedManifest(fn)); - } - return manifests.get(fn); - } - - // The map of loaded system packages - private static final ConcurrentHashMap pkgs - = new ConcurrentHashMap<>(); - - // Maps each directory or zip file name to its corresponding manifest, if - // it exists - private static final ConcurrentHashMap manifests - = new ConcurrentHashMap<>(); - - private static class CachedManifest { - private static final Manifest EMPTY_MANIFEST = new Manifest(); - private final String fileName; - private final URL url; - private volatile Manifest manifest; - - CachedManifest(final String fileName) { - this.fileName = fileName; - this.url = AccessController.doPrivileged(new PrivilegedAction<>() { - public URL run() { - final File file = new File(fileName); - if (file.isFile()) { - try { - return ParseUtil.fileToEncodedURL(file); - } catch (MalformedURLException e) { - } - } - return null; - } - }); + return new VersionInfo(spectitle, specversion, specvendor, + impltitle, implversion, implvendor, + sealbase); } - public URL getURL() { - return url; - } - - public Manifest getManifest() { - if (url == null) { - return EMPTY_MANIFEST; - } - Manifest m = manifest; - if (m != null) { - return m; - } - synchronized (this) { - m = manifest; - if (m != null) { - return m; - } - m = AccessController.doPrivileged(new PrivilegedAction<>() { - public Manifest run() { - try (FileInputStream fis = new FileInputStream(fileName); - JarInputStream jis = new JarInputStream(fis, false)) { - return jis.getManifest(); - } catch (IOException e) { - return null; - } - } - }); - manifest = m = (m == null ? EMPTY_MANIFEST : m); - } - return m; + private VersionInfo(String spectitle, String specversion, + String specvendor, String impltitle, + String implversion, String implvendor, + URL sealbase) + { + this.implTitle = impltitle; + this.implVersion = implversion; + this.implVendor = implvendor; + this.specTitle = spectitle; + this.specVersion = specversion; + this.specVendor = specvendor; + this.sealBase = sealbase; } } - private static native String getSystemPackage0(String name); - private static native String[] getSystemPackages0(); - - /* - * Private storage for the package name and attributes. - */ - private final String pkgName; - private final String specTitle; - private final String specVersion; - private final String specVendor; - private final String implTitle; - private final String implVersion; - private final String implVendor; - private final URL sealBase; - private final transient ClassLoader loader; - private transient Class packageInfo; + private final VersionInfo versionInfo; + private Class packageInfo; } diff --git a/jdk/src/java.base/share/classes/java/lang/StackFrameInfo.java b/jdk/src/java.base/share/classes/java/lang/StackFrameInfo.java index 5c7fbde5810..0750af15b03 100644 --- a/jdk/src/java.base/share/classes/java/lang/StackFrameInfo.java +++ b/jdk/src/java.base/share/classes/java/lang/StackFrameInfo.java @@ -29,6 +29,7 @@ import jdk.internal.misc.SharedSecrets; import static java.lang.StackWalker.Option.*; import java.lang.StackWalker.StackFrame; +import java.lang.reflect.Module; import java.util.Optional; import java.util.OptionalInt; @@ -133,4 +134,21 @@ class StackFrameInfo implements StackFrame { static native void fillInStackFrames(int startIndex, Object[] stackframes, int fromIndex, int toIndex); + + @Override + public StackTraceElement toStackTraceElement() { + ensureMethodInfoInitialized(); + + Module module = declaringClass.getModule(); + String moduleName = module.isNamed() ? module.getName() : null; + String moduleVersion = null; + if (module.isNamed() && module.getDescriptor().version().isPresent()) { + moduleVersion = module.getDescriptor().version().get().toString(); + } + return new StackTraceElement(moduleName, moduleVersion, + getClassName(), getMethodName(), + fileName, + lineNumber); + } + } diff --git a/jdk/src/java.base/share/classes/java/lang/StackTraceElement.java b/jdk/src/java.base/share/classes/java/lang/StackTraceElement.java index 58738d83c8c..1ad9a9cd9ac 100644 --- a/jdk/src/java.base/share/classes/java/lang/StackTraceElement.java +++ b/jdk/src/java.base/share/classes/java/lang/StackTraceElement.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 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 @@ -41,6 +41,8 @@ import java.util.Objects; */ public final class StackTraceElement implements java.io.Serializable { // Normally initialized by VM (public constructor added in 1.5) + private String moduleName; + private String moduleVersion; private String declaringClass; private String methodName; private String fileName; @@ -48,7 +50,9 @@ public final class StackTraceElement implements java.io.Serializable { /** * Creates a stack trace element representing the specified execution - * point. + * point. The {@link #getModuleName module name} and {@link + * #getModuleVersion module version} of the stack trace element will + * be {@code null}. * * @param declaringClass the fully qualified name of the class containing * the execution point represented by the stack trace element @@ -68,6 +72,40 @@ public final class StackTraceElement implements java.io.Serializable { */ public StackTraceElement(String declaringClass, String methodName, String fileName, int lineNumber) { + this(null, null, declaringClass, methodName, fileName, lineNumber); + } + + /** + * Creates a stack trace element representing the specified execution + * point. + * + * @param moduleName the module name if the class containing the + * execution point represented by the stack trace is in a named + * module; can be {@code null} + * @param moduleVersion the module version if the class containing the + * execution point represented by the stack trace is in a named + * module that has a version; can be {@code null} + * @param declaringClass the fully qualified name of the class containing + * the execution point represented by the stack trace element + * @param methodName the name of the method containing the execution point + * represented by the stack trace element + * @param fileName the name of the file containing the execution point + * represented by the stack trace element, or {@code null} if + * this information is unavailable + * @param lineNumber the line number of the source line containing the + * execution point represented by this stack trace element, or + * a negative number if this information is unavailable. A value + * of -2 indicates that the method containing the execution point + * is a native method + * @throws NullPointerException if {@code declaringClass} is {@code null} + * or {@code methodName} is {@code null} + * @since 9 + */ + public StackTraceElement(String moduleName, String moduleVersion, + String declaringClass, String methodName, + String fileName, int lineNumber) { + this.moduleName = moduleName; + this.moduleVersion = moduleVersion; this.declaringClass = Objects.requireNonNull(declaringClass, "Declaring class is null"); this.methodName = Objects.requireNonNull(methodName, "Method name is null"); this.fileName = fileName; @@ -105,6 +143,34 @@ public final class StackTraceElement implements java.io.Serializable { return lineNumber; } + /** + * Returns the module name of the module containing the execution point + * represented by this stack trace element. + * + * @return the module name of the {@code Module} containing the execution + * point represented by this stack trace element; {@code null} + * if the module name is not available. + * @since 9 + * @see java.lang.reflect.Module#getName() + */ + public String getModuleName() { + return moduleName; + } + + /** + * Returns the module version of the module containing the execution point + * represented by this stack trace element. + * + * @return the module version of the {@code Module} containing the execution + * point represented by this stack trace element; {@code null} + * if the module version is not available. + * @since 9 + * @see java.lang.module.ModuleDescriptor.Version + */ + public String getModuleVersion() { + return moduleVersion; + } + /** * Returns the fully qualified name of the class containing the * execution point represented by this stack trace element. @@ -148,32 +214,42 @@ public final class StackTraceElement implements java.io.Serializable { * examples may be regarded as typical: *

    *
  • - * {@code "MyClass.mash(MyClass.java:9)"} - Here, {@code "MyClass"} - * is the fully-qualified name of the class containing the - * execution point represented by this stack trace element, + * {@code "MyClass.mash(my.module@9.0/MyClass.java:101)"} - Here, + * {@code "MyClass"} is the fully-qualified name of the class + * containing the execution point represented by this stack trace element, * {@code "mash"} is the name of the method containing the execution - * point, {@code "MyClass.java"} is the source file containing the - * execution point, and {@code "9"} is the line number of the source + * point, {@code "my.module"} is the module name, {@code "9.0"} is the + * module version, and {@code "101"} is the line number of the source * line containing the execution point. *
  • - * {@code "MyClass.mash(MyClass.java)"} - As above, but the line - * number is unavailable. + * {@code "MyClass.mash(my.module@9.0/MyClass.java)"} - As above, but the + * line number is unavailable. *
  • - * {@code "MyClass.mash(Unknown Source)"} - As above, but neither - * the file name nor the line number are available. + * {@code "MyClass.mash(my.module@9.0/Unknown Source)"} - As above, but + * neither the file name nor the line number are available. *
  • - * {@code "MyClass.mash(Native Method)"} - As above, but neither - * the file name nor the line number are available, and the method - * containing the execution point is known to be a native method. + * {@code "MyClass.mash(my.module@9.0/Native Method)"} - As above, but + * neither the file name nor the line number are available, and the + * method containing the execution point is known to be a native method. *
+ * If the execution point is not in a named module, {@code "my.module@9.0/"} + * will be omitted from the above. + * * @see Throwable#printStackTrace() */ public String toString() { - return getClassName() + "." + methodName + - (isNativeMethod() ? "(Native Method)" : - (fileName != null && lineNumber >= 0 ? - "(" + fileName + ":" + lineNumber + ")" : - (fileName != null ? "("+fileName+")" : "(Unknown Source)"))); + String mid = ""; + if (moduleName != null) { + mid = moduleName; + if (moduleVersion != null) + mid += "@" + moduleVersion; + mid += "/"; + } + return getClassName() + "." + methodName + "(" + mid + + (isNativeMethod() ? "Native Method)" : + (fileName != null && lineNumber >= 0 ? + fileName + ":" + lineNumber + ")" : + (fileName != null ? ""+fileName+")" : "Unknown Source)"))); } /** @@ -184,6 +260,8 @@ public final class StackTraceElement implements java.io.Serializable { *
{@code
      *     equals(a.getFileName(), b.getFileName()) &&
      *     a.getLineNumber() == b.getLineNumber()) &&
+     *     equals(a.getModuleName(), b.getModuleName()) &&
+     *     equals(a.getModuleVersion(), b.getModuleVersion()) &&
      *     equals(a.getClassName(), b.getClassName()) &&
      *     equals(a.getMethodName(), b.getMethodName())
      * }
@@ -202,6 +280,8 @@ public final class StackTraceElement implements java.io.Serializable { return false; StackTraceElement e = (StackTraceElement)obj; return e.declaringClass.equals(declaringClass) && + Objects.equals(moduleName, e.moduleName) && + Objects.equals(moduleVersion, e.moduleVersion) && e.lineNumber == lineNumber && Objects.equals(methodName, e.methodName) && Objects.equals(fileName, e.fileName); @@ -212,6 +292,8 @@ public final class StackTraceElement implements java.io.Serializable { */ public int hashCode() { int result = 31*declaringClass.hashCode() + methodName.hashCode(); + result = 31*result + Objects.hashCode(moduleName); + result = 31*result + Objects.hashCode(moduleVersion); result = 31*result + Objects.hashCode(fileName); result = 31*result + lineNumber; return result; diff --git a/jdk/src/java.base/share/classes/java/lang/StackWalker.java b/jdk/src/java.base/share/classes/java/lang/StackWalker.java index f628459dfb3..636762e17d4 100644 --- a/jdk/src/java.base/share/classes/java/lang/StackWalker.java +++ b/jdk/src/java.base/share/classes/java/lang/StackWalker.java @@ -170,15 +170,8 @@ public final class StackWalker { * Gets a {@code StackTraceElement} for this stack frame. * * @return {@code StackTraceElement} for this stack frame. - * - * */ - public default StackTraceElement toStackTraceElement() { - int lineNumber = isNativeMethod() ? -2 - : getLineNumber().orElse(-1); - return new StackTraceElement(getClassName(), getMethodName(), - getFileName().orElse(null), - lineNumber); - } + */ + public StackTraceElement toStackTraceElement(); } /** diff --git a/jdk/src/java.base/share/classes/java/lang/System.java b/jdk/src/java.base/share/classes/java/lang/System.java index c0c7c05156a..92fee1a4337 100644 --- a/jdk/src/java.base/share/classes/java/lang/System.java +++ b/jdk/src/java.base/share/classes/java/lang/System.java @@ -24,9 +24,23 @@ */ package java.lang; -import java.io.*; -import java.lang.reflect.Executable; +import java.io.BufferedInputStream; +import java.io.BufferedOutputStream; +import java.io.Console; +import java.io.FileDescriptor; +import java.io.FileInputStream; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.PrintStream; +import java.io.UnsupportedEncodingException; import java.lang.annotation.Annotation; +import java.lang.reflect.Constructor; +import java.lang.reflect.Executable; +import java.lang.reflect.Layer; +import java.lang.reflect.Modifier; +import java.lang.reflect.Module; +import java.net.URL; import java.security.AccessControlContext; import java.util.Properties; import java.util.PropertyPermission; @@ -35,6 +49,8 @@ import java.security.AccessController; import java.security.PrivilegedAction; import java.nio.channels.Channel; import java.nio.channels.spi.SelectorProvider; +import java.util.stream.Stream; + import java.util.Objects; import java.util.ResourceBundle; import java.util.function.Supplier; @@ -51,6 +67,10 @@ import jdk.internal.logger.LoggerFinderLoader; import jdk.internal.logger.LazyLoggers; import jdk.internal.logger.LocalizedLoggerWrapper; +import jdk.internal.module.ModuleBootstrap; +import jdk.internal.module.Modules; +import jdk.internal.module.ServicesCatalog; + /** * The System class contains several useful class fields * and methods. It cannot be instantiated. @@ -65,7 +85,6 @@ import jdk.internal.logger.LocalizedLoggerWrapper; * @since 1.0 */ public final class System { - /* register the natives via the static initializer. * * VM will invoke the initializeSystemClass method to complete @@ -619,6 +638,24 @@ public final class System { * getProperties operation, it may choose to permit the * {@link #getProperty(String)} operation. * + * @implNote In addition to the standard system properties, the {@code + * java} launcher may create the Java Virtual Machine with system + * properties that have the following keys: + * + * + * + * + * + * + * + * + * + * + * + *
KeyDescription of Associated Value
{@code jdk.module.path}Application module path
{@code jdk.upgrade.module.path}The upgrade module path
{@code jdk.module.main}The module name of the initial/main module
{@code jdk.module.main.class}The main class name of the initial module
+ * These properties may also be set by custom launchers that use the JNI + * invocation API to create the Java Virtual Machine. + * * @return the system properties * @exception SecurityException if a security manager exists and its * checkPropertiesAccess method doesn't allow access @@ -1795,11 +1832,10 @@ public final class System { return new PrintStream(new BufferedOutputStream(fos, 128), true); } - /** * Initialize the system class. Called after thread initialization. */ - private static void initializeSystemClass() { + private static void initPhase1() { // VM might invoke JNU_NewStringPlatform() to set those encoding // sensitive properties (user.home, user.name, boot.class.path, etc.) @@ -1828,7 +1864,6 @@ public final class System { // certain system properties that are not intended for public access. VM.saveAndRemoveProperties(props); - lineSeparator = props.getProperty("line.separator"); VersionProps.init(); @@ -1862,9 +1897,81 @@ public final class System { // Subsystems that are invoked during initialization can invoke // VM.isBooted() in order to avoid doing things that should - // wait until the application class loader has been set up. + // wait until the VM is fully initialized. The initialization level + // is incremented from 0 to 1 here to indicate the first phase of + // initialization has completed. // IMPORTANT: Ensure that this remains the last initialization action! - VM.booted(); + VM.initLevel(1); + } + + // @see #initPhase2() + private static Layer bootLayer; + + /* + * Invoked by VM. Phase 2 module system initialization. + * Only classes in java.base can be loaded in this phase. + */ + private static void initPhase2() { + // initialize the module system + System.bootLayer = ModuleBootstrap.boot(); + + // base module needs to be loose (CODETOOLS-7901619) + Module base = Object.class.getModule(); + Modules.addReads(base, null); + + // module system initialized + VM.initLevel(2); + } + + /* + * Invoked by VM. Phase 3 is the final system initialization: + * 1. set security manager + * 2. set system class loader + * 3. set TCCL + * + * This method must be called after the module system initialization. + * The security manager and system class loader may be custom class from + * the application classpath or modulepath. + */ + private static void initPhase3() { + // set security manager + String cn = System.getProperty("java.security.manager"); + if (cn != null) { + if (cn.isEmpty() || "default".equals(cn)) { + System.setSecurityManager(new SecurityManager()); + } else { + try { + Class c = Class.forName(cn, false, ClassLoader.getBuiltinAppClassLoader()); + Constructor ctor = c.getConstructor(); + // Must be a public subclass of SecurityManager with + // a public no-arg constructor + if (!SecurityManager.class.isAssignableFrom(c) || + !Modifier.isPublic(c.getModifiers()) || + !Modifier.isPublic(ctor.getModifiers())) { + throw new Error("Could not create SecurityManager: " + ctor.toString()); + } + // custom security manager implementation may be in unnamed module + // or a named module but non-exported package + ctor.setAccessible(true); + SecurityManager sm = (SecurityManager) ctor.newInstance(); + System.setSecurityManager(sm); + } catch (Exception e) { + throw new Error("Could not create SecurityManager", e); + } + } + } + + // initializing the system class loader + VM.initLevel(3); + + // system class loader initialized + ClassLoader scl = ClassLoader.initSystemClassLoader(); + + // set TCCL + Thread.currentThread().setContextClassLoader(scl); + + // system is fully initialized + VM.initLevel(4); } private static void setJavaLangAccess() { @@ -1910,6 +2017,27 @@ public final class System { public void invokeFinalize(Object o) throws Throwable { o.finalize(); } + public Layer getBootLayer() { + return bootLayer; + } + public ServicesCatalog getServicesCatalog(ClassLoader cl) { + return cl.getServicesCatalog(); + } + public ServicesCatalog createOrGetServicesCatalog(ClassLoader cl) { + return cl.createOrGetServicesCatalog(); + } + public Class findBootstrapClassOrNull(ClassLoader cl, String name) { + return cl.findBootstrapClassOrNull(name); + } + public URL findResource(ClassLoader cl, String mn, String name) throws IOException { + return cl.findResource(mn, name); + } + public Stream packages(ClassLoader cl) { + return cl.packages(); + } + public Package definePackage(ClassLoader cl, String name, Module module) { + return cl.definePackage(name, module); + } public String fastUUID(long lsb, long msb) { return Long.fastUUID(lsb, msb); } diff --git a/jdk/src/java.base/share/classes/java/lang/invoke/BoundMethodHandle.java b/jdk/src/java.base/share/classes/java/lang/invoke/BoundMethodHandle.java index 90d0d590697..3d9d59fc89e 100644 --- a/jdk/src/java.base/share/classes/java/lang/invoke/BoundMethodHandle.java +++ b/jdk/src/java.base/share/classes/java/lang/invoke/BoundMethodHandle.java @@ -788,7 +788,8 @@ import static jdk.internal.org.objectweb.asm.Opcodes.*; static void setSpeciesDataToConcreteBMHClass(Class cbmh, SpeciesData speciesData) { try { Field F_SPECIES_DATA = cbmh.getDeclaredField("SPECIES_DATA"); - assert F_SPECIES_DATA.getDeclaredAnnotation(Stable.class) != null; + // ## FIXME: annotation parser can't create proxy classes until module system is fully initialzed + // assert F_SPECIES_DATA.getDeclaredAnnotation(Stable.class) != null; F_SPECIES_DATA.set(null, speciesData); } catch (ReflectiveOperationException ex) { throw newInternalError(ex); diff --git a/jdk/src/java.base/share/classes/java/lang/invoke/MemberName.java b/jdk/src/java.base/share/classes/java/lang/invoke/MemberName.java index 50e28e56cab..3c6b0d14d50 100644 --- a/jdk/src/java.base/share/classes/java/lang/invoke/MemberName.java +++ b/jdk/src/java.base/share/classes/java/lang/invoke/MemberName.java @@ -33,6 +33,7 @@ import java.lang.reflect.Field; import java.lang.reflect.Method; import java.lang.reflect.Member; import java.lang.reflect.Modifier; +import java.lang.reflect.Module; import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; @@ -475,8 +476,9 @@ import java.util.Objects; /** Utility method to query whether this member is accessible from a given lookup class. */ public boolean isAccessibleFrom(Class lookupClass) { + int mode = (ALL_ACCESS|MethodHandles.Lookup.PACKAGE|MethodHandles.Lookup.MODULE); return VerifyAccess.isMemberAccessible(this.getDeclaringClass(), this.getDeclaringClass(), flags, - lookupClass, ALL_ACCESS|MethodHandles.Lookup.PACKAGE); + lookupClass, mode); } /** Initialize a query. It is not resolved. */ @@ -851,7 +853,20 @@ import java.util.Objects; public IllegalAccessException makeAccessException(String message, Object from) { message = message + ": "+ toString(); - if (from != null) message += ", from " + from; + if (from != null) { + if (from == MethodHandles.publicLookup()) { + message += ", from public Lookup"; + } else { + Module m; + if (from instanceof MethodHandles.Lookup) { + MethodHandles.Lookup lookup = (MethodHandles.Lookup)from; + m = lookup.lookupClass().getModule(); + } else { + m = from.getClass().getModule(); + } + message += ", from " + from + " (" + m + ")"; + } + } return new IllegalAccessException(message); } private String message() { diff --git a/jdk/src/java.base/share/classes/java/lang/invoke/MethodHandleImpl.java b/jdk/src/java.base/share/classes/java/lang/invoke/MethodHandleImpl.java index 9b4029cc7b5..ed1dccc8041 100644 --- a/jdk/src/java.base/share/classes/java/lang/invoke/MethodHandleImpl.java +++ b/jdk/src/java.base/share/classes/java/lang/invoke/MethodHandleImpl.java @@ -1191,14 +1191,9 @@ import static java.lang.invoke.MethodHandles.Lookup.IMPL_LOOKUP; Class tClass = T.class; String tName = tClass.getName(); String tResource = tName.substring(tName.lastIndexOf('.')+1)+".class"; - java.net.URLConnection uconn = tClass.getResource(tResource).openConnection(); - int len = uconn.getContentLength(); - byte[] bytes = new byte[len]; - try (java.io.InputStream str = uconn.getInputStream()) { - int nr = str.read(bytes); - if (nr != len) throw new java.io.IOException(tResource); + try (java.io.InputStream in = tClass.getResourceAsStream(tResource)) { + values[0] = in.readAllBytes(); } - values[0] = bytes; } catch (java.io.IOException ex) { throw new InternalError(ex); } diff --git a/jdk/src/java.base/share/classes/java/lang/invoke/MethodHandles.java b/jdk/src/java.base/share/classes/java/lang/invoke/MethodHandles.java index 70791273212..9bf86d9d2c3 100644 --- a/jdk/src/java.base/share/classes/java/lang/invoke/MethodHandles.java +++ b/jdk/src/java.base/share/classes/java/lang/invoke/MethodHandles.java @@ -26,7 +26,13 @@ package java.lang.invoke; import java.lang.reflect.*; -import java.util.*; +import java.util.ArrayList; +import java.util.BitSet; +import java.util.List; +import java.util.Arrays; +import java.util.Objects; +import java.security.AccessController; +import java.security.PrivilegedAction; import sun.invoke.util.ValueConversions; import sun.invoke.util.VerifyAccess; @@ -44,6 +50,9 @@ import java.util.concurrent.ConcurrentHashMap; import java.util.stream.Collectors; import java.util.stream.Stream; +import jdk.internal.org.objectweb.asm.ClassWriter; +import jdk.internal.org.objectweb.asm.Opcodes; + /** * This class consists exclusively of static methods that operate on or return * method handles. They fall into several categories: @@ -94,18 +103,16 @@ public class MethodHandles { /** * Returns a {@link Lookup lookup object} which is trusted minimally. - * It can only be used to create method handles to - * publicly accessible fields and methods. + * It can only be used to create method handles to public members in + * public classes in packages that are exported unconditionally. *

- * As a matter of pure convention, the {@linkplain Lookup#lookupClass lookup class} - * of this lookup object will be {@link java.lang.Object}. + * For now, the {@linkplain Lookup#lookupClass lookup class} of this lookup + * object is in an unnamed module. * *

* Discussion: * The lookup class can be changed to any other class {@code C} using an expression of the form * {@link Lookup#in publicLookup().in(C.class)}. - * Since all classes have equal access to public names, - * such a change would confer no new access rights. * A public lookup object is always subject to * security manager checks. * Also, it cannot access @@ -113,7 +120,15 @@ public class MethodHandles { * @return a lookup object which is trusted minimally */ public static Lookup publicLookup() { - return Lookup.PUBLIC_LOOKUP; + // During VM startup then only classes in the java.base module can be + // loaded and linked. This is because java.base exports aren't setup until + // the module system is initialized, hence types in the unnamed module + // (or any named module) can't link to java/lang/Object. + if (!jdk.internal.misc.VM.isModuleSystemInited()) { + return new Lookup(Object.class, Lookup.PUBLIC); + } else { + return LookupHelper.PUBLIC_LOOKUP; + } } /** @@ -386,7 +401,7 @@ public class MethodHandles { * to a subset of members normally accessible to the lookup class. * For example, the {@link MethodHandles#publicLookup publicLookup} * method produces a lookup object which is only allowed to access - * public members in public classes. + * public members in public classes of exported packages. * The caller sensitive method {@link MethodHandles#lookup lookup} * produces a lookup object with full capabilities relative to * its caller class, to emulate all supported bytecode behaviors. @@ -558,12 +573,24 @@ public class MethodHandles { */ public static final int PACKAGE = Modifier.STATIC; - private static final int ALL_MODES = (PUBLIC | PRIVATE | PROTECTED | PACKAGE); + /** A single-bit mask representing {@code module} access (default access), + * which may contribute to the result of {@link #lookupModes lookupModes}. + * The value is {@code 0x10}, which does not correspond meaningfully to + * any particular {@linkplain java.lang.reflect.Modifier modifier bit}. + * In conjunction with the {@code PUBLIC} modifier bit, a {@code Lookup} + * with this lookup mode can access all public types in the module of the + * lookup class and public types in packages exported by other modules + * to the module of the lookup class. + * @since 9 + */ + public static final int MODULE = PACKAGE << 1; + + private static final int ALL_MODES = (PUBLIC | PRIVATE | PROTECTED | PACKAGE | MODULE); private static final int TRUSTED = -1; private static int fixmods(int mods) { - mods &= (ALL_MODES - PACKAGE); - return (mods != 0) ? mods : PACKAGE; + mods &= (ALL_MODES - PACKAGE - MODULE); + return (mods != 0) ? mods : (PACKAGE | MODULE); } /** Tells which class is performing the lookup. It is this class against @@ -589,11 +616,14 @@ public class MethodHandles { * {@linkplain #PUBLIC PUBLIC (0x01)}, * {@linkplain #PRIVATE PRIVATE (0x02)}, * {@linkplain #PROTECTED PROTECTED (0x04)}, - * and {@linkplain #PACKAGE PACKAGE (0x08)}. + * {@linkplain #PACKAGE PACKAGE (0x08)}, + * and {@linkplain #MODULE MODULE (0x10)}. *

* A freshly-created lookup object * on the {@linkplain java.lang.invoke.MethodHandles#lookup() caller's class} - * has all possible bits set, since the caller class can access all its own members. + * has all possible bits set, since the caller class can access all its own members, + * all public types in the caller's module, and all public types in packages exported + * by other modules to the caller's module. * A lookup object on a new lookup class * {@linkplain java.lang.invoke.MethodHandles.Lookup#in created from a previous lookup object} * may have some mode bits set to zero. @@ -630,6 +660,12 @@ public class MethodHandles { * However, the resulting {@code Lookup} object is guaranteed * to have no more access capabilities than the original. * In particular, access capabilities can be lost as follows: