5 Commits

Author SHA1 Message Date
sstone
7a33c81c01 Use Kotlin 1.4 2020-08-18 09:51:20 +02:00
Romain Boisselle
2615187b87 Kotlin 1.4.0-rc (#10)
* Move to kotlin 1.4.0-rc

* Removing broken WSL bash

see https://github.com/actions/virtual-environments/issues/50

Co-authored-by: Salomon BRYS <salomon@kodein.net>
2020-08-02 20:08:02 +02:00
Salomon BRYS
dbf8301f34 Android loading may fallback to standard JVM (#9)
* Android loading may fallback to standard JVM

* Java code must target JVM 1.8 in order to be compatible Android.

Co-authored-by: Salomon BRYS <salomon@kodein.net>
2020-07-16 22:21:30 +02:00
Fabrice Drouin
0cc4c251f9 Fixup for #6 (native signature format detection) (#8)
* Fixup for #6

Behaviour was changed in the JNI wapper but not in Kotlin native code.

* Set version to 0.2.1-1.4-M3
2020-07-09 20:16:39 +02:00
Salomon BRYS
eeac972785 Update README.md (#7)
The package in bintray has lost the `-kmp`.
The badge needs to be updated to avoid image not found.
2020-07-09 17:54:22 +02:00
12 changed files with 87 additions and 35 deletions

View File

@@ -37,7 +37,9 @@ jobs:
run: brew install automake
- name: Set up shell
if: matrix.os == 'windows-latest'
run: echo ::add-path::C:\msys64\usr\bin\
run: |
echo ::add-path::C:\msys64\usr\bin\
rm.exe "C:/WINDOWS/system32/bash.EXE"
- name: Check JVM
shell: bash
run: ./gradlew jvmTest

View File

@@ -45,7 +45,9 @@ jobs:
run: brew install automake
- name: Set up shell
if: matrix.os == 'windows-latest'
run: echo ::add-path::C:\msys64\usr\bin\
run: |
echo ::add-path::C:\msys64\usr\bin\
rm.exe "C:/WINDOWS/system32/bash.EXE"
- name: Check JVM
shell: bash
run: ./gradlew jvmTest

View File

@@ -51,7 +51,9 @@ jobs:
run: brew install automake
- name: Set up shell
if: matrix.os == 'windows-latest'
run: echo ::add-path::C:\msys64\usr\bin\
run: |
echo ::add-path::C:\msys64\usr\bin\
rm.exe "C:/WINDOWS/system32/bash.EXE"
- name: Check JVM
shell: bash
run: ./gradlew jvmTest

View File

@@ -4,7 +4,7 @@ Kotlin/Multiplatform wrapper for Bitcoin Core's secp256k1 library. Targets: JVM,
## Installation
[ ![Download](https://api.bintray.com/packages/acinq/libs/secp256k1-kmp/images/download.svg) ](https://bintray.com/acinq/libs/secp256k1-kmp/)
[ ![Download](https://api.bintray.com/packages/acinq/libs/secp256k1/images/download.svg) ](https://bintray.com/acinq/libs/secp256k1-kmp/)
First, you need to add the ACINQ libraries repository:

View File

@@ -6,7 +6,7 @@ import org.apache.http.impl.auth.BasicScheme
import org.apache.http.auth.UsernamePasswordCredentials
plugins {
kotlin("multiplatform") version "1.4-M3"
kotlin("multiplatform") version "1.4.0"
`maven-publish`
}
@@ -24,7 +24,7 @@ buildscript {
allprojects {
group = "fr.acinq.secp256k1"
version = "0.2.0-1.4-M3"
version = "0.3.0-1.4"
repositories {
jcenter()

View File

@@ -0,0 +1,31 @@
package fr.acinq.secp256k1.jni
import android.util.Log
import fr.acinq.secp256k1.Secp256k1
import fr.acinq.secp256k1.NativeSecp256k1
import java.util.*
public object NativeSecp256k1AndroidLoader {
@JvmStatic
@Synchronized
@Throws(Exception::class)
fun load(): Secp256k1 {
try {
System.loadLibrary("secp256k1-jni")
return NativeSecp256k1
} catch (ex: UnsatisfiedLinkError) {
// Purposefully not using Android Log
println("Could not load Android Secp256k1. Trying to extract JVM platform specific version.")
try {
val cls = Class.forName("fr.acinq.secp256k1.jni.NativeSecp256k1JvmLoader")
val load = cls.getMethod("load")
return load.invoke(null) as Secp256k1
} catch (_: ClassNotFoundException) {
throw ex
}
}
}
}

View File

@@ -1,16 +0,0 @@
package fr.acinq.secp256k1.jni
import fr.acinq.secp256k1.Secp256k1
import fr.acinq.secp256k1.NativeSecp256k1
public object NativeSecp256k1Loader {
@JvmStatic
@Synchronized
@Throws(Exception::class)
fun load(): Secp256k1 {
System.loadLibrary("secp256k1-jni")
return NativeSecp256k1
}
}

View File

@@ -3,6 +3,11 @@ plugins {
`maven-publish`
}
java {
sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8
}
kotlin {
explicitApi()
}

View File

@@ -14,7 +14,7 @@ import java.util.*
*
* @author leo
*/
public object NativeSecp256k1Loader {
public object NativeSecp256k1JvmLoader {
private var extracted = false
/**
@@ -106,7 +106,7 @@ public object NativeSecp256k1Loader {
val extractedLckFile = File(targetDirectory, extractedLckFileName)
return try {
// Extract a native library file into the target directory
val reader = NativeSecp256k1Loader::class.java.getResourceAsStream(libPath)
val reader = NativeSecp256k1JvmLoader::class.java.getResourceAsStream(libPath)
if (!extractedLckFile.exists()) {
FileOutputStream(extractedLckFile).close()
}
@@ -132,7 +132,7 @@ public object NativeSecp256k1Loader {
extractedLibFile.setExecutable(true)
// Check whether the contents are properly copied from the resource folder
NativeSecp256k1Loader::class.java.getResourceAsStream(libPath).use { nativeIn ->
NativeSecp256k1JvmLoader::class.java.getResourceAsStream(libPath).use { nativeIn ->
FileInputStream(extractedLibFile).use { extractedLibIn ->
if (!nativeIn.contentsEquals(extractedLibIn)) {
throw RuntimeException(
@@ -196,9 +196,9 @@ public object NativeSecp256k1Loader {
}
// Load the os-dependent library from the jar file
val packagePath = NativeSecp256k1Loader::class.java.getPackage().name.replace("\\.".toRegex(), "/")
val packagePath = NativeSecp256k1JvmLoader::class.java.getPackage().name.replace("\\.".toRegex(), "/")
val embeddedLibraryPath = "/$packagePath/native/${OSInfo.nativeSuffix}"
val hasNativeLib = NativeSecp256k1Loader::class.java.getResource("$embeddedLibraryPath/$libraryName") != null
val hasNativeLib = NativeSecp256k1JvmLoader::class.java.getResource("$embeddedLibraryPath/$libraryName") != null
if (!hasNativeLib) {
error("No native library found: at $embeddedLibraryPath/$libraryName")
}

View File

@@ -16,15 +16,20 @@
package fr.acinq.secp256k1
import java.lang.IllegalStateException
import java.util.*
internal actual fun getSecpk256k1(): Secp256k1 {
private fun tryLoad(platform: String): Secp256k1? {
try {
val cls = Class.forName("fr.acinq.secp256k1.jni.NativeSecp256k1Loader")
val cls = Class.forName("fr.acinq.secp256k1.jni.NativeSecp256k1${platform.capitalize(Locale.ROOT)}Loader")
val load = cls.getMethod("load")
return load.invoke(null) as Secp256k1
} catch (ex: ClassNotFoundException) {
throw IllegalStateException("Could not load native Secp256k1 JNI library. Have you added the JNI dependency?", ex)
return null
}
}
internal actual fun getSecpk256k1(): Secp256k1 =
tryLoad("android")
?: tryLoad("jvm")
?: error("Could not load native Secp256k1 JNI library. Have you added the JNI dependency?")

View File

@@ -18,10 +18,10 @@ public object Secp256k1Native : Secp256k1 {
val sig = alloc<secp256k1_ecdsa_signature>()
val nativeBytes = toNat(input)
val result = when (input.size) {
64 -> secp256k1_ecdsa_signature_parse_compact(ctx, sig.ptr, nativeBytes)
in 70..73 -> secp256k1_ecdsa_signature_parse_der(ctx, sig.ptr, nativeBytes, input.size.convert())
else -> throw Secp256k1Exception("Unknown signature format")
val result = when {
input.size == 64 -> secp256k1_ecdsa_signature_parse_compact(ctx, sig.ptr, nativeBytes)
input.size < 64 -> throw Secp256k1Exception("Unknown signature format")
else -> secp256k1_ecdsa_signature_parse_der(ctx, sig.ptr, nativeBytes, input.size.convert())
}
result.requireSuccess("cannot parse signature (size = ${input.size} sig = ${Hex.encode(input)}")
return sig

View File

@@ -1,5 +1,6 @@
package fr.acinq.secp256k1
import kotlin.random.Random
import kotlin.test.*
@@ -265,4 +266,24 @@ class Secp256k1Test {
Hex.encode(der).toUpperCase(),
)
}
@Test
fun testFormatConversion() {
val random = Random.Default
fun randomBytes(length: Int): ByteArray {
val buffer = ByteArray(length)
random.nextBytes(buffer)
return buffer
}
repeat(200) {
val priv = randomBytes(32)
val pub = Secp256k1.pubkeyCreate(priv)
val data = randomBytes(32)
val sig = Secp256k1.sign(data, priv)
val der = Secp256k1.compact2der(sig)
Secp256k1.verify(der, data, pub)
}
}
}