test/jdk/java/net/MulticastSocket/UnreferencedMulticastSockets.java
author rriggs
Fri, 02 Feb 2018 14:17:07 -0500
changeset 48737 7c12219870fd
child 51830 d2c72de3cf83
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) 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 -Djava.net.preferIPv4Stack=true UnreferencedMulticastSockets
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    28
 * @run main/othervm UnreferencedMulticastSockets
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    29
 * @summary Check that unreferenced multicast 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.MulticastSocket;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    44
import java.net.UnknownHostException;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    45
import java.nio.file.Files;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    46
import java.nio.file.Path;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    47
import java.nio.file.Paths;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    48
import java.util.ArrayDeque;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    49
import java.util.List;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    50
import java.util.Optional;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    51
import java.util.concurrent.TimeUnit;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    52
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    53
import com.sun.management.UnixOperatingSystemMXBean;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    54
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    55
public class UnreferencedMulticastSockets {
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
    /**
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    58
     * 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
    59
     */
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    60
    final static ArrayDeque<NamedWeak> pendingSockets = new ArrayDeque<>(5);
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
    /**
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    63
     * Queued objects when they are unreferenced.
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    64
     */
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    65
    final static ReferenceQueue<Object> pendingQueue = new ReferenceQueue<>();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    66
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    67
    // Server to echo a datagram packet
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    68
    static class Server implements Runnable {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    69
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    70
        MulticastSocket ss;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    71
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    72
        Server() throws IOException {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    73
            ss = new MulticastSocket(0);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    74
            System.out.printf("  DatagramServer addr: %s: %d%n",
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    75
                    this.getHost(), this.getPort());
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    76
            pendingSockets.add(new NamedWeak(ss, pendingQueue, "serverMulticastSocket"));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    77
            extractRefs(ss, "serverMulticastSocket");
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    80
        InetAddress getHost() throws UnknownHostException {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    81
            InetAddress localhost = InetAddress.getByName("localhost"); //.getLocalHost();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    82
            return localhost;
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    85
        int getPort() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    86
            return ss.getLocalPort();
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    89
        // Receive a byte and send back a byte
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    90
        public void run() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    91
            try {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    92
                byte[] buffer = new byte[50];
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    93
                DatagramPacket p = new DatagramPacket(buffer, buffer.length);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    94
                ss.receive(p);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    95
                buffer[0] += 1;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    96
                ss.send(p);         // send back +1
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    97
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    98
                // do NOT close but 'forget' the socket reference
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
    99
                ss = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   100
            } catch (Exception ioe) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   101
                ioe.printStackTrace();
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   106
    public static void main(String args[]) throws Exception {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   107
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   108
        // Create and close a MulticastSocket 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
   109
        try (MulticastSocket s = new MulticastSocket(0)) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   110
            // no-op; close immediately
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   111
            s.getLocalPort();   // no-op
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   114
        long fdCount0 = getFdCount();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   115
        listProcFD();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   116
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   117
        // start a server
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   118
        Server svr = new Server();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   119
        Thread thr = new Thread(svr);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   120
        thr.start();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   121
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   122
        MulticastSocket client = new MulticastSocket(0);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   123
        client.connect(svr.getHost(), svr.getPort());
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   124
        pendingSockets.add(new NamedWeak(client, pendingQueue, "clientMulticastSocket"));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   125
        extractRefs(client, "clientMulticastSocket");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   126
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   127
        byte[] msg = new byte[1];
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   128
        msg[0] = 1;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   129
        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
   130
        client.send(p);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   131
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   132
        p = new DatagramPacket(msg, msg.length);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   133
        client.receive(p);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   134
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   135
        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
   136
        if (msg[0] != 2) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   137
            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
   138
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   139
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   140
        // Do NOT close the MulticastSocket; forget it
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   141
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   142
        Object ref;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   143
        int loops = 20;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   144
        while (!pendingSockets.isEmpty() && loops-- > 0) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   145
            ref = pendingQueue.remove(1000L);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   146
            if (ref != null) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   147
                pendingSockets.remove(ref);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   148
                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
   149
            } else {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   150
                client = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   151
                p = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   152
                msg = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   153
                System.gc();
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   157
        thr.join();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   158
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   159
        // List the open file descriptors
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   160
        long fdCount = getFdCount();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   161
        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
   162
        listProcFD();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   163
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   164
        if (loops == 0) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   165
            throw new AssertionError("Not all references reclaimed");
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   169
    // 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
   170
    private static long getFdCount() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   171
        OperatingSystemMXBean mxBean = ManagementFactory.getOperatingSystemMXBean();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   172
        return (mxBean instanceof UnixOperatingSystemMXBean)
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   173
                ? ((UnixOperatingSystemMXBean) mxBean).getOpenFileDescriptorCount()
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   174
                : -1L;
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   177
    // Reflect to find references in the socket implementation that will be gc'd
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   178
    private static void extractRefs(MulticastSocket s, String name) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   179
        try {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   180
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   181
            Field socketImplField = DatagramSocket.class.getDeclaredField("impl");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   182
            socketImplField.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   183
            Object socketImpl = socketImplField.get(s);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   184
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   185
            Field fileDescriptorField = DatagramSocketImpl.class.getDeclaredField("fd");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   186
            fileDescriptorField.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   187
            FileDescriptor fileDescriptor = (FileDescriptor) fileDescriptorField.get(socketImpl);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   188
            extractRefs(fileDescriptor, name);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   189
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   190
            Class<?> socketImplClass = socketImpl.getClass();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   191
            System.out.printf("socketImplClass: %s%n", socketImplClass);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   192
            if (socketImplClass.getName().equals("java.net.TwoStacksPlainDatagramSocketImpl")) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   193
                Field fileDescriptor1Field = socketImplClass.getDeclaredField("fd1");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   194
                fileDescriptor1Field.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   195
                FileDescriptor fileDescriptor1 = (FileDescriptor) fileDescriptor1Field.get(socketImpl);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   196
                extractRefs(fileDescriptor1, name + "::twoStacksFd1");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   197
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   198
            } else {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   199
                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
   200
                        socketImplClass.getName(), "java.net.TwoStacksPlainDatagramSocketImpl");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   201
            }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   202
        } catch (NoSuchFieldException | IllegalAccessException ex) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   203
            ex.printStackTrace();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   204
            throw new AssertionError("missing field", ex);
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   208
    private static void extractRefs(FileDescriptor fileDescriptor, String name) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   209
        Object cleanup = null;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   210
        int rawfd = -1;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   211
        try {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   212
            if (fileDescriptor != null) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   213
                Field fd1Field = FileDescriptor.class.getDeclaredField("fd");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   214
                fd1Field.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   215
                rawfd = fd1Field.getInt(fileDescriptor);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   216
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   217
                Field cleanupfdField = FileDescriptor.class.getDeclaredField("cleanup");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   218
                cleanupfdField.setAccessible(true);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   219
                cleanup = cleanupfdField.get(fileDescriptor);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   220
                pendingSockets.add(new NamedWeak(fileDescriptor, pendingQueue,
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   221
                        name + "::fileDescriptor: " + rawfd));
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   222
                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
   223
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   224
            }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   225
        } catch (NoSuchFieldException | IllegalAccessException ex) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   226
            ex.printStackTrace();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   227
            throw new AssertionError("missing field", ex);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   228
        } finally {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   229
            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
   230
                    name, fileDescriptor, rawfd, cleanup));
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
    /**
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   235
     * 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
   236
     */
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   237
    static void listProcFD() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   238
        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
   239
        Optional<Path> lsof = lsofDirs.stream()
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   240
                .map(s -> Paths.get(s, "lsof"))
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   241
                .filter(f -> Files.isExecutable(f))
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   242
                .findFirst();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   243
        lsof.ifPresent(exe -> {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   244
            try {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   245
                System.out.printf("Open File Descriptors:%n");
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   246
                long pid = ProcessHandle.current().pid();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   247
                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
   248
                pb.inheritIO();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   249
                Process p = pb.start();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   250
                p.waitFor(10, TimeUnit.SECONDS);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   251
            } catch (IOException | InterruptedException ie) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   252
                ie.printStackTrace();
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
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   257
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   258
    // 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
   259
    static class NamedWeak extends WeakReference<Object> {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   260
        private final String name;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   261
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   262
        NamedWeak(Object o, ReferenceQueue<Object> queue, String name) {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   263
            super(o, queue);
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   264
            this.name = name;
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   265
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   266
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   267
        public String toString() {
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   268
            return name + "; " + super.toString();
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   269
        }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   270
    }
7c12219870fd 8195059: Update java.net Socket and DatagramSocket implementations to use Cleaner
rriggs
parents:
diff changeset
   271
}