From a6426413115aac7c90692c9ee99c37198b8f9521 Mon Sep 17 00:00:00 2001 From: Goetz Lindenmaier Date: Thu, 16 Oct 2025 11:15:02 +0200 Subject: [PATCH 1/2] backport 45180633d34b6cbb679bae0753d9f422e76d6297 --- .../internal/WindowsAppImageBuilder.java | 7 +- .../internal/WixAppImageFragmentBuilder.java | 6 - .../helpers/jdk/jpackage/test/Functional.java | 5 +- .../jpackage/test/LauncherIconVerifier.java | 142 +++++++++++++++++- .../tools/jpackage/share/AddLauncherTest.java | 8 +- test/jdk/tools/jpackage/share/IconTest.java | 4 +- .../share/MultiLauncherTwoPhaseTest.java | 4 +- 7 files changed, 159 insertions(+), 17 deletions(-) diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WindowsAppImageBuilder.java b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WindowsAppImageBuilder.java index ef7139b61cd..9a9a7006550 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WindowsAppImageBuilder.java +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WindowsAppImageBuilder.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2022, 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 @@ -117,8 +117,9 @@ private void createLauncherForEntryPoint(Map params, mainParams); Path iconTarget = null; if (iconResource != null) { - iconTarget = appLayout.destktopIntegrationDirectory().resolve( - APP_NAME.fetchFrom(params) + ".ico"); + Path iconDir = StandardBundlerParam.TEMP_ROOT.fetchFrom(params).resolve( + "icons"); + iconTarget = iconDir.resolve(APP_NAME.fetchFrom(params) + ".ico"); if (null == iconResource.saveToFile(iconTarget)) { iconTarget = null; } diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixAppImageFragmentBuilder.java b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixAppImageFragmentBuilder.java index ac926232144..eccbd3c7ba7 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixAppImageFragmentBuilder.java +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixAppImageFragmentBuilder.java @@ -443,17 +443,11 @@ private String addShortcutComponent(XMLStreamWriter xml, Path launcherPath, Path shortcutPath = folder.getPath(this).resolve(launcherBasename); return addComponent(xml, shortcutPath, Component.Shortcut, unused -> { - final Path icoFile = IOUtils.addSuffix( - installedAppImage.destktopIntegrationDirectory().resolve( - launcherBasename), ".ico"); - xml.writeAttribute("Name", launcherBasename); xml.writeAttribute("WorkingDirectory", INSTALLDIR.toString()); xml.writeAttribute("Advertise", "no"); - xml.writeAttribute("IconIndex", "0"); xml.writeAttribute("Target", String.format("[#%s]", Component.File.idOf(launcherPath))); - xml.writeAttribute("Icon", Id.Icon.of(icoFile)); }); } diff --git a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/Functional.java b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/Functional.java index 6fe7e56ffb7..87718c1394c 100644 --- a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/Functional.java +++ b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/Functional.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2019, 2021, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, 2022, 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 @@ -149,7 +149,8 @@ public ExceptionBox(Throwable throwable) { } @SuppressWarnings("unchecked") - public static void rethrowUnchecked(Throwable throwable) throws ExceptionBox { + public static RuntimeException rethrowUnchecked(Throwable throwable) throws + ExceptionBox { if (throwable instanceof RuntimeException) { throw (RuntimeException)throwable; } diff --git a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LauncherIconVerifier.java b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LauncherIconVerifier.java index 77f71f01673..cf9dec86d35 100644 --- a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LauncherIconVerifier.java +++ b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LauncherIconVerifier.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, 2022, 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 @@ -24,8 +24,12 @@ package jdk.jpackage.test; import java.io.IOException; +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; import java.nio.file.Files; import java.nio.file.Path; +import java.util.List; +import java.util.Optional; public final class LauncherIconVerifier { public LauncherIconVerifier() { @@ -60,7 +64,11 @@ public void applyTo(JPackageCommand cmd) throws IOException { Path iconPath = cmd.appLayout().destktopIntegrationDirectory().resolve( curLauncherName + TKit.ICON_SUFFIX); - if (expectedDefault) { + if (TKit.isWindows()) { + TKit.assertPathExists(iconPath, false); + WinIconVerifier.instance.verifyLauncherIcon(cmd, launcherName, + expectedIcon, expectedDefault); + } else if (expectedDefault) { TKit.assertPathExists(iconPath, true); } else if (expectedIcon == null) { TKit.assertPathExists(iconPath, false); @@ -73,6 +81,136 @@ public void applyTo(JPackageCommand cmd) throws IOException { } } + private static class WinIconVerifier { + + void verifyLauncherIcon(JPackageCommand cmd, String launcherName, + Path expectedIcon, boolean expectedDefault) { + TKit.withTempDirectory("icons", tmpDir -> { + Path launcher = cmd.appLauncherPath(launcherName); + Path iconWorkDir = tmpDir.resolve(launcher.getFileName()); + Path iconContainer = iconWorkDir.resolve("container.exe"); + Files.createDirectories(iconContainer.getParent()); + Files.copy(getDefaultAppLauncher(expectedIcon == null + && !expectedDefault), iconContainer); + if (expectedIcon != null) { + setIcon(expectedIcon, iconContainer); + } + + Path extractedExpectedIcon = extractIconFromExecutable( + iconWorkDir, iconContainer, "expected"); + Path extractedActualIcon = extractIconFromExecutable(iconWorkDir, + launcher, "actual"); + TKit.assertTrue(-1 == Files.mismatch(extractedExpectedIcon, + extractedActualIcon), + String.format( + "Check icon file [%s] of %s launcher is a copy of source icon file [%s]", + extractedActualIcon, + Optional.ofNullable(launcherName).orElse("main"), + extractedExpectedIcon)); + }); + } + + private WinIconVerifier() { + try { + executableRebranderClass = Class.forName( + "jdk.jpackage.internal.ExecutableRebrander"); + + lockResource = executableRebranderClass.getDeclaredMethod( + "lockResource", String.class); + // Note: this reflection call requires + // --add-opens jdk.jpackage/jdk.jpackage.internal=ALL-UNNAMED + lockResource.setAccessible(true); + + unlockResource = executableRebranderClass.getDeclaredMethod( + "unlockResource", long.class); + unlockResource.setAccessible(true); + + iconSwap = executableRebranderClass.getDeclaredMethod("iconSwap", + long.class, String.class); + iconSwap.setAccessible(true); + } catch (ClassNotFoundException | NoSuchMethodException + | SecurityException ex) { + throw Functional.rethrowUnchecked(ex); + } + } + + private Path extractIconFromExecutable(Path outputDir, Path executable, + String label) { + Path psScript = outputDir.resolve(label + ".ps1"); + Path extractedIcon = outputDir.resolve(label + ".bmp"); + TKit.createTextFile(psScript, List.of( + "[System.Reflection.Assembly]::LoadWithPartialName('System.Drawing')", + String.format( + "[System.Drawing.Icon]::ExtractAssociatedIcon(\"%s\").ToBitmap().Save(\"%s\", [System.Drawing.Imaging.ImageFormat]::Bmp)", + executable.toAbsolutePath().normalize(), + extractedIcon.toAbsolutePath().normalize()), + "exit 0")); + + Executor.of("powershell", "-NoLogo", "-NoProfile", "-File", + psScript.toAbsolutePath().normalize().toString()).execute(); + + return extractedIcon; + } + + private Path getDefaultAppLauncher(boolean noIcon) { + // Create app image with the sole purpose to get the default app launcher + Path defaultAppOutputDir = TKit.workDir().resolve(String.format( + "out-%d", ProcessHandle.current().pid())); + JPackageCommand cmd = JPackageCommand.helloAppImage().setFakeRuntime().setArgumentValue( + "--dest", defaultAppOutputDir); + + String launcherName; + if (noIcon) { + launcherName = "no-icon"; + new AdditionalLauncher(launcherName).setNoIcon().applyTo(cmd); + } else { + launcherName = null; + } + + if (!Files.isExecutable(cmd.appLauncherPath(launcherName))) { + cmd.execute(); + } + return cmd.appLauncherPath(launcherName); + } + + private void setIcon(Path iconPath, Path launcherPath) { + TKit.trace(String.format("Set icon of [%s] launcher to [%s] file", + launcherPath, iconPath)); + try { + launcherPath.toFile().setWritable(true, true); + try { + long lock = 0; + try { + lock = (Long) lockResource.invoke(null, new Object[]{ + launcherPath.toAbsolutePath().normalize().toString()}); + if (lock == 0) { + throw new RuntimeException(String.format( + "Failed to lock [%s] executable", + launcherPath)); + } + iconSwap.invoke(null, new Object[]{lock, + iconPath.toAbsolutePath().normalize().toString()}); + } finally { + if (lock != 0) { + unlockResource.invoke(null, new Object[]{lock}); + } + } + } catch (IllegalAccessException | InvocationTargetException ex) { + throw Functional.rethrowUnchecked(ex); + } + } finally { + launcherPath.toFile().setWritable(false, true); + } + } + + final static WinIconVerifier instance = new WinIconVerifier(); + + private final Class executableRebranderClass; + private final Method lockResource; + private final Method unlockResource; + private final Method iconSwap; + } + private String launcherName; private Path expectedIcon; private boolean expectedDefault; diff --git a/test/jdk/tools/jpackage/share/AddLauncherTest.java b/test/jdk/tools/jpackage/share/AddLauncherTest.java index 2dbab36b9c9..1fd930117fc 100644 --- a/test/jdk/tools/jpackage/share/AddLauncherTest.java +++ b/test/jdk/tools/jpackage/share/AddLauncherTest.java @@ -51,7 +51,9 @@ * @library /test/jdk/tools/jpackage/helpers * @build jdk.jpackage.test.* * @compile AddLauncherTest.java - * @run main/othervm/timeout=360 -Xmx512m jdk.jpackage.test.Main + * @run main/othervm/timeout=360 -Xmx512m + * --add-opens jdk.jpackage/jdk.jpackage.internal=ALL-UNNAMED + * jdk.jpackage.test.Main * --jpt-run=AddLauncherTest.test */ @@ -63,7 +65,9 @@ * @library /test/jdk/tools/jpackage/helpers * @build jdk.jpackage.test.* * @compile AddLauncherTest.java - * @run main/othervm/timeout=540 -Xmx512m jdk.jpackage.test.Main + * @run main/othervm/timeout=540 -Xmx512m + * --add-opens jdk.jpackage/jdk.jpackage.internal=ALL-UNNAMED + * jdk.jpackage.test.Main * --jpt-run=AddLauncherTest */ diff --git a/test/jdk/tools/jpackage/share/IconTest.java b/test/jdk/tools/jpackage/share/IconTest.java index 096f9bc21b3..c2a70022db5 100644 --- a/test/jdk/tools/jpackage/share/IconTest.java +++ b/test/jdk/tools/jpackage/share/IconTest.java @@ -52,7 +52,9 @@ * @library /test/jdk/tools/jpackage/helpers * @build jdk.jpackage.test.* * @compile IconTest.java - * @run main/othervm/timeout=540 -Xmx512m jdk.jpackage.test.Main + * @run main/othervm/timeout=540 -Xmx512m + * --add-opens jdk.jpackage/jdk.jpackage.internal=ALL-UNNAMED + * jdk.jpackage.test.Main * --jpt-run=IconTest */ diff --git a/test/jdk/tools/jpackage/share/MultiLauncherTwoPhaseTest.java b/test/jdk/tools/jpackage/share/MultiLauncherTwoPhaseTest.java index 535d9851e45..9dc6705936a 100644 --- a/test/jdk/tools/jpackage/share/MultiLauncherTwoPhaseTest.java +++ b/test/jdk/tools/jpackage/share/MultiLauncherTwoPhaseTest.java @@ -47,7 +47,9 @@ * @key jpackagePlatformPackage * @build jdk.jpackage.test.* * @compile MultiLauncherTwoPhaseTest.java - * @run main/othervm/timeout=360 -Xmx512m jdk.jpackage.test.Main + * @run main/othervm/timeout=360 -Xmx512m + * --add-opens jdk.jpackage/jdk.jpackage.internal=ALL-UNNAMED + * jdk.jpackage.test.Main * --jpt-run=MultiLauncherTwoPhaseTest */ From da0b09fb4ebcb6b3b3969cf75e0b3c09f9423d39 Mon Sep 17 00:00:00 2001 From: Goetz Lindenmaier Date: Thu, 16 Oct 2025 13:36:30 +0200 Subject: [PATCH 2/2] backport c254c9d095d0473282ad74e66239a790912a3d76 --- .../jdk/jpackage/test/LauncherIconVerifier.java | 13 +++++-------- 1 file changed, 5 insertions(+), 8 deletions(-) diff --git a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LauncherIconVerifier.java b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LauncherIconVerifier.java index cf9dec86d35..4dfe5167940 100644 --- a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LauncherIconVerifier.java +++ b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LauncherIconVerifier.java @@ -28,7 +28,6 @@ import java.lang.reflect.Method; import java.nio.file.Files; import java.nio.file.Path; -import java.util.List; import java.util.Optional; public final class LauncherIconVerifier { @@ -136,18 +135,16 @@ private WinIconVerifier() { private Path extractIconFromExecutable(Path outputDir, Path executable, String label) { - Path psScript = outputDir.resolve(label + ".ps1"); Path extractedIcon = outputDir.resolve(label + ".bmp"); - TKit.createTextFile(psScript, List.of( + String script = String.join(";", "[System.Reflection.Assembly]::LoadWithPartialName('System.Drawing')", String.format( - "[System.Drawing.Icon]::ExtractAssociatedIcon(\"%s\").ToBitmap().Save(\"%s\", [System.Drawing.Imaging.ImageFormat]::Bmp)", + "[System.Drawing.Icon]::ExtractAssociatedIcon('%s').ToBitmap().Save('%s', [System.Drawing.Imaging.ImageFormat]::Bmp)", executable.toAbsolutePath().normalize(), - extractedIcon.toAbsolutePath().normalize()), - "exit 0")); + extractedIcon.toAbsolutePath().normalize())); - Executor.of("powershell", "-NoLogo", "-NoProfile", "-File", - psScript.toAbsolutePath().normalize().toString()).execute(); + Executor.of("powershell", "-NoLogo", "-NoProfile", "-Command", + script).execute(); return extractedIcon; }