diff --git a/hotspot/test/applications/ctw/Modules.java b/hotspot/test/applications/ctw/Modules.java index 4d628edcf02..8d100e1a4e7 100644 --- a/hotspot/test/applications/ctw/Modules.java +++ b/hotspot/test/applications/ctw/Modules.java @@ -33,6 +33,6 @@ * @build sun.hotspot.WhiteBox * @run driver ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission - * @run driver/timeout=0 sun.hotspot.tools.ctw.CtwRunner modules + * @run main/timeout=0 sun.hotspot.tools.ctw.CtwRunner modules */ diff --git a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathDirEntry.java b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathDirEntry.java index bafe2dd1532..0831e537cb0 100644 --- a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathDirEntry.java +++ b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathDirEntry.java @@ -66,6 +66,16 @@ public class ClassPathDirEntry extends PathHandler { } } + @Override + public long classCount() { + try { + return Files.walk(root, FileVisitOption.FOLLOW_LINKS).count(); + } catch (IOException e) { + throw new Error("can not walk dir " + root + " : " + + e.getMessage(), e); + } + } + private void processFile(Path file) { if (Utils.isClassFile(file.toString())) { processClass(pathToClassName(file)); diff --git a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJarEntry.java b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJarEntry.java index dd31c77a7a8..f93126a1732 100644 --- a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJarEntry.java +++ b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJarEntry.java @@ -69,7 +69,20 @@ public class ClassPathJarEntry extends PathHandler { } } - private void processJarEntry(JarEntry entry) { + @Override + public long classCount() { + try (JarFile jarFile = new JarFile(root.toFile())) { + return jarFile.stream() + .map(JarEntry::getName) + .filter(Utils::isClassFile) + .count(); + } catch (IOException e) { + throw new Error("can not open jar file " + root + " : " + + e.getMessage() , e); + } + } + + private void processJarEntry(JarEntry entry) { String filename = entry.getName(); if (Utils.isClassFile(filename)) { processClass(Utils.fileNameToClassName(filename)); diff --git a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJarInDirEntry.java b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJarInDirEntry.java index c9bdcb19f08..1996bc39370 100644 --- a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJarInDirEntry.java +++ b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJarInDirEntry.java @@ -56,5 +56,18 @@ public class ClassPathJarInDirEntry extends PathHandler { ioe.printStackTrace(); } } + + @Override + public long classCount() { + try { + return Files.list(root) + .filter(p -> p.getFileName().toString().endsWith(".jar")) + .map(p -> new ClassPathJarEntry(p, executor)) + .mapToLong(ClassPathJarEntry::classCount).sum(); + } catch (IOException e) { + throw new Error("can not walk dir " + root + " : " + + e.getMessage(), e); + } + } } diff --git a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJimageEntry.java b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJimageEntry.java index e768aac579f..e09085dbeb9 100644 --- a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJimageEntry.java +++ b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassPathJimageEntry.java @@ -64,4 +64,18 @@ public class ClassPathJimageEntry extends PathHandler { ioe.printStackTrace(); } } + + @Override + public long classCount() { + try (ImageReader reader = ImageReader.open(root)) { + return Arrays.stream(reader.getEntryNames()) + .filter(name -> name.endsWith(".class")) + .filter(name -> !name.endsWith("module-info.class")) + .map(Utils::fileNameToClassName) + .count(); + } catch (IOException e) { + throw new Error("can not open jimage file " + root + " : " + + e.getMessage() , e); + } + } } diff --git a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassesListInFile.java b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassesListInFile.java index 7c810dd3840..ca61beeb07b 100644 --- a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassesListInFile.java +++ b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/ClassesListInFile.java @@ -45,8 +45,7 @@ public class ClassesListInFile extends PathHandler { return; } try { - try (BufferedReader reader = Files.newBufferedReader(root, - StandardCharsets.UTF_8)) { + try (BufferedReader reader = Files.newBufferedReader(root)) { String line; while (!isFinished() && ((line = reader.readLine()) != null)) { processClass(line); @@ -56,4 +55,16 @@ public class ClassesListInFile extends PathHandler { e.printStackTrace(); } } + + @Override + public long classCount() { + try { + try (BufferedReader reader = Files.newBufferedReader(root)) { + return reader.lines().count(); + } + } catch (IOException e) { + throw new Error("can not read list " + root + " : " + + e.getMessage(), e); + } + } } diff --git a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/CompileTheWorld.java b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/CompileTheWorld.java index 68282f9ac83..6799a112f08 100644 --- a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/CompileTheWorld.java +++ b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/CompileTheWorld.java @@ -83,7 +83,7 @@ public class CompileTheWorld { await(executor); } CompileTheWorld.OUT.printf("Done (%d classes, %d methods, %d ms)%n", - PathHandler.getClassCount(), + PathHandler.getProcessedClassCount(), Compiler.getMethodCount(), System.currentTimeMillis() - start); passed = true; diff --git a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/CtwRunner.java b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/CtwRunner.java index fd6585dfe2d..c84059b2d8f 100644 --- a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/CtwRunner.java +++ b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/CtwRunner.java @@ -23,6 +23,7 @@ package sun.hotspot.tools.ctw; +import jdk.test.lib.Asserts; import jdk.test.lib.Utils; import jdk.test.lib.process.ProcessTools; import jdk.test.lib.util.Pair; @@ -34,6 +35,8 @@ import java.nio.file.Path; import java.nio.file.Paths; import java.util.ArrayList; import java.util.List; +import java.util.concurrent.Executor; +import java.util.concurrent.ExecutorService; import java.util.concurrent.TimeUnit; import java.util.function.Predicate; import java.util.regex.Pattern; @@ -98,7 +101,10 @@ public class CtwRunner { private void startCtwforAllClasses() { - long classStart = 0; + long classStart = 0L; + long classCount = classCount(); + Asserts.assertGreaterThan(classCount, 0L, + targetPath + " does not have any classes"); boolean done = false; while (!done) { String[] cmd = cmd(classStart); @@ -124,6 +130,16 @@ public class CtwRunner { exitCode); Pair lastClass = getLastClass(out); if (exitCode == 0) { + long lastIndex = lastClass == null ? -1 : lastClass.second; + if (lastIndex != classCount) { + errors.add(new Error(phase + ": Unexpected zero exit code" + + "before finishing all compilations." + + " lastClass[" + lastIndex + + "] != classCount[" + classCount + "]")); + } else { + System.out.println("Executed CTW for all " + classCount + + " classes in " + targetPath); + } done = true; } else { if (lastClass == null) { @@ -145,6 +161,11 @@ public class CtwRunner { } } + private long classCount() { + return PathHandler.create(targetPath.toString(), Runnable::run) + .classCount(); + } + private Pair getLastClass(Path errFile) { try (BufferedReader reader = Files.newBufferedReader(errFile)) { String line = reader.lines() @@ -167,7 +188,7 @@ public class CtwRunner { private String[] cmd(long classStart) { String phase = phaseName(classStart); - return new String[]{ + return new String[] { "-Xbatch", "-XX:-UseCounterDecay", "-XX:-ShowMessageBoxOnError", diff --git a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/PathHandler.java b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/PathHandler.java index f5347cd8242..c2ac839049b 100644 --- a/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/PathHandler.java +++ b/hotspot/test/testlibrary/ctw/src/sun/hotspot/tools/ctw/PathHandler.java @@ -62,7 +62,7 @@ public abstract class PathHandler { this.loader = ClassLoader.getSystemClassLoader(); } - /** + /** * Factory method. Construct concrete handler in depends from {@code path}. * * @param path the path to process @@ -118,11 +118,16 @@ public abstract class PathHandler { } /** - * Processes all classes in specified path. + * Processes all classes in the specified path. */ public abstract void process(); - /** + /** + * @return count of all classes in the specified path. + */ + public abstract long classCount(); + + /** * Sets class loader, that will be used to define class at * {@link #processClass(String)}. * @@ -168,7 +173,7 @@ public abstract class PathHandler { /** * @return count of processed classes */ - public static long getClassCount() { + public static long getProcessedClassCount() { long id = CLASS_COUNT.get(); if (id < Utils.COMPILE_THE_WORLD_START_AT) { return 0;