/** * Copyright 2013 Google Inc. * * 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 org.devcoin; import java.nio.ByteBuffer; import java.nio.ByteOrder; import com.google.common.base.Preconditions; /** * <p>This class holds native methods to handle ECDSA verification.</p> * * <p>You can find an example library that can be used for this at https://github.com/sipa/secp256k1</p> * * <p>To build secp256k1 for use with bitcoinj, run `./configure` and `make libjavasecp256k1.so` then copy * libjavasecp256k1.so to your system library path or point the JVM to the folder containing it with -Djava.library.path * </p> */ public class NativeSecp256k1 { public static boolean enabled = false; static { try { System.loadLibrary("javasecp256k1"); } catch (UnsatisfiedLinkError e) { enabled = false; } } private static ThreadLocal<ByteBuffer> nativeECDSABuffer = new ThreadLocal<ByteBuffer>(); /** * Verifies the given secp256k1 signature in native code. * Calling when enabled == false is undefined (probably library not loaded) * * @param data The data which was signed, must be exactly 32 bytes * @param signature The signature * @param pub The public key which did the signing */ public static boolean verify(byte[] data, byte[] signature, byte[] pub) { Preconditions.checkArgument(data.length == 32 && signature.length <= 520 && pub.length <= 520); ByteBuffer byteBuff = nativeECDSABuffer.get(); if (byteBuff == null) { byteBuff = ByteBuffer.allocateDirect(32 + 8 + 520 + 520); byteBuff.order(ByteOrder.nativeOrder()); nativeECDSABuffer.set(byteBuff); } byteBuff.rewind(); byteBuff.put(data); byteBuff.putInt(signature.length); byteBuff.putInt(pub.length); byteBuff.put(signature); byteBuff.put(pub); return secp256k1_ecdsa_verify(byteBuff) == 1; } /** * @param byteBuff signature format is byte[32] data, * native-endian int signatureLength, native-endian int pubkeyLength, * byte[signatureLength] signature, byte[pubkeyLength] pub * @returns 1 for valid signature, anything else for invalid */ private static native int secp256k1_ecdsa_verify(ByteBuffer byteBuff); }