Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

GraalVM updates to extract the shared lib with native image instead of package as resource #816

Merged
merged 41 commits into from
Sep 13, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
41 commits
Select commit Hold shift + click to select a range
2903314
add extract lib API
TingDaoK Aug 9, 2024
ab8c823
revamp
TingDaoK Aug 12, 2024
a8fe21f
Merge branch 'main' into add-extract-lib-api
TingDaoK Aug 12, 2024
5588c94
add comments
TingDaoK Aug 12, 2024
04fef59
use the old version of graalvm SDK to support java 8
TingDaoK Aug 27, 2024
e46c230
provided as scope
TingDaoK Aug 27, 2024
653a9a2
optional?
TingDaoK Aug 27, 2024
2109a3d
add dependency for gradle
TingDaoK Aug 28, 2024
5cf9feb
Merge branch 'main' into add-extract-lib-api
TingDaoK Aug 28, 2024
97b2d07
exclude the native feature from source
TingDaoK Aug 28, 2024
e8e4363
I cannot build it locally...
TingDaoK Aug 30, 2024
bae83a8
chatgpt
TingDaoK Aug 30, 2024
63ff90a
submodule
TingDaoK Aug 30, 2024
cc6a944
submodule
TingDaoK Aug 30, 2024
d059836
move it to internal
TingDaoK Aug 30, 2024
4106dfa
what about this?
TingDaoK Aug 30, 2024
4fd604f
let it compile
TingDaoK Sep 4, 2024
86303b5
ignore those warnings?
TingDaoK Sep 4, 2024
8d85cba
Move the API to internal
TingDaoK Sep 4, 2024
296a649
Use maven to build Java doc
TingDaoK Sep 4, 2024
c214e60
Merge branch 'main' into add-extract-lib-api
TingDaoK Sep 4, 2024
8fce325
maven config the output path
TingDaoK Sep 4, 2024
b8bb695
don't ignore any warnings now
TingDaoK Sep 4, 2024
db4e92d
:) you complaining about the thing you are not generating.
TingDaoK Sep 4, 2024
4c2a979
trivial
TingDaoK Sep 6, 2024
a7ac437
format
TingDaoK Sep 6, 2024
0e4c097
more trivial
TingDaoK Sep 6, 2024
b8998b7
Merge branch 'main' into add-extract-lib-api
TingDaoK Sep 6, 2024
b86e67b
add graalvm to the class name as native is nto very clear in our code…
TingDaoK Sep 6, 2024
2a5dc32
address comments
TingDaoK Sep 7, 2024
e444bfc
maybe
TingDaoK Sep 7, 2024
79ac10e
trivial
TingDaoK Sep 9, 2024
5854997
keep the options at one place
TingDaoK Sep 9, 2024
833daba
no needs for -missing on java 11
TingDaoK Sep 9, 2024
3953dee
just cp instead of redirect it
TingDaoK Sep 9, 2024
814357b
update comments
TingDaoK Sep 9, 2024
cf326f3
docs tweaks
graebm Sep 11, 2024
56763e6
Apply suggestions from code review
TingDaoK Sep 13, 2024
9159d2f
move the comments
TingDaoK Sep 13, 2024
f52a410
run it on arm-mac
TingDaoK Sep 13, 2024
a912994
fine
TingDaoK Sep 13, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 3 additions & 3 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -192,7 +192,7 @@ jobs:
python -c "from urllib.request import urlretrieve; urlretrieve('${{ env.BUILDER_HOST }}/${{ env.BUILDER_SOURCE }}/${{ env.BUILDER_VERSION }}/builder.pyz?run=${{ env.RUN }}', 'builder.pyz')"
python builder.pyz build -p ${{ env.PACKAGE_NAME }} downstream

macos:
macos:
runs-on: macos-14 #latest
steps:
- name: Checkout Sources
Expand All @@ -204,7 +204,7 @@ jobs:
python3 -c "from urllib.request import urlretrieve; urlretrieve('${{ env.BUILDER_HOST }}/${{ env.BUILDER_SOURCE }}/${{ env.BUILDER_VERSION }}/builder.pyz?run=${{ env.RUN }}', 'builder')"
chmod a+x builder
./builder build -p ${{ env.PACKAGE_NAME }} --spec=downstream
python3 codebuild/macos_compatibility_check.py
python3 codebuild/macos_compatibility_check.py

