Skip to content

Commit 93b6ab5

Browse files
committedDec 2, 2020
8256818: SSLSocket that is never bound or connected leaks socket resources
Reviewed-by: xuelei
1 parent 692b273 commit 93b6ab5

File tree

5 files changed

+114
-32
lines changed

5 files changed

+114
-32
lines changed
 

‎src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java

+23-14
Original file line numberDiff line numberDiff line change
@@ -553,7 +553,7 @@ public boolean isClosed() {
553553
// locks may be deadlocked.
554554
@Override
555555
public void close() throws IOException {
556-
if (tlsIsClosed) {
556+
if (isClosed()) {
557557
return;
558558
}
559559

@@ -562,27 +562,36 @@ public void close() throws IOException {
562562
}
563563

564564
try {
565-
// shutdown output bound, which may have been closed previously.
566-
if (!isOutputShutdown()) {
567-
duplexCloseOutput();
568-
}
569-
570-
// shutdown input bound, which may have been closed previously.
571-
if (!isInputShutdown()) {
572-
duplexCloseInput();
573-
}
565+
if (isConnected()) {
566+
// shutdown output bound, which may have been closed previously.
567+
if (!isOutputShutdown()) {
568+
duplexCloseOutput();
569+
}
574570

575-
if (!isClosed()) {
576-
// close the connection directly
577-
closeSocket(false);
571+
// shutdown input bound, which may have been closed previously.
572+
if (!isInputShutdown()) {
573+
duplexCloseInput();
574+
}
578575
}
579576
} catch (IOException ioe) {
580577
// ignore the exception
581578
if (SSLLogger.isOn && SSLLogger.isOn("ssl")) {
582579
SSLLogger.warning("SSLSocket duplex close failed", ioe);
583580
}
584581
} finally {
585-
tlsIsClosed = true;
582+
if (!isClosed()) {
583+
// close the connection directly
584+
try {
585+
closeSocket(false);
586+
} catch (IOException ioe) {
587+
// ignore the exception
588+
if (SSLLogger.isOn && SSLLogger.isOn("ssl")) {
589+
SSLLogger.warning("SSLSocket close failed", ioe);
590+
}
591+
} finally {
592+
tlsIsClosed = true;
593+
}
594+
}
586595
}
587596
}
588597

‎test/jdk/java/lang/ProcessBuilder/checkHandles/CheckHandles.java

+7-9
Original file line numberDiff line numberDiff line change
@@ -27,32 +27,30 @@
2727
import java.io.InputStreamReader;
2828
import java.lang.ProcessHandle;
2929

30+
import jdk.test.lib.util.FileUtils;
31+
3032
/*
3133
* @test
3234
* @bug 8239893
3335
* @summary Verify that handles for processes that terminate do not accumulate
3436
* @requires ((os.family == "windows") & (vm.compMode != "Xcomp"))
37+
* @library /test/lib
3538
* @run main/othervm/native -Xint CheckHandles
3639
*/
3740
public class CheckHandles {
3841

39-
// Return the current process handle count
40-
private static native long getProcessHandleCount();
41-
4242
public static void main(String[] args) throws Exception {
43-
System.loadLibrary("CheckHandles");
44-
4543
System.out.println("mypid: " + ProcessHandle.current().pid());
4644

4745
// Warmup the process launch mechanism and vm to stabilize the number of handles in use
4846
int MAX_WARMUP = 20;
49-
long prevCount = getProcessHandleCount();
47+
long prevCount = FileUtils.getProcessHandleCount();
5048
for (int i = 0; i < MAX_WARMUP; i++) {
5149
oneProcess();
5250
System.gc(); // an opportunity to close unreferenced handles
5351
sleep(10);
5452

55-
long count = getProcessHandleCount();
53+
long count = FileUtils.getProcessHandleCount();
5654
if (count < 0)
5755
throw new AssertionError("getProcessHandleCount failed");
5856
System.out.println("warmup handle delta: " + (count - prevCount));
@@ -61,7 +59,7 @@ public static void main(String[] args) throws Exception {
6159
System.out.println("Warmup done");
6260
System.out.println();
6361

64-
prevCount = getProcessHandleCount();
62+
prevCount = FileUtils.getProcessHandleCount();
6563
long startHandles = prevCount;
6664
long maxHandles = startHandles;
6765
int MAX_SPAWN = 50;
@@ -70,7 +68,7 @@ public static void main(String[] args) throws Exception {
7068
System.gc(); // an opportunity to close unreferenced handles
7169
sleep(10);
7270

73-
long count = getProcessHandleCount();
71+
long count = FileUtils.getProcessHandleCount();
7472
if (count < 0)
7573
throw new AssertionError("getProcessHandleCount failed");
7674
System.out.println("handle delta: " + (count - prevCount));
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
/*
2+
* Copyright (c) 2020 SAP SE. 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.io.IOException;
25+
26+
import javax.net.SocketFactory;
27+
import javax.net.ssl.SSLSocketFactory;
28+
29+
import jdk.test.lib.util.FileUtils;
30+
31+
/*
32+
* @test
33+
* @bug 8256818
34+
* @summary Test that creating and closing SSL Sockets without bind/connect
35+
* will not leave leaking socket file descriptors
36+
* @library /test/lib
37+
* @run main/othervm SSLSocketLeak
38+
*/
39+
public class SSLSocketLeak {
40+
41+
private static final int NUM_TEST_SOCK = 500;
42+
43+
public static void main(String[] args) throws IOException {
44+
long fds_start = FileUtils.getProcessHandleCount();
45+
System.out.println("FDs at the beginning: " + fds_start);
46+
47+
SocketFactory f = SSLSocketFactory.getDefault();
48+
for (int i = 0; i < NUM_TEST_SOCK; i++) {
49+
f.createSocket().close();
50+
}
51+
52+
long fds_end = FileUtils.getProcessHandleCount();
53+
System.out.println("FDs in the end: " + fds_end);
54+
55+
if ((fds_end - fds_start) > (NUM_TEST_SOCK / 10)) {
56+
throw new RuntimeException("Too many open file descriptors. Looks leaky.");
57+
}
58+
}
59+
}

‎test/lib/jdk/test/lib/util/FileUtils.java

+24-8
Original file line numberDiff line numberDiff line change
@@ -23,34 +23,34 @@
2323

2424
package jdk.test.lib.util;
2525

26-
import jdk.test.lib.Platform;
27-
2826
import java.io.BufferedReader;
29-
import java.io.InputStreamReader;
3027
import java.io.IOException;
28+
import java.io.InputStreamReader;
3129
import java.io.PrintStream;
3230
import java.io.UncheckedIOException;
3331
import java.lang.ProcessBuilder.Redirect;
32+
import java.lang.management.ManagementFactory;
3433
import java.nio.file.DirectoryNotEmptyException;
3534
import java.nio.file.FileVisitResult;
3635
import java.nio.file.Files;
3736
import java.nio.file.NoSuchFileException;
3837
import java.nio.file.Path;
39-
import java.nio.file.Paths;
4038
import java.nio.file.SimpleFileVisitor;
4139
import java.nio.file.attribute.BasicFileAttributes;
4240
import java.time.Instant;
43-
import java.time.Duration;
44-
import java.util.Arrays;
4541
import java.util.ArrayList;
46-
import java.util.ArrayDeque;
42+
import java.util.Arrays;
4743
import java.util.HashSet;
4844
import java.util.List;
4945
import java.util.Optional;
5046
import java.util.Set;
47+
import java.util.concurrent.TimeUnit;
5148
import java.util.concurrent.atomic.AtomicBoolean;
5249
import java.util.concurrent.atomic.AtomicReference;
53-
import java.util.concurrent.TimeUnit;
50+
51+
import jdk.test.lib.Platform;
52+
53+
import com.sun.management.UnixOperatingSystemMXBean;
5454

5555
/**
5656
* Common library for various test file utility functions.
@@ -59,6 +59,7 @@ public final class FileUtils {
5959
private static final boolean IS_WINDOWS = Platform.isWindows();
6060
private static final int RETRY_DELETE_MILLIS = IS_WINDOWS ? 500 : 0;
6161
private static final int MAX_RETRY_DELETE_TIMES = IS_WINDOWS ? 15 : 0;
62+
private static volatile boolean nativeLibLoaded;
6263

6364
/**
6465
* Deletes a file, retrying if necessary.
@@ -363,6 +364,21 @@ public static void listFileDescriptors(PrintStream ps) {
363364
});
364365
}
365366

367+
// Return the current process handle count
368+
public static long getProcessHandleCount() {
369+
if (IS_WINDOWS) {
370+
if (!nativeLibLoaded) {
371+
System.loadLibrary("FileUtils");
372+
nativeLibLoaded = true;
373+
}
374+
return getWinProcessHandleCount();
375+
} else {
376+
return ((UnixOperatingSystemMXBean)ManagementFactory.getOperatingSystemMXBean()).getOpenFileDescriptorCount();
377+
}
378+
}
379+
380+
private static native long getWinProcessHandleCount();
381+
366382
// Possible command locations and arguments
367383
static String[][] lsCommands = new String[][] {
368384
{"/usr/bin/lsof", "-p"},

‎test/jdk/java/lang/ProcessBuilder/checkHandles/libCheckHandles.c ‎test/lib/jdk/test/lib/util/libFileUtils.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@
2929
#include "jni.h"
3030
#include <windows.h>
3131

32-
JNIEXPORT jlong JNICALL Java_CheckHandles_getProcessHandleCount(JNIEnv *env)
32+
JNIEXPORT jlong JNICALL Java_jdk_test_lib_util_FileUtils_getWinProcessHandleCount(JNIEnv *env)
3333
{
3434
DWORD handleCount;
3535
HANDLE handle = GetCurrentProcess();

0 commit comments

Comments
 (0)
Please sign in to comment.