mirror of
https://github.com/openjdk/jdk.git
synced 2025-08-28 15:24:43 +02:00
8208583: Better management of internal KeyStore buffers
Reviewed-by: weijun
This commit is contained in:
parent
4555c28590
commit
7a791910c3
6 changed files with 105 additions and 67 deletions
|
@ -1,5 +1,5 @@
|
|||
/*
|
||||
* Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1997, 2018, 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
|
||||
|
@ -133,18 +133,20 @@ public abstract class JavaKeyStore extends KeyStoreSpi {
|
|||
throw new UnrecoverableKeyException("Password must not be null");
|
||||
}
|
||||
|
||||
KeyProtector keyProtector = new KeyProtector(password);
|
||||
byte[] passwordBytes = convertToBytes(password);
|
||||
KeyProtector keyProtector = new KeyProtector(passwordBytes);
|
||||
byte[] encrBytes = ((KeyEntry)entry).protectedPrivKey;
|
||||
EncryptedPrivateKeyInfo encrInfo;
|
||||
byte[] plain;
|
||||
try {
|
||||
encrInfo = new EncryptedPrivateKeyInfo(encrBytes);
|
||||
return keyProtector.recover(encrInfo);
|
||||
} catch (IOException ioe) {
|
||||
throw new UnrecoverableKeyException("Private key not stored as "
|
||||
+ "PKCS #8 "
|
||||
+ "EncryptedPrivateKeyInfo");
|
||||
} finally {
|
||||
Arrays.fill(passwordBytes, (byte) 0x00);
|
||||
}
|
||||
return keyProtector.recover(encrInfo);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -253,7 +255,8 @@ public abstract class JavaKeyStore extends KeyStoreSpi {
|
|||
Certificate[] chain)
|
||||
throws KeyStoreException
|
||||
{
|
||||
KeyProtector keyProtector = null;
|
||||
KeyProtector keyProtector;
|
||||
byte[] passwordBytes = null;
|
||||
|
||||
if (!(key instanceof java.security.PrivateKey)) {
|
||||
throw new KeyStoreException("Cannot store non-PrivateKeys");
|
||||
|
@ -264,7 +267,8 @@ public abstract class JavaKeyStore extends KeyStoreSpi {
|
|||
entry.date = new Date();
|
||||
|
||||
// Protect the encoding of the key
|
||||
keyProtector = new KeyProtector(password);
|
||||
passwordBytes = convertToBytes(password);
|
||||
keyProtector = new KeyProtector(passwordBytes);
|
||||
entry.protectedPrivKey = keyProtector.protect(key);
|
||||
|
||||
// clone the chain
|
||||
|
@ -280,7 +284,8 @@ public abstract class JavaKeyStore extends KeyStoreSpi {
|
|||
} catch (NoSuchAlgorithmException nsae) {
|
||||
throw new KeyStoreException("Key protection algorithm not found");
|
||||
} finally {
|
||||
keyProtector = null;
|
||||
if (passwordBytes != null)
|
||||
Arrays.fill(passwordBytes, (byte) 0x00);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -793,19 +798,27 @@ public abstract class JavaKeyStore extends KeyStoreSpi {
|
|||
private MessageDigest getPreKeyedHash(char[] password)
|
||||
throws NoSuchAlgorithmException, UnsupportedEncodingException
|
||||
{
|
||||
int i, j;
|
||||
|
||||
MessageDigest md = MessageDigest.getInstance("SHA");
|
||||
byte[] passwdBytes = convertToBytes(password);
|
||||
md.update(passwdBytes);
|
||||
Arrays.fill(passwdBytes, (byte) 0x00);
|
||||
md.update("Mighty Aphrodite".getBytes("UTF8"));
|
||||
return md;
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper method to convert char[] to byte[]
|
||||
*/
|
||||
|
||||
private byte[] convertToBytes(char[] password) {
|
||||
int i, j;
|
||||
byte[] passwdBytes = new byte[password.length * 2];
|
||||
for (i=0, j=0; i<password.length; i++) {
|
||||
passwdBytes[j++] = (byte)(password[i] >> 8);
|
||||
passwdBytes[j++] = (byte)password[i];
|
||||
}
|
||||
md.update(passwdBytes);
|
||||
for (i=0; i<passwdBytes.length; i++)
|
||||
passwdBytes[i] = 0;
|
||||
md.update("Mighty Aphrodite".getBytes("UTF8"));
|
||||
return md;
|
||||
return passwdBytes;
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
/*
|
||||
* Copyright (c) 1997, 2017, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1997, 2018, 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
|
||||
|
@ -35,7 +35,6 @@ import java.security.SecureRandom;
|
|||
import java.security.UnrecoverableKeyException;
|
||||
import java.util.*;
|
||||
|
||||
import jdk.internal.ref.CleanerFactory;
|
||||
import sun.security.pkcs.PKCS8Key;
|
||||
import sun.security.pkcs.EncryptedPrivateKeyInfo;
|
||||
import sun.security.x509.AlgorithmId;
|
||||
|
@ -120,32 +119,15 @@ final class KeyProtector {
|
|||
/**
|
||||
* Creates an instance of this class, and initializes it with the given
|
||||
* password.
|
||||
*
|
||||
* <p>The password is expected to be in printable ASCII.
|
||||
* Normal rules for good password selection apply: at least
|
||||
* seven characters, mixed case, with punctuation encouraged.
|
||||
* Phrases or words which are easily guessed, for example by
|
||||
* being found in dictionaries, are bad.
|
||||
*/
|
||||
public KeyProtector(char[] password)
|
||||
public KeyProtector(byte[] passwordBytes)
|
||||
throws NoSuchAlgorithmException
|
||||
{
|
||||
int i, j;
|
||||
|
||||
if (password == null) {
|
||||
if (passwordBytes == null) {
|
||||
throw new IllegalArgumentException("password can't be null");
|
||||
}
|
||||
md = MessageDigest.getInstance(DIGEST_ALG);
|
||||
// Convert password to byte array, so that it can be digested
|
||||
passwdBytes = new byte[password.length * 2];
|
||||
for (i=0, j=0; i<password.length; i++) {
|
||||
passwdBytes[j++] = (byte)(password[i] >> 8);
|
||||
passwdBytes[j++] = (byte)password[i];
|
||||
}
|
||||
// Use the cleaner to zero the password when no longer referenced
|
||||
final byte[] k = this.passwdBytes;
|
||||
CleanerFactory.cleaner().register(this,
|
||||
() -> java.util.Arrays.fill(k, (byte)0x00));
|
||||
this.passwdBytes = passwordBytes;
|
||||
}
|
||||
|
||||
/*
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue