test/jdk/java/nio/channels/DatagramChannel/AdaptorConcurrentIO.java
author phh
Sat, 30 Nov 2019 14:33:05 -0800
changeset 59330 5b96c12f909d
parent 58899 5573a7098439
permissions -rw-r--r--
8234541: C1 emits an empty message when it inlines successfully Summary: Use "inline" as the message when successfull Reviewed-by: thartmann, mdoerr Contributed-by: navy.xliu@gmail.com

/*
 * Copyright (c) 2019, 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 8232673
 * @summary Test DatagramChannel socket adaptor with concurrent send/receive
 */

import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.nio.channels.DatagramChannel;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;

public class AdaptorConcurrentIO {

    public static void main(String[] args) throws Exception {
        testConcurrentSendReceive(0);
        testConcurrentSendReceive(60_000);
    }

    /**
     * Starts a task that blocks in the adaptor's receive method, then invokes
     * the adaptor's send method to send a datagram. If the adaptor were using
     * the channel's blockingLock then send without be blocked waiting for
     * the receive to complete.
     */
    static void testConcurrentSendReceive(int timeout) throws Exception {
        try (DatagramChannel dc = DatagramChannel.open()) {
            InetAddress lb = InetAddress.getLoopbackAddress();
            dc.bind(new InetSocketAddress(lb, 0));
            DatagramSocket s = dc.socket();
            s.setSoTimeout(timeout);

            ExecutorService pool = Executors.newSingleThreadExecutor();
            try {
                Future<String> result = pool.submit(() -> {
                    byte[] data = new byte[100];
                    DatagramPacket p = new DatagramPacket(data, 0, data.length);
                    s.receive(p);
                    return new String(p.getData(), p.getOffset(), p.getLength(), "UTF-8");
                });

                Thread.sleep(200); // give chance for thread to block

                byte[] data = "hello".getBytes("UTF-8");
                DatagramPacket p = new DatagramPacket(data, 0, data.length);
                p.setSocketAddress(s.getLocalSocketAddress());
                s.send(p);

                String msg = result.get();
                if (!msg.equals("hello"))
                    throw new RuntimeException("Unexpected message: " + msg);
            } finally {
                pool.shutdown();
            }
        }
    }
}