Skip to content

Commit 72b251d

Browse files
committedMar 17, 2020
8226806: [macOS 10.14] Methods of Java Robot should be called from appropriate thread
Reviewed-by: psadhukhan, prr
1 parent bca2465 commit 72b251d

File tree

2 files changed

+149
-54
lines changed

2 files changed

+149
-54
lines changed
 

‎src/java.desktop/macosx/native/libawt_lwawt/awt/CRobot.m

+60-54
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2016, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2016, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -23,7 +23,6 @@
2323
* questions.
2424
*/
2525

26-
2726
#import "jni_util.h"
2827

2928
#import <JavaNativeFoundation/JavaNativeFoundation.h>
@@ -35,6 +34,7 @@
3534
#import "java_awt_event_InputEvent.h"
3635
#import "java_awt_event_KeyEvent.h"
3736
#import "sizecalc.h"
37+
#import "ThreadUtilities.h"
3838

3939
// Starting number for event numbers generated by Robot.
4040
// Apple docs don't mention at all what are the requirements
@@ -97,38 +97,40 @@ static void PostMouseEvent(const CGPoint point, CGMouseButton button,
9797
// Always set all states, in case Apple ever changes default behaviors.
9898
static int setupDone = 0;
9999
if (!setupDone) {
100-
int i;
101-
jint* tmp;
102-
jboolean copy = JNI_FALSE;
103-
104-
setupDone = 1;
105-
// Don't block local events after posting ours
106-
CGSetLocalEventsSuppressionInterval(0.0);
107-
108-
// Let our event's modifier key state blend with local hardware events
109-
CGEnableEventStateCombining(TRUE);
110-
111-
// Don't let our events block local hardware events
112-
CGSetLocalEventsFilterDuringSupressionState(
113-
kCGEventFilterMaskPermitAllEvents,
114-
kCGEventSupressionStateSupressionInterval);
115-
CGSetLocalEventsFilterDuringSupressionState(
116-
kCGEventFilterMaskPermitAllEvents,
117-
kCGEventSupressionStateRemoteMouseDrag);
118-
119-
gsClickCount = 0;
120-
gsLastClickTime = 0;
121-
gsEventNumber = ROBOT_EVENT_NUMBER_START;
100+
[ThreadUtilities performOnMainThreadWaiting:NO block:^(){
101+
int i;
102+
jint* tmp;
103+
jboolean copy = JNI_FALSE;
104+
105+
setupDone = 1;
106+
// Don't block local events after posting ours
107+
CGSetLocalEventsSuppressionInterval(0.0);
108+
109+
// Let our event's modifier key state blend with local hardware events
110+
CGEnableEventStateCombining(TRUE);
111+
112+
// Don't let our events block local hardware events
113+
CGSetLocalEventsFilterDuringSupressionState(
114+
kCGEventFilterMaskPermitAllEvents,
115+
kCGEventSupressionStateSupressionInterval);
116+
CGSetLocalEventsFilterDuringSupressionState(
117+
kCGEventFilterMaskPermitAllEvents,
118+
kCGEventSupressionStateRemoteMouseDrag);
122119

123-
gsButtonEventNumber = (int*)SAFE_SIZE_ARRAY_ALLOC(malloc, sizeof(int), gNumberOfButtons);
124-
if (gsButtonEventNumber == NULL) {
125-
JNU_ThrowOutOfMemoryError(env, NULL);
126-
return;
127-
}
120+
gsClickCount = 0;
121+
gsLastClickTime = 0;
122+
gsEventNumber = ROBOT_EVENT_NUMBER_START;
128123

129-
for (i = 0; i < gNumberOfButtons; ++i) {
130-
gsButtonEventNumber[i] = ROBOT_EVENT_NUMBER_START;
131-
}
124+
gsButtonEventNumber = (int*)SAFE_SIZE_ARRAY_ALLOC(malloc, sizeof(int), gNumberOfButtons);
125+
if (gsButtonEventNumber == NULL) {
126+
JNU_ThrowOutOfMemoryError(env, NULL);
127+
return;
128+
}
129+
130+
for (i = 0; i < gNumberOfButtons; ++i) {
131+
gsButtonEventNumber[i] = ROBOT_EVENT_NUMBER_START;
132+
}
133+
}];
132134
}
133135
}
134136

