blob: bcccda7212085583b6447267991657c52a462334 [file] [log] [blame]
/*
* Copyright (C) 2008 The Android Open Source Project
*
* 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.apache.harmony.xnet.provider.jsse;
import java.security.InvalidKeyException;
import java.security.InvalidParameterException;
import java.security.NoSuchAlgorithmException;
import java.security.PrivateKey;
import java.security.PublicKey;
import java.security.Signature;
import java.security.SignatureException;
import java.security.interfaces.DSAParams;
import java.security.interfaces.DSAPublicKey;
import java.security.interfaces.RSAPublicKey;
import java.util.HashMap;
import java.util.Locale;
import java.util.Map;
/**
* Implements the subset of the JDK Signature interface needed for
* signature verification using OpenSSL.
*/
public class OpenSSLSignature extends Signature {
private static Map<String,Class<? extends OpenSSLSignature>> jdkToOpenSsl
= new HashMap<String,Class<? extends OpenSSLSignature>>();
private static void register(String algorithm, Class implementation) {
jdkToOpenSsl.put(algorithm.toUpperCase(Locale.US), implementation);
}
private static Class lookup(String algorithm) {
return jdkToOpenSsl.get(algorithm.toUpperCase(Locale.US));
}
static {
// TODO Finish OpenSSLSignature implementation and move
// registration information to the OpenSSLProvider
register("MD5WithRSAEncryption", MD5RSA.class);
register("MD5WithRSA", MD5RSA.class);
register("MD5/RSA", MD5RSA.class);
register("1.2.840.113549.1.1.4", MD5RSA.class);
register("1.2.840.113549.2.5with1.2.840.113549.1.1.1", MD5RSA.class);
register("SHA1WithRSAEncryption", SHA1RSA.class);
register("SHA1WithRSA", SHA1RSA.class);
register("SHA1/RSA", SHA1RSA.class);
register("SHA-1/RSA", SHA1RSA.class);
register("1.2.840.113549.1.1.5", SHA1RSA.class);
register("1.3.14.3.2.26with1.2.840.113549.1.1.1", SHA1RSA.class);
register("1.3.14.3.2.26with1.2.840.113549.1.1.5", SHA1RSA.class);
register("1.3.14.3.2.29", SHA1RSA.class);
register("SHA256WithRSAEncryption", SHA256RSA.class);
register("SHA256WithRSA", SHA256RSA.class);
register("1.2.840.113549.1.1.11", SHA256RSA.class);
register("SHA384WithRSAEncryption", SHA384RSA.class);
register("SHA384WithRSA", SHA384RSA.class);
register("1.2.840.113549.1.1.12", SHA384RSA.class);
register("SHA512WithRSAEncryption", SHA512RSA.class);
register("SHA512WithRSA", SHA512RSA.class);
register("1.2.840.113549.1.1.13", SHA512RSA.class);
register("SHA1withDSA", SHA1DSA.class);
register("SHA/DSA", SHA1DSA.class);
register("DSA", SHA1DSA.class);
register("1.3.14.3.2.26with1.2.840.10040.4.1", SHA1DSA.class);
register("1.3.14.3.2.26with1.2.840.10040.4.3", SHA1DSA.class);
register("DSAWithSHA1", SHA1DSA.class);
register("1.2.840.10040.4.3", SHA1DSA.class);
}
/**
* Holds a pointer to the native message digest context.
*/
private int ctx;
/**
* Holds a pointer to the native DSA key.
*/
private int dsa;
/**
* Holds a pointer to the native RSA key.
*/
private int rsa;
/**
* Holds the OpenSSL name of the algorithm (lower case, no dashes).
*/
private final String evpAlgorithm;
/**
* Holds a dummy buffer for writing single bytes to the digest.
*/
private final byte[] singleByte = new byte[1];
/**
* Creates a new OpenSSLSignature instance for the given algorithm name.
*
* @param algorithm The name of the algorithm, e.g. "SHA1WithRSA".
*
* @return The new OpenSSLSignature instance.
*
* @throws RuntimeException In case of problems.
*/
public static OpenSSLSignature getInstance(String algorithm) throws NoSuchAlgorithmException {
// System.out.println("getInstance() invoked with " + algorithm);
Class <? extends OpenSSLSignature> clazz = lookup(algorithm);
if (clazz == null) {
throw new NoSuchAlgorithmException(algorithm);
}
try {
return clazz.newInstance();
} catch (InstantiationException e) {
throw new NoSuchAlgorithmException(algorithm, e);
} catch (IllegalAccessException e) {
throw new NoSuchAlgorithmException(algorithm, e);
}
}
/**
* Creates a new OpenSSLSignature instance for the given algorithm name.
*
* @param algorithm OpenSSL name of the algorithm, e.g. "RSA-SHA1".
*/
private OpenSSLSignature(String algorithm) throws NoSuchAlgorithmException {
super(algorithm);
// We don't support MD2
if ("RSA-MD2".equals(algorithm)) {
throw new NoSuchAlgorithmException(algorithm);
}
this.evpAlgorithm = algorithm;
}
@Override
protected void engineUpdate(byte input) {
singleByte[0] = input;
engineUpdate(singleByte, 0, 1);
}
@Override
protected void engineUpdate(byte[] input, int offset, int len) {
if (state == SIGN) {
throw new UnsupportedOperationException();
} else {
NativeCrypto.EVP_VerifyUpdate(ctx, input, offset, len);
}
}
@Override
protected Object engineGetParameter(String param) throws InvalidParameterException {
return null;
}
@Override
protected void engineInitSign(PrivateKey privateKey) throws InvalidKeyException {
throw new UnsupportedOperationException();
}
@Override
protected void engineInitVerify(PublicKey publicKey) throws InvalidKeyException {
// System.out.println("engineInitVerify() invoked with "
// + publicKey.getClass().getCanonicalName());
if (publicKey instanceof DSAPublicKey) {
try {
DSAPublicKey dsaPublicKey = (DSAPublicKey)publicKey;
DSAParams dsaParams = dsaPublicKey.getParams();
dsa = NativeCrypto.EVP_PKEY_new_DSA(dsaParams.getP().toByteArray(),
dsaParams.getQ().toByteArray(), dsaParams.getG().toByteArray(),
dsaPublicKey.getY().toByteArray(), null);
} catch (Exception e) {
throw new InvalidKeyException(e);
}
} else if (publicKey instanceof RSAPublicKey) {
try {
RSAPublicKey rsaPublicKey = (RSAPublicKey)publicKey;
rsa = NativeCrypto.EVP_PKEY_new_RSA(rsaPublicKey.getModulus().toByteArray(),
rsaPublicKey.getPublicExponent().toByteArray(), null, null, null);
} catch (Exception e) {
throw new InvalidKeyException(e);
}
} else {
throw new InvalidKeyException("Need DSA or RSA public key");
}
try {
ctx = NativeCrypto.EVP_VerifyInit(evpAlgorithm);
} catch (Exception ex) {
throw new RuntimeException(ex);
}
}
@Override
protected void engineSetParameter(String param, Object value) throws InvalidParameterException {
}
@Override
protected byte[] engineSign() throws SignatureException {
throw new UnsupportedOperationException();
}
@Override
protected boolean engineVerify(byte[] sigBytes) throws SignatureException {
int handle = (rsa != 0) ? rsa : dsa;
if (handle == 0) {
// This can't actually happen, but you never know...
throw new SignatureException("Need DSA or RSA public key");
}
try {
int result = NativeCrypto.EVP_VerifyFinal(ctx, sigBytes, 0, sigBytes.length, handle);
return result == 1;
} catch (Exception ex) {
throw new SignatureException(ex);
}
}
@Override protected void finalize() throws Throwable {
try {
if (dsa != 0) {
NativeCrypto.EVP_PKEY_free(dsa);
}
if (rsa != 0) {
NativeCrypto.EVP_PKEY_free(rsa);
}
if (ctx != 0) {
NativeCrypto.EVP_MD_CTX_destroy(ctx);
}
} finally {
super.finalize();
}
}
public static final class MD5RSA extends OpenSSLSignature {
public MD5RSA() throws NoSuchAlgorithmException {
super("RSA-MD5");
}
}
public static final class SHA1RSA extends OpenSSLSignature {
public SHA1RSA() throws NoSuchAlgorithmException {
super("RSA-SHA1");
}
}
public static final class SHA256RSA extends OpenSSLSignature {
public SHA256RSA() throws NoSuchAlgorithmException {
super("RSA-SHA256");
}
}
public static final class SHA384RSA extends OpenSSLSignature {
public SHA384RSA() throws NoSuchAlgorithmException {
super("RSA-SHA384");
}
}
public static final class SHA512RSA extends OpenSSLSignature {
public SHA512RSA() throws NoSuchAlgorithmException {
super("RSA-SHA512");
}
}
public static final class SHA1DSA extends OpenSSLSignature {
public SHA1DSA() throws NoSuchAlgorithmException {
super("DSA-SHA1");
}
}
}