blob: 5ba0a8a05138cbc3c548799ea94f614173eea607 [file] [log] [blame]
// Copyright 2019 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.jni;
import static com.google.common.base.Preconditions.checkArgument;
import com.google.common.flogger.GoogleLogger;
import com.google.common.io.ByteStreams;
import com.google.devtools.build.lib.util.OS;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.file.Files;
import java.nio.file.Path;
import javax.annotation.Nullable;
/** Generic code to interact with the platform-specific JNI code bundle. */
public final class JniLoader {
private static final GoogleLogger logger = GoogleLogger.forEnclosingClass();
@Nullable private static final Throwable JNI_LOAD_ERROR;
static {
Throwable jniLoadError;
try {
switch (OS.getCurrent()) {
case LINUX:
case FREEBSD:
case OPENBSD:
case UNKNOWN:
loadLibrary("main/native/libunix_jni.so");
break;
case DARWIN:
loadLibrary("main/native/libunix_jni.dylib");
break;
case WINDOWS:
try {
// TODO(jmmv): This is here only for the bootstrapping process, which builds the JNI
// library and passes a -Djava.library.path to the JVM to find it. I'm sure that this
// can be replaced by properly bundling the library as a resource in the JAR. For some
// strange reason that I haven't fully understood yet, this also must come first.
System.loadLibrary("windows_jni");
} catch (UnsatisfiedLinkError e) {
try {
loadLibrary("main/native/windows/windows_jni.dll");
} catch (IOException e2) {
e2.addSuppressed(e);
throw e2;
}
}
break;
}
jniLoadError = null;
} catch (IOException | UnsatisfiedLinkError e) {
logger.atWarning().withCause(e).log("Failed to load JNI library");
jniLoadError = e;
}
JNI_LOAD_ERROR = jniLoadError;
}
/**
* Loads a resource as a shared library.
*
* @param resourceName the name of the shared library to load, specified as a slash-separated
* relative path within the JAR with at least two components
* @throws IOException if the resource cannot be extracted or loading the library fails for any
* other reason
*/
private static void loadLibrary(String resourceName) throws IOException {
Path dir = null;
Path tempFile = null;
try {
dir = Files.createTempDirectory("bazel-jni.");
int slash = resourceName.lastIndexOf('/');
checkArgument(slash != -1, "resourceName must contain two path components");
tempFile = dir.resolve(resourceName.substring(slash + 1));
ClassLoader loader = JniLoader.class.getClassLoader();
try (InputStream resource = loader.getResourceAsStream(resourceName)) {
if (resource == null) {
throw new UnsatisfiedLinkError("Resource " + resourceName + " not in JAR");
}
try (OutputStream diskFile = new FileOutputStream(tempFile.toString())) {
ByteStreams.copy(resource, diskFile);
}
}
System.load(tempFile.toString());
// Remove the temporary file now that we have loaded it. If we keep it short-lived, we can
// avoid the file system from persisting it to disk, avoiding an unnecessary cost.
//
// Unfortunately, we cannot do this on Windows because the DLL remains open and we don't have
// a way to specify FILE_SHARE_DELETE in the System.load() call.
if (OS.getCurrent() != OS.WINDOWS) {
Files.delete(tempFile);
tempFile = null;
Files.delete(dir);
dir = null;
}
} catch (IOException e) {
try {
if (tempFile != null) {
Files.deleteIfExists(tempFile);
}
if (dir != null) {
Files.delete(dir);
}
} catch (IOException e2) {
// Nothing else we can do. Rely on "delete on exit" to try clean things up later on.
}
throw e;
}
}
private JniLoader() {}
/**
* Triggers the load of the JNI bundle in a platform-independent basis.
*
* <p>This does <b>not</b> fail if the JNI bundle cannot be loaded because there are scenarios in
* which we want to run Bazel without JNI (e.g. during bootstrapping) or are able to fall back to
* an alternative implementation (e.g. in some filesystem implementations).
*
* <p>Callers can check if the JNI bundle was successfully loaded via {@link #isJniAvailable()}
* and obtain the load error via {@link #getJniLoadError()}.
*/
public static void loadJni() {}
/** Returns whether the JNI bundle was successfully loaded. */
public static boolean isJniAvailable() {
return JNI_LOAD_ERROR == null;
}
/** Returns the exception thrown while loading the JNI bundle, if it failed. */
@Nullable
public static Throwable getJniLoadError() {
return JNI_LOAD_ERROR;
}
}