@@ -239,14 +241,15 @@ static void PostMouseEvent(const CGPoint point, CGMouseButton button,
239241
Java_sun_lwawt_macosx_CRobot_mouseWheel
240242
(JNIEnv *env, jobject peer, jint wheelAmt)
241243
{
242-
CGEventRef event = CGEventCreateScrollWheelEvent(NULL,
243-
kCGScrollEventUnitLine,
244-
k_JAVA_ROBOT_WHEEL_COUNT, wheelAmt);
245-
246-
if (event != NULL) {
247-
CGEventPost(kCGSessionEventTap, event);
248-
CFRelease(event);
249-
}
244+
[ThreadUtilities performOnMainThreadWaiting:YES block:^(){
245+
CGEventRef event = CGEventCreateScrollWheelEvent(NULL,
246+
kCGScrollEventUnitLine,
247+
k_JAVA_ROBOT_WHEEL_COUNT, wheelAmt);
248+
if (event != NULL) {
249+
CGEventPost(kCGSessionEventTap, event);
250+
CFRelease(event);
251+
}
252+
}];
250253
}
251254

252255
/*
@@ -258,13 +261,14 @@ static void PostMouseEvent(const CGPoint point, CGMouseButton button,
258261
Java_sun_lwawt_macosx_CRobot_keyEvent
259262
(JNIEnv *env, jobject peer, jint javaKeyCode, jboolean keyPressed)
260263
{
261-
CGKeyCode keyCode = GetCGKeyCode(javaKeyCode);
262-
263-
CGEventRef event = CGEventCreateKeyboardEvent(NULL, keyCode, keyPressed);
264-
if (event != NULL) {
265-
CGEventPost(kCGSessionEventTap, event);
266-
CFRelease(event);
267-
}
264+
[ThreadUtilities performOnMainThreadWaiting:YES block:^(){
265+
CGKeyCode keyCode = GetCGKeyCode(javaKeyCode);
266+
CGEventRef event = CGEventCreateKeyboardEvent(NULL, keyCode, keyPressed);
267+
if (event != NULL) {
268+
CGEventPost(kCGSessionEventTap, event);
269+
CFRelease(event);
270+
}
271+
}];
268272
}
269273

270274
/*
@@ -333,13 +337,15 @@ static void PostMouseEvent(const CGPoint point, CGMouseButton button,
333337
static void PostMouseEvent(const CGPoint point, CGMouseButton button,
334338
CGEventType type, int clickCount, int eventNumber)
335339
{
336-
CGEventRef mouseEvent = CGEventCreateMouseEvent(NULL, type, point, button);
337-
if (mouseEvent != NULL) {
338-
CGEventSetIntegerValueField(mouseEvent, kCGMouseEventClickState, clickCount);
339-
CGEventSetIntegerValueField(mouseEvent, kCGMouseEventNumber, eventNumber);
340-
CGEventPost(kCGSessionEventTap, mouseEvent);
341-
CFRelease(mouseEvent);
342-
}
340+
[ThreadUtilities performOnMainThreadWaiting:YES block:^(){
341+
CGEventRef mouseEvent = CGEventCreateMouseEvent(NULL, type, point, button);
342+
if (mouseEvent != NULL) {
343+
CGEventSetIntegerValueField(mouseEvent, kCGMouseEventClickState, clickCount);
344+
CGEventSetIntegerValueField(mouseEvent, kCGMouseEventNumber, eventNumber);
345+
CGEventPost(kCGSessionEventTap, mouseEvent);
346+
CFRelease(mouseEvent);
347+
}
348+
}];
343349
}
344350

345351
static inline CGKeyCode GetCGKeyCode(jint javaKeyCode)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
/*
2+
* Copyright (c) 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.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import java.awt.Frame;
25+
import java.awt.Robot;
26+
import java.awt.*;
27+
import java.awt.event.InputEvent;
28+
import java.awt.event.KeyEvent;
29+
import java.util.concurrent.TimeUnit;
30+
31+
import jdk.test.lib.process.OutputAnalyzer;
32+
import jdk.test.lib.process.ProcessTools;
33+
34+
/**
35+
* @test
36+
* @bug 8226806
37+
* @key headful
38+
* @library /test/lib
39+
* @summary checks for unexpected output in stderr and stdout
40+
*/
41+
public final class NonEmptyErrorStream {
42+
43+
public static void main(String[] args) throws Exception {
44+
if (args.length == 0) {
45+
ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(
46+
NonEmptyErrorStream.class.getSimpleName(),"run");
47+
Process p = pb.start();
48+
OutputAnalyzer output = new OutputAnalyzer(p);
49+
p.waitFor(20, TimeUnit.SECONDS);
50+
p.destroy();
51+
output.shouldHaveExitValue(0);
52+
output.stdoutShouldBeEmpty();
53+
output.stderrShouldBeEmpty();
54+
return;
55+
}
56+
Frame frame = new Frame();
57+
frame.setSize(400, 300);
58+
frame.setLocationRelativeTo(null);
59+
frame.setVisible(true);
60+
61+
Robot robot = new Robot();
62+
robot.setAutoDelay(50);
63+
robot.waitForIdle();
64+
frame.toFront();
65+
66+
Rectangle rect = frame.getBounds();
67+
int x = (int) rect.getCenterX();
68+
int y = (int) rect.getCenterY();
69+
70+
for (int i = 0; i < 20; i++) {
71+
robot.getPixelColor(x, y);
72+
robot.createScreenCapture(rect);
73+
}
74+
for (int i = 0; i < 20; i++) {
75+
robot.mouseMove(x + 50, y + 50);
76+
robot.mouseMove(x - 50, y - 50);
77+
}
78+
for (int i = 0; i < 20; i++) {
79+
robot.keyPress(KeyEvent.VK_ESCAPE);
80+
robot.keyRelease(KeyEvent.VK_ESCAPE);
81+
}
82+
for (int i = 0; i < 20; i++) {
83+
robot.mousePress(InputEvent.BUTTON1_MASK);
84+
robot.mouseRelease(InputEvent.BUTTON1_MASK);
85+
}
86+
robot.waitForIdle();
87+
frame.dispose();
88+
}
89+
}

0 commit comments

Comments
 (0)
Please sign in to comment.