Skip to content

Commit e58f33a

Browse files
committedMar 14, 2025
8280468: Crashes in getConfigColormap, getConfigVisualId, XVisualIDFromVisual on Linux
Backport-of: 05dac5a23ed2813b2f4f2e4f007ebb93b4ae23ef
1 parent a16057a commit e58f33a

File tree

3 files changed

+51
-45
lines changed

3 files changed

+51
-45
lines changed
 

‎src/java.desktop/unix/classes/sun/awt/X11/XCanvasPeer.java

+19-22
Original file line numberDiff line numberDiff line change
@@ -62,36 +62,33 @@ public GraphicsConfiguration getAppropriateGraphicsConfiguration(
6262
if (graphicsConfig == null || gc == null) {
6363
return gc;
6464
}
65-
// Opt: Only need to do if we're not using the default GC
6665

67-
int screenNum = ((X11GraphicsDevice)gc.getDevice()).getScreen();
66+
final X11GraphicsDevice newDev = getSameScreenDevice(gc);
67+
final int visualToLookFor = graphicsConfig.getVisual();
6868

69-
X11GraphicsConfig parentgc;
70-
// save vis id of current gc
71-
int visual = graphicsConfig.getVisual();
72-
73-
X11GraphicsDevice newDev = (X11GraphicsDevice) GraphicsEnvironment.
74-
getLocalGraphicsEnvironment().
75-
getScreenDevices()[screenNum];
76-
77-
for (int i = 0; i < newDev.getNumConfigs(screenNum); i++) {
78-
if (visual == newDev.getConfigVisualId(i, screenNum)) {
79-
// use that
80-
graphicsConfig = (X11GraphicsConfig)newDev.getConfigurations()[i];
81-
break;
69+
final GraphicsConfiguration[] configurations = newDev.getConfigurations();
70+
for (final GraphicsConfiguration config : configurations) {
71+
final X11GraphicsConfig x11gc = (X11GraphicsConfig) config;
72+
if (visualToLookFor == x11gc.getVisual()) {
73+
graphicsConfig = x11gc;
8274
}
8375
}
84-
// just in case...
85-
if (graphicsConfig == null) {
86-
graphicsConfig = (X11GraphicsConfig) GraphicsEnvironment.
87-
getLocalGraphicsEnvironment().
88-
getScreenDevices()[screenNum].
89-
getDefaultConfiguration();
90-
}
9176

9277
return graphicsConfig;
9378
}
9479

80+
private X11GraphicsDevice getSameScreenDevice(GraphicsConfiguration gc) {
81+
XToolkit.awtLock(); // so that the number of screens doesn't change during
82+
try {
83+
final int screenNum = ((X11GraphicsDevice) gc.getDevice()).getScreen();
84+
return (X11GraphicsDevice) GraphicsEnvironment.
85+
getLocalGraphicsEnvironment().
86+
getScreenDevices()[screenNum];
87+
} finally {
88+
XToolkit.awtUnlock();
89+
}
90+
}
91+
9592
protected boolean shouldFocusOnClick() {
9693
// Canvas should always be able to be focused by mouse clicks.
9794
return true;

‎src/java.desktop/unix/classes/sun/awt/X11GraphicsDevice.java

+11-3
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@
4141
import sun.awt.util.ThreadGroupUtils;
4242
import sun.java2d.SunGraphicsEnvironment;
4343
import sun.java2d.loops.SurfaceType;
44+
import sun.awt.X11.XToolkit;
4445
import sun.java2d.opengl.GLXGraphicsConfig;
4546
import sun.java2d.pipe.Region;
4647
import sun.java2d.xr.XRGraphicsConfig;
@@ -63,7 +64,6 @@ public final class X11GraphicsDevice extends GraphicsDevice
6364

6465
private static AWTPermission fullScreenExclusivePermission;
6566
private static Boolean xrandrExtSupported;
66-
private final Object configLock = new Object();
6767
private SunDisplayChanger topLevels = new SunDisplayChanger();
6868
private DisplayMode origDisplayMode;
6969
private boolean shutdownHookRegistered;
@@ -150,8 +150,11 @@ public String getIDstring() {
150150
@Override
151151
public GraphicsConfiguration[] getConfigurations() {
152152
if (configs == null) {
153-
synchronized (configLock) {
153+
XToolkit.awtLock();
154+
try {
154155
makeConfigurations();
156+
} finally {
157+
XToolkit.awtUnlock();
155158
}
156159
}
157160
return configs.clone();
@@ -238,8 +241,11 @@ private void addDoubleBufferVisual(int visNum) {
238241
@Override
239242
public GraphicsConfiguration getDefaultConfiguration() {
240243
if (defaultConfig == null) {
241-
synchronized (configLock) {
244+
XToolkit.awtLock();
245+
try {
242246
makeDefaultConfiguration();
247+
} finally {
248+
XToolkit.awtUnlock();
243249
}
244250
}
245251
return defaultConfig;
@@ -571,6 +577,8 @@ public String toString() {
571577
}
572578

573579
public void invalidate(X11GraphicsDevice device) {
580+
assert XToolkit.isAWTLockHeldByCurrentThread();
581+
574582
screen = device.screen;
575583
}
576584
}

‎src/java.desktop/unix/native/libawt_xawt/awt/awt_GraphicsEnv.c

+21-20
Original file line numberDiff line numberDiff line change
@@ -292,6 +292,8 @@ makeDefaultConfig(JNIEnv *env, int screen) {
292292

293293
static void
294294
getAllConfigs (JNIEnv *env, int screen, AwtScreenDataPtr screenDataPtr) {
295+
// NB: should be invoked only while holding the AWT lock
296+
DASSERT(screen >= 0 && screen < awt_numScreens);
295297

296298
int i;
297299
int n8p=0, n12p=0, n8s=0, n8gs=0, n8sg=0, n1sg=0, nTrue=0;
@@ -314,8 +316,6 @@ getAllConfigs (JNIEnv *env, int screen, AwtScreenDataPtr screenDataPtr) {
314316
xinawareScreen = screen;
315317
}
316318

317-
AWT_LOCK ();
318-
319319
viTmp.screen = xinawareScreen;
320320

321321
viTmp.depth = 8;
@@ -372,7 +372,6 @@ getAllConfigs (JNIEnv *env, int screen, AwtScreenDataPtr screenDataPtr) {
372372
if (graphicsConfigs == NULL) {
373373
JNU_ThrowOutOfMemoryError((JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2),
374374
NULL);
375-
AWT_UNLOCK();
376375
return;
377376
}
378377

@@ -382,6 +381,9 @@ getAllConfigs (JNIEnv *env, int screen, AwtScreenDataPtr screenDataPtr) {
382381
* been reset, so we need to recreate the default config here.
383382
*/
384383
screenDataPtr->defaultConfig = makeDefaultConfig(env, screen);
384+
if (screenDataPtr->defaultConfig == NULL) {
385+
return;
386+
}
385387
}
386388

387389
defaultConfig = screenDataPtr->defaultConfig;
@@ -581,7 +583,6 @@ getAllConfigs (JNIEnv *env, int screen, AwtScreenDataPtr screenDataPtr) {
581583
XFree (pVI1sg);
582584
if (nTrue != 0)
583585
XFree (pVITrue);
584-
AWT_UNLOCK ();
585586
}
586587

587588
/*
@@ -770,6 +771,7 @@ JNIEnv *env, jobject this)
770771
}
771772

772773
static void ensureConfigsInited(JNIEnv* env, int screen) {
774+
// NB: should be invoked only while holding the AWT lock
773775
if (x11Screens[screen].numConfigs == 0) {
774776
if (env == NULL) {
775777
env = (JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
@@ -780,6 +782,8 @@ static void ensureConfigsInited(JNIEnv* env, int screen) {
780782

781783
AwtGraphicsConfigDataPtr
782784
getDefaultConfig(int screen) {
785+
// NB: should be invoked only while holding the AWT lock
786+
DASSERT(screen >= 0 && screen < awt_numScreens);
783787
ensureConfigsInited(NULL, screen);
784788
return x11Screens[screen].defaultConfig;
785789
}
@@ -973,11 +977,11 @@ JNIEXPORT jint JNICALL
973977
Java_sun_awt_X11GraphicsDevice_getNumConfigs(
974978
JNIEnv *env, jobject this, jint screen)
975979
{
976-
AWT_LOCK();
980+
// NB: should be invoked only while holding the AWT lock
981+
DASSERT(screen >= 0 && screen < awt_numScreens);
977982
ensureConfigsInited(env, screen);
978-
int configs = x11Screens[screen].numConfigs;
979-
AWT_UNLOCK();
980-
return configs;
983+
return x11Screens[screen].numConfigs;
984+
981985
}
982986

983987
/*
@@ -989,12 +993,11 @@ JNIEXPORT jint JNICALL
989993
Java_sun_awt_X11GraphicsDevice_getConfigVisualId(
990994
JNIEnv *env, jobject this, jint index, jint screen)
991995
{
992-
int visNum;
993-
AWT_LOCK();
996+
// NB: should be invoked only while holding the AWT lock
997+
DASSERT(screen >= 0 && screen < awt_numScreens);
994998
ensureConfigsInited(env, screen);
995999
jint id = (jint) (index == 0 ? x11Screens[screen].defaultConfig
9961000
: x11Screens[screen].configs[index])->awt_visInfo.visualid;
997-
AWT_UNLOCK();
9981001
return id;
9991002
}
10001003

@@ -1007,12 +1010,11 @@ JNIEXPORT jint JNICALL
10071010
Java_sun_awt_X11GraphicsDevice_getConfigDepth(
10081011
JNIEnv *env, jobject this, jint index, jint screen)
10091012
{
1010-
int visNum;
1011-
AWT_LOCK();
1013+
// NB: should be invoked only while holding the AWT lock
1014+
DASSERT(screen >= 0 && screen < awt_numScreens);
10121015
ensureConfigsInited(env, screen);
10131016
jint depth = (jint) (index == 0 ? x11Screens[screen].defaultConfig
10141017
: x11Screens[screen].configs[index])->awt_visInfo.depth;
1015-
AWT_UNLOCK();
10161018
return depth;
10171019
}
10181020

@@ -1025,12 +1027,11 @@ JNIEXPORT jint JNICALL
10251027
Java_sun_awt_X11GraphicsDevice_getConfigColormap(
10261028
JNIEnv *env, jobject this, jint index, jint screen)
10271029
{
1028-
int visNum;
1029-
AWT_LOCK();
1030+
// NB: should be invoked only while holding the AWT lock
1031+
DASSERT(screen >= 0 && screen < awt_numScreens);
10301032
ensureConfigsInited(env, screen);
10311033
jint colormap = (jint) (index == 0 ? x11Screens[screen].defaultConfig
10321034
: x11Screens[screen].configs[index])->awt_cmap;
1033-
AWT_UNLOCK();
10341035
return colormap;
10351036
}
10361037

@@ -1140,8 +1141,10 @@ JNIEXPORT void JNICALL
11401141
Java_sun_awt_X11GraphicsConfig_init(
11411142
JNIEnv *env, jobject this, jint visualNum, jint screen)
11421143
{
1144+
// NB: should be invoked only while holding the AWT lock
1145+
DASSERT(screen >= 0 && screen < awt_numScreens);
1146+
11431147
AwtGraphicsConfigData *adata = NULL;
1144-
AWT_LOCK();
11451148
AwtScreenData asd = x11Screens[screen];
11461149
int i, n;
11471150
int depth;
@@ -1163,7 +1166,6 @@ JNIEnv *env, jobject this, jint visualNum, jint screen)
11631166

11641167
/* If didn't find the visual, throw an exception... */
11651168
if (adata == (AwtGraphicsConfigData *) NULL) {
1166-
AWT_UNLOCK();
11671169
JNU_ThrowIllegalArgumentException(env, "Unknown Visual Specified");
11681170
return;
11691171
}
@@ -1182,7 +1184,6 @@ JNIEnv *env, jobject this, jint visualNum, jint screen)
11821184
(*env)->SetIntField(env, this, x11GraphicsConfigIDs.bitsPerPixel,
11831185
(jint)tempImage->bits_per_pixel);
11841186
XDestroyImage(tempImage);
1185-
AWT_UNLOCK();
11861187
}
11871188

11881189
/*

1 commit comments

Comments
 (1)

openjdk-notifier[bot] commented on Mar 14, 2025

@openjdk-notifier[bot]
Please sign in to comment.