Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/mobile
Path: blob/master/src/java.desktop/unix/native/libawt_xawt/awt/awt_Robot.c
41154 views
1
/*
2
* Copyright (c) 1999, 2020, 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__)
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
71
static Bool checkXCompositeFunctions(void) {
72
return (compositeQueryExtension != NULL &&
73
compositeQueryVersion != NULL &&
74
compositeGetOverlayWindow != NULL);
75
}
76
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
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
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
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", 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
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
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
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
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
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
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
*/
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
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
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
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
523
Java_sun_awt_X11_XRobotPeer_loadNativeLibraries (JNIEnv *env, jclass cls) {
524
initXCompositeFunctions();
525
}
526
527