diff --git a/src/jdk.incubator.jpackage/macosx/classes/jdk/incubator/jpackage/internal/MacDmgBundler.java b/src/jdk.incubator.jpackage/macosx/classes/jdk/incubator/jpackage/internal/MacDmgBundler.java index eeaf30b5f01..c599bc1e040 100644 --- a/src/jdk.incubator.jpackage/macosx/classes/jdk/incubator/jpackage/internal/MacDmgBundler.java +++ b/src/jdk.incubator.jpackage/macosx/classes/jdk/incubator/jpackage/internal/MacDmgBundler.java @@ -446,9 +446,22 @@ public class MacDmgBundler extends MacBaseInstallerBundler { // "hdiutil detach" might not work right away due to resource busy error, so // repeat detach several times. RetryExecutor retryExecutor = new RetryExecutor(); - // 10 times with 3 second delays. - retryExecutor.setMaxAttemptsCount(10).setAttemptTimeoutMillis(3000) - .execute(pb); + // Image can get detach even if we got resource busy error, so stop + // trying to detach it if it is no longer attached. + retryExecutor.setExecutorInitializer(exec -> { + if (!Files.exists(mountedRoot)) { + retryExecutor.abort(); + } + }); + try { + // 10 times with 3 second delays. + retryExecutor.setMaxAttemptsCount(10).setAttemptTimeoutMillis(3000) + .execute(pb); + } catch (IOException ex) { + if (!retryExecutor.isAborted()) { + throw ex; + } + } } // Compress it to a new image diff --git a/src/jdk.incubator.jpackage/share/classes/jdk/incubator/jpackage/internal/RetryExecutor.java b/src/jdk.incubator.jpackage/share/classes/jdk/incubator/jpackage/internal/RetryExecutor.java index 586477abb27..2370389d52b 100644 --- a/src/jdk.incubator.jpackage/share/classes/jdk/incubator/jpackage/internal/RetryExecutor.java +++ b/src/jdk.incubator.jpackage/share/classes/jdk/incubator/jpackage/internal/RetryExecutor.java @@ -53,6 +53,10 @@ public final class RetryExecutor { aborted = true; } + boolean isAborted() { + return aborted; + } + static RetryExecutor retryOnKnownErrorMessage(String v) { RetryExecutor result = new RetryExecutor(); return result.setExecutorInitializer(exec -> { @@ -75,6 +79,10 @@ public final class RetryExecutor { private void executeLoop(Supplier execSupplier) throws IOException { aborted = false; for (;;) { + if (aborted) { + break; + } + try { Executor exec = execSupplier.get(); if (executorInitializer != null) {