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
57static Bool (*compositeQueryExtension) (Display*, int*, int*);
58static Status (*compositeQueryVersion) (Display*, int*, int*);
59static Window (*compositeGetOverlayWindow) (Display *, Window);
60
61extern struct X11GraphicsConfigIDs x11GraphicsConfigIDs;
62
63static jint * masks;
64static jint num_buttons;
65
66static void *xCompositeHandle;
67
68static const char* XCOMPOSITE = JNI_LIB_NAME("Xcomposite");
69static const char* XCOMPOSITE_VERSIONED = VERSIONED_JNI_LIB_NAME("Xcomposite", "1");
70
71static Bool checkXCompositeFunctions(void) {
72 return (compositeQueryExtension != NULL &&
73 compositeQueryVersion != NULL &&
74 compositeGetOverlayWindow != NULL);
75}
76
77static 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
97static 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
131static 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
150static 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
161static 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
236JNIEXPORT void JNICALL
237Java_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
272JNIEXPORT void JNICALL
273Java_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
373JNIEXPORT void JNICALL
374Java_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
393JNIEXPORT void JNICALL
394Java_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
411JNIEXPORT void JNICALL
412Java_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 */
436void 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
480JNIEXPORT void JNICALL
481Java_sun_awt_X11_XRobotPeer_mousePressImpl (JNIEnv *env,
482 jclass cls,
483 jint buttonMask) {
484 mouseAction(env, cls, buttonMask, True);
485}
486
487JNIEXPORT void JNICALL
488Java_sun_awt_X11_XRobotPeer_mouseReleaseImpl (JNIEnv *env,
489 jclass cls,
490 jint buttonMask) {
491 mouseAction(env, cls, buttonMask, False);
492}
493
494JNIEXPORT void JNICALL
495Java_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
522JNIEXPORT void JNICALL
523Java_sun_awt_X11_XRobotPeer_loadNativeLibraries (JNIEnv *env, jclass cls) {
524 initXCompositeFunctions();
525}
526