/* * Copyright (c) 2019, Red Hat, Inc. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ import java.io.BufferedWriter; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.PrintWriter; import java.io.StringWriter; import java.nio.file.FileVisitResult; import java.nio.file.Files; import java.nio.file.NoSuchFileException; import java.nio.file.Path; import java.nio.file.Paths; import java.nio.file.SimpleFileVisitor; import java.nio.file.attribute.BasicFileAttributes; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import java.util.Map; import java.util.Scanner; import java.util.spi.ToolProvider; import java.util.stream.Collectors; import java.util.stream.Stream; import jdk.test.lib.compiler.CompilerUtils; import jdk.tools.jlink.internal.ResourcePoolManager; import jdk.tools.jlink.internal.plugins.StripNativeDebugSymbolsPlugin; import jdk.tools.jlink.internal.plugins.StripNativeDebugSymbolsPlugin.ObjCopyCmdBuilder; import jdk.tools.jlink.plugin.ResourcePool; import jdk.tools.jlink.plugin.ResourcePoolEntry; /* * @test * @requires os.family == "linux" * @bug 8214796 * @summary Test --strip-native-debug-symbols plugin * @library /test/lib * @modules jdk.compiler * jdk.jlink/jdk.tools.jlink.internal.plugins * jdk.jlink/jdk.tools.jlink.internal * jdk.jlink/jdk.tools.jlink.plugin * @build jdk.test.lib.compiler.CompilerUtils FakeObjCopy * @run main/othervm -Xmx1g StripNativeDebugSymbolsPluginTest */ public class StripNativeDebugSymbolsPluginTest { private static final String OBJCOPY = "objcopy"; private static final String DEFAULT_OBJCOPY_CMD = OBJCOPY; private static final String PLUGIN_NAME = "strip-native-debug-symbols"; private static final String MODULE_NAME_WITH_NATIVE = "fib"; private static final String JAVA_HOME = System.getProperty("java.home"); private static final String NATIVE_LIB_NAME = "libFib.so"; private static final Path JAVA_LIB_PATH = Paths.get(System.getProperty("java.library.path")); private static final Path LIB_FIB_SRC = JAVA_LIB_PATH.resolve(NATIVE_LIB_NAME); private static final String FIBJNI_CLASS_NAME = "FibJNI.java"; private static final Path JAVA_SRC_DIR = Paths.get(System.getProperty("test.src")) .resolve("src") .resolve(MODULE_NAME_WITH_NATIVE); private static final Path FIBJNI_JAVA_CLASS = JAVA_SRC_DIR.resolve(FIBJNI_CLASS_NAME); private static final String DEBUG_EXTENSION = "debug"; private static final long ORIG_LIB_FIB_SIZE = LIB_FIB_SRC.toFile().length(); private static final String FAKE_OBJ_COPY_LOG_FILE = "objcopy.log"; private static final String OBJCOPY_ONLY_DEBUG_SYMS_OPT = "-g"; private static final String OBJCOPY_ONLY_KEEP_DEBUG_SYMS_OPT = "--only-keep-debug"; private static final String OBJCOPY_ADD_DEBUG_LINK_OPT = "--add-gnu-debuglink"; /////////////////////////////////////////////////////////////////////////// // // Tests which do NOT rely on objcopy being present on the test system // /////////////////////////////////////////////////////////////////////////// public void testPluginLoaded() { List output = JLink.run("--list-plugins").output(); if (output.stream().anyMatch(s -> s.contains(PLUGIN_NAME))) { System.out.println("DEBUG: " + PLUGIN_NAME + " plugin loaded as expected."); } else { throw new AssertionError("strip-native-debug-symbols plugin not in " + "--list-plugins output."); } } public void testConfigureFakeObjCopy() throws Exception { configureConflictingOptions(); configureObjcopyWithOmit(); configureObjcopyWithKeep(); configureUnknownOptions(); configureMultipleTimesSamePlugin(); System.out.println("Test testConfigureFakeObjCopy() PASSED!"); } private void configureMultipleTimesSamePlugin() throws Exception { Map keepDebug = Map.of( StripNativeDebugSymbolsPlugin.NAME, "keep-debuginfo-files" ); Map excludeDebug = Map.of( StripNativeDebugSymbolsPlugin.NAME, "exclude-debuginfo-files" ); StripNativeDebugSymbolsPlugin plugin = createAndConfigPlugin(keepDebug); try { plugin.doConfigure(false, excludeDebug); throw new AssertionError("should have thrown IAE for broken config: " + keepDebug + " and " + excludeDebug); } catch (IllegalArgumentException e) { // pass System.out.println("DEBUG: test threw IAE " + e.getMessage() + " as expected."); } } private void configureUnknownOptions() throws Exception { Map config = Map.of( StripNativeDebugSymbolsPlugin.NAME, "foobar" ); doConfigureUnknownOption(config); config = Map.of( StripNativeDebugSymbolsPlugin.NAME, "keep-debuginfo-files", "foo", "bar" // unknown value ); doConfigureUnknownOption(config); } private void doConfigureUnknownOption(Map config) throws Exception { try { createAndConfigPlugin(config); throw new AssertionError("should have thrown IAE for broken config: " + config); } catch (IllegalArgumentException e) { // pass System.out.println("DEBUG: test threw IAE " + e.getMessage() + " as expected."); } } private void configureObjcopyWithKeep() throws Exception { String objcopyPath = "foobar"; String debugExt = "debuginfo"; // that's the default value Map config = Map.of( StripNativeDebugSymbolsPlugin.NAME, "keep-debuginfo-files", "objcopy", objcopyPath ); doKeepDebugInfoFakeObjCopyTest(config, debugExt, objcopyPath); // Do it again combining options the other way round debugExt = "testme"; config = Map.of( StripNativeDebugSymbolsPlugin.NAME, "objcopy=" + objcopyPath, "keep-debuginfo-files", debugExt ); doKeepDebugInfoFakeObjCopyTest(config, debugExt, objcopyPath); System.out.println("DEBUG: configureObjcopyWithKeep() PASSED!"); } private void configureObjcopyWithOmit() throws Exception { String objcopyPath = "something-non-standard"; Map config = Map.of( StripNativeDebugSymbolsPlugin.NAME, "exclude-debuginfo-files", "objcopy", objcopyPath ); doOmitDebugInfoFakeObjCopyTest(config, objcopyPath); System.out.println("DEBUG: configureObjcopyWithOmit() PASSED!"); } private void configureConflictingOptions() throws Exception { Map config = Map.of( StripNativeDebugSymbolsPlugin.NAME, "exclude-debuginfo-files", "keep-debuginfo-files", "foo-ext" ); doConfigureConflictingOptions(config); config = Map.of( StripNativeDebugSymbolsPlugin.NAME, "exclude-debuginfo-files=bar", "keep-debuginfo-files", "foo-ext" ); doConfigureConflictingOptions(config); } private void doConfigureConflictingOptions(Map config) throws Exception { try { createAndConfigPlugin(config); throw new AssertionError("keep-debuginfo-files and exclude-debuginfo-files " + " should have conflicted!"); } catch (IllegalArgumentException e) { // pass if (e.getMessage().contains("keep-debuginfo-files") && e.getMessage().contains("exclude-debuginfo-files")) { System.out.println("DEBUG: test threw IAE " + e.getMessage() + " as expected."); } else { throw new AssertionError("Unexpected IAE", e); } } } public void testTransformFakeObjCopyNoDebugInfoFiles() throws Exception { Map defaultConfig = Map.of( StripNativeDebugSymbolsPlugin.NAME, "exclude-debuginfo-files" ); doOmitDebugInfoFakeObjCopyTest(defaultConfig, DEFAULT_OBJCOPY_CMD); System.out.println("testTransformFakeObjCopyNoDebugInfoFiles() PASSED!"); } private void doOmitDebugInfoFakeObjCopyTest(Map config, String expectedObjCopy) throws Exception { StripNativeDebugSymbolsPlugin plugin = createAndConfigPlugin(config, expectedObjCopy); String binFile = "mybin"; String path = "/fib/bin/" + binFile; ResourcePoolEntry debugEntry = createMockEntry(path, ResourcePoolEntry.Type.NATIVE_CMD); ResourcePoolManager inResources = new ResourcePoolManager(); ResourcePoolManager outResources = new ResourcePoolManager(); inResources.add(debugEntry); ResourcePool output = plugin.transform( inResources.resourcePool(), outResources.resourcePoolBuilder()); // expect entry to be present if (output.findEntry(path).isPresent()) { System.out.println("DEBUG: File " + path + " present as exptected."); } else { throw new AssertionError("Test failed. Binary " + path + " not present after stripping!"); } verifyFakeObjCopyCalled(binFile); } public void testTransformFakeObjCopyKeepDebugInfoFiles() throws Exception { Map defaultConfig = Map.of( StripNativeDebugSymbolsPlugin.NAME, "keep-debuginfo-files=" + DEBUG_EXTENSION ); doKeepDebugInfoFakeObjCopyTest(defaultConfig, DEBUG_EXTENSION, DEFAULT_OBJCOPY_CMD); System.out.println("testTransformFakeObjCopyKeepDebugInfoFiles() PASSED!"); } private void doKeepDebugInfoFakeObjCopyTest(Map config, String debugExt, String expectedObjCopy) throws Exception { StripNativeDebugSymbolsPlugin plugin = createAndConfigPlugin(config, expectedObjCopy); String sharedLib = "myLib.so"; String path = "/fib/lib/" + sharedLib; ResourcePoolEntry debugEntry = createMockEntry(path, ResourcePoolEntry.Type.NATIVE_LIB); ResourcePoolManager inResources = new ResourcePoolManager(); ResourcePoolManager outResources = new ResourcePoolManager(); inResources.add(debugEntry); ResourcePool output = plugin.transform( inResources.resourcePool(), outResources.resourcePoolBuilder()); // expect entry + debug info entry to be present String debugPath = path + "." + debugExt; if (output.findEntry(path).isPresent() && output.findEntry(debugPath).isPresent()) { System.out.println("DEBUG: Files " + path + "{,." + debugExt + "} present as exptected."); } else { throw new AssertionError("Test failed. Binary files " + path + "{,." + debugExt +"} not present after " + "stripping!"); } verifyFakeObjCopyCalledMultiple(sharedLib, debugExt); } /////////////////////////////////////////////////////////////////////////// // // Tests which DO rely on objcopy being present on the test system. // Skipped otherwise. // /////////////////////////////////////////////////////////////////////////// public void testStripNativeLibraryDefaults() throws Exception { if (!hasJmods()) return; Path libFibJmod = createLibFibJmod(); Path imageDir = Paths.get("stripped-native-libs"); JLink.run("--output", imageDir.toString(), "--verbose", "--module-path", modulePathWith(libFibJmod), "--add-modules", MODULE_NAME_WITH_NATIVE, "--strip-native-debug-symbols=exclude-debuginfo-files").output(); Path libDir = imageDir.resolve("lib"); Path postStripLib = libDir.resolve(NATIVE_LIB_NAME); long postStripSize = postStripLib.toFile().length(); if (postStripSize == 0) { throw new AssertionError("Lib file size 0. Test error?!"); } // Heuristic: libLib.so is smaller post debug info stripping if (postStripSize >= ORIG_LIB_FIB_SIZE) { throw new AssertionError("Expected native library stripping to " + "reduce file size. Expected < " + ORIG_LIB_FIB_SIZE + ", got: " + postStripSize); } else { System.out.println("DEBUG: File size of " + postStripLib.toString() + " " + postStripSize + " < " + ORIG_LIB_FIB_SIZE + " as expected." ); } verifyFibModule(imageDir); // Sanity check fib module which got libFib.so stripped System.out.println("DEBUG: testStripNativeLibraryDefaults() PASSED!"); } public void testOptionsInvalidObjcopy() throws Exception { if (!hasJmods()) return; Path libFibJmod = createLibFibJmod(); String notExists = "/do/not/exist/objcopy"; Path imageDir = Paths.get("invalid-objcopy-command"); String[] jlinkCmdArray = new String[] { JAVA_HOME + File.separator + "bin" + File.separator + "jlink", "--output", imageDir.toString(), "--verbose", "--module-path", modulePathWith(libFibJmod), "--add-modules", MODULE_NAME_WITH_NATIVE, "--strip-native-debug-symbols", "objcopy=" + notExists, }; List jlinkCmd = Arrays.asList(jlinkCmdArray); System.out.println("Debug: command: " + jlinkCmd.stream().collect( Collectors.joining(" "))); ProcessBuilder builder = new ProcessBuilder(jlinkCmd); Process p = builder.start(); int status = p.waitFor(); if (status == 0) { throw new AssertionError("Expected jlink to fail!"); } else { verifyInvalidObjcopyError(p.getInputStream(), notExists); System.out.println("DEBUG: testOptionsInvalidObjcopy() PASSED!"); } } public void testStripNativeLibsDebugSymsIncluded() throws Exception { if (!hasJmods()) return; Path libFibJmod = createLibFibJmod(); Path imageDir = Paths.get("stripped-native-libs-with-debug"); JLink.run("--output", imageDir.toString(), "--verbose", "--module-path", modulePathWith(libFibJmod), "--add-modules", MODULE_NAME_WITH_NATIVE, "--strip-native-debug-symbols", "keep-debuginfo-files=" + DEBUG_EXTENSION); Path libDir = imageDir.resolve("lib"); Path postStripLib = libDir.resolve(NATIVE_LIB_NAME); long postStripSize = postStripLib.toFile().length(); if (postStripSize == 0) { throw new AssertionError("Lib file size 0. Test error?!"); } // Heuristic: libLib.so is smaller post debug info stripping if (postStripSize >= ORIG_LIB_FIB_SIZE) { throw new AssertionError("Expected native library stripping to " + "reduce file size. Expected < " + ORIG_LIB_FIB_SIZE + ", got: " + postStripSize); } else { System.out.println("DEBUG: File size of " + postStripLib.toString() + " " + postStripSize + " < " + ORIG_LIB_FIB_SIZE + " as expected." ); } // stripped with option to preserve debug symbols file verifyDebugInfoSymbolFilePresent(imageDir); System.out.println("DEBUG: testStripNativeLibsDebugSymsIncluded() PASSED!"); } private void verifyFakeObjCopyCalledMultiple(String expectedFile, String dbgExt) throws Exception { // transform of the StripNativeDebugSymbolsPlugin created objcopy.log // with our stubbed FakeObjCopy. See FakeObjCopy.java List allLines = Files.readAllLines(Paths.get(FAKE_OBJ_COPY_LOG_FILE)); if (allLines.size() != 3) { throw new AssertionError("Expected 3 calls to objcopy"); } // 3 calls to objcopy are as follows: // 1. Only keep debug symbols // 2. Strip debug symbols // 3. Add debug link to stripped file String onlyKeepDebug = allLines.get(0); String stripSymbolsLine = allLines.get(1); String addGnuDebugLink = allLines.get(2); System.out.println("DEBUG: Inspecting fake objcopy calls: " + allLines); boolean passed = stripSymbolsLine.startsWith(OBJCOPY_ONLY_DEBUG_SYMS_OPT); passed &= stripSymbolsLine.endsWith(expectedFile); String[] tokens = onlyKeepDebug.split("\\s"); passed &= tokens[0].equals(OBJCOPY_ONLY_KEEP_DEBUG_SYMS_OPT); passed &= tokens[1].endsWith(expectedFile); passed &= tokens[2].endsWith(expectedFile + "." + dbgExt); tokens = addGnuDebugLink.split("\\s"); String[] addDbgTokens = tokens[0].split("="); passed &= addDbgTokens[1].equals(expectedFile + "." + dbgExt); passed &= addDbgTokens[0].equals(OBJCOPY_ADD_DEBUG_LINK_OPT); passed &= tokens[1].endsWith(expectedFile); if (!passed) { throw new AssertionError("Test failed! objcopy not properly called " + "with expected options!"); } } private void verifyFakeObjCopyCalled(String expectedFile) throws Exception { // transform of the StripNativeDebugSymbolsPlugin created objcopy.log // with our stubbed FakeObjCopy. See FakeObjCopy.java List allLines = Files.readAllLines(Paths.get(FAKE_OBJ_COPY_LOG_FILE)); if (allLines.size() != 1) { throw new AssertionError("Expected 1 call to objcopy only"); } String optionLine = allLines.get(0); System.out.println("DEBUG: Inspecting fake objcopy arguments: " + optionLine); boolean passed = optionLine.startsWith(OBJCOPY_ONLY_DEBUG_SYMS_OPT); passed &= optionLine.endsWith(expectedFile); if (!passed) { throw new AssertionError("Test failed! objcopy not called with " + "expected options!"); } } private ResourcePoolEntry createMockEntry(String path, ResourcePoolEntry.Type type) { byte[] mockContent = new byte[] { 0, 1, 2, 3 }; ResourcePoolEntry entry = ResourcePoolEntry.create( path, type, mockContent); return entry; } private StripNativeDebugSymbolsPlugin createAndConfigPlugin( Map config, String expectedObjcopy) throws IOException { TestObjCopyCmdBuilder cmdBuilder = new TestObjCopyCmdBuilder(expectedObjcopy); return createAndConfigPlugin(config, cmdBuilder); } private StripNativeDebugSymbolsPlugin createAndConfigPlugin( Map config) throws IOException { TestObjCopyCmdBuilder cmdBuilder = new TestObjCopyCmdBuilder(); return createAndConfigPlugin(config, cmdBuilder); } private StripNativeDebugSymbolsPlugin createAndConfigPlugin( Map config, TestObjCopyCmdBuilder builder) throws IOException { StripNativeDebugSymbolsPlugin plugin = new StripNativeDebugSymbolsPlugin(builder); plugin.doConfigure(false, config); return plugin; } // Create the jmod with the native library private Path createLibFibJmod() throws IOException { JmodFileBuilder jmodBuilder = new JmodFileBuilder(MODULE_NAME_WITH_NATIVE); jmodBuilder.javaClass(FIBJNI_JAVA_CLASS); jmodBuilder.nativeLib(LIB_FIB_SRC); return jmodBuilder.build(); } private String modulePathWith(Path jmod) { return Paths.get(JAVA_HOME, "jmods").toString() + File.pathSeparator + jmod.getParent().toString(); } private boolean hasJmods() { if (!Files.exists(Paths.get(JAVA_HOME, "jmods"))) { System.err.println("Test skipped. NO jmods directory"); return false; } return true; } private void verifyInvalidObjcopyError(InputStream errInput, String match) { boolean foundMatch = false; try (Scanner scanner = new Scanner(errInput)) { while (scanner.hasNextLine()) { String line = scanner.nextLine(); System.out.println("DEBUG: >>>> " + line); if (line.contains(match)) { foundMatch = true; break; } } } if (!foundMatch) { throw new AssertionError("Expected to find " + match + " in error stream."); } else { System.out.println("DEBUG: Found string " + match + " as expected."); } } private void verifyDebugInfoSymbolFilePresent(Path image) throws IOException, InterruptedException { Path debugSymsFile = image.resolve("lib/libFib.so.debug"); if (!Files.exists(debugSymsFile)) { throw new AssertionError("Expected stripped debug info file " + debugSymsFile.toString() + " to exist."); } long debugSymsSize = debugSymsFile.toFile().length(); if (debugSymsSize <= 0) { throw new AssertionError("sanity check for fib.FibJNI failed " + "post-stripping!"); } else { System.out.println("DEBUG: Debug symbols stripped from libFib.so " + "present (" + debugSymsFile.toString() + ") as expected."); } } private void verifyFibModule(Path image) throws IOException, InterruptedException { System.out.println("DEBUG: sanity checking fib module..."); Path launcher = image.resolve("bin/java"); List args = new ArrayList<>(); args.add(launcher.toString()); args.add("--add-modules"); args.add(MODULE_NAME_WITH_NATIVE); args.add("fib.FibJNI"); args.add("7"); args.add("13"); // fib(7) == 13 System.out.println("DEBUG: [command] " + args.stream().collect(Collectors.joining(" "))); Process proc = new ProcessBuilder(args).inheritIO().start(); int status = proc.waitFor(); if (status == 0) { System.out.println("DEBUG: sanity checking fib module... PASSED!"); } else { throw new AssertionError("sanity check for fib.FibJNI failed post-" + "stripping!"); } } private static boolean isObjcopyPresent() throws Exception { String[] objcopyVersion = new String[] { OBJCOPY, "--version", }; List command = Arrays.asList(objcopyVersion); try { ProcessBuilder builder = new ProcessBuilder(command); builder.inheritIO(); Process p = builder.start(); int status = p.waitFor(); if (status != 0) { System.out.println("Debug: objcopy binary doesn't seem to be " + "present or functional."); return false; } } catch (IOException e) { System.out.println("Debug: objcopy binary doesn't seem to be present " + "or functional."); return false; } return true; } public static void main(String[] args) throws Exception { StripNativeDebugSymbolsPluginTest test = new StripNativeDebugSymbolsPluginTest(); if (isObjcopyPresent()) { test.testStripNativeLibraryDefaults(); test.testStripNativeLibsDebugSymsIncluded(); test.testOptionsInvalidObjcopy(); } else { System.out.println("DEBUG: objcopy binary not available. " + "Running reduced set of tests."); } test.testTransformFakeObjCopyNoDebugInfoFiles(); test.testTransformFakeObjCopyKeepDebugInfoFiles(); test.testConfigureFakeObjCopy(); test.testPluginLoaded(); } static class JLink { static final ToolProvider JLINK_TOOL = ToolProvider.findFirst("jlink") .orElseThrow(() -> new RuntimeException("jlink tool not found") ); static JLink run(String... options) { JLink jlink = new JLink(); if (jlink.execute(options) != 0) { throw new AssertionError("Jlink expected to exit with 0 return code"); } return jlink; } final List output = new ArrayList<>(); private int execute(String... options) { System.out.println("jlink " + Stream.of(options).collect(Collectors.joining(" "))); StringWriter writer = new StringWriter(); PrintWriter pw = new PrintWriter(writer); int rc = JLINK_TOOL.run(pw, pw, options); System.out.println(writer.toString()); Stream.of(writer.toString().split("\\v")) .map(String::trim) .forEach(output::add); return rc; } boolean contains(String s) { return output.contains(s); } List output() { return output; } } /** * Builder to create JMOD file */ private static class JmodFileBuilder { private static final ToolProvider JMOD_TOOL = ToolProvider .findFirst("jmod") .orElseThrow(() -> new RuntimeException("jmod tool not found") ); private static final Path SRC_DIR = Paths.get("src"); private static final Path MODS_DIR = Paths.get("mod"); private static final Path JMODS_DIR = Paths.get("jmods"); private static final Path LIBS_DIR = Paths.get("libs"); private final String name; private final List nativeLibs = new ArrayList<>(); private final List javaClasses = new ArrayList<>(); private JmodFileBuilder(String name) throws IOException { this.name = name; deleteDirectory(MODS_DIR); deleteDirectory(SRC_DIR); deleteDirectory(LIBS_DIR); deleteDirectory(JMODS_DIR); Path msrc = SRC_DIR.resolve(name); if (Files.exists(msrc)) { deleteDirectory(msrc); } } JmodFileBuilder nativeLib(Path libFileSrc) { nativeLibs.add(libFileSrc); return this; } JmodFileBuilder javaClass(Path srcPath) { javaClasses.add(srcPath); return this; } Path build() throws IOException { compileModule(); return createJmodFile(); } private void compileModule() throws IOException { Path msrc = SRC_DIR.resolve(name); Files.createDirectories(msrc); // copy class using native lib to expected path if (javaClasses.size() > 0) { for (Path srcPath: javaClasses) { Path targetPath = msrc.resolve(srcPath.getFileName()); Files.copy(srcPath, targetPath); } } // generate module-info file. Path minfo = msrc.resolve("module-info.java"); try (BufferedWriter bw = Files.newBufferedWriter(minfo); PrintWriter writer = new PrintWriter(bw)) { writer.format("module %s { }%n", name); } if (!CompilerUtils.compile(msrc, MODS_DIR, "--module-source-path", SRC_DIR.toString())) { } } private Path createJmodFile() throws IOException { Path mclasses = MODS_DIR.resolve(name); Files.createDirectories(JMODS_DIR); Path outfile = JMODS_DIR.resolve(name + ".jmod"); List args = new ArrayList<>(); args.add("create"); // add classes args.add("--class-path"); args.add(mclasses.toString()); // native libs if (nativeLibs.size() > 0) { // Copy the JNI library to the expected path Files.createDirectories(LIBS_DIR); for (Path srcLib: nativeLibs) { Path targetLib = LIBS_DIR.resolve(srcLib.getFileName()); Files.copy(srcLib, targetLib); } args.add("--libs"); args.add(LIBS_DIR.toString()); } args.add(outfile.toString()); if (Files.exists(outfile)) { Files.delete(outfile); } System.out.println("jmod " + args.stream().collect(Collectors.joining(" "))); int rc = JMOD_TOOL.run(System.out, System.out, args.toArray(new String[args.size()])); if (rc != 0) { throw new AssertionError("jmod failed: rc = " + rc); } return outfile; } private static void deleteDirectory(Path dir) throws IOException { try { Files.walkFileTree(dir, new SimpleFileVisitor() { @Override public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException { Files.delete(file); return FileVisitResult.CONTINUE; } @Override public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException { Files.delete(dir); return FileVisitResult.CONTINUE; } }); } catch (NoSuchFileException e) { // ignore non-existing files } } } private static class TestObjCopyCmdBuilder implements ObjCopyCmdBuilder { private final String expectedObjCopy; private final String logFile; TestObjCopyCmdBuilder() { this(DEFAULT_OBJCOPY_CMD); } TestObjCopyCmdBuilder(String exptectedObjCopy) { Path logFilePath = Paths.get(FAKE_OBJ_COPY_LOG_FILE); try { Files.deleteIfExists(logFilePath); } catch (Exception e) { e.printStackTrace(); } this.logFile = logFilePath.toFile().getAbsolutePath(); this.expectedObjCopy = exptectedObjCopy; } @Override public List build(String objCopy, String... options) { if (!expectedObjCopy.equals(objCopy)) { throw new AssertionError("Expected objcopy to be '" + expectedObjCopy + "' but was '" + objCopy); } List fakeObjCopy = new ArrayList<>(); fakeObjCopy.add(JAVA_HOME + File.separator + "bin" + File.separator + "java"); fakeObjCopy.add("-cp"); fakeObjCopy.add(System.getProperty("test.classes")); fakeObjCopy.add("FakeObjCopy"); // Note that adding the gnu debug link changes the PWD of the // java process calling FakeObjCopy. As such we need to pass in the // log file path this way. Relative paths won't work as it would be // relative to the temporary directory which gets deleted post // adding the debug link fakeObjCopy.add(logFile); if (options.length > 0) { fakeObjCopy.addAll(Arrays.asList(options)); } return fakeObjCopy; } } }