Index: tools/telemetry/telemetry/core/platform/win_platform_backend.py |
diff --git a/tools/telemetry/telemetry/core/platform/win_platform_backend.py b/tools/telemetry/telemetry/core/platform/win_platform_backend.py |
index de9ec475741195492fa9b9b1674a84f39e95f404..e9d36a429904b4a2f5575108b1701ea9176c4607 100644 |
--- a/tools/telemetry/telemetry/core/platform/win_platform_backend.py |
+++ b/tools/telemetry/telemetry/core/platform/win_platform_backend.py |
@@ -6,6 +6,7 @@ import atexit |
import collections |
import contextlib |
import ctypes |
+import logging |
import os |
import platform |
import re |
@@ -31,6 +32,7 @@ try: |
from win32com.shell import shell # pylint: disable=F0401 |
from win32com.shell import shellcon # pylint: disable=F0401 |
import win32con # pylint: disable=F0401 |
+ import win32gui # pylint: disable=F0401 |
import win32process # pylint: disable=F0401 |
import win32security # pylint: disable=F0401 |
except ImportError: |
@@ -39,6 +41,7 @@ except ImportError: |
shellcon = None |
win32api = None |
win32con = None |
+ win32gui = None |
win32process = None |
win32security = None |
@@ -362,3 +365,38 @@ class WinPlatformBackend(desktop_platform_backend.DesktopPlatformBackend): |
finally: |
sock.close() |
return struct.unpack('Q', response)[0] >> start & ((1 << length) - 1) |
+ |
+ def IsCooperativeShutdownSupported(self): |
+ return True |
+ |
+ def CooperativelyShutdown(self, proc, app_name): |
+ pid = proc.pid |
+ |
+ # http://timgolden.me.uk/python/win32_how_do_i/ |
+ # find-the-window-for-my-subprocess.html |
+ # |
+ # It seems that intermittently this code manages to find windows |
+ # that don't belong to Chrome -- for example, the cmd.exe window |
+ # running slave.bat on the tryservers. Try to be careful about |
+ # finding only Chrome's windows. This works for both the browser |
+ # and content_shell. |
+ # |
+ # It seems safest to send the WM_CLOSE messages after discovering |
+ # all of the sub-process's windows. |
+ def find_chrome_windows(hwnd, hwnds): |
+ _, win_pid = win32process.GetWindowThreadProcessId(hwnd) |
+ if (pid == win_pid and |
+ win32gui.IsWindowVisible(hwnd) and |
+ win32gui.IsWindowEnabled(hwnd) and |
+ win32gui.GetClassName(hwnd).lower().startswith(app_name)): |
+ hwnds.append(hwnd) |
+ return True |
+ hwnds = [] |
+ win32gui.EnumWindows(find_chrome_windows, hwnds) |
+ if hwnds: |
+ for hwnd in hwnds: |
+ win32gui.SendMessage(hwnd, win32con.WM_CLOSE, 0, 0) |
+ return True |
+ else: |
+ logging.info('Did not find any windows owned by target process') |
+ return False |