jdk-24/test/jdk/tools/jlink/plugins/StripNativeDebugSymbolsPlugin/StripNativeDebugSymbolsPluginTest.java
2019-03-14 14:04:39 +01:00

808 lines
34 KiB
Java

/*
* 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<String> 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<String, String> keepDebug = Map.of(
StripNativeDebugSymbolsPlugin.NAME, "keep-debuginfo-files"
);
Map<String, String> 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<String, String> 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<String, String> 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<String, String> 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<String, String> 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<String, String> 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<String, String> 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<String, String> defaultConfig = Map.of(
StripNativeDebugSymbolsPlugin.NAME, "exclude-debuginfo-files"
);
doOmitDebugInfoFakeObjCopyTest(defaultConfig, DEFAULT_OBJCOPY_CMD);
System.out.println("testTransformFakeObjCopyNoDebugInfoFiles() PASSED!");
}
private void doOmitDebugInfoFakeObjCopyTest(Map<String, String> 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<String, String> 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<String, String> 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<String> 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<String> 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<String> 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<String, String> config,
String expectedObjcopy)
throws IOException {
TestObjCopyCmdBuilder cmdBuilder = new TestObjCopyCmdBuilder(expectedObjcopy);
return createAndConfigPlugin(config, cmdBuilder);
}
private StripNativeDebugSymbolsPlugin createAndConfigPlugin(
Map<String, String> config) throws IOException {
TestObjCopyCmdBuilder cmdBuilder = new TestObjCopyCmdBuilder();
return createAndConfigPlugin(config, cmdBuilder);
}
private StripNativeDebugSymbolsPlugin createAndConfigPlugin(
Map<String, String> 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<String> 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<String> 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<String> 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<String> 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<Path> nativeLibs = new ArrayList<>();
private final List<Path> 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<String> 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<Path>() {
@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<String> build(String objCopy, String... options) {
if (!expectedObjCopy.equals(objCopy)) {
throw new AssertionError("Expected objcopy to be '" +
expectedObjCopy + "' but was '" +
objCopy);
}
List<String> 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;
}
}
}