X11 JNA坏窗口(无效窗口参数)导致JVM退出



下面的JNA使用JNA 4.2.2崩溃-如果您在最小化和最大化窗口时运行它,JVM将退出。在Ubuntu 16.04上发生,但在Redhat 6u2上更常见。

X error of failed request: BadWindow (invalid Window parameter)
Major opcode of failed request: 20 (X_GetProperty)
...

我猜窗口id是有效的,当他们从XQueryTree返回,但在调用XGetWMName之前变得无效。问题是这会导致JVM退出,而不仅仅是一个我可以很好地处理的异常。如何避免这个问题?

public class X11WindowFinder {
    private X11 x11;
    public X11WindowFinder() {
        x11 = X11.INSTANCE;
    }
    public List<Window> find(Pattern title) {
        Display display = x11.XOpenDisplay(null);
        Window root = x11.XDefaultRootWindow(display);
        List<Window> windows = recurse(x11, display, root, title);
        x11.XCloseDisplay(display);
        return windows;
    }
    private synchronized List<Window> recurse(X11 x11, Display display, Window root, Pattern pattern) {
        List<Window> windows = new ArrayList<>(1);
        X11.WindowByReference windowRef = new X11.WindowByReference();
        X11.WindowByReference parentRef = new X11.WindowByReference();
        PointerByReference childrenRef = new PointerByReference();
        IntByReference childCountRef = new IntByReference();
        x11.XQueryTree(display, root, windowRef, parentRef, childrenRef, childCountRef);
        if (childrenRef.getValue() == null) {
            return Collections.emptyList();
        }

        long[] ids = {};
        if (Native.LONG_SIZE == Long.BYTES) {
            ids = childrenRef.getValue().getLongArray(0, childCountRef.getValue());
        } else if (Native.LONG_SIZE == Integer.BYTES) {
            int[] intIds = childrenRef.getValue().getIntArray(0, childCountRef.getValue());
            ids = new long[intIds.length];
            for (int i = 0; i < intIds.length; i++) {
                ids[i] = intIds[i];
            }
        }
        for (long id : ids) {
            Window child = new Window(id);
            X11.XTextProperty name = new X11.XTextProperty();
            x11.XGetWMName(display, child, name);
            String value = name.value;
            if (value != null) {
                System.out.println(String.format("Found window %s free %s", value, name));
            }
            if (value != null && pattern.matcher(value).matches()) {
                windows.add(child);
            }
            windows.addAll(recurse(x11, display, child, pattern));
        }
        return windows;
    }
    public static void main(String[] args) {
        X11WindowFinder finder = new X11WindowFinder();
        while (true) {
            finder.find(Pattern.compile(".*Firefox.*"));
        }
    }
}

看起来像默认的错误处理程序导致退出,但可以被覆盖。例如

x11.XSetErrorHandler(new XErrorHandler() {
    @Override
    public int apply(Display display, XErrorEvent errorEvent) {
        System.err.println("bad fish " + errorEvent);
        return 0;
    }
});

最新更新