macos-x64:
runs-on: macos-14-large #latest
Expand Down Expand Up @@ -274,7 +274,7 @@ jobs:

# check that docs can still build
check-docs:
runs-on: ubuntu-22.04 # latest
runs-on: ubuntu-22.04 # use same version as docs.yml
steps:
- uses: actions/checkout@v3
with:
Expand Down
3 changes: 1 addition & 2 deletions .github/workflows/docs.yml
Original file line number Diff line number Diff line change
Expand Up @@ -9,15 +9,14 @@ on:

jobs:
update-docs-branch:
runs-on: ubuntu-20.04 # latest
runs-on: ubuntu-22.04 # use same version as ci.yml's check-docs
permissions:
contents: write # allow push
steps:
- name: Checkout
uses: actions/checkout@v3
with:
submodules: true

- name: Update docs branch
run: |
./make-docs.sh
Expand Down
7 changes: 7 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -159,6 +159,13 @@ Platforms without FIPS compliance are also included in this jar, for compatibili
> [!WARNING]
> The classifier, and platforms with FIPS compliance are subject to change in the future.

## GraalVM support

Since version v0.29.20, GraalVM native image was supported. You can compile your application with AWS CRT in a GraalVM native image project without any additional configuration.

Since version v0.30.12, GraalVM support was updated. Instead of packaging the JNI shared lib with native image as resource, the corresponding shared lib will be written to the same directory as the native image.
In this way, it reduces the native image size around 30%, and avoids the extra loading time needed for extracting the JNI lib to the temporary path for load. No additional configuration needed.
**Note**: the JNI shared lib must be in the same directory as the GraalVM native image. If you move the native image, you must move this file too. It is `aws-crt-jni.dll` on Windows, `libaws-crt-jni.dylib` on macOS, and `libaws-crt-jni.so` on Unix.

## System Properties

