Skip to content

Commit ffca23a

Browse files
committedMay 3, 2022
8284490: Remove finalizer method in java.security.jgss
Reviewed-by: rriggs, dfuchs, weijun
1 parent 0f62cb6 commit ffca23a

File tree

11 files changed

+418
-37
lines changed

11 files changed

+418
-37
lines changed
 

‎src/java.security.jgss/share/classes/sun/security/jgss/wrapper/GSSCredElement.java

+16-12
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, 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
@@ -25,6 +25,7 @@
2525
package sun.security.jgss.wrapper;
2626

2727
import org.ietf.jgss.*;
28+
import java.lang.ref.Cleaner;
2829
import java.security.Provider;
2930
import sun.security.jgss.GSSUtil;
3031
import sun.security.jgss.spi.GSSCredentialSpi;
@@ -37,11 +38,12 @@
3738
* @since 1.6
3839
*/
3940
public class GSSCredElement implements GSSCredentialSpi {
41+
private final Cleaner.Cleanable cleanable;
4042

4143
private int usage;
42-
long pCred; // Pointer to the gss_cred_id_t structure
44+
final long pCred; // Pointer to the gss_cred_id_t structure
4345
private GSSNameElement name = null;
44-
private GSSLibStub cStub;
46+
private final GSSLibStub cStub;
4547

4648
// Perform the necessary ServicePermission check on this cred
4749
@SuppressWarnings("removal")
@@ -69,6 +71,7 @@ void doServicePermCheck() throws GSSException {
6971
cStub = GSSLibStub.getInstance(mech);
7072
usage = GSSCredential.INITIATE_ONLY;
7173
name = srcName;
74+
cleanable = Krb5Util.cleaner.register(this, disposerFor(cStub, pCred));
7275
}
7376

7477
GSSCredElement(GSSNameElement name, int lifetime, int usage,
@@ -85,17 +88,23 @@ void doServicePermCheck() throws GSSException {
8588
this.name = new GSSNameElement(cStub.getCredName(pCred), cStub);
8689
doServicePermCheck();
8790
}
91+
92+
cleanable = Krb5Util.cleaner.register(this, disposerFor(cStub, pCred));
8893
}
8994

9095
public Provider getProvider() {
9196
return SunNativeProvider.INSTANCE;
9297
}
9398

94-
public void dispose() throws GSSException {
99+
public void dispose() {
95100
name = null;
96-
if (pCred != 0) {
97-
pCred = cStub.releaseCred(pCred);
98-
}
101+
cleanable.clean();
102+
}
103+
104+
private static Runnable disposerFor(GSSLibStub stub, long pCredentials) {
105+
return () -> {
106+
stub.releaseCred(pCredentials);
107+
};
99108
}
100109

101110
public GSSNameElement getName() throws GSSException {
@@ -132,11 +141,6 @@ public String toString() {
132141
return "N/A";
133142
}
134143

135-
@SuppressWarnings("removal")
136-
protected void finalize() throws Throwable {
137-
dispose();
138-
}
139-
140144
@Override
141145
public GSSCredentialSpi impersonate(GSSNameSpi name) throws GSSException {
142146
throw new GSSException(GSSException.FAILURE, -1,

‎src/java.security.jgss/share/classes/sun/security/jgss/wrapper/GSSNameElement.java

+18-9
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, 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
@@ -26,6 +26,7 @@
2626
package sun.security.jgss.wrapper;
2727

2828
import org.ietf.jgss.*;
29+
import java.lang.ref.Cleaner;
2930
import java.security.Provider;
3031
import java.security.Security;
3132
import java.io.IOException;
@@ -48,11 +49,12 @@
4849
*/
4950

5051
public class GSSNameElement implements GSSNameSpi {
52+
private final Cleaner.Cleanable cleanable;
5153

52-
long pName = 0; // Pointer to the gss_name_t structure
54+
final long pName; // Pointer to the gss_name_t structure
5355
private String printableName;
5456
private Oid printableType;
55-
private GSSLibStub cStub;
57+
final private GSSLibStub cStub;
5658

5759
static final GSSNameElement DEF_ACCEPTOR = new GSSNameElement();
5860

@@ -94,6 +96,9 @@ private static Oid getNativeNameType(Oid nameType, GSSLibStub stub) {
9496

9597
private GSSNameElement() {
9698
printableName = "<DEFAULT ACCEPTOR>";
99+
pName = 0;
100+
cleanable = null;
101+
cStub = null;
97102
}
98103

99104
// Warning: called by NativeUtil.c
@@ -106,6 +111,8 @@ private GSSNameElement() {
106111
pName = pNativeName;
107112
cStub = stub;
108113
setPrintables();
114+
115+
cleanable = Krb5Util.cleaner.register(this, disposerFor(stub, pName));
109116
}
110117

111118
GSSNameElement(byte[] nameBytes, Oid nameType, GSSLibStub stub)
@@ -151,6 +158,8 @@ private GSSNameElement() {
151158
}
152159
}
153160
pName = cStub.importName(name, nameType);
161+
cleanable = Krb5Util.cleaner.register(this, disposerFor(stub, pName));
162+
154163
setPrintables();
155164

156165
@SuppressWarnings("removal")
@@ -284,14 +293,14 @@ public boolean isAnonymousName() {
284293
}
285294

286295
public void dispose() {
287-
if (pName != 0) {
288-
cStub.releaseName(pName);
289-
pName = 0;
296+
if (cleanable != null) {
297+
cleanable.clean();
290298
}
291299
}
292300

293-
@SuppressWarnings("removal")
294-
protected void finalize() throws Throwable {
295-
dispose();
301+
private static Runnable disposerFor(GSSLibStub stub, long pName) {
302+
return () -> {
303+
stub.releaseName(pName);
304+
};
296305
}
297306
}

‎src/java.security.jgss/share/classes/sun/security/jgss/wrapper/Krb5Util.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, 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
@@ -25,6 +25,7 @@
2525
package sun.security.jgss.wrapper;
2626

2727
import org.ietf.jgss.*;
28+
import java.lang.ref.Cleaner;
2829
import javax.security.auth.kerberos.ServicePermission;
2930

3031
/**
@@ -33,6 +34,8 @@
3334
* @since 1.6
3435
*/
3536
class Krb5Util {
37+
// A cleaner, shared within this module.
38+
static final Cleaner cleaner = Cleaner.create();
3639

3740
// Return the Kerberos TGS principal name using the domain
3841
// of the specified <code>name</code>

‎src/java.security.jgss/share/classes/sun/security/jgss/wrapper/NativeGSSContext.java

+30-9
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@
2626
package sun.security.jgss.wrapper;
2727

2828
import org.ietf.jgss.*;
29+
import java.lang.ref.Cleaner;
2930
import java.security.Provider;
3031
import sun.security.jgss.GSSHeader;
3132
import sun.security.jgss.GSSUtil;
@@ -46,6 +47,7 @@
4647
* @since 1.6
4748
*/
4849
class NativeGSSContext implements GSSContextSpi {
50+
private Cleaner.Cleanable cleanable;
4951

5052
private static final int GSS_C_DELEG_FLAG = 1;
5153
private static final int GSS_C_MUTUAL_FLAG = 2;
@@ -238,8 +240,8 @@ private byte[] retrieveToken(InputStream is, int mechTokenLen)
238240
// Warning: called by NativeUtil.c
239241
NativeGSSContext(long pCtxt, GSSLibStub stub) throws GSSException {
240242
assert(pCtxt != 0);
241-
pContext = pCtxt;
242243
cStub = stub;
244+
setContext(pCtxt);
243245

244246
// Set everything except cred, cb, delegatedCred
245247
long[] info = cStub.inquireContext(pContext);
@@ -359,7 +361,7 @@ public boolean isEstablished() {
359361
return isEstablished;
360362
}
361363

362-
public void dispose() throws GSSException {
364+
public void dispose() {
363365
if (disposeCred != null) {
364366
disposeCred.dispose();
365367
}
@@ -370,12 +372,36 @@ public void dispose() throws GSSException {
370372
srcName = null;
371373
targetName = null;
372374
delegatedCred = null;
373-
if (pContext != 0) {
374-
pContext = cStub.deleteContext(pContext);
375+
376+
if (pContext != 0 && cleanable != null) {
375377
pContext = 0;
378+
cleanable.clean();
379+
}
380+
}
381+
382+
// Note: this method is also used in native code.
383+
private void setContext(long pContext) {
384+
// Dispose the existing context.
385+
if (this.pContext != 0L && cleanable != null) {
386+
cleanable.clean();
387+
}
388+
389+
// Reset the context
390+
this.pContext = pContext;
391+
392+
// Register the cleaner.
393+
if (pContext != 0L) {
394+
cleanable = Krb5Util.cleaner.register(this,
395+
disposerFor(cStub, pContext));
376396
}
377397
}
378398

399+
private static Runnable disposerFor(GSSLibStub stub, long pContext) {
400+
return () -> {
401+
stub.deleteContext(pContext);
402+
};
403+
}
404+
379405
public int getWrapSizeLimit(int qop, boolean confReq,
380406
int maxTokenSize)
381407
throws GSSException {
@@ -639,11 +665,6 @@ public boolean isInitiator() {
639665
return isInitiator;
640666
}
641667

642-
@SuppressWarnings("removal")
643-
protected void finalize() throws Throwable {
644-
dispose();
645-
}
646-
647668
public Object inquireSecContext(String type)
648669
throws GSSException {
649670
throw new GSSException(GSSException.UNAVAILABLE, -1,

‎src/java.security.jgss/share/native/libj2gss/GSSLibStub.c

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, 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
@@ -937,7 +937,7 @@ Java_sun_security_jgss_wrapper_GSSLibStub_initContext(JNIEnv *env,
937937
// this is to work with both MIT and Solaris. Former deletes half-built
938938
// context if error occurs
939939
if (contextHdl != contextHdlSave) {
940-
(*env)->SetLongField(env, jcontextSpi, FID_NativeGSSContext_pContext,
940+
(*env)->CallVoidMethod(env, jcontextSpi, MID_NativeGSSContext_setContext,
941941
ptr_to_jlong(contextHdl));
942942
TRACE1("[GSSLibStub_initContext] set pContext=%" PRIuPTR "", (uintptr_t)contextHdl);
943943
}
@@ -1057,7 +1057,7 @@ Java_sun_security_jgss_wrapper_GSSLibStub_acceptContext(JNIEnv *env,
10571057
// this is to work with both MIT and Solaris. Former deletes half-built
10581058
// context if error occurs
10591059
if (contextHdl != contextHdlSave) {
1060-
(*env)->SetLongField(env, jcontextSpi, FID_NativeGSSContext_pContext,
1060+
(*env)->CallVoidMethod(env, jcontextSpi, MID_NativeGSSContext_setContext,
10611061
ptr_to_jlong(contextHdl));
10621062
TRACE1("[GSSLibStub_acceptContext] set pContext=%" PRIuPTR "", (uintptr_t)contextHdl);
10631063
}

‎src/java.security.jgss/share/native/libj2gss/NativeUtil.c

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, 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
@@ -81,6 +81,7 @@ jmethodID MID_InetAddress_getAddr;
8181
jmethodID MID_GSSNameElement_ctor;
8282
jmethodID MID_GSSCredElement_ctor;
8383
jmethodID MID_NativeGSSContext_ctor;
84+
jmethodID MID_NativeGSSContext_setContext;
8485
jfieldID FID_GSSLibStub_pMech;
8586
jfieldID FID_NativeGSSContext_pContext;
8687
jfieldID FID_NativeGSSContext_srcName;
@@ -290,6 +291,15 @@ DEF_JNI_OnLoad(JavaVM *jvm, void *reserved) {
290291
printf("Couldn't find NativeGSSContext(long, GSSLibStub) constructor\n");
291292
return JNI_ERR;
292293
}
294+
295+
MID_NativeGSSContext_setContext =
296+
(*env)->GetMethodID(env, CLS_NativeGSSContext, "setContext",
297+
"(J)V");
298+
if (MID_NativeGSSContext_setContext == NULL) {
299+
printf("Couldn't find NativeGSSContext.setContext(long) method\n");
300+
return JNI_ERR;
301+
}
302+
293303
/* Compute and cache the field ID */
294304
cls = (*env)->FindClass(env, "sun/security/jgss/wrapper/GSSLibStub");
295305
if (cls == NULL) {

‎src/java.security.jgss/share/native/libj2gss/NativeUtil.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, 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
@@ -73,6 +73,7 @@ extern "C" {
7373
extern jmethodID MID_GSSNameElement_ctor;
7474
extern jmethodID MID_GSSCredElement_ctor;
7575
extern jmethodID MID_NativeGSSContext_ctor;
76+
extern jmethodID MID_NativeGSSContext_setContext;
7677
extern jfieldID FID_GSSLibStub_pMech;
7778
extern jfieldID FID_NativeGSSContext_pContext;
7879
extern jfieldID FID_NativeGSSContext_srcName;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
/*
2+
* Copyright (C) 2022 THL A29 Limited, a Tencent company. 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+
/*
25+
* @test
26+
* @bug 8284490
27+
* @summary Remove finalizer method in java.security.jgss
28+
* @key intermittent
29+
* @run main/othervm GssContextCleanup
30+
*/
31+
32+
import org.ietf.jgss.GSSContext;
33+
import org.ietf.jgss.GSSCredential;
34+
import org.ietf.jgss.GSSManager;
35+
36+
import java.util.WeakHashMap;
37+
38+
public final class GssContextCleanup {
39+
private final static WeakHashMap<GSSContext, ?> whm = new WeakHashMap<>();
40+
41+
public static void main(String[] args) throws Exception {
42+
// Enable debug log so that the failure analysis could be easier.
43+
System.setProperty("sun.security.nativegss.debug", "true");
44+
45+
// Use native provider
46+
System.setProperty("sun.security.jgss.native", "true");
47+
48+
// Create an object
49+
GSSManager manager = GSSManager.getInstance();
50+
GSSContext context = manager.createContext((GSSCredential)null);
51+
whm.put(context, null);
52+
context = null;
53+
54+
// Wait to trigger the cleanup.
55+
for (int i = 0; i < 10 && whm.size() > 0; i++) {
56+
System.gc();
57+
Thread.sleep(100);
58+
}
59+
60+
// Check if the object has been collected.
61+
if (whm.size() > 0) {
62+
throw new RuntimeException("GSSContext object is not released");
63+
}
64+
}
65+
}
66+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
/*
2+
* Copyright (C) 2022 THL A29 Limited, a Tencent company. 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+
/*
25+
* @test
26+
* @bug 8284490
27+
* @summary Remove finalizer method in java.security.jgss
28+
* @key intermittent
29+
* @run main/othervm GssNameCleanup
30+
*/
31+
32+
import java.util.WeakHashMap;
33+
import org.ietf.jgss.GSSManager;
34+
import org.ietf.jgss.GSSName;
35+
import org.ietf.jgss.GSSException;
36+
37+
public final class GssNameCleanup {
38+
private final static WeakHashMap<GSSName, ?> whm = new WeakHashMap<>();
39+
40+
public static void main(String[] args) throws Exception {
41+
// Enable debug log so that the failure analysis could be easier.
42+
System.setProperty("sun.security.nativegss.debug", "true");
43+
44+
// Use native provider
45+
System.setProperty("sun.security.jgss.native", "true");
46+
47+
// Create an object
48+
GSSManager manager = GSSManager.getInstance();
49+
try {
50+
GSSName name =
51+
manager.createName("u1", GSSName.NT_USER_NAME);
52+
whm.put(name, null);
53+
name = null;
54+
} catch (GSSException gsse) {
55+
// createName() could fail if the local default realm
56+
// cannot be located. Just ignore the test case for
57+
// such circumstances.
58+
System.out.println("Ignore this test case: " + gsse);
59+
}
60+
61+
// Wait to trigger the cleanup.
62+
for (int i = 0; i < 10 && whm.size() > 0; i++) {
63+
System.gc();
64+
Thread.sleep(100);
65+
}
66+
67+
// Check if the object has been collected.
68+
if (whm.size() > 0) {
69+
throw new RuntimeException("GSSName object is not released");
70+
}
71+
}
72+
}
73+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,180 @@
1+
/*
2+
* Copyright (c) 2022, 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+
/*
25+
* @test
26+
* @bug 8284490
27+
* @summary Remove finalizer method in java.security.jgss
28+
* @key intermittent
29+
* @requires os.family != "windows"
30+
* @library /test/lib
31+
* @compile -XDignore.symbol.file Cleaners.java
32+
* @run main/othervm Cleaners launcher
33+
*/
34+
35+
import java.nio.charset.StandardCharsets;
36+
import java.nio.file.Files;
37+
import java.nio.file.Paths;
38+
import java.nio.file.attribute.PosixFilePermission;
39+
import java.util.Arrays;
40+
import java.util.Set;
41+
42+
import jdk.test.lib.Asserts;
43+
import jdk.test.lib.process.Proc;
44+
import org.ietf.jgss.Oid;
45+
import sun.security.krb5.Config;
46+
47+
public class Cleaners {
48+
49+
private static final String CONF = "krb5.conf";
50+
private static final String KTAB_S = "server.ktab";
51+
private static final String KTAB_B = "backend.ktab";
52+
53+
private static final String HOST = "localhost";
54+
private static final String SERVER = "server/" + HOST;
55+
private static final String BACKEND = "backend/" + HOST;
56+
private static final String USER = "user";
57+
private static final char[] PASS = "password".toCharArray();
58+
private static final String REALM = "REALM";
59+
60+
private static final byte[] MSG = "12345678".repeat(128)
61+
.getBytes(StandardCharsets.UTF_8);
62+
63+
public static void main(String[] args) throws Exception {
64+
65+
Oid oid = new Oid("1.2.840.113554.1.2.2");
66+
byte[] token, msg;
67+
68+
switch (args[0]) {
69+
case "launcher" -> {
70+
KDC kdc = KDC.create(REALM, HOST, 0, true);
71+
kdc.addPrincipal(USER, PASS);
72+
kdc.addPrincipalRandKey("krbtgt/" + REALM);
73+
kdc.addPrincipalRandKey(SERVER);
74+
kdc.addPrincipalRandKey(BACKEND);
75+
76+
// Native lib might do some name lookup
77+
KDC.saveConfig(CONF, kdc,
78+
"dns_lookup_kdc = no",
79+
"ticket_lifetime = 1h",
80+
"dns_lookup_realm = no",
81+
"dns_canonicalize_hostname = false",
82+
"forwardable = true");
83+
System.setProperty("java.security.krb5.conf", CONF);
84+
Config.refresh();
85+
86+
// Create kaytab and ccache files for native clients
87+
kdc.writeKtab(KTAB_S, false, SERVER);
88+
kdc.writeKtab(KTAB_B, false, BACKEND);
89+
kdc.kinit(USER, "ccache");
90+
Files.setPosixFilePermissions(Paths.get("ccache"),
91+
Set.of(PosixFilePermission.OWNER_READ,
92+
PosixFilePermission.OWNER_WRITE));
93+
94+
Proc pc = proc("client")
95+
.env("KRB5CCNAME", "FILE:ccache")
96+
.env("KRB5_KTNAME", "none") // Do not try system ktab if ccache fails
97+
.start();
98+
Proc ps = proc("server")
99+
.env("KRB5_KTNAME", KTAB_S)
100+
.start();
101+
Proc pb = proc("backend")
102+
.env("KRB5_KTNAME", KTAB_B)
103+
.start();
104+
105+
// Client and server
106+
ps.println(pc.readData()); // AP-REQ
107+
pc.println(ps.readData()); // AP-REP, mutual auth
108+
ps.println(pc.readData()); // wrap msg
109+
ps.println(pc.readData()); // mic msg
110+
111+
// Server and backend
112+
pb.println(ps.readData()); // AP-REQ
113+
ps.println(pb.readData()); // wrap msg
114+
ps.println(pb.readData()); // mic msg
115+
116+
ensureCleanersCalled(pc);
117+
ensureCleanersCalled(ps);
118+
ensureCleanersCalled(pb);
119+
}
120+
case "client" -> {
121+
Context c = Context.fromThinAir();
122+
c.startAsClient(SERVER, oid);
123+
c.x().requestCredDeleg(true);
124+
c.x().requestMutualAuth(true);
125+
Proc.binOut(c.take(new byte[0])); // AP-REQ
126+
c.take(Proc.binIn()); // AP-REP
127+
Proc.binOut(c.wrap(MSG, true));
128+
Proc.binOut(c.getMic(MSG));
129+
}
130+
case "server" -> {
131+
Context s = Context.fromThinAir();
132+
s.startAsServer(oid);
133+
token = Proc.binIn(); // AP-REQ
134+
Proc.binOut(s.take(token)); // AP-REP
135+
msg = s.unwrap(Proc.binIn(), true);
136+
Asserts.assertTrue(Arrays.equals(msg, MSG));
137+
s.verifyMic(Proc.binIn(), msg);
138+
Context s2 = s.delegated();
139+
s2.startAsClient(BACKEND, oid);
140+
s2.x().requestMutualAuth(false);
141+
Proc.binOut(s2.take(new byte[0])); // AP-REQ
142+
msg = s2.unwrap(Proc.binIn(), true);
143+
Asserts.assertTrue(Arrays.equals(msg, MSG));
144+
s2.verifyMic(Proc.binIn(), msg);
145+
}
146+
case "backend" -> {
147+
Context b = Context.fromThinAir();
148+
b.startAsServer(oid);
149+
token = b.take(Proc.binIn()); // AP-REQ
150+
Asserts.assertTrue(token == null);
151+
Proc.binOut(b.wrap(MSG, true));
152+
Proc.binOut(b.getMic(MSG));
153+
}
154+
}
155+
System.out.println("Prepare for GC");
156+
for (int i = 0; i < 10; i++) {
157+
System.gc();
158+
Thread.sleep(100);
159+
}
160+
}
161+
162+
private static void ensureCleanersCalled(Proc p) throws Exception {
163+
p.output()
164+
.shouldHaveExitValue(0)
165+
.stdoutShouldMatch("Prepare for GC(.|\\n)*GSSLibStub_deleteContext")
166+
.stdoutShouldMatch("Prepare for GC(.|\\n)*GSSLibStub_releaseName")
167+
.stdoutShouldMatch("Prepare for GC(.|\\n)*GSSLibStub_releaseCred");
168+
}
169+
170+
private static Proc proc(String type) throws Exception {
171+
return Proc.create("Cleaners")
172+
.args(type)
173+
.debug(type)
174+
.env("KRB5_CONFIG", CONF)
175+
.env("KRB5_TRACE", "/dev/stderr")
176+
.prop("sun.security.jgss.native", "true")
177+
.prop("javax.security.auth.useSubjectCredsOnly", "false")
178+
.prop("sun.security.nativegss.debug", "true");
179+
}
180+
}

‎test/lib/jdk/test/lib/process/Proc.java

+15-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2013, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2013, 2022, 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
@@ -123,6 +123,7 @@ public class Proc {
123123
private String debug; // debug flag, controller will show data
124124
// transfer between procs. If debug is set,
125125
// it MUST be different between Procs.
126+
private final StringBuilder stdout = new StringBuilder();
126127

127128
final private static String PREFIX = "PROCISFUN:";
128129

@@ -358,6 +359,9 @@ String getId(String suffix) {
358359
// Reads a line from stdout of proc
359360
public String readLine() throws IOException {
360361
String s = br.readLine();
362+
if (s != null) {
363+
stdout.append(s).append('\n');
364+
}
361365
if (debug != null) {
362366
System.out.println("PROC: " + debug + " readline: " +
363367
(s == null ? "<EOF>" : s));
@@ -402,6 +406,16 @@ public int waitFor() throws Exception {
402406
}
403407
return p.waitFor();
404408
}
409+
410+
// Returns an OutputAnalyzer
411+
public OutputAnalyzer output() throws Exception {
412+
int exitCode = waitFor();
413+
Path stderr = Path.of(getId("stderr"));
414+
return new OutputAnalyzer(stdout.toString(),
415+
Files.exists(stderr) ? Files.readString(stderr) : "",
416+
exitCode);
417+
}
418+
405419
// Wait for process end with expected exit code
406420
public void waitFor(int expected) throws Exception {
407421
if (p.waitFor() != expected) {

0 commit comments

Comments
 (0)
Please sign in to comment.