1 /*
2  * Copyright (c) 1999, 2017, Oracle and/or its affiliates. All rights reserved.
3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4  *
5  * This code is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License version 2 only, as
7  * published by the Free Software Foundation.  Oracle designates this
8  * particular file as subject to the "Classpath" exception as provided
9  * by Oracle in the LICENSE file that accompanied this code.
10  *
11  * This code is distributed in the hope that it will be useful, but WITHOUT
12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
14  * version 2 for more details (a copy is included in the LICENSE file that
15  * accompanied this code).
16  *
17  * You should have received a copy of the GNU General Public License version
18  * 2 along with this work; if not, write to the Free Software Foundation,
19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20  *
21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22  * or visit www.oracle.com if you need additional information or have any
23  * questions.
24  */
25 
26 #ifdef HEADLESS
27     #error This file should not be included in headless library
28 #endif
29 
30 #include "jvm_md.h"
31 #include <dlfcn.h>
32 
33 #include "awt_p.h"
34 #include "awt_GraphicsEnv.h"
35 #define XK_MISCELLANY
36 #include <X11/keysymdef.h>
37 #include <X11/Xutil.h>
38 #include <X11/Xmd.h>
39 #include <X11/extensions/xtestext1.h>
40 #include <X11/extensions/XTest.h>
41 #include <X11/extensions/XInput.h>
42 #include <X11/extensions/XI.h>
43 #include <jni.h>
44 #include <sizecalc.h>
45 #include "canvas.h"
46 #include "wsutils.h"
47 #include "list.h"
48 #include "multiVis.h"
49 #include "gtk_interface.h"
50 
51 #include "java_awt_event_InputEvent.h"
52 
53 #if defined(__linux__) || defined(MACOSX)
54 #include <sys/socket.h>
55 #endif
56 
57 static Bool   (*compositeQueryExtension)   (Display*, int*, int*);
58 static Status (*compositeQueryVersion)     (Display*, int*, int*);
59 static Window (*compositeGetOverlayWindow) (Display *, Window);
60 
61 extern struct X11GraphicsConfigIDs x11GraphicsConfigIDs;
62 
63 static jint * masks;
64 static jint num_buttons;
65 
66 static void *xCompositeHandle;
67 
68 static const char* XCOMPOSITE = JNI_LIB_NAME("Xcomposite");
69 static const char* XCOMPOSITE_VERSIONED = VERSIONED_JNI_LIB_NAME("Xcomposite", "1");
70 
checkXCompositeFunctions(void)71 static Bool checkXCompositeFunctions(void) {
72     return (compositeQueryExtension   != NULL   &&
73             compositeQueryVersion     != NULL   &&
74             compositeGetOverlayWindow != NULL);
75 }
76 
initXCompositeFunctions(void)77 static void initXCompositeFunctions(void) {
78 
79     if (xCompositeHandle == NULL) {
80         xCompositeHandle = dlopen(XCOMPOSITE, RTLD_LAZY | RTLD_GLOBAL);
81         if (xCompositeHandle == NULL) {
82             xCompositeHandle = dlopen(XCOMPOSITE_VERSIONED, RTLD_LAZY | RTLD_GLOBAL);
83         }
84     }
85     //*(void **)(&asyncGetCallTraceFunction)
86     if (xCompositeHandle != NULL) {
87         *(void **)(&compositeQueryExtension) = dlsym(xCompositeHandle, "XCompositeQueryExtension");
88         *(void **)(&compositeQueryVersion) = dlsym(xCompositeHandle, "XCompositeQueryVersion");
89         *(void **)(&compositeGetOverlayWindow) = dlsym(xCompositeHandle, "XCompositeGetOverlayWindow");
90     }
91 
92     if (xCompositeHandle && !checkXCompositeFunctions()) {
93         dlclose(xCompositeHandle);
94     }
95 }
96 
isXTestAvailable()97 static int32_t isXTestAvailable() {
98     int32_t major_opcode, first_event, first_error;
99     int32_t  event_basep, error_basep, majorp, minorp;
100     int32_t isXTestAvailable;
101 
102     /* check if XTest is available */
103     isXTestAvailable = XQueryExtension(awt_display, XTestExtensionName, &major_opcode, &first_event, &first_error);
104     if (isXTestAvailable) {
105         DTRACE_PRINTLN3("RobotPeer: XQueryExtension(XTEST) returns major_opcode = %d, first_event = %d, first_error = %d",
106                         major_opcode, first_event, first_error);
107         /* check if XTest version is OK */
108         XTestQueryExtension(awt_display, &event_basep, &error_basep, &majorp, &minorp);
109         DTRACE_PRINTLN4("RobotPeer: XTestQueryExtension returns event_basep = %d, error_basep = %d, majorp = %d, minorp = %d",
110                         event_basep, error_basep, majorp, minorp);
111         if (majorp < 2 || (majorp == 2 && minorp < 2)) {
112             /* bad version*/
113             DTRACE_PRINTLN2("XRobotPeer: XTEST version is %d.%d \n", majorp, minorp);
114             if (majorp == 2 && minorp == 1) {
115                 DTRACE_PRINTLN("XRobotPeer: XTEST is 2.1 - no grab is available\n");
116             } else {
117                 isXTestAvailable = False;
118             }
119         } else {
120             /* allow XTest calls even if someone else has the grab; e.g. during
121              * a window resize operation. Works only with XTEST2.2*/
122             XTestGrabControl(awt_display, True);
123         }
124     } else {
125         DTRACE_PRINTLN("RobotPeer: XTEST extension is unavailable");
126     }
127 
128     return isXTestAvailable;
129 }
130 
hasXCompositeOverlayExtension(Display * display)131 static Bool hasXCompositeOverlayExtension(Display *display) {
132 
133     int xoverlay = False;
134     int eventBase, errorBase;
135     if (checkXCompositeFunctions() &&
136         compositeQueryExtension(display, &eventBase, &errorBase))
137     {
138         int major = 0;
139         int minor = 0;
140 
141         compositeQueryVersion(display, &major, &minor);
142         if (major > 0 || minor >= 3) {
143             xoverlay = True;
144         }
145     }
146 
147     return xoverlay;
148 }
149 
isXCompositeDisplay(Display * display,int screenNumber)150 static jboolean isXCompositeDisplay(Display *display, int screenNumber) {
151 
152     char NET_WM_CM_Sn[25];
153     snprintf(NET_WM_CM_Sn, sizeof(NET_WM_CM_Sn), "_NET_WM_CM_S%d\0", screenNumber);
154 
155     Atom managerSelection = XInternAtom(display, NET_WM_CM_Sn, 0);
156     Window owner = XGetSelectionOwner(display, managerSelection);
157 
158     return owner != 0;
159 }
160 
getWindowImage(Display * display,Window window,int32_t x,int32_t y,int32_t w,int32_t h)161 static XImage *getWindowImage(Display * display, Window window,
162                               int32_t x, int32_t y,
163                               int32_t w, int32_t h) {
164     XImage         *image;
165     int32_t        transparentOverlays;
166     int32_t        numVisuals;
167     XVisualInfo    *pVisuals;
168     int32_t        numOverlayVisuals;
169     OverlayInfo    *pOverlayVisuals;
170     int32_t        numImageVisuals;
171     XVisualInfo    **pImageVisuals;
172     list_ptr       vis_regions;    /* list of regions to read from */
173     list_ptr       vis_image_regions ;
174     int32_t        allImage = 0 ;
175     int32_t        format = ZPixmap;
176 
177     /* prevent user from moving stuff around during the capture */
178     XGrabServer(display);
179 
180     /*
181      * The following two functions live in multiVis.c-- they are pretty
182      * much verbatim taken from the source to the xwd utility from the
183      * X11 source. This version of the xwd source was somewhat better written
184      * for reuse compared to Sun's version.
185      *
186      *        ftp.x.org/pub/R6.3/xc/programs/xwd
187      *
188      * We use these functions since they do the very tough job of capturing
189      * the screen correctly when it contains multiple visuals. They take into
190      * account the depth/colormap of each visual and produce a capture as a
191      * 24-bit RGB image so we don't have to fool around with colormaps etc.
192      */
193 
194     GetMultiVisualRegions(
195         display,
196         window,
197         x, y, w, h,
198         &transparentOverlays,
199         &numVisuals,
200         &pVisuals,
201         &numOverlayVisuals,
202         &pOverlayVisuals,
203         &numImageVisuals,
204         &pImageVisuals,
205         &vis_regions,
206         &vis_image_regions,
207         &allImage );
208 
209     image = ReadAreaToImage(
210         display,
211         window,
212         x, y, w, h,
213         numVisuals,
214         pVisuals,
215         numOverlayVisuals,
216         pOverlayVisuals,
217         numImageVisuals,
218         pImageVisuals,
219         vis_regions,
220         vis_image_regions,
221         format,
222         allImage );
223 
224     /* allow user to do stuff again */
225     XUngrabServer(display);
226 
227     /* make sure the grab/ungrab is flushed */
228     XSync(display, False);
229 
230     return image;
231 }
232 
233 /*********************************************************************************************/
234 
235 // this should be called from XRobotPeer constructor
236 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_setup(JNIEnv * env,jclass cls,jint numberOfButtons,jintArray buttonDownMasks)237 Java_sun_awt_X11_XRobotPeer_setup (JNIEnv * env, jclass cls, jint numberOfButtons, jintArray buttonDownMasks)
238 {
239     int32_t xtestAvailable;
240     jint *tmp;
241     int i;
242 
243     DTRACE_PRINTLN("RobotPeer: setup()");
244 
245     num_buttons = numberOfButtons;
246     tmp = (*env)->GetIntArrayElements(env, buttonDownMasks, JNI_FALSE);
247     CHECK_NULL(tmp);
248 
249     masks = (jint *)SAFE_SIZE_ARRAY_ALLOC(malloc, sizeof(jint), num_buttons);
250     if (masks == (jint *) NULL) {
251         (*env)->ExceptionClear(env);
252         (*env)->ReleaseIntArrayElements(env, buttonDownMasks, tmp, 0);
253         JNU_ThrowOutOfMemoryError((JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2), NULL);
254         return;
255     }
256     for (i = 0; i < num_buttons; i++) {
257         masks[i] = tmp[i];
258     }
259     (*env)->ReleaseIntArrayElements(env, buttonDownMasks, tmp, 0);
260 
261     AWT_LOCK();
262     xtestAvailable = isXTestAvailable();
263     DTRACE_PRINTLN1("RobotPeer: XTest available = %d", xtestAvailable);
264     if (!xtestAvailable) {
265         JNU_ThrowByName(env, "java/awt/AWTException", "java.awt.Robot requires your X server support the XTEST extension version 2.2");
266     }
267 
268     AWT_UNLOCK();
269 }
270 
271 
272 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_getRGBPixelsImpl(JNIEnv * env,jclass cls,jobject xgc,jint jx,jint jy,jint jwidth,jint jheight,jintArray pixelArray,jboolean useGtk)273 Java_sun_awt_X11_XRobotPeer_getRGBPixelsImpl( JNIEnv *env,
274                              jclass cls,
275                              jobject xgc,
276                              jint jx,
277                              jint jy,
278                              jint jwidth,
279                              jint jheight,
280                              jintArray pixelArray,
281                              jboolean useGtk) {
282     XImage *image;
283     jint *ary;               /* Array of jints for sending pixel values back
284                               * to parent process.
285                               */
286     Window rootWindow;
287     XWindowAttributes attr;
288     AwtGraphicsConfigDataPtr adata;
289 
290     DTRACE_PRINTLN6("RobotPeer: getRGBPixelsImpl(%lx, %d, %d, %d, %d, %x)", xgc, jx, jy, jwidth, jheight, pixelArray);
291 
292     if (jwidth <= 0 || jheight <= 0) {
293         return;
294     }
295 
296     adata = (AwtGraphicsConfigDataPtr) JNU_GetLongFieldAsPtr(env, xgc, x11GraphicsConfigIDs.aData);
297     DASSERT(adata != NULL);
298 
299     AWT_LOCK();
300 
301     rootWindow = XRootWindow(awt_display, adata->awt_visInfo.screen);
302 
303     if (!useGtk) {
304         if (hasXCompositeOverlayExtension(awt_display) &&
305             isXCompositeDisplay(awt_display, adata->awt_visInfo.screen))
306         {
307             rootWindow = compositeGetOverlayWindow(awt_display, rootWindow);
308         }
309     }
310 
311     if (!XGetWindowAttributes(awt_display, rootWindow, &attr)
312             || jx + jwidth <= attr.x
313             || attr.x + attr.width <= jx
314             || jy + jheight <= attr.y
315             || attr.y + attr.height <= jy) {
316 
317         AWT_UNLOCK();
318         return; // Does not intersect with root window
319     }
320 
321     gboolean gtk_failed = TRUE;
322     jint _x, _y;
323 
324     jint x = MAX(jx, attr.x);
325     jint y = MAX(jy, attr.y);
326     jint width = MIN(jx + jwidth, attr.x + attr.width) - x;
327     jint height = MIN(jy + jheight, attr.y + attr.height) - y;
328 
329     int dx = attr.x > jx ? attr.x - jx : 0;
330     int dy = attr.y > jy ? attr.y - jy : 0;
331 
332     int index;
333 
334     if (useGtk) {
335         gtk->gdk_threads_enter();
336         gtk_failed = gtk->get_drawable_data(env, pixelArray, x, y, width,
337                                             height, jwidth, dx, dy, 1);
338         gtk->gdk_threads_leave();
339     }
340 
341     if (gtk_failed) {
342         image = getWindowImage(awt_display, rootWindow, x, y, width, height);
343 
344         ary = (*env)->GetPrimitiveArrayCritical(env, pixelArray, NULL);
345 
346         if (!ary) {
347             XDestroyImage(image);
348             AWT_UNLOCK();
349             return;
350         }
351 
352         /* convert to Java ARGB pixels */
353         for (_y = 0; _y < height; _y++) {
354             for (_x = 0; _x < width; _x++) {
355                 jint pixel = (jint) XGetPixel(image, _x, _y);
356                                                               /* Note ignore upper
357                                                                * 32-bits on 64-bit
358                                                                * OSes.
359                                                                */
360                 pixel |= 0xff000000; /* alpha - full opacity */
361 
362                 index = (_y + dy) * jwidth + (_x + dx);
363                 ary[index] = pixel;
364             }
365         }
366 
367         XDestroyImage(image);
368         (*env)->ReleasePrimitiveArrayCritical(env, pixelArray, ary, 0);
369     }
370     AWT_UNLOCK();
371 }
372 
373 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_keyPressImpl(JNIEnv * env,jclass cls,jint keycode)374 Java_sun_awt_X11_XRobotPeer_keyPressImpl (JNIEnv *env,
375                          jclass cls,
376                          jint keycode) {
377 
378     AWT_LOCK();
379 
380     DTRACE_PRINTLN1("RobotPeer: keyPressImpl(%i)", keycode);
381 
382     XTestFakeKeyEvent(awt_display,
383                       XKeysymToKeycode(awt_display, awt_getX11KeySym(keycode)),
384                       True,
385                       CurrentTime);
386 
387     XSync(awt_display, False);
388 
389     AWT_UNLOCK();
390 
391 }
392 
393 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_keyReleaseImpl(JNIEnv * env,jclass cls,jint keycode)394 Java_sun_awt_X11_XRobotPeer_keyReleaseImpl (JNIEnv *env,
395                            jclass cls,
396                            jint keycode) {
397     AWT_LOCK();
398 
399     DTRACE_PRINTLN1("RobotPeer: keyReleaseImpl(%i)", keycode);
400 
401     XTestFakeKeyEvent(awt_display,
402                       XKeysymToKeycode(awt_display, awt_getX11KeySym(keycode)),
403                       False,
404                       CurrentTime);
405 
406     XSync(awt_display, False);
407 
408     AWT_UNLOCK();
409 }
410 
411 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_mouseMoveImpl(JNIEnv * env,jclass cls,jobject xgc,jint root_x,jint root_y)412 Java_sun_awt_X11_XRobotPeer_mouseMoveImpl (JNIEnv *env,
413                           jclass cls,
414                           jobject xgc,
415                           jint root_x,
416                           jint root_y) {
417 
418     AwtGraphicsConfigDataPtr adata;
419 
420     AWT_LOCK();
421 
422     DTRACE_PRINTLN3("RobotPeer: mouseMoveImpl(%lx, %i, %i)", xgc, root_x, root_y);
423 
424     adata = (AwtGraphicsConfigDataPtr) JNU_GetLongFieldAsPtr(env, xgc, x11GraphicsConfigIDs.aData);
425     DASSERT(adata != NULL);
426 
427     XWarpPointer(awt_display, None, XRootWindow(awt_display, adata->awt_visInfo.screen), 0, 0, 0, 0, root_x, root_y);
428     XSync(awt_display, False);
429 
430     AWT_UNLOCK();
431 }
432 
433 /*
434   * Function joining the code of mousePressImpl and mouseReleaseImpl
435   */
mouseAction(JNIEnv * env,jclass cls,jint buttonMask,Bool isMousePress)436 void mouseAction(JNIEnv *env,
437                  jclass cls,
438                  jint buttonMask,
439                  Bool isMousePress)
440 {
441     AWT_LOCK();
442 
443     DTRACE_PRINTLN1("RobotPeer: mouseAction(%i)", buttonMask);
444     DTRACE_PRINTLN1("RobotPeer: mouseAction, press = %d", isMousePress);
445 
446     if (buttonMask & java_awt_event_InputEvent_BUTTON1_MASK ||
447         buttonMask & java_awt_event_InputEvent_BUTTON1_DOWN_MASK )
448     {
449         XTestFakeButtonEvent(awt_display, 1, isMousePress, CurrentTime);
450     }
451     if ((buttonMask & java_awt_event_InputEvent_BUTTON2_MASK ||
452          buttonMask & java_awt_event_InputEvent_BUTTON2_DOWN_MASK) &&
453         (num_buttons >= 2)) {
454         XTestFakeButtonEvent(awt_display, 2, isMousePress, CurrentTime);
455     }
456     if ((buttonMask & java_awt_event_InputEvent_BUTTON3_MASK ||
457          buttonMask & java_awt_event_InputEvent_BUTTON3_DOWN_MASK) &&
458         (num_buttons >= 3)) {
459         XTestFakeButtonEvent(awt_display, 3, isMousePress, CurrentTime);
460     }
461 
462     if (num_buttons > 3){
463         int32_t i;
464         int32_t button = 0;
465         for (i = 3; i<num_buttons; i++){
466             if ((buttonMask & masks[i])) {
467                 // arrays starts from zero index => +1
468                 // users wants to affect 4 or 5 button but they are assigned
469                 // to the wheel so => we have to shift it to the right by 2.
470                 button = i + 3;
471                 XTestFakeButtonEvent(awt_display, button, isMousePress, CurrentTime);
472             }
473         }
474     }
475 
476     XSync(awt_display, False);
477     AWT_UNLOCK();
478 }
479 
480 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_mousePressImpl(JNIEnv * env,jclass cls,jint buttonMask)481 Java_sun_awt_X11_XRobotPeer_mousePressImpl (JNIEnv *env,
482                            jclass cls,
483                            jint buttonMask) {
484     mouseAction(env, cls, buttonMask, True);
485 }
486 
487 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_mouseReleaseImpl(JNIEnv * env,jclass cls,jint buttonMask)488 Java_sun_awt_X11_XRobotPeer_mouseReleaseImpl (JNIEnv *env,
489                              jclass cls,
490                              jint buttonMask) {
491     mouseAction(env, cls, buttonMask, False);
492 }
493 
494 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_mouseWheelImpl(JNIEnv * env,jclass cls,jint wheelAmt)495 Java_sun_awt_X11_XRobotPeer_mouseWheelImpl (JNIEnv *env,
496                            jclass cls,
497                            jint wheelAmt) {
498 /* Mouse wheel is implemented as a button press of button 4 and 5, so it */
499 /* probably could have been hacked into robot_mouseButtonEvent, but it's */
500 /* cleaner to give it its own command type, in case the implementation   */
501 /* needs to be changed later.  -bchristi, 6/20/01                        */
502 
503     int32_t repeat = abs(wheelAmt);
504     int32_t button = wheelAmt < 0 ? 4 : 5;  /* wheel up:   button 4 */
505                                                  /* wheel down: button 5 */
506     int32_t loopIdx;
507 
508     AWT_LOCK();
509 
510     DTRACE_PRINTLN1("RobotPeer: mouseWheelImpl(%i)", wheelAmt);
511 
512     for (loopIdx = 0; loopIdx < repeat; loopIdx++) { /* do nothing for   */
513                                                      /* wheelAmt == 0    */
514         XTestFakeButtonEvent(awt_display, button, True, CurrentTime);
515         XTestFakeButtonEvent(awt_display, button, False, CurrentTime);
516     }
517     XSync(awt_display, False);
518 
519     AWT_UNLOCK();
520 }
521 
522 JNIEXPORT void JNICALL
Java_sun_awt_X11_XRobotPeer_loadNativeLibraries(JNIEnv * env,jclass cls)523 Java_sun_awt_X11_XRobotPeer_loadNativeLibraries (JNIEnv *env, jclass cls) {
524     initXCompositeFunctions();
525 }
526