From 904a71eb8ab7cc81a59b08ed36d8e25e021b7c57 Mon Sep 17 00:00:00 2001 From: Keith Harp Date: Tue, 22 Jan 2019 13:49:22 -0500 Subject: [PATCH] Update W32Service.stopService() to be more resilient to large dwWaitHint values (#1058) -- make stopService(long timeout) public so that 30 seconds is not the only valid timeout -- never wait more than the current timeout -- if the service stops during the last sleep, do not throw an exception -- use the preexisting timing logic in waitForNonPendingState to sleep fractions of the total dwWaitHint instead of the entire time --- CHANGES.md | 1 + .../sun/jna/platform/win32/W32Service.java | 49 +++++++++++-------- 2 files changed, 29 insertions(+), 21 deletions(-) diff --git a/CHANGES.md b/CHANGES.md index 3a8380af0c..4cdf34c6d6 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -7,6 +7,7 @@ Next release (5.3.0) Features -------- +* [#1058](https://github.com/java-native-access/jna/pull/1058): Add selectable timeout to stopService() and improve timeout handling - [@keithharp](https://github.com/keithharp). * [#1050](https://github.com/java-native-access/jna/pull/1050): Add `c.s.j.p.win32.VersionHelpers` and supporting functions - [@dbwiddis](https://github.com/dbwiddis). Bug Fixes diff --git a/contrib/platform/src/com/sun/jna/platform/win32/W32Service.java b/contrib/platform/src/com/sun/jna/platform/win32/W32Service.java index ab9d5c36ef..8f0b09158f 100644 --- a/contrib/platform/src/com/sun/jna/platform/win32/W32Service.java +++ b/contrib/platform/src/com/sun/jna/platform/win32/W32Service.java @@ -221,7 +221,7 @@ public void stopService() { * @param timeout timeout in ms until the service must report to be * stopped */ - void stopService(long timeout) { + public void stopService(long timeout) { long startTime = System.currentTimeMillis(); waitForNonPendingState(); // If the service is already stopped - return @@ -236,22 +236,23 @@ void stopService(long timeout) { // the previouos implementation queried the service status and // failed if the application did not correctly update its state while(status.dwCurrentState != Winsvc.SERVICE_STOPPED) { - try { - Thread.sleep( status.dwWaitHint ); + long msRemainingBeforeTimeout = timeout - (System.currentTimeMillis() - startTime); + + if (msRemainingBeforeTimeout < 0) { + throw new RuntimeException(String.format("Service stop exceeded timeout time of %d ms", timeout)); + } + + long dwWaitTime = Math.min(sanitizeWaitTime(status.dwWaitHint), msRemainingBeforeTimeout); + + try { + Thread.sleep( dwWaitTime ); } catch (InterruptedException e){ throw new RuntimeException(e); } if(! Advapi32.INSTANCE.QueryServiceStatus(_handle, status)) { throw new Win32Exception(Kernel32.INSTANCE.GetLastError()); } - if((System.currentTimeMillis() - startTime) > timeout) { - throw new RuntimeException(String.format("Service stop exceeded timeout time of %d ms", timeout)); - } } - waitForNonPendingState(); - if (queryStatus().dwCurrentState != Winsvc.SERVICE_STOPPED) { - throw new RuntimeException("Unable to stop the service"); - } } /** @@ -292,7 +293,22 @@ public void pauseService() { } } - /** + /** + * do not wait longer than the wait hint. A good interval is + * one-tenth the wait hint, but no less than 1 second and no + * more than 10 seconds. + */ + int sanitizeWaitTime(int dwWaitHint) { + int dwWaitTime = dwWaitHint / 10; + + if (dwWaitTime < 1000) + dwWaitTime = 1000; + else if (dwWaitTime > 10000) + dwWaitTime = 10000; + return dwWaitTime; + } + + /** * Wait for the state to change to something other than a pending state. */ public void waitForNonPendingState() { @@ -315,16 +331,7 @@ public void waitForNonPendingState() { throw new RuntimeException("Timeout waiting for service to change to a non-pending state."); } - // do not wait longer than the wait hint. A good interval is - // one-tenth the wait hint, but no less than 1 second and no - // more than 10 seconds. - - int dwWaitTime = status.dwWaitHint / 10; - - if (dwWaitTime < 1000) - dwWaitTime = 1000; - else if (dwWaitTime > 10000) - dwWaitTime = 10000; + int dwWaitTime = sanitizeWaitTime(status.dwWaitHint); try { Thread.sleep( dwWaitTime );