Skip to content

Commit ce2d38c

Browse files
committed
WIP: Graal native-image build of Java Schnorr Example
This is now rebased for JDK 25-ea and can compile because only native-image is run on JDK 25 and it is not run directly from Gradle. To build native image use: ./gradlew secp-examples-java:nativeCompile Running does not work on aarch64-macOS and has not been tested on Linux. Note: the Arena.ofConfined() change should probably not be merged to `master`, but is currently necessary to work with native-image. # Conflicts: # secp-examples-java/src/main/java/org/bitcoinj/secp/examples/ForeignRegistrationFeature.java # secp-integration-test/src/test/java/org/bitcoinj/secp/integration/package-info.java # secp256k1-examples-java/build.gradle
1 parent 3aa410e commit ce2d38c

File tree

6 files changed

+91
-2
lines changed

6 files changed

+91
-2
lines changed

README.adoc

+14
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,20 @@ Run the script:
100100

101101
* `./secp-examples-java/build/install/secp-examples-java/bin/secp-examples-java`
102102

103+
== Build and run a native image (Currently targets GraalVM for JDK 25-ea)
104+
105+
To build using GraalVM `native-image`:
106+
107+
. Make sure you have GraalVM 25 or later installed
108+
. Make sure `GRAALVM_HOME` points to the Graal JDK 25 installation
109+
. `./gradlew secp-examples-java:nativeCompile`
110+
111+
To run the compiled, native executable:
112+
113+
. `export LD_LIBRARY_PATH="$HOME/.nix-profile/lib:$LD_LIBRARY_PATH"`
114+
. `./secp-examples-java/build/schnorr`
115+
. Don't blink!
116+
103117
== Building with Nix
104118

105119
NOTE:: We currently only support setting up a development environment with Nix. In the future we hope to support a full Nix build.

secp-examples-java/build.gradle

+28
Original file line numberDiff line numberDiff line change
@@ -13,12 +13,16 @@ dependencies {
1313
implementation project(':secp-api')
1414
runtimeOnly project(':secp-bouncy')
1515
runtimeOnly project(':secp-ffm')
16+
17+
// This is only needed for ForeignRegistrationFeature and the native-image build
18+
implementation group: 'org.graalvm.sdk', name: 'nativeimage', version: '24.0.0'
1619
}
1720

1821
jar {
1922
inputs.property("moduleName", moduleName)
2023
manifest {
2124
attributes 'Implementation-Title': 'Example secp256k1-jdk Apps',
25+
'Main-Class': application.mainClass,
2226
'Implementation-Version': archiveVersion.get()
2327
}
2428
}
@@ -36,6 +40,10 @@ run {
3640
jvmArgs += '--enable-native-access=org.bitcoinj.secp.ffm'
3741
}
3842

43+
configurations {
44+
nativeToolImplementation.extendsFrom implementation
45+
}
46+
3947
tasks.register('runEcdsa', JavaExec) {
4048
javaLauncher = javaToolchains.launcherFor {
4149
languageVersion = JavaLanguageVersion.of(javaToolchainVersion)
@@ -46,3 +54,23 @@ tasks.register('runEcdsa', JavaExec) {
4654
mainClass = 'org.bitcoinj.secp.examples.Ecdsa'
4755
jvmArgs += '--enable-native-access=org.bitcoinj.secp.ffm'
4856
}
57+
58+
// Compile a native image using GraalVM's native-image tool
59+
// Graal must be installed at $GRAALVM_HOME
60+
tasks.register('nativeCompile', Exec) {
61+
dependsOn jar
62+
workingDir = projectDir
63+
executable = "${System.env.GRAALVM_HOME}/bin/native-image"
64+
args = ['--verbose',
65+
'--no-fallback',
66+
'-cp', "${-> configurations.nativeToolImplementation.asPath}", // Lazy configuration resolution
67+
'-jar', jar.archiveFile.get(),
68+
'-H:Path=build',
69+
'-H:Name=schnorr',
70+
'-H:+ForeignAPISupport',
71+
'--features=org.bitcoinj.secp.examples.ForeignRegistrationFeature',
72+
'--enable-native-access=ALL-UNNAMED',
73+
'-H:+ReportUnsupportedElementsAtRuntime',
74+
'-H:+ReportExceptionStackTraces'
75+
]
76+
}

secp-examples-java/src/main/java/module-info.java

+1
Original file line numberDiff line numberDiff line change
@@ -17,4 +17,5 @@
1717
module org.bitcoinj.secp.examples {
1818
requires org.bitcoinj.secp.api;
1919
requires org.jspecify;
20+
requires org.graalvm.nativeimage;
2021
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
/*
2+
* Copyright 2023-2024 secp256k1-jdk Developers.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.bitcoinj.secp.examples;
17+
18+
import org.graalvm.nativeimage.hosted.Feature;
19+
import org.graalvm.nativeimage.hosted.RuntimeForeignAccess;
20+
21+
import java.lang.foreign.FunctionDescriptor;
22+
import java.lang.foreign.Linker;
23+
24+
import static java.lang.foreign.ValueLayout.*;
25+
26+
/**
27+
*
28+
*/
29+
public class ForeignRegistrationFeature implements Feature {
30+
public void duringSetup(Feature.DuringSetupAccess access) {
31+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.ofVoid());
32+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(JAVA_INT, JAVA_INT, JAVA_INT));
33+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(JAVA_LONG, JAVA_INT));
34+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(JAVA_INT, JAVA_INT, JAVA_INT, JAVA_INT));
35+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(JAVA_INT, JAVA_LONG, JAVA_LONG, JAVA_LONG));
36+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(JAVA_INT, JAVA_LONG, JAVA_LONG, JAVA_LONG, JAVA_LONG, JAVA_LONG));
37+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(JAVA_INT, JAVA_LONG, JAVA_LONG, JAVA_LONG, JAVA_LONG, JAVA_LONG, JAVA_LONG));
38+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(JAVA_INT, JAVA_LONG, JAVA_LONG, JAVA_LONG, JAVA_LONG, JAVA_INT));
39+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(JAVA_INT, JAVA_LONG, JAVA_LONG));
40+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.ofVoid(JAVA_LONG, JAVA_LONG));
41+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.ofVoid(JAVA_LONG));
42+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.ofVoid(JAVA_INT, JAVA_INT));
43+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.of(ADDRESS, JAVA_INT, JAVA_INT), Linker.Option.firstVariadicArg(1));
44+
RuntimeForeignAccess.registerForDowncall(FunctionDescriptor.ofVoid(JAVA_INT), Linker.Option.captureCallState("errno"));
45+
}
46+
}

secp-ffm/src/main/java/org/bitcoinj/secp/ffm/Secp256k1Foreign.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ public Secp256k1Foreign(int flags) {
7373
}
7474

7575
public Secp256k1Foreign(int flags, boolean randomize) {
76-
arena = Arena.ofShared();
76+
arena = Arena.ofConfined(); // Changed from `ofShared` for use in Graal native-image tools
7777
/* Before we can call actual API functions, we need to create a "context". */
7878
ctx = secp256k1_h.secp256k1_context_create(flags);
7979

secp-integration-test/src/test/java/org/bitcoinj/secp/integration/package-info.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,4 +14,4 @@
1414
* limitations under the License.
1515
*/
1616
@org.jspecify.annotations.NullMarked
17-
package org.bitcoinj.secp.integration;
17+
package org.bitcoinj.secp.integration;

0 commit comments

Comments
 (0)