提交 3d8c4fbc 编写于 作者: I igerasim

8163896: Finalizing one key of a KeyPair invalidates the other key

Reviewed-by: coffeys, vinnie
上级 702718aa
/* /*
* Copyright (c) 2005, 2013, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2005, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -30,7 +30,6 @@ import sun.security.util.Length; ...@@ -30,7 +30,6 @@ import sun.security.util.Length;
/** /**
* The handle for an RSA or DSA key using the Microsoft Crypto API. * The handle for an RSA or DSA key using the Microsoft Crypto API.
* *
* @see DSAPrivateKey
* @see RSAPrivateKey * @see RSAPrivateKey
* @see RSAPublicKey * @see RSAPublicKey
* *
...@@ -41,9 +40,35 @@ abstract class Key implements java.security.Key, Length ...@@ -41,9 +40,35 @@ abstract class Key implements java.security.Key, Length
{ {
private static final long serialVersionUID = -1088859394025049194L; private static final long serialVersionUID = -1088859394025049194L;
// Native handle static class NativeHandles {
protected long hCryptProv = 0; long hCryptProv = 0;
protected long hCryptKey = 0; long hCryptKey = 0;
public NativeHandles(long hCryptProv, long hCryptKey) {
this.hCryptProv = hCryptProv;
this.hCryptKey = hCryptKey;
}
/**
* Finalization method
*/
protected void finalize() throws Throwable
{
try {
synchronized(this)
{
cleanUp(hCryptProv, hCryptKey);
hCryptProv = 0;
hCryptKey = 0;
}
} finally {
super.finalize();
}
}
}
protected NativeHandles handles;
// Key length // Key length
protected int keyLength = 0; protected int keyLength = 0;
...@@ -51,31 +76,12 @@ abstract class Key implements java.security.Key, Length ...@@ -51,31 +76,12 @@ abstract class Key implements java.security.Key, Length
/** /**
* Construct a Key object. * Construct a Key object.
*/ */
protected Key(long hCryptProv, long hCryptKey, int keyLength) protected Key(NativeHandles handles, int keyLength)
{ {
this.hCryptProv = hCryptProv; this.handles = handles;
this.hCryptKey = hCryptKey;
this.keyLength = keyLength; this.keyLength = keyLength;
} }
/**
* Finalization method
*/
protected void finalize() throws Throwable
{
try {
synchronized(this)
{
cleanUp(hCryptProv, hCryptKey);
hCryptProv = 0;
hCryptKey = 0;
}
} finally {
super.finalize();
}
}
/** /**
* Native method to cleanup the key handle. * Native method to cleanup the key handle.
*/ */
...@@ -96,7 +102,7 @@ abstract class Key implements java.security.Key, Length ...@@ -96,7 +102,7 @@ abstract class Key implements java.security.Key, Length
*/ */
public long getHCryptKey() public long getHCryptKey()
{ {
return hCryptKey; return handles.hCryptKey;
} }
/** /**
...@@ -104,12 +110,12 @@ abstract class Key implements java.security.Key, Length ...@@ -104,12 +110,12 @@ abstract class Key implements java.security.Key, Length
*/ */
public long getHCryptProvider() public long getHCryptProvider()
{ {
return hCryptProv; return handles.hCryptProv;
} }
/** /**
* Returns the standard algorithm name for this key. For * Returns the standard algorithm name for this key. For
* example, "DSA" would indicate that this key is a DSA key. * example, "RSA" would indicate that this key is a RSA key.
* See Appendix A in the <a href= * See Appendix A in the <a href=
* "../../../guide/security/CryptoSpec.html#AppA"> * "../../../guide/security/CryptoSpec.html#AppA">
* Java Cryptography Architecture API Specification &amp; Reference </a> * Java Cryptography Architecture API Specification &amp; Reference </a>
......
...@@ -169,7 +169,7 @@ abstract class KeyStore extends KeyStoreSpi { ...@@ -169,7 +169,7 @@ abstract class KeyStore extends KeyStoreSpi {
} }
certChain = chain; certChain = chain;
} }
}; }
/* /*
* An X.509 certificate factory. * An X.509 certificate factory.
...@@ -800,7 +800,8 @@ abstract class KeyStore extends KeyStoreSpi { ...@@ -800,7 +800,8 @@ abstract class KeyStore extends KeyStoreSpi {
} }
storeWithUniqueAlias(alias, new KeyEntry(alias, storeWithUniqueAlias(alias, new KeyEntry(alias,
new RSAPrivateKey(hCryptProv, hCryptKey, keyLength), new RSAPrivateKey(new Key.NativeHandles(hCryptProv,
hCryptKey), keyLength),
certChain)); certChain));
} }
catch (Throwable e) catch (Throwable e)
...@@ -856,7 +857,6 @@ abstract class KeyStore extends KeyStoreSpi { ...@@ -856,7 +857,6 @@ abstract class KeyStore extends KeyStoreSpi {
* Load keys and/or certificates from keystore into Collection. * Load keys and/or certificates from keystore into Collection.
* *
* @param name Name of keystore. * @param name Name of keystore.
* @param entries Collection of key/certificate.
*/ */
private native void loadKeysOrCertificateChains(String name) private native void loadKeysOrCertificateChains(String name)
throws KeyStoreException; throws KeyStoreException;
......
/* /*
* Copyright (c) 2005, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2005, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -41,8 +41,9 @@ class RSAKeyPair { ...@@ -41,8 +41,9 @@ class RSAKeyPair {
*/ */
RSAKeyPair(long hCryptProv, long hCryptKey, int keyLength) RSAKeyPair(long hCryptProv, long hCryptKey, int keyLength)
{ {
privateKey = new RSAPrivateKey(hCryptProv, hCryptKey, keyLength); Key.NativeHandles handles = new Key.NativeHandles(hCryptProv, hCryptKey);
publicKey = new RSAPublicKey(hCryptProv, hCryptKey, keyLength); privateKey = new RSAPrivateKey(handles, keyLength);
publicKey = new RSAPublicKey(handles, keyLength);
} }
public RSAPrivateKey getPrivate() { public RSAPrivateKey getPrivate() {
......
/* /*
* Copyright (c) 2005, 2011, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2005, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -42,7 +42,15 @@ class RSAPrivateKey extends Key implements PrivateKey ...@@ -42,7 +42,15 @@ class RSAPrivateKey extends Key implements PrivateKey
*/ */
RSAPrivateKey(long hCryptProv, long hCryptKey, int keyLength) RSAPrivateKey(long hCryptProv, long hCryptKey, int keyLength)
{ {
super(hCryptProv, hCryptKey, keyLength); super(new NativeHandles(hCryptProv, hCryptKey), keyLength);
}
/**
* Construct an RSAPrivateKey object.
*/
RSAPrivateKey(NativeHandles handles, int keyLength)
{
super(handles, keyLength);
} }
/** /**
...@@ -63,8 +71,8 @@ class RSAPrivateKey extends Key implements PrivateKey ...@@ -63,8 +71,8 @@ class RSAPrivateKey extends Key implements PrivateKey
public String toString() public String toString()
{ {
return "RSAPrivateKey [size=" + keyLength + " bits, type=" + return "RSAPrivateKey [size=" + keyLength + " bits, type=" +
getKeyType(hCryptKey) + ", container=" + getKeyType(handles.hCryptKey) + ", container=" +
getContainerName(hCryptProv) + "]"; getContainerName(handles.hCryptProv) + "]";
} }
// This class is not serializable // This class is not serializable
......
/* /*
* Copyright (c) 2005, 2011, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2005, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -51,7 +51,15 @@ class RSAPublicKey extends Key implements java.security.interfaces.RSAPublicKey ...@@ -51,7 +51,15 @@ class RSAPublicKey extends Key implements java.security.interfaces.RSAPublicKey
*/ */
RSAPublicKey(long hCryptProv, long hCryptKey, int keyLength) RSAPublicKey(long hCryptProv, long hCryptKey, int keyLength)
{ {
super(hCryptProv, hCryptKey, keyLength); super(new NativeHandles(hCryptProv, hCryptKey), keyLength);
}
/**
* Construct an RSAPublicKey object.
*/
RSAPublicKey(NativeHandles handles, int keyLength)
{
super(handles, keyLength);
} }
/** /**
...@@ -77,8 +85,8 @@ class RSAPublicKey extends Key implements java.security.interfaces.RSAPublicKey ...@@ -77,8 +85,8 @@ class RSAPublicKey extends Key implements java.security.interfaces.RSAPublicKey
StringBuffer sb = new StringBuffer(); StringBuffer sb = new StringBuffer();
sb.append("RSAPublicKey [size=").append(keyLength) sb.append("RSAPublicKey [size=").append(keyLength)
.append(" bits, type=").append(getKeyType(hCryptKey)) .append(" bits, type=").append(getKeyType(handles.hCryptKey))
.append(", container=").append(getContainerName(hCryptProv)) .append(", container=").append(getContainerName(handles.hCryptProv))
.append("]\n modulus: ").append(getModulus()) .append("]\n modulus: ").append(getModulus())
.append("\n public exponent: ").append(getPublicExponent()); .append("\n public exponent: ").append(getPublicExponent());
...@@ -93,7 +101,7 @@ class RSAPublicKey extends Key implements java.security.interfaces.RSAPublicKey ...@@ -93,7 +101,7 @@ class RSAPublicKey extends Key implements java.security.interfaces.RSAPublicKey
if (exponent == null) { if (exponent == null) {
try { try {
publicKeyBlob = getPublicKeyBlob(hCryptKey); publicKeyBlob = getPublicKeyBlob(handles.hCryptKey);
exponent = new BigInteger(1, getExponent(publicKeyBlob)); exponent = new BigInteger(1, getExponent(publicKeyBlob));
} catch (KeyException e) { } catch (KeyException e) {
...@@ -112,7 +120,7 @@ class RSAPublicKey extends Key implements java.security.interfaces.RSAPublicKey ...@@ -112,7 +120,7 @@ class RSAPublicKey extends Key implements java.security.interfaces.RSAPublicKey
if (modulus == null) { if (modulus == null) {
try { try {
publicKeyBlob = getPublicKeyBlob(hCryptKey); publicKeyBlob = getPublicKeyBlob(handles.hCryptKey);
modulus = new BigInteger(1, getModulus(publicKeyBlob)); modulus = new BigInteger(1, getModulus(publicKeyBlob));
} catch (KeyException e) { } catch (KeyException e) {
......
/*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @test
* @bug 8163896
* @summary Finalizing one key of a KeyPair invalidates the other key
*/
import java.security.Key;
import java.security.KeyPair;
import java.security.KeyPairGenerator;
import java.security.NoSuchAlgorithmException;
import java.security.Provider;
import java.security.ProviderException;
import java.security.Security;
import java.util.function.Consumer;
import java.util.ArrayList;
import java.util.List;
public class FinalizeHalf {
static int failures = 0;
public static void main(String[] args) throws Throwable {
List<Consumer<Key>> methods = new ArrayList<>();
methods.add((Key k) -> k.getEncoded());
methods.add((Key k) -> k.toString());
for (String algo : new String[] {"DiffieHellman", "DSA", "RSA"}) {
for (Provider provider : Security.getProviders()) {
for (boolean priv : new boolean[] {true, false}) {
for (Consumer<Key> method : methods) {
test(algo, provider, priv, method);
}
}
}
}
if (failures > 0) {
throw new RuntimeException(failures + " test(s) failed.");
}
}
static void test(String algo, Provider provider, boolean priv,
Consumer<Key> method) throws Exception {
KeyPairGenerator generator;
try {
generator = KeyPairGenerator.getInstance(algo, provider);
} catch (NoSuchAlgorithmException nsae) {
return;
}
System.out.println("Checking " + provider.getName() + ", " + algo);
KeyPair pair = generator.generateKeyPair();
Key key = priv ? pair.getPrivate() : pair.getPublic();
pair = null;
for (int i = 0; i < 32; ++i) {
System.gc();
}
try {
method.accept(key);
} catch (ProviderException pe) {
failures++;
}
}
}
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册