src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.serviceprovider/src/org/graalvm/compiler/serviceprovider/GraalServices.java
author dlong
Tue, 24 Apr 2018 09:04:57 -0700
changeset 49873 26ebfe8ce852
parent 49451 e06f9607f370
child 50858 2d3e99a72541
permissions -rw-r--r--
8199755: Update Graal Reviewed-by: kvn
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     1
/*
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     2
 * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     4
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     7
 * published by the Free Software Foundation.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     8
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    13
 * accompanied this code).
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    14
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    18
 *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    21
 * questions.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    22
 */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    23
package org.graalvm.compiler.serviceprovider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    24
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    25
import static java.lang.Thread.currentThread;
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    26
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    27
import java.io.IOException;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    28
import java.io.InputStream;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    29
import java.util.Iterator;
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    30
import java.util.List;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    31
import java.util.ServiceConfigurationError;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    32
import java.util.ServiceLoader;
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    33
import java.util.concurrent.atomic.AtomicLong;
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    34
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    35
import jdk.vm.ci.services.JVMCIPermission;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    36
import jdk.vm.ci.services.Services;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    37
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    38
/**
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    39
 * Interface to functionality that abstracts over which JDK version Graal is running on.
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    40
 */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    41
public final class GraalServices {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    42
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    43
    private static int getJavaSpecificationVersion() {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    44
        String value = System.getProperty("java.specification.version");
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    45
        if (value.startsWith("1.")) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    46
            value = value.substring(2);
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    47
        }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    48
        return Integer.parseInt(value);
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    49
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    50
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    51
    /**
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    52
     * The integer value corresponding to the value of the {@code java.specification.version} system
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    53
     * property after any leading {@code "1."} has been stripped.
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    54
     */
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    55
    public static final int JAVA_SPECIFICATION_VERSION = getJavaSpecificationVersion();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    56
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    57
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    58
     * Determines if the Java runtime is version 8 or earlier.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    59
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    60
    public static final boolean Java8OrEarlier = JAVA_SPECIFICATION_VERSION <= 8;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    61
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    62
    private GraalServices() {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    63
    }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    64
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    65
    /**
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    66
     * Gets an {@link Iterable} of the providers available for a given service.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    67
     *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    68
     * @throws SecurityException if on JDK8 and a security manager is present and it denies
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    69
     *             {@link JVMCIPermission}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    70
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    71
    public static <S> Iterable<S> load(Class<S> service) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    72
        assert !service.getName().startsWith("jdk.vm.ci") : "JVMCI services must be loaded via " + Services.class.getName();
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    73
        Iterable<S> iterable = ServiceLoader.load(service);
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    74
        return new Iterable<>() {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    75
            @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    76
            public Iterator<S> iterator() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    77
                Iterator<S> iterator = iterable.iterator();
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
    78
                return new Iterator<>() {
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    79
                    @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    80
                    public boolean hasNext() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    81
                        return iterator.hasNext();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    82
                    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    83
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    84
                    @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    85
                    public S next() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    86
                        S provider = iterator.next();
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    87
                        // Allow Graal extensions to access JVMCI
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
    88
                        openJVMCITo(provider.getClass());
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    89
                        return provider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    90
                    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    91
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    92
                    @Override
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    93
                    public void remove() {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    94
                        iterator.remove();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    95
                    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    96
                };
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    97
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    98
        };
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
    99
    }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   100
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   101
    /**
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   102
     * Opens all JVMCI packages to the module of a given class. This relies on JVMCI already having
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   103
     * opened all its packages to the module defining {@link GraalServices}.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   104
     *
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   105
     * @param other all JVMCI packages will be opened to the module defining this class
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   106
     */
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   107
    static void openJVMCITo(Class<?> other) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   108
        Module jvmciModule = JVMCI_MODULE;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   109
        Module otherModule = other.getModule();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   110
        if (jvmciModule != otherModule) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   111
            for (String pkg : jvmciModule.getPackages()) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   112
                if (!jvmciModule.isOpen(pkg, otherModule)) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   113
                    jvmciModule.addOpens(pkg, otherModule);
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   114
                }
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   115
            }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   116
        }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   117
    }
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   118
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   119
    /**
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   120
     * Gets the provider for a given service for which at most one provider must be available.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   121
     *
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   122
     * @param service the service whose provider is being requested
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   123
     * @param required specifies if an {@link InternalError} should be thrown if no provider of
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   124
     *            {@code service} is available
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   125
     * @return the requested provider if available else {@code null}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   126
     * @throws SecurityException if on JDK8 and a security manager is present and it denies
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   127
     *             {@link JVMCIPermission}
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   128
     */
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   129
    public static <S> S loadSingle(Class<S> service, boolean required) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   130
        assert !service.getName().startsWith("jdk.vm.ci") : "JVMCI services must be loaded via " + Services.class.getName();
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   131
        Iterable<S> providers = load(service);
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   132
        S singleProvider = null;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   133
        try {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   134
            for (Iterator<S> it = providers.iterator(); it.hasNext();) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   135
                singleProvider = it.next();
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   136
                if (it.hasNext()) {
46459
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   137
                    S other = it.next();
7d4e637d3f21 8180267: Update Graal
kvn
parents: 46458
diff changeset
   138
                    throw new InternalError(String.format("Multiple %s providers found: %s, %s", service.getName(), singleProvider.getClass().getName(), other.getClass().getName()));
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   139
                }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   140
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   141
        } catch (ServiceConfigurationError e) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   142
            // If the service is required we will bail out below.
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   143
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   144
        if (singleProvider == null) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   145
            if (required) {
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   146
                throw new InternalError(String.format("No provider for %s found", service.getName()));
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   147
            }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   148
        }
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   149
        return singleProvider;
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   150
    }