Expand Down
1 change: 1 addition & 0 deletions android/crt/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ android {
main {
java.srcDir '../../src/main/java'
java.srcDir 'src/main/java'
java.exclude '**/GraalVMNativeFeature.java'
}
androidTest {
setRoot '../../src/test'
Expand Down
12 changes: 0 additions & 12 deletions javadoc.options

This file was deleted.

5 changes: 4 additions & 1 deletion make-docs.sh
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,9 @@ pushd $(dirname $0) > /dev/null
rm -rf docs/

# build
javadoc @javadoc.options
mvn javadoc:javadoc -Dmaven.javadoc.failOnWarnings=true

# copy to docs/
cp -r target/site/apidocs/ docs/

popd > /dev/null
28 changes: 26 additions & 2 deletions pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -162,7 +162,7 @@
<!-- cmake configure -->
<execution>
<id>cmake-configure</id>
<phase>generate-sources</phase>
<phase>generate-resources</phase>
<goals>
<goal>exec</goal>
</goals>
Expand Down Expand Up @@ -292,7 +292,7 @@
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-javadoc-plugin</artifactId>
<version>2.9.1</version>
<version>3.4.0</version>
<executions>
<execution>
<id>attach-javadocs</id>
Expand Down Expand Up @@ -351,6 +351,13 @@
<version>1.4</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.graalvm.sdk</groupId>
<artifactId>graal-sdk</artifactId>
<version>21.3.11</version><!-- 21.3.11 is the last version to support JDK8 -->
<scope>provided</scope>
<optional>true</optional>
</dependency>
</dependencies>
<build>
<resources>
Expand Down Expand Up @@ -484,6 +491,23 @@
<forkCount>0</forkCount>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-javadoc-plugin</artifactId>
<version>3.4.0</version>
<configuration>
<windowtitle>AWS Common Runtime for Java/JVM</windowtitle>
<doctitle>AWS Common Runtime for Java/JVM</doctitle>
<header>AWS Common Runtime for Java/JVM</header>
<bottom>Copyright © Amazon.com, Inc. or its affiliates. All Rights Reserved.</bottom>
<show>public</show>
<sourcepath>src/main/java</sourcepath>
<notimestamp>true</notimestamp>
<quiet>true</quiet>
<doclint>all</doclint>
<excludePackageNames>software.amazon.awssdk.crt.internal</excludePackageNames>
</configuration>
</plugin>
</plugins>
</build>
</project>
159 changes: 59 additions & 100 deletions src/main/java/software/amazon/awssdk/crt/CRT.java
Original file line number Diff line number Diff line change
Expand Up @@ -4,16 +4,15 @@
*/
package software.amazon.awssdk.crt;

import software.amazon.awssdk.crt.internal.ExtractLib;
import software.amazon.awssdk.crt.io.ClientBootstrap;
import software.amazon.awssdk.crt.io.EventLoopGroup;
import software.amazon.awssdk.crt.io.HostResolver;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileOutputStream;
import java.io.FilenameFilter;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.lang.reflect.InvocationTargetException;
import java.util.*;
Expand All @@ -29,7 +28,7 @@ public final class CRT {
private static final String CRT_ARCH_OVERRIDE_SYSTEM_PROPERTY = "aws.crt.arch";
private static final String CRT_ARCH_OVERRIDE_ENVIRONMENT_VARIABLE = "AWS_CRT_ARCH";

private static final String CRT_LIB_NAME = "aws-crt-jni";
public static final String CRT_LIB_NAME = "aws-crt-jni";
graebm marked this conversation as resolved.
Show resolved Hide resolved
public static final int AWS_CRT_SUCCESS = 0;
private static final CrtPlatform s_platform;

Expand All @@ -41,8 +40,15 @@ public final class CRT {
// If the lib is already present/loaded or is in java.library.path, just use it
System.loadLibrary(CRT_LIB_NAME);
} catch (UnsatisfiedLinkError e) {
// otherwise, load from the jar this class is in
loadLibraryFromJar();
String graalVMImageCode = System.getProperty("org.graalvm.nativeimage.imagecode");
if (graalVMImageCode != null && graalVMImageCode == "runtime") {
throw new CrtRuntimeException(
"Failed to load '" + System.mapLibraryName(CRT_LIB_NAME) +
"'. Make sure this file is in the same directory as the GraalVM native image. ");
} else {
// otherwise, load from the jar this class is in
loadLibraryFromJar();
}
}

// Initialize the CRT
Expand Down Expand Up @@ -246,112 +252,65 @@ private static List<String> runProcess(String[] cmdArray) throws IOException {
}

private static void extractAndLoadLibrary(String path) {
// Check java.io.tmpdir
String tmpdirPath;
File tmpdirFile;
try {
// Check java.io.tmpdir
String tmpdirPath;
File tmpdirFile;
try {
tmpdirFile = new File(path).getAbsoluteFile();
tmpdirPath = tmpdirFile.getAbsolutePath();
if (tmpdirFile.exists()) {
if (!tmpdirFile.isDirectory()) {
throw new IOException("not a directory: " + tmpdirPath);
}
} else {
tmpdirFile.mkdirs();
}

if (!tmpdirFile.canRead() || !tmpdirFile.canWrite()) {
throw new IOException("access denied: " + tmpdirPath);
}
} catch (Exception ex) {
String msg = "Invalid directory: " + path;
throw new IOException(msg, ex);
}

String libraryName = System.mapLibraryName(CRT_LIB_NAME);

// Prefix the lib we'll extract to disk
String tempSharedLibPrefix = "AWSCRT_";

File tempSharedLib = File.createTempFile(tempSharedLibPrefix, libraryName, tmpdirFile);
if (!tempSharedLib.setExecutable(true, true)) {
throw new CrtRuntimeException("Unable to make shared library executable by owner only");
}
if (!tempSharedLib.setWritable(true, true)) {
throw new CrtRuntimeException("Unable to make shared library writeable by owner only");
}
if (!tempSharedLib.setReadable(true, true)) {
throw new CrtRuntimeException("Unable to make shared library readable by owner only");
}

// The temp lib file should be deleted when we're done with it.
// Ask Java to try and delete it on exit. We call this immediately
// so that if anything goes wrong writing the file to disk, or
// loading it as a shared lib, it will still get cleaned up.
tempSharedLib.deleteOnExit();

// Unfortunately File.deleteOnExit() won't work on Windows, where
// files cannot be deleted while they're in use. On Windows, once
// our .dll is loaded, it can't be deleted by this process.
//
// The Windows-only solution to this problem is to scan on startup
// for old instances of the .dll and try to delete them. If another
// process is still using the .dll, the delete will fail, which is fine.
String os = getOSIdentifier();
if (os.equals("windows")) {
tryDeleteOldLibrariesFromTempDir(tmpdirFile, tempSharedLibPrefix, libraryName);
}

// open a stream to read the shared lib contents from this JAR
String libResourcePath = "/" + os + "/" + getArchIdentifier() + "/" + getCRuntime(os) + "/" + libraryName;
// Check whether there is a platform specific resource path to use
CrtPlatform platform = getPlatformImpl();
if (platform != null){
String platformLibResourcePath = platform.getResourcePath(getCRuntime(os), libraryName);
if (platformLibResourcePath != null){
libResourcePath = platformLibResourcePath;
tmpdirFile = new File(path).getAbsoluteFile();
tmpdirPath = tmpdirFile.getAbsolutePath();
if (tmpdirFile.exists()) {
if (!tmpdirFile.isDirectory()) {
throw new IOException("not a directory: " + tmpdirPath);
}
} else {
tmpdirFile.mkdirs();
}

try (InputStream in = CRT.class.getResourceAsStream(libResourcePath)) {
if (in == null) {
throw new IOException("Unable to open library in jar for AWS CRT: " + libResourcePath);
}

// Copy from jar stream to temp file
try (FileOutputStream out = new FileOutputStream(tempSharedLib)) {
int read;
byte [] bytes = new byte[1024];
while ((read = in.read(bytes)) != -1){
out.write(bytes, 0, read);
}
}
if (!tmpdirFile.canRead() || !tmpdirFile.canWrite()) {
throw new IOException("access denied: " + tmpdirPath);
}
} catch (IOException ex) {
CrtRuntimeException rex = new CrtRuntimeException("Invalid directory: " + path);
rex.initCause(ex);
throw rex;
}

if (!tempSharedLib.setWritable(false)) {
throw new CrtRuntimeException("Unable to make shared library read-only");
}
String libraryName = System.mapLibraryName(CRT_LIB_NAME);

// load the shared lib from the temp path
System.load(tempSharedLib.getAbsolutePath());
} catch (CrtRuntimeException crtex) {
System.err.println("Unable to initialize AWS CRT: " + crtex);
crtex.printStackTrace();
throw crtex;
} catch (UnknownPlatformException upe) {
System.err.println("Unable to determine platform for AWS CRT: " + upe);
upe.printStackTrace();
CrtRuntimeException rex = new CrtRuntimeException("Unable to determine platform for AWS CRT");
rex.initCause(upe);
throw rex;
} catch (Exception ex) {
System.err.println("Unable to unpack AWS CRT lib: " + ex);
// Prefix the lib we'll extract to disk
String tempSharedLibPrefix = "AWSCRT_";
File tempSharedLib = null;
try{
tempSharedLib = File.createTempFile(tempSharedLibPrefix, libraryName, tmpdirFile);
}
catch (IOException ex){
System.err.println("Unable to create temp file to extract AWS CRT library: " + ex);
ex.printStackTrace();
CrtRuntimeException rex = new CrtRuntimeException("Unable to unpack AWS CRT library");
CrtRuntimeException rex = new CrtRuntimeException("Unable to create temp file to extract AWS CRT library");
rex.initCause(ex);
throw rex;
}
// The temp lib file should be deleted when we're done with it.
// Ask Java to try and delete it on exit. We call this immediately
// so that if anything goes wrong writing the file to disk, or
// loading it as a shared lib, it will still get cleaned up.
tempSharedLib.deleteOnExit();

// Unfortunately File.deleteOnExit() won't work on Windows, where
// files cannot be deleted while they're in use. On Windows, once
// our .dll is loaded, it can't be deleted by this process.
//
// The Windows-only solution to this problem is to scan on startup
// for old instances of the .dll and try to delete them. If another
// process is still using the .dll, the delete will fail, which is fine.
String os = getOSIdentifier();
if (os.equals("windows")) {
tryDeleteOldLibrariesFromTempDir(tmpdirFile, tempSharedLibPrefix, libraryName);
}
ExtractLib.extractLibrary(tempSharedLib);
// load the shared lib from the temp path
System.load(tempSharedLib.getAbsolutePath());

}

private static void loadLibraryFromJar() {
Expand Down
Loading
Loading