blob: 7543d0180f5e8e500233d73f48a81f7efaf73656 [file] [log] [blame]
// +build android
package security
import (
"crypto/ecdsa"
"runtime"
"veyron/jni/runtimes/google/util"
"veyron2/security"
)
// #cgo LDFLAGS: -ljniwrapper
// #include "jni_wrapper.h"
import "C"
// newSigner creates an instance of security.Signer that uses the provided
// Java Signer as its underlying implementation.
// NOTE: Because CGO creates package-local types and because this method may be
// invoked from a different package, Java types are passed in an empty interface
// and then cast into their package local types.
func newSigner(env *C.JNIEnv, jSigner C.jobject) security.Signer {
// We cannot cache Java environments as they are only valid in the current
// thread. We can, however, cache the Java VM and obtain an environment
// from it in whatever thread happens to be running at the time.
var jVM *C.JavaVM
if status := C.GetJavaVM(env, &jVM); status != 0 {
panic("couldn't get Java VM from the (Java) environment")
}
// Reference Java Signer; it will be de-referenced when the Go Signer
// created below is garbage-collected (through the finalizer callback we
// setup just below).
jSigner = C.NewGlobalRef(env, jSigner)
s := &signer{
jVM: jVM,
jSigner: jSigner,
}
runtime.SetFinalizer(s, func(s *signer) {
envPtr, freeFunc := util.GetEnv(s.jVM)
env := (*C.JNIEnv)(envPtr)
defer freeFunc()
C.DeleteGlobalRef(env, s.jSigner)
})
return s
}
type signer struct {
jVM *C.JavaVM
jSigner C.jobject
}
func (s *signer) Sign(message []byte) (security.Signature, error) {
envPtr, freeFunc := util.GetEnv(s.jVM)
env := (*C.JNIEnv)(envPtr)
defer freeFunc()
signatureSign := util.ClassSign("com.veyron2.security.Signature")
jSig, err := util.CallObjectMethod(env, s.jSigner, "sign", []util.Sign{util.ArraySign(util.ByteSign)}, signatureSign, message)
if err != nil {
return security.Signature{}, err
}
jHash := util.CallObjectMethodOrCatch(env, jSig, "getHash", nil, util.ClassSign("com.veyron2.security.Hash"))
sig := security.Signature{
Hash: security.Hash(util.CallStringMethodOrCatch(env, jHash, "getValue", nil)),
R: util.CallByteArrayMethodOrCatch(env, jSig, "getR", nil),
S: util.CallByteArrayMethodOrCatch(env, jSig, "getS", nil),
}
return sig, nil
}
func (s *signer) PublicKey() *ecdsa.PublicKey {
envPtr, freeFunc := util.GetEnv(s.jVM)
env := (*C.JNIEnv)(envPtr)
defer freeFunc()
publicKeySign := util.ClassSign("java.security.interfaces.ECPublicKey")
jPublicKey := C.jobject(util.CallObjectMethodOrCatch(env, s.jSigner, "publicKey", nil, publicKeySign))
// Get the encoded version of the public key.
encoded := util.CallByteArrayMethodOrCatch(env, jPublicKey, "getEncoded", nil)
key, err := parsePKIXPublicKey(encoded)
if err != nil {
panic("couldn't parse Java ECDSA public key: " + err.Error())
}
return key
}