49873
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   151
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   152
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   153
     * Gets the class file bytes for {@code c}.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   154
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   155
    public static InputStream getClassfileAsStream(Class<?> c) throws IOException {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   156
        String classfilePath = c.getName().replace('.', '/') + ".class";
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   157
        return c.getModule().getResourceAsStream(classfilePath);
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   158
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   159
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   160
    private static final Module JVMCI_MODULE = Services.class.getModule();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   161
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   162
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   163
     * A JVMCI package dynamically exported to trusted modules.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   164
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   165
    private static final String JVMCI_RUNTIME_PACKAGE = "jdk.vm.ci.runtime";
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   166
    static {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   167
        assert JVMCI_MODULE.getPackages().contains(JVMCI_RUNTIME_PACKAGE);
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   168
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   169
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   170
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   171
     * Determines if invoking {@link Object#toString()} on an instance of {@code c} will only run
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   172
     * trusted code.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   173
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   174
    public static boolean isToStringTrusted(Class<?> c) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   175
        Module module = c.getModule();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   176
        Module jvmciModule = JVMCI_MODULE;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   177
        assert jvmciModule.getPackages().contains("jdk.vm.ci.runtime");
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   178
        if (module == jvmciModule || jvmciModule.isOpen(JVMCI_RUNTIME_PACKAGE, module)) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   179
            // Can access non-statically-exported package in JVMCI
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   180
            return true;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   181
        }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   182
        return false;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   183
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   184
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   185
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   186
     * Gets a unique identifier for this execution such as a process ID or a
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   187
     * {@linkplain #getGlobalTimeStamp() fixed timestamp}.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   188
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   189
    public static String getExecutionID() {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   190
        return Long.toString(ProcessHandle.current().pid());
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   191
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   192
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   193
    private static final AtomicLong globalTimeStamp = new AtomicLong();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   194
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   195
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   196
     * Gets a time stamp for the current process. This method will always return the same value for
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   197
     * the current VM execution.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   198
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   199
    public static long getGlobalTimeStamp() {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   200
        if (globalTimeStamp.get() == 0) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   201
            globalTimeStamp.compareAndSet(0, System.currentTimeMillis());
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   202
        }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   203
        return globalTimeStamp.get();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   204
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   205
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   206
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   207
     * Access to thread specific information made available via Java Management Extensions (JMX).
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   208
     * Using this abstraction enables avoiding a dependency to the {@code java.management} and
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   209
     * {@code jdk.management} modules on JDK 9 and later.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   210
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   211
    public abstract static class JMXService {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   212
        protected abstract long getThreadAllocatedBytes(long id);
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   213
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   214
        protected abstract long getCurrentThreadCpuTime();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   215
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   216
        protected abstract boolean isThreadAllocatedMemorySupported();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   217
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   218
        protected abstract boolean isCurrentThreadCpuTimeSupported();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   219
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   220
        protected abstract List<String> getInputArguments();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   221
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   222
        // Placing this static field in JMXService (instead of GraalServices)
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   223
        // allows for lazy initialization.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   224
        static final JMXService instance = loadSingle(JMXService.class, false);
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   225
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   226
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   227
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   228
     * Returns an approximation of the total amount of memory, in bytes, allocated in heap memory
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   229
     * for the thread of the specified ID. The returned value is an approximation because some Java
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   230
     * virtual machine implementations may use object allocation mechanisms that result in a delay
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   231
     * between the time an object is allocated and the time its size is recorded.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   232
     * <p>
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   233
     * If the thread of the specified ID is not alive or does not exist, this method returns
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   234
     * {@code -1}. If thread memory allocation measurement is disabled, this method returns
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   235
     * {@code -1}. A thread is alive if it has been started and has not yet died.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   236
     * <p>
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   237
     * If thread memory allocation measurement is enabled after the thread has started, the Java
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   238
     * virtual machine implementation may choose any time up to and including the time that the
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   239
     * capability is enabled as the point where thread memory allocation measurement starts.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   240
     *
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   241
     * @param id the thread ID of a thread
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   242
     * @return an approximation of the total memory allocated, in bytes, in heap memory for a thread
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   243
     *         of the specified ID if the thread of the specified ID exists, the thread is alive,
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   244
     *         and thread memory allocation measurement is enabled; {@code -1} otherwise.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   245
     *
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   246
     * @throws IllegalArgumentException if {@code id} {@code <=} {@code 0}.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   247
     * @throws UnsupportedOperationException if the Java virtual machine implementation does not
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   248
     *             {@linkplain #isThreadAllocatedMemorySupported() support} thread memory allocation
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   249
     *             measurement.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   250
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   251
    public static long getThreadAllocatedBytes(long id) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   252
        JMXService jmx = JMXService.instance;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   253
        if (jmx == null) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   254
            throw new UnsupportedOperationException();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   255
        }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   256
        return jmx.getThreadAllocatedBytes(id);
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   257
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   258
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   259
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   260
     * Convenience method for calling {@link #getThreadAllocatedBytes(long)} with the id of the
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   261
     * current thread.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   262
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   263
    public static long getCurrentThreadAllocatedBytes() {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   264
        return getThreadAllocatedBytes(currentThread().getId());
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   265
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   266
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   267
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   268
     * Returns the total CPU time for the current thread in nanoseconds. The returned value is of
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   269
     * nanoseconds precision but not necessarily nanoseconds accuracy. If the implementation
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   270
     * distinguishes between user mode time and system mode time, the returned CPU time is the
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   271
     * amount of time that the current thread has executed in user mode or system mode.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   272
     *
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   273
     * @return the total CPU time for the current thread if CPU time measurement is enabled;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   274
     *         {@code -1} otherwise.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   275
     *
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   276
     * @throws UnsupportedOperationException if the Java virtual machine does not
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   277
     *             {@linkplain #isCurrentThreadCpuTimeSupported() support} CPU time measurement for
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   278
     *             the current thread
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   279
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   280
    public static long getCurrentThreadCpuTime() {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   281
        JMXService jmx = JMXService.instance;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   282
        if (jmx == null) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   283
            throw new UnsupportedOperationException();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   284
        }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   285
        return jmx.getCurrentThreadCpuTime();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   286
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   287
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   288
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   289
     * Determines if the Java virtual machine implementation supports thread memory allocation
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   290
     * measurement.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   291
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   292
    public static boolean isThreadAllocatedMemorySupported() {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   293
        JMXService jmx = JMXService.instance;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   294
        if (jmx == null) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   295
            return false;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   296
        }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   297
        return jmx.isThreadAllocatedMemorySupported();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   298
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   299
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   300
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   301
     * Determines if the Java virtual machine supports CPU time measurement for the current thread.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   302
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   303
    public static boolean isCurrentThreadCpuTimeSupported() {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   304
        JMXService jmx = JMXService.instance;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   305
        if (jmx == null) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   306
            return false;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   307
        }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   308
        return jmx.isCurrentThreadCpuTimeSupported();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   309
    }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   310
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   311
    /**
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   312
     * Gets the input arguments passed to the Java virtual machine which does not include the
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   313
     * arguments to the {@code main} method. This method returns an empty list if there is no input
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   314
     * argument to the Java virtual machine.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   315
     * <p>
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   316
     * Some Java virtual machine implementations may take input arguments from multiple different
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   317
     * sources: for examples, arguments passed from the application that launches the Java virtual
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   318
     * machine such as the 'java' command, environment variables, configuration files, etc.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   319
     * <p>
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   320
     * Typically, not all command-line options to the 'java' command are passed to the Java virtual
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   321
     * machine. Thus, the returned input arguments may not include all command-line options.
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   322
     *
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   323
     * @return the input arguments to the JVM or {@code null} if they are unavailable
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   324
     */
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   325
    public static List<String> getInputArguments() {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   326
        JMXService jmx = JMXService.instance;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   327
        if (jmx == null) {
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   328
            return null;
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   329
        }
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   330
        return jmx.getInputArguments();
26ebfe8ce852 8199755: Update Graal
dlong
parents: 49451
diff changeset
   331
    }
43972
1ade39b8381b 8174879: Rename jdk.vm.ci to jdk.internal.vm.ci
kvn
parents:
diff changeset
   332
}