test/jdk/java/nio/channels/DatagramChannel/EmptyBuffer.java
author bpb
Mon, 30 Apr 2018 13:40:39 -0700
changeset 49930 3aaaa5370999
parent 47216 71c04702a3d5
permissions -rw-r--r--
8202284: FileChannel and FileOutpuStream variants of AtomicAppend should fail silently on macOS >= 10.13 Reviewed-by: chegar

/*
 * Copyright (c) 2002, 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 4669040
 * @summary Test DatagramChannel receive with empty buffer
 * @author Mike McCloskey
 */

import java.io.IOException;
import java.io.PrintStream;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.channels.ClosedByInterruptException;
import java.nio.channels.DatagramChannel;

public class EmptyBuffer {

    private static final PrintStream log = System.err;

    public static void main(String[] args) throws Exception {
        test();
    }

    private static void test() throws Exception {
        DatagramChannel dc = DatagramChannel.open();
        InetAddress localHost = InetAddress.getLocalHost();
        dc.bind(new InetSocketAddress(localHost, 0));

        Server server = new Server(dc.getLocalAddress());
        Thread serverThread = new Thread(server);
        serverThread.start();

        try {
            InetSocketAddress isa = new InetSocketAddress(localHost, server.port());
            dc.connect(isa);

            ByteBuffer bb = ByteBuffer.allocateDirect(12);
            bb.order(ByteOrder.BIG_ENDIAN);
            bb.putInt(1).putLong(1);
            bb.flip();

            dc.write(bb);
            bb.rewind();
            dc.write(bb);
            bb.rewind();
            dc.write(bb);

            Thread.sleep(2000);

            serverThread.interrupt();
            server.throwException();
        } finally {
            dc.close();
        }
    }

    private static class Server implements Runnable {
        private final DatagramChannel dc;
        private final SocketAddress clientAddress;
        private Exception e = null;

        Server(SocketAddress clientAddress) throws IOException {
            this.dc = DatagramChannel.open().bind(new InetSocketAddress(0));
            this.clientAddress = clientAddress;
        }

        int port() {
            return dc.socket().getLocalPort();
        }

        void throwException() throws Exception {
            if (e != null)
                throw e;
        }

        void showBuffer(String s, ByteBuffer bb) {
            log.println(s);
            bb.rewind();
            for (int i=0; i<bb.limit(); i++) {
                byte element = bb.get();
                log.print(element);
            }
            log.println();
        }

        @Override
        public void run() {
            try {
                ByteBuffer bb = ByteBuffer.allocateDirect(12);
                bb.clear();
                // Only one clear. The buffer will be full after
                // the first receive, but it should still block
                // and receive and discard the next two
                int numberReceived = 0;
                while (!Thread.interrupted()) {
                    SocketAddress sa;
                    try {
                        sa = dc.receive(bb);
                    } catch (ClosedByInterruptException cbie) {
                        // Expected
                        log.println("Took expected exit");
                        // Verify that enough packets were received
                        if (numberReceived != 3)
                            throw new RuntimeException("Failed: Too few datagrams");
                        break;
                    }
                    if (sa != null) {
                        log.println("Client: " + sa);
                        // Check client address so as not to count stray packets
                        if (sa.equals(clientAddress)) {
                            showBuffer("RECV", bb);
                            numberReceived++;
                        }
                        if (numberReceived > 3)
                            throw new RuntimeException("Failed: Too many datagrams");
                        sa = null;
                    }
                }
            } catch (Exception ex) {
                e = ex;
            } finally {
                try { dc.close(); } catch (IOException ignore) { }
            }
        }
    }
}