/src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/SymbolLookup.java
https://github.com/openjdk/valhalla · Java · 82 lines · 23 code · 5 blank · 54 comment · 1 complexity · 800fbe114c6b21481a5563d17d38c432 MD5 · raw file
- /*
- * Copyright (c) 2021, 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. Oracle designates this
- * particular file as subject to the "Classpath" exception as provided
- * by Oracle in the LICENSE file that accompanied this code.
- *
- * 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.
- */
- package jdk.incubator.foreign;
- import jdk.internal.access.JavaLangAccess;
- import jdk.internal.access.SharedSecrets;
- import jdk.internal.reflect.CallerSensitive;
- import jdk.internal.reflect.Reflection;
- import java.util.Objects;
- import java.util.Optional;
- /**
- * A symbol lookup. Exposes a lookup operation for searching symbol addresses by name, see {@link SymbolLookup#lookup(String)}.
- * A symbol lookup can be used to lookup a symbol in a loaded library. Clients can obtain a {@linkplain #loaderLookup() loader lookup},
- * which can be used to search symbols in libraries loaded by the current classloader (e.g. using {@link System#load(String)},
- * or {@link System#loadLibrary(String)}).
- * Alternatively, clients can obtain a {@linkplain CLinker#systemLookup() platform-dependent lookup}, to search symbols
- * in the standard C library.
- * <p> Unless otherwise specified, passing a {@code null} argument, or an array argument containing one or more {@code null}
- * elements to a method in this class causes a {@link NullPointerException NullPointerException} to be thrown. </p>
- */
- @FunctionalInterface
- public interface SymbolLookup {
- /**
- * Looks up a symbol with given name in this lookup.
- *
- * @param name the symbol name.
- * @return the memory address associated with the symbol (if any).
- */
- Optional<MemoryAddress> lookup(String name);
- /**
- * Obtains a symbol lookup suitable to find symbols in native libraries associated with the caller's classloader
- * (that is, libraries loaded using {@link System#loadLibrary} or {@link System#load}).
- * <p>
- * This method is <a href="package-summary.html#restricted"><em>restricted</em></a>.
- * Restricted methods are unsafe, and, if used incorrectly, their use might crash
- * the JVM or, worse, silently result in memory corruption. Thus, clients should refrain from depending on
- * restricted methods, and use safe and supported functionalities, where possible.
- *
- * @return a symbol lookup suitable to find symbols in libraries loaded by the caller's classloader.
- * @throws IllegalCallerException if access to this method occurs from a module {@code M} and the command line option
- * {@code --enable-native-access} is either absent, or does not mention the module name {@code M}, or
- * {@code ALL-UNNAMED} in case {@code M} is an unnamed module.
- */
- @CallerSensitive
- static SymbolLookup loaderLookup() {
- Class<?> caller = Reflection.getCallerClass();
- Reflection.ensureNativeAccess(caller);
- ClassLoader loader = Objects.requireNonNull(caller.getClassLoader());
- return name -> {
- Objects.requireNonNull(name);
- JavaLangAccess javaLangAccess = SharedSecrets.getJavaLangAccess();
- MemoryAddress addr = MemoryAddress.ofLong(javaLangAccess.findNative(loader, name));
- return addr == MemoryAddress.NULL? Optional.empty() : Optional.of(addr);
- };
- }
- }