test/jdk/java/net/DatagramSocket/UnreferencedDatagramSockets.java
author rriggs
Fri, 02 Feb 2018 14:17:07 -0500
changeset 48737 7c12219870fd
child 54770 62b6e7587b1f
permissions -rw-r--r--
8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner Reviewed-by: chegar, plevart
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
48737
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     1
/*
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     2
 * Copyright (c) 2017, 2018, Oracle and/or its affiliates. All rights reserved.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     4
 *
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     7
 * published by the Free Software Foundation.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     8
 *
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    13
 * accompanied this code).
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    14
 *
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    18
 *
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    21
 * questions.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    22
 */
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    23
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    24
/**
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    25
 * @test
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    26
 * @modules java.management java.base/java.io:+open java.base/java.net:+open
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    27
 * @run main/othervm UnreferencedDatagramSockets
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    28
 * @run main/othervm -Djava.net.preferIPv4Stack=true UnreferencedDatagramSockets
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    29
 * @summary Check that unreferenced datagram sockets are closed
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    30
 */
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    31
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    32
import java.io.FileDescriptor;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    33
import java.lang.management.ManagementFactory;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    34
import java.lang.management.OperatingSystemMXBean;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    35
import java.lang.ref.ReferenceQueue;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    36
import java.lang.ref.WeakReference;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    37
import java.lang.reflect.Field;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    38
import java.io.IOException;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    39
import java.net.DatagramPacket;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    40
import java.net.DatagramSocket;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    41
import java.net.DatagramSocketImpl;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    42
import java.net.InetAddress;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    43
import java.net.UnknownHostException;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    44
import java.nio.file.Files;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    45
import java.nio.file.Path;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    46
import java.nio.file.Paths;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    47
import java.util.ArrayDeque;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    48
import java.util.List;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    49
import java.util.Optional;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    50
import java.util.concurrent.TimeUnit;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    51
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    52
import com.sun.management.UnixOperatingSystemMXBean;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    53
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    54
public class UnreferencedDatagramSockets {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    55
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    56
    /**
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    57
     * The set of sockets we have to check up on.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    58
     */
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    59
    final static ArrayDeque<NamedWeak> pendingSockets = new ArrayDeque<>(5);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    60
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    61
    /**
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    62
     * Queued objects when they are unreferenced.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    63
     */
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    64
    final static ReferenceQueue<Object> pendingQueue = new ReferenceQueue<>();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    65
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    66
    // Server to echo a datagram packet
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    67
    static class Server implements Runnable {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    68
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    69
        DatagramSocket ss;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    70
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    71
        Server() throws IOException {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    72
            ss = new DatagramSocket(0);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    73
            System.out.printf("  DatagramServer addr: %s: %d%n",
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    74
                    this.getHost(), this.getPort());
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    75
            pendingSockets.add(new NamedWeak(ss, pendingQueue, "serverDatagramSocket"));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    76
            extractRefs(ss, "serverDatagramSocket");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    77
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    78
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    79
        InetAddress getHost() throws UnknownHostException {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    80
            InetAddress localhost = InetAddress.getByName("localhost"); //.getLocalHost();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    81
            return localhost;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    82
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    83
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    84
        int getPort() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    85
            return ss.getLocalPort();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    86
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    87
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    88
        // Receive a byte and send back a byte
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    89
        public void run() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    90
            try {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    91
                byte[] buffer = new byte[50];
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    92
                DatagramPacket p = new DatagramPacket(buffer, buffer.length);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    93
                ss.receive(p);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    94
                buffer[0] += 1;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    95
                ss.send(p);         // send back +1
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    96
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    97
                // do NOT close but 'forget' the datagram socket reference
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    98
                ss = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    99
            } catch (Exception ioe) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   100
                ioe.printStackTrace();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   101
            }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   102
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   103
    }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   104
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   105
    public static void main(String args[]) throws Exception {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   106
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   107
        // Create and close a DatagramSocket to warm up the FD count for side effects.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   108
        try (DatagramSocket s = new DatagramSocket(0)) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   109
            // no-op; close immediately
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   110
            s.getLocalPort();   // no-op
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   111
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   112
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   113
        long fdCount0 = getFdCount();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   114
        listProcFD();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   115
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   116
        // start a server
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   117
        Server svr = new Server();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   118
        Thread thr = new Thread(svr);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   119
        thr.start();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   120
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   121
        DatagramSocket client = new DatagramSocket(0);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   122
        client.connect(svr.getHost(), svr.getPort());
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   123
        pendingSockets.add(new NamedWeak(client, pendingQueue, "clientDatagramSocket"));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   124
        extractRefs(client, "clientDatagramSocket");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   125
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   126
        byte[] msg = new byte[1];
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   127
        msg[0] = 1;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   128
        DatagramPacket p = new DatagramPacket(msg, msg.length, svr.getHost(), svr.getPort());
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   129
        client.send(p);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   130
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   131
        p = new DatagramPacket(msg, msg.length);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   132
        client.receive(p);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   133
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   134
        System.out.printf("echo received from: %s%n", p.getSocketAddress());
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   135
        if (msg[0] != 2) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   136
            throw new AssertionError("incorrect data received: expected: 2, actual: " + msg[0]);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   137
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   138
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   139
        // Do NOT close the DatagramSocket; forget it
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   140
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   141
        Object ref;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   142
        int loops = 20;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   143
        while (!pendingSockets.isEmpty() && loops-- > 0) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   144
            ref = pendingQueue.remove(1000L);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   145
            if (ref != null) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   146
                pendingSockets.remove(ref);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   147
                System.out.printf("  ref freed: %s, remaining: %d%n", ref, pendingSockets.size());
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   148
            } else {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   149
                client = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   150
                p = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   151
                msg = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   152
                System.gc();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   153
            }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   154
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   155
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   156
        thr.join();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   157
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   158
        // List the open file descriptors
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   159
        long fdCount = getFdCount();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   160
        System.out.printf("Initial fdCount: %d, final fdCount: %d%n", fdCount0, fdCount);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   161
        listProcFD();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   162
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   163
        if (loops == 0) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   164
            throw new AssertionError("Not all references reclaimed");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   165
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   166
    }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   167
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   168
    // Get the count of open file descriptors, or -1 if not available
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   169
    private static long getFdCount() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   170
        OperatingSystemMXBean mxBean = ManagementFactory.getOperatingSystemMXBean();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   171
        return (mxBean instanceof UnixOperatingSystemMXBean)
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   172
                ? ((UnixOperatingSystemMXBean) mxBean).getOpenFileDescriptorCount()
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   173
                : -1L;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   174
    }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   175
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   176
    // Reflect to find references in the datagram implementation that will be gc'd
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   177
    private static void extractRefs(DatagramSocket s, String name) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   178
        try {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   179
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   180
            Field socketImplField = DatagramSocket.class.getDeclaredField("impl");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   181
            socketImplField.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   182
            Object socketImpl = socketImplField.get(s);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   183
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   184
            Field fileDescriptorField = DatagramSocketImpl.class.getDeclaredField("fd");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   185
            fileDescriptorField.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   186
            FileDescriptor fileDescriptor = (FileDescriptor) fileDescriptorField.get(socketImpl);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   187
            extractRefs(fileDescriptor, name);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   188
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   189
            Class<?> socketImplClass = socketImpl.getClass();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   190
            System.out.printf("socketImplClass: %s%n", socketImplClass);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   191
            if (socketImplClass.getName().equals("java.net.TwoStacksPlainDatagramSocketImpl")) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   192
                Field fileDescriptor1Field = socketImplClass.getDeclaredField("fd1");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   193
                fileDescriptor1Field.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   194
                FileDescriptor fileDescriptor1 = (FileDescriptor) fileDescriptor1Field.get(socketImpl);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   195
                extractRefs(fileDescriptor1, name + "::twoStacksFd1");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   196
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   197
            } else {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   198
                System.out.printf("socketImpl class name not matched: %s != %s%n",
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   199
                        socketImplClass.getName(), "java.net.TwoStacksPlainDatagramSocketImpl");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   200
            }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   201
        } catch (NoSuchFieldException | IllegalAccessException ex) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   202
            ex.printStackTrace();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   203
            throw new AssertionError("missing field", ex);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   204
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   205
    }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   206
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   207
    private static void extractRefs(FileDescriptor fileDescriptor, String name) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   208
        Object cleanup = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   209
        int rawfd = -1;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   210
        try {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   211
            if (fileDescriptor != null) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   212
                Field fd1Field = FileDescriptor.class.getDeclaredField("fd");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   213
                fd1Field.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   214
                rawfd = fd1Field.getInt(fileDescriptor);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   215
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   216
                Field cleanupfdField = FileDescriptor.class.getDeclaredField("cleanup");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   217
                cleanupfdField.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   218
                cleanup = cleanupfdField.get(fileDescriptor);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   219
                pendingSockets.add(new NamedWeak(fileDescriptor, pendingQueue,
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   220
                        name + "::fileDescriptor: " + rawfd));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   221
                pendingSockets.add(new NamedWeak(cleanup, pendingQueue, name + "::fdCleanup: " + rawfd));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   222
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   223
            }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   224
        } catch (NoSuchFieldException | IllegalAccessException ex) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   225
            ex.printStackTrace();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   226
            throw new AssertionError("missing field", ex);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   227
        } finally {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   228
            System.out.print(String.format("  %s:: fd: %s, fd: %d, cleanup: %s%n",
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   229
                    name, fileDescriptor, rawfd, cleanup));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   230
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   231
    }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   232
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   233
    /**
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   234
     * Method to list the open file descriptors (if supported by the 'lsof' command).
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   235
     */
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   236
    static void listProcFD() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   237
        List<String> lsofDirs = List.of("/usr/bin", "/usr/sbin");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   238
        Optional<Path> lsof = lsofDirs.stream()
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   239
                .map(s -> Paths.get(s, "lsof"))
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   240
                .filter(f -> Files.isExecutable(f))
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   241
                .findFirst();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   242
        lsof.ifPresent(exe -> {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   243
            try {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   244
                System.out.printf("Open File Descriptors:%n");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   245
                long pid = ProcessHandle.current().pid();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   246
                ProcessBuilder pb = new ProcessBuilder(exe.toString(), "-p", Integer.toString((int) pid));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   247
                pb.inheritIO();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   248
                Process p = pb.start();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   249
                p.waitFor(10, TimeUnit.SECONDS);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   250
            } catch (IOException | InterruptedException ie) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   251
                ie.printStackTrace();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   252
            }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   253
        });
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   254
    }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   255
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   256
    // Simple class to identify which refs have been queued
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   257
    static class NamedWeak extends WeakReference<Object> {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   258
        private final String name;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   259
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   260
        NamedWeak(Object o, ReferenceQueue<Object> queue, String name) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   261
            super(o, queue);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   262
            this.name = name;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   263
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   264
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   265
        public String toString() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   266
            return name + "; " + super.toString();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   267
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   268
    }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   269
}