jdk/src/java.base/share/classes/jdk/internal/jimage/concurrent/ConcurrentPReader.java
author chegar
Wed, 03 Dec 2014 14:22:58 +0000
changeset 27565 729f9700483a
permissions -rw-r--r--
8049367: Modular Run-Time Images Reviewed-by: chegar, dfuchs, ihse, joehw, mullan, psandoz, wetmore Contributed-by: alan.bateman@oracle.com, alex.buckley@oracle.com, bradford.wetmore@oracle.com, chris.hegarty@oracle.com, erik.joelsson@oracle.com, james.laskey@oracle.com, jonathan.gibbons@oracle.com, karen.kinnear@oracle.com, magnus.ihse.bursie@oracle.com, mandy.chung@oracle.com, mark.reinhold@oracle.com, paul.sandoz@oracle.com, sundararajan.athijegannathan@oracle.com
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
27565
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     1
/*
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     2
 * Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     4
 *
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    10
 *
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    15
 * accompanied this code).
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    16
 *
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    20
 *
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    23
 * questions.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    24
 */
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    25
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    26
package jdk.internal.jimage.concurrent;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    27
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    28
import java.io.FileDescriptor;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    29
import java.io.FileInputStream;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    30
import java.io.IOException;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    31
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    32
import jdk.internal.jimage.PReader;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    33
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    34
import sun.misc.Unsafe;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    35
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    36
/**
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    37
 * A PReader implementation that supports concurrent pread operations.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    38
 */
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    39
public class ConcurrentPReader extends PReader {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    40
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    41
    private static final Unsafe UNSAFE = Unsafe.getUnsafe();
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    42
    private static final long BA_OFFSET = (long) UNSAFE.arrayBaseOffset(byte[].class);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    43
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    44
    /**
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    45
     * A temporary buffer that is cached on a per-thread basis.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    46
     */
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    47
    private static class TemporaryBuffer {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    48
        static final ThreadLocal<TemporaryBuffer> CACHED_BUFFER =
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    49
             new ThreadLocal<TemporaryBuffer>() {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    50
                @Override
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    51
                protected TemporaryBuffer initialValue() { return null; }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    52
             };
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    53
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    54
        static final TemporaryBuffer NOT_AVAILABLE = new TemporaryBuffer(0L, 0);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    55
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    56
        final long address;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    57
        final int size;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    58
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    59
        TemporaryBuffer(long address, int size) {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    60
            this.address = address;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    61
            this.size = size;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    62
        }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    63
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    64
        long address() { return address; }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    65
        int size() { return size; }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    66
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    67
        /**
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    68
         * Returns the {@code TemporaryBuffer} for the current thread. The buffer
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    69
         * is guaranteed to be of at least the given size. Returns {@code null}
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    70
         * if a buffer cannot be cached for this thread.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    71
         */
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    72
        static TemporaryBuffer get(int len) {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    73
            TemporaryBuffer buffer = CACHED_BUFFER.get();
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    74
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    75
            // cached buffer large enough?
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    76
            if (buffer != null && buffer.size() >= len) {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    77
                return buffer;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    78
            }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    79
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    80
            // if this is an InnocuousThread then don't return anything
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    81
            if (buffer == NOT_AVAILABLE)
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    82
                return null;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    83
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    84
            if (buffer != null) {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    85
                // replace buffer in cache with a larger buffer
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    86
                long originalAddress = buffer.address();
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    87
                long address = UNSAFE.allocateMemory(len);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    88
                buffer = new TemporaryBuffer(address, len);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    89
                CACHED_BUFFER.set(buffer);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    90
                UNSAFE.freeMemory(originalAddress);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    91
            } else {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    92
                // first usage.
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    93
                if (Thread.currentThread() instanceof sun.misc.InnocuousThread) {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    94
                    buffer = NOT_AVAILABLE;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    95
                } else {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    96
                    long address = UNSAFE.allocateMemory(len);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    97
                    buffer = new TemporaryBuffer(address, len);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    98
                }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
    99
                CACHED_BUFFER.set(buffer);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   100
            }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   101
            return buffer;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   102
        }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   103
    }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   104
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   105
    private final FileDescriptor fd;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   106
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   107
    private ConcurrentPReader(FileInputStream fis) throws IOException {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   108
        super(fis.getChannel());
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   109
        this.fd = fis.getFD();
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   110
    }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   111
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   112
    public ConcurrentPReader(String file) throws IOException {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   113
        this(new FileInputStream(file));
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   114
    }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   115
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   116
    @Override
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   117
    public byte[] read(int len, long position) throws IOException {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   118
        // need a temporary area of memory to read into
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   119
        TemporaryBuffer buffer = TemporaryBuffer.get(len);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   120
        long address;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   121
        if (buffer == null) {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   122
            address = UNSAFE.allocateMemory(len);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   123
        } else {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   124
            address = buffer.address();
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   125
        }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   126
        try {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   127
            int n = pread(fd, address, len, position);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   128
            if (n != len)
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   129
                throw new InternalError("short read, not handled yet");
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   130
            byte[] result = new byte[n];
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   131
            UNSAFE.copyMemory(null, address, result, BA_OFFSET, len);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   132
            return result;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   133
        } finally {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   134
            if (buffer == null) {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   135
                UNSAFE.freeMemory(address);
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   136
            }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   137
        }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   138
    }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   139
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   140
    private static native int pread(FileDescriptor fd, long address, int len, long pos)
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   141
        throws IOException;
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   142
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   143
    private static native void initIDs();
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   144
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   145
    static {
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   146
        System.loadLibrary("java");
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   147
        initIDs();
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   148
    }
729f9700483a 8049367: Modular Run-Time Images
chegar
parents:
diff changeset
   149
}