From c1de0b8ec9fea0cb08c3f5ec851517217dd21b07 Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 10:07:22 +0100 Subject: [PATCH 01/10] Attempt to build aarch64 so files --- .github/workflows/deploy-javadoc.yml | 10 +-- .github/workflows/gradle-build.yml | 19 +++--- .github/workflows/gradle-publish.yml | 15 +---- crypto/build.gradle | 23 ++++--- crypto/readme.md | 3 +- .../crypto/sr25519/Native.java | 9 ++- flake.lock | 61 +++++++++++++++++++ flake.nix | 40 ++++++++++++ rust-toolchain.toml | 5 ++ 9 files changed, 145 insertions(+), 40 deletions(-) create mode 100644 flake.lock create mode 100644 flake.nix create mode 100644 rust-toolchain.toml diff --git a/.github/workflows/deploy-javadoc.yml b/.github/workflows/deploy-javadoc.yml index de0d8e1f..c6a39a6f 100644 --- a/.github/workflows/deploy-javadoc.yml +++ b/.github/workflows/deploy-javadoc.yml @@ -12,14 +12,10 @@ jobs: steps: - uses: actions/checkout@v3 - - name: Set up JDK 8 - uses: actions/setup-java@v3 - with: - java-version: '8' - distribution: 'temurin' - + - uses: DeterminateSystems/nix-installer-action@main + - uses: DeterminateSystems/magic-nix-cache-action@main - name: Generate Javadoc - run: gradle aggregateJavadoc + run: nix develop --command bash -c "./gradlew aggregateJavadoc" - name: Deploy uses: JamesIves/github-pages-deploy-action@v4 diff --git a/.github/workflows/gradle-build.yml b/.github/workflows/gradle-build.yml index 088123f0..532be85b 100644 --- a/.github/workflows/gradle-build.yml +++ b/.github/workflows/gradle-build.yml @@ -4,7 +4,8 @@ name: Gradle Build on: - push + workflow_dispatch: + # push jobs: build: @@ -15,14 +16,12 @@ jobs: packages: write steps: - - name: Output GLIBC Version - run: ldd --version - uses: actions/checkout@v3 - - name: Set up JDK 8 - uses: actions/setup-java@v3 - with: - java-version: '8' - distribution: 'temurin' - + - uses: DeterminateSystems/nix-installer-action@main + - uses: DeterminateSystems/magic-nix-cache-action@main - name: Build with Gradle - run: gradle -i build + run: nix develop --ignore-environment --command bash -c "./gradlew build" + env: + USERNAME: ${{ github.actor }} + TOKEN: ${{ secrets.GITHUB_TOKEN }} + diff --git a/.github/workflows/gradle-publish.yml b/.github/workflows/gradle-publish.yml index 185ec5ca..bf1ace90 100644 --- a/.github/workflows/gradle-publish.yml +++ b/.github/workflows/gradle-publish.yml @@ -18,19 +18,10 @@ jobs: steps: - uses: actions/checkout@v3 - - name: Set up JDK 8 - uses: actions/setup-java@v3 - with: - java-version: '8' - distribution: 'temurin' - + - uses: DeterminateSystems/nix-installer-action@main + - uses: DeterminateSystems/magic-nix-cache-action@main - name: Build with Gradle - run: gradle -i build - - # The USERNAME and TOKEN need to correspond to the credentials environment variables used in - # the publishing section of your build.gradle - - name: Publish to GitHub Packages - run: gradle publish + run: nix develop --ignore-environment --command bash -c "./gradlew build && ./gradlew publishToMavenLocal" env: USERNAME: ${{ github.actor }} TOKEN: ${{ secrets.GITHUB_TOKEN }} diff --git a/crypto/build.gradle b/crypto/build.gradle index e7896913..22c47587 100644 --- a/crypto/build.gradle +++ b/crypto/build.gradle @@ -13,21 +13,24 @@ test { task compileRust { project.logger.info("#####################" + System.getProperty("os.name").toLowerCase()) doLast { - if(System.getProperty("os.name").toLowerCase().contains("linux")) { + // nix build detection + if (!System.getenv('NIX_CC').empty) { + exec { + workingDir 'src/jni-crypto' + commandLine 'cargo', 'build', '--release', '--target=aarch64-unknown-linux-gnu', '--target-dir=../../build/jni-crypto' + } exec { workingDir 'src/jni-crypto' commandLine 'cargo', 'build', '--release', '--target=x86_64-unknown-linux-gnu', '--target-dir=../../build/jni-crypto' } - } - else if (System.getProperty("os.name").toLowerCase().contains("windows")) { exec { workingDir 'src/jni-crypto' commandLine 'cargo', 'build', '--release', '--target=x86_64-pc-windows-gnu', '--target-dir=../../build/jni-crypto' + environment RUSTFLAGS: "-L ${System.getenv('PTHREAD_LOCATION')}" } - } - // Does not cover solaris, just MacOS. See potential return types here: - // https://docs.gradle.org/current/javadoc/org/gradle/nativeplatform/platform/OperatingSystem.html - else { + } else { + // Does not cover solaris, just MacOS. See potential return types here: + // https://docs.gradle.org/current/javadoc/org/gradle/nativeplatform/platform/OperatingSystem.html exec { workingDir 'src/jni-crypto' commandLine 'cargo', 'build', '--release', '--target-dir=../../build/jni-crypto' @@ -44,7 +47,11 @@ processResources { include '*.dylib' } from("${buildDir}/jni-crypto/x86_64-unknown-linux-gnu/release") { - into "include/linux" + into "include/x86_64-linux" + include '*.so' + } + from("${buildDir}/jni-crypto/aarch64-unknown-linux-gnu/release") { + into "include/aarch64-linux" include '*.so' } from("${buildDir}/jni-crypto/x86_64-pc-windows-gnu/release") { diff --git a/crypto/readme.md b/crypto/readme.md index f001d4d6..9cafc47e 100644 --- a/crypto/readme.md +++ b/crypto/readme.md @@ -1,4 +1,5 @@ // TODO write readme.md :) Preparation: `rustup target add x86_64-pc-windows-gnu` -`sudo apt install gcc-mingw-w64-x86-64` \ No newline at end of file +`rustup target add aarch64-unknown-linux-gnu` +`sudo apt install gcc-mingw-w64-x86-64` diff --git a/crypto/src/main/java/com/strategyobject/substrateclient/crypto/sr25519/Native.java b/crypto/src/main/java/com/strategyobject/substrateclient/crypto/sr25519/Native.java index 817cae37..ab9be027 100644 --- a/crypto/src/main/java/com/strategyobject/substrateclient/crypto/sr25519/Native.java +++ b/crypto/src/main/java/com/strategyobject/substrateclient/crypto/sr25519/Native.java @@ -22,9 +22,12 @@ class Native { private static final String LIB_PATH = "SUBSTRATE_CLIENT_CRYPTO_LIB_PATH"; private static final String LIB_NAME = "jni_crypto"; private static final String OS_NAME = System.getProperty("os.name").toLowerCase(); + private static final String ARCH_NAME = System.getProperty("os.arch").toLowerCase(); private static final boolean IS_WINDOWS = OS_NAME.contains("win"); private static final boolean IS_MAC = OS_NAME.contains("mac"); private static final boolean IS_LINUX = OS_NAME.contains("linux"); + private static final boolean IS_AMD64 = ARCH_NAME.contains("amd64") || ARCH_NAME.contains("x86_64"); + private static final boolean IS_AARCH64 = ARCH_NAME.contains("aarch64"); static { try { @@ -49,8 +52,10 @@ private static String copyLibraryFromResourcesToTempDir() throws IOException { String osDir; if (IS_WINDOWS) { osDir = "windows"; - } else if (IS_LINUX) { - osDir = "linux"; + } else if (IS_LINUX && IS_AMD64) { + osDir = "x86_64-linux"; + } else if (IS_LINUX && IS_AARCH64) { + osDir = "aarch64-linux"; } else if (IS_MAC) { // TODO 'MasOS support problem' osDir = "macos"; } else { diff --git a/flake.lock b/flake.lock new file mode 100644 index 00000000..35cfdb9f --- /dev/null +++ b/flake.lock @@ -0,0 +1,61 @@ +{ + "nodes": { + "flake-utils": { + "inputs": { + "systems": "systems" + }, + "locked": { + "lastModified": 1710146030, + "narHash": "sha256-SZ5L6eA7HJ/nmkzGG7/ISclqe6oZdOZTNoesiInkXPQ=", + "owner": "numtide", + "repo": "flake-utils", + "rev": "b1d9ab70662946ef0850d488da1c9019f3a9752a", + "type": "github" + }, + "original": { + "owner": "numtide", + "repo": "flake-utils", + "type": "github" + } + }, + "nixpkgs": { + "locked": { + "lastModified": 1709961763, + "narHash": "sha256-6H95HGJHhEZtyYA3rIQpvamMKAGoa8Yh2rFV29QnuGw=", + "owner": "nixos", + "repo": "nixpkgs", + "rev": "3030f185ba6a4bf4f18b87f345f104e6a6961f34", + "type": "github" + }, + "original": { + "owner": "nixos", + "ref": "nixos-unstable", + "repo": "nixpkgs", + "type": "github" + } + }, + "root": { + "inputs": { + "flake-utils": "flake-utils", + "nixpkgs": "nixpkgs" + } + }, + "systems": { + "locked": { + "lastModified": 1681028828, + "narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=", + "owner": "nix-systems", + "repo": "default", + "rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e", + "type": "github" + }, + "original": { + "owner": "nix-systems", + "repo": "default", + "type": "github" + } + } + }, + "root": "root", + "version": 7 +} diff --git a/flake.nix b/flake.nix new file mode 100644 index 00000000..9d7e7747 --- /dev/null +++ b/flake.nix @@ -0,0 +1,40 @@ +{ + description = "Environment flake"; + + inputs = { + nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable"; + flake-utils.url = "github:numtide/flake-utils"; + }; + + outputs = { self, nixpkgs, flake-utils }: + flake-utils.lib.eachDefaultSystem (system: + let + pkgs = import nixpkgs { inherit system; }; + + mkLinkerPath = {cc, ...}: + # let + # inherit (stdenv) cc; + # in + "${cc}/bin/${cc.targetPrefix}cc"; + + in + { + devShells.default = with pkgs; mkShell { + + CARGO_TARGET_AARCH64_UNKNOWN_LINUX_GNU_LINKER = mkLinkerPath pkgsCross.aarch64-multiplatform.stdenv; + CARGO_TARGET_X86_64_UNKNOWN_LINUX_GNU_LINKER = mkLinkerPath pkgsCross.gnu64.stdenv; + CARGO_TARGET_X86_64_PC_WINDOWS_GNU_LINKER = mkLinkerPath pkgsCross.mingwW64.stdenv; + + # for mingw compilation + PTHREAD_LOCATION = "${pkgs.pkgsCross.mingwW64.windows.pthreads}/lib"; + + shellHook = "rustup show"; + + packages = [ + temurin-bin-8 + rustup + ]; + }; + } + ); +} diff --git a/rust-toolchain.toml b/rust-toolchain.toml new file mode 100644 index 00000000..37d505d7 --- /dev/null +++ b/rust-toolchain.toml @@ -0,0 +1,5 @@ +[toolchain] +channel = "nightly-2024-02-04" +components = [ "rustc" ] +targets = [ "x86_64-pc-windows-gnu", "aarch64-unknown-linux-gnu", "x86_64-pc-windows-gnu" ] +profile = "minimal" \ No newline at end of file From e0386bf3e7ce466c1fcb9f68d306a1d2310afc82 Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 10:16:05 +0100 Subject: [PATCH 02/10] Fix publish task --- .github/workflows/gradle-publish.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/gradle-publish.yml b/.github/workflows/gradle-publish.yml index bf1ace90..48e72d34 100644 --- a/.github/workflows/gradle-publish.yml +++ b/.github/workflows/gradle-publish.yml @@ -21,7 +21,7 @@ jobs: - uses: DeterminateSystems/nix-installer-action@main - uses: DeterminateSystems/magic-nix-cache-action@main - name: Build with Gradle - run: nix develop --ignore-environment --command bash -c "./gradlew build && ./gradlew publishToMavenLocal" + run: nix develop --ignore-environment --command bash -c "./gradlew build && ./gradlew publish" env: USERNAME: ${{ github.actor }} TOKEN: ${{ secrets.GITHUB_TOKEN }} From a6bc2790a3c5a57d95d325d993f6d255df452168 Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 10:27:29 +0100 Subject: [PATCH 03/10] Add github cache --- .github/workflows/deploy-javadoc.yml | 10 ++++++++++ .github/workflows/gradle-build.yml | 12 +++++++++++- .github/workflows/gradle-publish.yml | 12 +++++++++++- 3 files changed, 32 insertions(+), 2 deletions(-) diff --git a/.github/workflows/deploy-javadoc.yml b/.github/workflows/deploy-javadoc.yml index c6a39a6f..e49c5b04 100644 --- a/.github/workflows/deploy-javadoc.yml +++ b/.github/workflows/deploy-javadoc.yml @@ -12,6 +12,16 @@ jobs: steps: - uses: actions/checkout@v3 + - uses: actions/cache@v4 + with: + path: | + ~/.cargo/bin/ + ~/.cargo/registry/index/ + ~/.cargo/registry/cache/ + ~/.cargo/git/db/ + ~/.rustup + target/ + key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }} - uses: DeterminateSystems/nix-installer-action@main - uses: DeterminateSystems/magic-nix-cache-action@main - name: Generate Javadoc diff --git a/.github/workflows/gradle-build.yml b/.github/workflows/gradle-build.yml index 532be85b..38897bfc 100644 --- a/.github/workflows/gradle-build.yml +++ b/.github/workflows/gradle-build.yml @@ -16,7 +16,17 @@ jobs: packages: write steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 + - uses: actions/cache@v4 + with: + path: | + ~/.cargo/bin/ + ~/.cargo/registry/index/ + ~/.cargo/registry/cache/ + ~/.cargo/git/db/ + ~/.rustup + target/ + key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }} - uses: DeterminateSystems/nix-installer-action@main - uses: DeterminateSystems/magic-nix-cache-action@main - name: Build with Gradle diff --git a/.github/workflows/gradle-publish.yml b/.github/workflows/gradle-publish.yml index 48e72d34..a3d2fc2d 100644 --- a/.github/workflows/gradle-publish.yml +++ b/.github/workflows/gradle-publish.yml @@ -17,7 +17,17 @@ jobs: packages: write steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 + - uses: actions/cache@v4 + with: + path: | + ~/.cargo/bin/ + ~/.cargo/registry/index/ + ~/.cargo/registry/cache/ + ~/.cargo/git/db/ + ~/.rustup + target/ + key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }} - uses: DeterminateSystems/nix-installer-action@main - uses: DeterminateSystems/magic-nix-cache-action@main - name: Build with Gradle From 7f9fc8bdc6efcafa165c42d909ffde9315d390e8 Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 10:27:44 +0100 Subject: [PATCH 04/10] Change mvn url --- build.gradle | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index 7e81f31b..afdf4911 100644 --- a/build.gradle +++ b/build.gradle @@ -54,7 +54,8 @@ subprojects { repositories { maven { name = "GitHubPackages" - url = "https://maven.pkg.github.com/LibertyDSNP/substrate-client-java" + // url = "https://maven.pkg.github.com/LibertyDSNP/substrate-client-java" + url = "https://maven.pkg.github.com/kczulko/substrate-client-java" credentials { username = System.getenv("USERNAME") password = System.getenv("TOKEN") From 9229d0f680ed56b1f24c8744bffbcb9957f9c1b6 Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 10:37:21 +0100 Subject: [PATCH 05/10] Keep token and username env variables --- .github/workflows/gradle-publish.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/gradle-publish.yml b/.github/workflows/gradle-publish.yml index a3d2fc2d..ba76c617 100644 --- a/.github/workflows/gradle-publish.yml +++ b/.github/workflows/gradle-publish.yml @@ -31,7 +31,7 @@ jobs: - uses: DeterminateSystems/nix-installer-action@main - uses: DeterminateSystems/magic-nix-cache-action@main - name: Build with Gradle - run: nix develop --ignore-environment --command bash -c "./gradlew build && ./gradlew publish" + run: nix develop --ignore-environment --keep USERNAME --keep TOKEN --command bash -c "./gradlew build && ./gradlew publish" env: USERNAME: ${{ github.actor }} TOKEN: ${{ secrets.GITHUB_TOKEN }} From 07a31ed0d0fd9c68314ac36984d277b2976536fb Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 10:56:58 +0100 Subject: [PATCH 06/10] Minor changes --- flake.nix | 7 +------ rust-toolchain.toml | 2 +- 2 files changed, 2 insertions(+), 7 deletions(-) diff --git a/flake.nix b/flake.nix index 9d7e7747..21a4484e 100644 --- a/flake.nix +++ b/flake.nix @@ -11,12 +11,7 @@ let pkgs = import nixpkgs { inherit system; }; - mkLinkerPath = {cc, ...}: - # let - # inherit (stdenv) cc; - # in - "${cc}/bin/${cc.targetPrefix}cc"; - + mkLinkerPath = {cc, ...}: "${cc}/bin/${cc.targetPrefix}cc"; in { devShells.default = with pkgs; mkShell { diff --git a/rust-toolchain.toml b/rust-toolchain.toml index 37d505d7..84037350 100644 --- a/rust-toolchain.toml +++ b/rust-toolchain.toml @@ -2,4 +2,4 @@ channel = "nightly-2024-02-04" components = [ "rustc" ] targets = [ "x86_64-pc-windows-gnu", "aarch64-unknown-linux-gnu", "x86_64-pc-windows-gnu" ] -profile = "minimal" \ No newline at end of file +profile = "minimal" From f9dad69f9fe06576403c273b726c778171ce894b Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 12:28:37 +0100 Subject: [PATCH 07/10] Revert "Change mvn url" This reverts commit 7f9fc8bdc6efcafa165c42d909ffde9315d390e8. --- build.gradle | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/build.gradle b/build.gradle index afdf4911..7e81f31b 100644 --- a/build.gradle +++ b/build.gradle @@ -54,8 +54,7 @@ subprojects { repositories { maven { name = "GitHubPackages" - // url = "https://maven.pkg.github.com/LibertyDSNP/substrate-client-java" - url = "https://maven.pkg.github.com/kczulko/substrate-client-java" + url = "https://maven.pkg.github.com/LibertyDSNP/substrate-client-java" credentials { username = System.getenv("USERNAME") password = System.getenv("TOKEN") From 43fe5aee346e00de096115e3f7e2e74a9f9f323d Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 12:30:05 +0100 Subject: [PATCH 08/10] Revert gradle-build.yml workflow_dispatch --- .github/workflows/gradle-build.yml | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/.github/workflows/gradle-build.yml b/.github/workflows/gradle-build.yml index 38897bfc..98365962 100644 --- a/.github/workflows/gradle-build.yml +++ b/.github/workflows/gradle-build.yml @@ -4,8 +4,7 @@ name: Gradle Build on: - workflow_dispatch: - # push + push jobs: build: From 873858537e9c302e37cfa97b912bb54fce3a1a41 Mon Sep 17 00:00:00 2001 From: kczulko Date: Tue, 12 Mar 2024 12:42:39 +0100 Subject: [PATCH 09/10] Adjust comment --- crypto/build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crypto/build.gradle b/crypto/build.gradle index 22c47587..73d2a5a3 100644 --- a/crypto/build.gradle +++ b/crypto/build.gradle @@ -13,7 +13,7 @@ test { task compileRust { project.logger.info("#####################" + System.getProperty("os.name").toLowerCase()) doLast { - // nix build detection + // nix build detection if (!System.getenv('NIX_CC').empty) { exec { workingDir 'src/jni-crypto' From 648e60ef1dca7a88a2aed72d15d4ed252cb42f58 Mon Sep 17 00:00:00 2001 From: kczulko Date: Fri, 15 Mar 2024 08:42:53 +0100 Subject: [PATCH 10/10] Fix nix shell detection + keep the non-nix build behavior --- crypto/build.gradle | 24 +++++++++++++++---- .../crypto/sr25519/Native.java | 2 +- 2 files changed, 20 insertions(+), 6 deletions(-) diff --git a/crypto/build.gradle b/crypto/build.gradle index 73d2a5a3..8f055957 100644 --- a/crypto/build.gradle +++ b/crypto/build.gradle @@ -14,7 +14,7 @@ task compileRust { project.logger.info("#####################" + System.getProperty("os.name").toLowerCase()) doLast { // nix build detection - if (!System.getenv('NIX_CC').empty) { + if (System.getenv('NIX_CC') != null) { exec { workingDir 'src/jni-crypto' commandLine 'cargo', 'build', '--release', '--target=aarch64-unknown-linux-gnu', '--target-dir=../../build/jni-crypto' @@ -29,11 +29,25 @@ task compileRust { environment RUSTFLAGS: "-L ${System.getenv('PTHREAD_LOCATION')}" } } else { + // keep the old build behavior while not running in nix shell + if (System.getProperty("os.name").toLowerCase().contains("linux")) { + exec { + workingDir 'src/jni-crypto' + commandLine 'cargo', 'build', '--release', '--target=x86_64-unknown-linux-gnu', '--target-dir=../../build/jni-crypto' + } + } else if (System.getProperty("os.name").toLowerCase().contains("windows")) { + exec { + workingDir 'src/jni-crypto' + commandLine 'cargo', 'build', '--release', '--target=x86_64-pc-windows-gnu', '--target-dir=../../build/jni-crypto' + } + } // Does not cover solaris, just MacOS. See potential return types here: - // https://docs.gradle.org/current/javadoc/org/gradle/nativeplatform/platform/OperatingSystem.html - exec { - workingDir 'src/jni-crypto' - commandLine 'cargo', 'build', '--release', '--target-dir=../../build/jni-crypto' + // https://docs.gradle.org/current/javadoc/org/gradle/nativeplatform/platform/OperatingSystem.html + else { + exec { + workingDir 'src/jni-crypto' + commandLine 'cargo', 'build', '--release', '--target-dir=../../build/jni-crypto' + } } } } diff --git a/crypto/src/main/java/com/strategyobject/substrateclient/crypto/sr25519/Native.java b/crypto/src/main/java/com/strategyobject/substrateclient/crypto/sr25519/Native.java index ab9be027..fae76da1 100644 --- a/crypto/src/main/java/com/strategyobject/substrateclient/crypto/sr25519/Native.java +++ b/crypto/src/main/java/com/strategyobject/substrateclient/crypto/sr25519/Native.java @@ -56,7 +56,7 @@ private static String copyLibraryFromResourcesToTempDir() throws IOException { osDir = "x86_64-linux"; } else if (IS_LINUX && IS_AARCH64) { osDir = "aarch64-linux"; - } else if (IS_MAC) { // TODO 'MasOS support problem' + } else if (IS_MAC) { // TODO 'MacOS support problem' osDir = "macos"; } else { throw new RuntimeException("JNI library can't be loaded because OS wasn't detected as supported.");