jdk/test/javax/management/remote/mandatory/connection/DeadLockTest.java
author dbuck
Tue, 18 Aug 2015 04:29:28 -0700
changeset 32417 6859107fc6c3
parent 30376 2ccf2cf7ea48
child 44423 306c020eb154
permissions -rw-r--r--
8133666: OperatingSystemMXBean reports abnormally high machine CPU consumption on Linux Reviewed-by: sla, mgronlun

/*
 * Copyright (c) 2004, 2015, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

/*
 * @test
 * @bug 5039210
 * @summary test on a client notification deadlock.
 * @author Shanliang JIANG
 * @modules java.management
 * @run clean DeadLockTest
 * @run build DeadLockTest
 * @run main DeadLockTest
 */

import java.net.MalformedURLException;
import java.io.IOException;
import java.util.HashMap;

import javax.management.*;
import javax.management.remote.*;

public class DeadLockTest {
    private static final String[] protocols = {"rmi", "iiop", "jmxmp"};
    private static final MBeanServer mbs = MBeanServerFactory.createMBeanServer();

    public static void main(String[] args) {
        System.out.println(">>> test on a client notification deadlock.");

        boolean ok = true;
        for (int i = 0; i < protocols.length; i++) {
            try {
                test(protocols[i]);
            } catch (Exception e) {
                System.out.println(">>> Test failed for " + protocols[i]);
                e.printStackTrace(System.out);
            }
        }

        System.out.println(">>> Test passed");
    }

    private static void test(String proto)
            throws Exception {
        System.out.println(">>> Test for protocol " + proto);

        JMXServiceURL u = null;
        JMXConnectorServer server = null;

        HashMap env = new HashMap(2);
        // server will close a client connection after 1 second
        env.put("jmx.remote.x.server.connection.timeout", "1000");

        // disable the client ping
        env.put("jmx.remote.x.client.connection.check.period", "0");

        try {
            u = new JMXServiceURL(proto, null, 0);
            server = JMXConnectorServerFactory.newJMXConnectorServer(u, env, mbs);
        } catch (MalformedURLException e) {
            System.out.println(">>> Skipping unsupported URL " + proto);
        }

        server.start();

        JMXServiceURL addr = server.getAddress();

        long st = 2000;
        MyListener myListener;

        // a cycle to make sure that we test the blocking problem.
        do {
            JMXConnector client = JMXConnectorFactory.connect(addr, env);
            MBeanServerConnection conn = client.getMBeanServerConnection();
            myListener = new MyListener(conn);
            client.addConnectionNotificationListener(myListener, null, null);

            // wait the server to close the client connection
            Thread.sleep(st);

            // makes the listener to do a remote request via the connection
            // which should be closed by the server.
            conn.getDefaultDomain();

            // allow the listner to have time to work
            Thread.sleep(100);

            // get a closed notif, should no block.
            client.close();
            Thread.sleep(100);

            st += 2000;

        } while(!myListener.isDone());

        server.stop();
    }

    private static class MyListener implements NotificationListener {
        public MyListener(MBeanServerConnection conn) {
            this.conn = conn;
        }

        public void handleNotification(Notification n, Object h) {
            if (n instanceof JMXConnectionNotification) {
                JMXConnectionNotification jcn = (JMXConnectionNotification)n;
                final String type = jcn.getType();
                System.out.println(">>> The listener receives notif with the type:"+type);

                if (JMXConnectionNotification.CLOSED.equals(type) ||
                    JMXConnectionNotification.FAILED.equals(type)) {

                    synchronized(this) {
                        done = false;
                    }

                    try {
                        conn.getDefaultDomain();
                    } catch (IOException ioe) {
                        // Greate !
                    }

                    synchronized(this) {
                        done = true;
                    }

                    System.out.println(">>> The listener is not blocked!");
                }
            }
        }

        public boolean isDone() {
            synchronized(this) {
                return done;
            }
        }

        private boolean done = false;
        private MBeanServerConnection conn;
    }
}