8220513: Wrapper Key may get deleted when closing sessions in SunPKCS11 crypto provider
Do not close the session holding the Wrapper Key while in use. Delete the Wrapper Key when no longer needed. Reviewed-by: valeriep
This commit is contained in:
parent
73d7e8f86c
commit
0d35ef38e6
@ -1137,20 +1137,79 @@ final class NativeKeyHolder {
|
|||||||
|
|
||||||
private static long nativeKeyWrapperKeyID = 0;
|
private static long nativeKeyWrapperKeyID = 0;
|
||||||
private static CK_MECHANISM nativeKeyWrapperMechanism = null;
|
private static CK_MECHANISM nativeKeyWrapperMechanism = null;
|
||||||
|
private static long nativeKeyWrapperRefCount = 0;
|
||||||
|
private static Session nativeKeyWrapperSession = null;
|
||||||
|
|
||||||
private final P11Key p11Key;
|
private final P11Key p11Key;
|
||||||
private final byte[] nativeKeyInfo;
|
private final byte[] nativeKeyInfo;
|
||||||
|
private boolean wrapperKeyUsed;
|
||||||
|
|
||||||
// destroyed and recreated when refCount toggles to 1
|
// destroyed and recreated when refCount toggles to 1
|
||||||
private long keyID;
|
private long keyID;
|
||||||
|
|
||||||
private boolean isTokenObject;
|
|
||||||
|
|
||||||
// phantom reference notification clean up for session keys
|
// phantom reference notification clean up for session keys
|
||||||
private SessionKeyRef ref;
|
private SessionKeyRef ref;
|
||||||
|
|
||||||
private int refCount;
|
private int refCount;
|
||||||
|
|
||||||
|
private static void createNativeKeyWrapper(Token token)
|
||||||
|
throws PKCS11Exception {
|
||||||
|
assert(nativeKeyWrapperKeyID == 0);
|
||||||
|
assert(nativeKeyWrapperRefCount == 0);
|
||||||
|
assert(nativeKeyWrapperSession == null);
|
||||||
|
// Create a global wrapping/unwrapping key
|
||||||
|
CK_ATTRIBUTE[] wrappingAttributes = token.getAttributes(O_GENERATE,
|
||||||
|
CKO_SECRET_KEY, CKK_AES, new CK_ATTRIBUTE[] {
|
||||||
|
new CK_ATTRIBUTE(CKA_CLASS, CKO_SECRET_KEY),
|
||||||
|
new CK_ATTRIBUTE(CKA_VALUE_LEN, 256 >> 3)});
|
||||||
|
Session s = null;
|
||||||
|
try {
|
||||||
|
s = token.getObjSession();
|
||||||
|
nativeKeyWrapperKeyID = token.p11.C_GenerateKey(
|
||||||
|
s.id(), new CK_MECHANISM(CKM_AES_KEY_GEN),
|
||||||
|
wrappingAttributes);
|
||||||
|
nativeKeyWrapperSession = s;
|
||||||
|
nativeKeyWrapperSession.addObject();
|
||||||
|
byte[] iv = new byte[16];
|
||||||
|
JCAUtil.getSecureRandom().nextBytes(iv);
|
||||||
|
nativeKeyWrapperMechanism = new CK_MECHANISM(CKM_AES_CBC_PAD, iv);
|
||||||
|
} catch (PKCS11Exception e) {
|
||||||
|
// best effort
|
||||||
|
} finally {
|
||||||
|
token.releaseSession(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void deleteNativeKeyWrapper() {
|
||||||
|
Token token = nativeKeyWrapperSession.token;
|
||||||
|
if (token.isValid()) {
|
||||||
|
Session s = null;
|
||||||
|
try {
|
||||||
|
s = token.getOpSession();
|
||||||
|
token.p11.C_DestroyObject(s.id(), nativeKeyWrapperKeyID);
|
||||||
|
nativeKeyWrapperSession.removeObject();
|
||||||
|
} catch (PKCS11Exception e) {
|
||||||
|
// best effort
|
||||||
|
} finally {
|
||||||
|
token.releaseSession(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
nativeKeyWrapperKeyID = 0;
|
||||||
|
nativeKeyWrapperMechanism = null;
|
||||||
|
nativeKeyWrapperSession = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void decWrapperKeyRef() {
|
||||||
|
synchronized(NativeKeyHolder.class) {
|
||||||
|
assert(nativeKeyWrapperKeyID != 0);
|
||||||
|
assert(nativeKeyWrapperRefCount > 0);
|
||||||
|
nativeKeyWrapperRefCount--;
|
||||||
|
if (nativeKeyWrapperRefCount == 0) {
|
||||||
|
deleteNativeKeyWrapper();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
NativeKeyHolder(P11Key p11Key, long keyID, Session keySession,
|
NativeKeyHolder(P11Key p11Key, long keyID, Session keySession,
|
||||||
boolean extractKeyInfo, boolean isTokenObject) {
|
boolean extractKeyInfo, boolean isTokenObject) {
|
||||||
this.p11Key = p11Key;
|
this.p11Key = p11Key;
|
||||||
@ -1160,35 +1219,23 @@ final class NativeKeyHolder {
|
|||||||
if (isTokenObject) {
|
if (isTokenObject) {
|
||||||
this.ref = null;
|
this.ref = null;
|
||||||
} else {
|
} else {
|
||||||
this.ref = new SessionKeyRef(p11Key, keyID, keySession);
|
|
||||||
|
|
||||||
// Try extracting key info, if any error, disable it
|
// Try extracting key info, if any error, disable it
|
||||||
Token token = p11Key.token;
|
Token token = p11Key.token;
|
||||||
if (extractKeyInfo) {
|
if (extractKeyInfo) {
|
||||||
try {
|
try {
|
||||||
if (p11Key.sensitive && nativeKeyWrapperKeyID == 0) {
|
if (p11Key.sensitive) {
|
||||||
|
// p11Key native key information has to be wrapped
|
||||||
synchronized(NativeKeyHolder.class) {
|
synchronized(NativeKeyHolder.class) {
|
||||||
// Create a global wrapping/unwrapping key
|
if (nativeKeyWrapperKeyID == 0) {
|
||||||
CK_ATTRIBUTE[] wrappingAttributes = token.getAttributes
|
createNativeKeyWrapper(token);
|
||||||
(O_GENERATE, CKO_SECRET_KEY, CKK_AES, new CK_ATTRIBUTE[] {
|
}
|
||||||
new CK_ATTRIBUTE(CKA_CLASS, CKO_SECRET_KEY),
|
// If a wrapper-key was successfully created or
|
||||||
new CK_ATTRIBUTE(CKA_VALUE_LEN, 256 >> 3),
|
// already exists, increment its reference
|
||||||
});
|
// counter to keep it alive while native key
|
||||||
Session wrappingSession = null;
|
// information is being held.
|
||||||
try {
|
if (nativeKeyWrapperKeyID != 0) {
|
||||||
wrappingSession = token.getObjSession();
|
nativeKeyWrapperRefCount++;
|
||||||
nativeKeyWrapperKeyID = token.p11.C_GenerateKey
|
wrapperKeyUsed = true;
|
||||||
(wrappingSession.id(),
|
|
||||||
new CK_MECHANISM(CKM_AES_KEY_GEN),
|
|
||||||
wrappingAttributes);
|
|
||||||
byte[] iv = new byte[16];
|
|
||||||
JCAUtil.getSecureRandom().nextBytes(iv);
|
|
||||||
nativeKeyWrapperMechanism = new CK_MECHANISM
|
|
||||||
(CKM_AES_CBC_PAD, iv);
|
|
||||||
} catch (PKCS11Exception e) {
|
|
||||||
// best effort
|
|
||||||
} finally {
|
|
||||||
token.releaseSession(wrappingSession);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1196,7 +1243,8 @@ final class NativeKeyHolder {
|
|||||||
try {
|
try {
|
||||||
opSession = token.getOpSession();
|
opSession = token.getOpSession();
|
||||||
ki = p11Key.token.p11.getNativeKeyInfo(opSession.id(),
|
ki = p11Key.token.p11.getNativeKeyInfo(opSession.id(),
|
||||||
keyID, nativeKeyWrapperKeyID, nativeKeyWrapperMechanism);
|
keyID, nativeKeyWrapperKeyID,
|
||||||
|
nativeKeyWrapperMechanism);
|
||||||
} catch (PKCS11Exception e) {
|
} catch (PKCS11Exception e) {
|
||||||
// best effort
|
// best effort
|
||||||
} finally {
|
} finally {
|
||||||
@ -1206,6 +1254,8 @@ final class NativeKeyHolder {
|
|||||||
// best effort
|
// best effort
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
this.ref = new SessionKeyRef(p11Key, keyID, wrapperKeyUsed,
|
||||||
|
keySession);
|
||||||
}
|
}
|
||||||
this.nativeKeyInfo = ((ki == null || ki.length == 0)? null : ki);
|
this.nativeKeyInfo = ((ki == null || ki.length == 0)? null : ki);
|
||||||
}
|
}
|
||||||
@ -1222,18 +1272,15 @@ final class NativeKeyHolder {
|
|||||||
throw new RuntimeException(
|
throw new RuntimeException(
|
||||||
"Error: null keyID with non-zero refCount " + cnt);
|
"Error: null keyID with non-zero refCount " + cnt);
|
||||||
}
|
}
|
||||||
if (this.ref != null) {
|
|
||||||
throw new RuntimeException(
|
|
||||||
"Error: null keyID with non-null session ref");
|
|
||||||
}
|
|
||||||
Token token = p11Key.token;
|
Token token = p11Key.token;
|
||||||
// Create keyID using nativeKeyInfo
|
// Create keyID using nativeKeyInfo
|
||||||
Session session = null;
|
Session session = null;
|
||||||
try {
|
try {
|
||||||
session = token.getObjSession();
|
session = token.getObjSession();
|
||||||
this.keyID = token.p11.createNativeKey(session.id(),
|
this.keyID = token.p11.createNativeKey(session.id(),
|
||||||
nativeKeyInfo, nativeKeyWrapperKeyID, nativeKeyWrapperMechanism);
|
nativeKeyInfo, nativeKeyWrapperKeyID,
|
||||||
this.ref = new SessionKeyRef(p11Key, this.keyID, session);
|
nativeKeyWrapperMechanism);
|
||||||
|
this.ref.registerNativeKey(this.keyID, session);
|
||||||
} catch (PKCS11Exception e) {
|
} catch (PKCS11Exception e) {
|
||||||
this.refCount--;
|
this.refCount--;
|
||||||
throw new ProviderException("Error recreating native key", e);
|
throw new ProviderException("Error recreating native key", e);
|
||||||
@ -1263,12 +1310,9 @@ final class NativeKeyHolder {
|
|||||||
throw new RuntimeException("ERROR: null keyID can't be destroyed");
|
throw new RuntimeException("ERROR: null keyID can't be destroyed");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.ref == null) {
|
|
||||||
throw new RuntimeException("ERROR: null session ref can't be disposed");
|
|
||||||
}
|
|
||||||
// destroy
|
// destroy
|
||||||
this.keyID = 0;
|
this.keyID = 0;
|
||||||
this.ref = this.ref.dispose();
|
this.ref.removeNativeKey();
|
||||||
} else {
|
} else {
|
||||||
if (cnt < 0) {
|
if (cnt < 0) {
|
||||||
// should never happen as we start count at 1 and pair get/release calls
|
// should never happen as we start count at 1 and pair get/release calls
|
||||||
@ -1285,12 +1329,11 @@ final class NativeKeyHolder {
|
|||||||
* otherwise the key maybe cleared before other objects which
|
* otherwise the key maybe cleared before other objects which
|
||||||
* still use these keys during finalization such as SSLSocket.
|
* still use these keys during finalization such as SSLSocket.
|
||||||
*/
|
*/
|
||||||
final class SessionKeyRef extends PhantomReference<P11Key>
|
final class SessionKeyRef extends PhantomReference<P11Key> {
|
||||||
implements Comparable<SessionKeyRef> {
|
|
||||||
private static ReferenceQueue<P11Key> refQueue =
|
private static ReferenceQueue<P11Key> refQueue =
|
||||||
new ReferenceQueue<P11Key>();
|
new ReferenceQueue<P11Key>();
|
||||||
private static Set<SessionKeyRef> refList =
|
private static Set<SessionKeyRef> refSet =
|
||||||
Collections.synchronizedSortedSet(new TreeSet<SessionKeyRef>());
|
Collections.synchronizedSet(new HashSet<SessionKeyRef>());
|
||||||
|
|
||||||
static ReferenceQueue<P11Key> referenceQueue() {
|
static ReferenceQueue<P11Key> referenceQueue() {
|
||||||
return refQueue;
|
return refQueue;
|
||||||
@ -1307,47 +1350,69 @@ final class SessionKeyRef extends PhantomReference<P11Key>
|
|||||||
}
|
}
|
||||||
|
|
||||||
// handle to the native key and the session it is generated under
|
// handle to the native key and the session it is generated under
|
||||||
private final long keyID;
|
private long keyID;
|
||||||
private final Session session;
|
private Session session;
|
||||||
|
private boolean wrapperKeyUsed;
|
||||||
|
|
||||||
SessionKeyRef(P11Key p11Key, long keyID, Session session) {
|
SessionKeyRef(P11Key p11Key, long keyID, boolean wrapperKeyUsed,
|
||||||
|
Session session) {
|
||||||
super(p11Key, refQueue);
|
super(p11Key, refQueue);
|
||||||
if (session == null) {
|
if (session == null) {
|
||||||
throw new ProviderException("key must be associated with a session");
|
throw new ProviderException("key must be associated with a session");
|
||||||
}
|
}
|
||||||
this.keyID = keyID;
|
registerNativeKey(keyID, session);
|
||||||
this.session = session;
|
this.wrapperKeyUsed = wrapperKeyUsed;
|
||||||
this.session.addObject();
|
|
||||||
|
|
||||||
refList.add(this);
|
refSet.add(this);
|
||||||
drainRefQueueBounded();
|
drainRefQueueBounded();
|
||||||
}
|
}
|
||||||
|
|
||||||
SessionKeyRef dispose() {
|
void registerNativeKey(long newKeyID, Session newSession) {
|
||||||
Token token = session.token;
|
assert(newKeyID != 0);
|
||||||
// If the token is still valid, try to remove the key object
|
assert(newSession != null);
|
||||||
if (token.isValid()) {
|
updateNativeKey(newKeyID, newSession);
|
||||||
Session s = null;
|
|
||||||
try {
|
|
||||||
s = token.getOpSession();
|
|
||||||
token.p11.C_DestroyObject(s.id(), keyID);
|
|
||||||
} catch (PKCS11Exception e) {
|
|
||||||
// best effort
|
|
||||||
} finally {
|
|
||||||
token.releaseSession(s);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
refList.remove(this);
|
|
||||||
this.clear();
|
|
||||||
session.removeObject();
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public int compareTo(SessionKeyRef other) {
|
void removeNativeKey() {
|
||||||
if (this.keyID == other.keyID) {
|
assert(session != null);
|
||||||
return 0;
|
updateNativeKey(0, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateNativeKey(long newKeyID, Session newSession) {
|
||||||
|
if (newKeyID == 0) {
|
||||||
|
assert(newSession == null);
|
||||||
|
Token token = session.token;
|
||||||
|
// If the token is still valid, try to remove the key object
|
||||||
|
if (token.isValid()) {
|
||||||
|
Session s = null;
|
||||||
|
try {
|
||||||
|
s = token.getOpSession();
|
||||||
|
token.p11.C_DestroyObject(s.id(), this.keyID);
|
||||||
|
} catch (PKCS11Exception e) {
|
||||||
|
// best effort
|
||||||
|
} finally {
|
||||||
|
token.releaseSession(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
session.removeObject();
|
||||||
} else {
|
} else {
|
||||||
return (this.keyID < other.keyID) ? -1 : 1;
|
newSession.addObject();
|
||||||
}
|
}
|
||||||
|
keyID = newKeyID;
|
||||||
|
session = newSession;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called when the GC disposes a p11Key
|
||||||
|
void dispose() {
|
||||||
|
if (wrapperKeyUsed) {
|
||||||
|
// Wrapper-key no longer needed for
|
||||||
|
// p11Key native key information
|
||||||
|
NativeKeyHolder.decWrapperKeyRef();
|
||||||
|
}
|
||||||
|
if (keyID != 0) {
|
||||||
|
removeNativeKey();
|
||||||
|
}
|
||||||
|
refSet.remove(this);
|
||||||
|
this.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -308,7 +308,7 @@ Java_sun_security_pkcs11_wrapper_PKCS11_getNativeKeyInfo
|
|||||||
*(CK_BBOOL*)(((CK_ATTRIBUTE_PTR)(((CK_ATTRIBUTE_PTR)nativeKeyInfoArrayRawCkAttributes)
|
*(CK_BBOOL*)(((CK_ATTRIBUTE_PTR)(((CK_ATTRIBUTE_PTR)nativeKeyInfoArrayRawCkAttributes)
|
||||||
+sensitiveAttributePosition))->pValue) == CK_TRUE) {
|
+sensitiveAttributePosition))->pValue) == CK_TRUE) {
|
||||||
// Key is sensitive. Need to extract it wrapped.
|
// Key is sensitive. Need to extract it wrapped.
|
||||||
if (jWrappingKeyHandle != -1) {
|
if (jWrappingKeyHandle != 0) {
|
||||||
|
|
||||||
jMechanismToCKMechanism(env, jWrappingMech, &ckMechanism);
|
jMechanismToCKMechanism(env, jWrappingMech, &ckMechanism);
|
||||||
rv = (*ckpFunctions->C_WrapKey)(ckSessionHandle, &ckMechanism,
|
rv = (*ckpFunctions->C_WrapKey)(ckSessionHandle, &ckMechanism,
|
||||||
@ -351,6 +351,7 @@ Java_sun_security_pkcs11_wrapper_PKCS11_getNativeKeyInfo
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
ckAssertReturnValueOK(env, CKR_KEY_HANDLE_INVALID);
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
returnValue = nativeKeyInfoWrappedKeyArray;
|
returnValue = nativeKeyInfoWrappedKeyArray;
|
||||||
|
Loading…
Reference in New Issue
Block a user