mirror of
https://github.com/openjdk/jdk.git
synced 2025-08-28 07:14:30 +02:00
8300416: java.security.MessageDigestSpi clone can result in thread-unsafe clones
Reviewed-by: mullan
This commit is contained in:
parent
a73d012c72
commit
2e2e71e1fa
2 changed files with 64 additions and 6 deletions
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2022, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2023, 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
|
||||||
|
@ -205,7 +205,15 @@ public abstract class MessageDigestSpi {
|
||||||
*/
|
*/
|
||||||
public Object clone() throws CloneNotSupportedException {
|
public Object clone() throws CloneNotSupportedException {
|
||||||
if (this instanceof Cloneable) {
|
if (this instanceof Cloneable) {
|
||||||
return super.clone();
|
MessageDigestSpi o = (MessageDigestSpi)super.clone();
|
||||||
|
if (o.tempArray != null) {
|
||||||
|
// New byte arrays are allocated when the ByteBuffer argument
|
||||||
|
// to engineUpdate is not backed by a byte array.
|
||||||
|
// Here, the newly allocated byte array must also be cloned
|
||||||
|
// to prevent threads from sharing the same memory.
|
||||||
|
o.tempArray = tempArray.clone();
|
||||||
|
}
|
||||||
|
return o;
|
||||||
} else {
|
} else {
|
||||||
throw new CloneNotSupportedException();
|
throw new CloneNotSupportedException();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2020, 2023, 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
|
||||||
|
@ -23,12 +23,16 @@
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @test
|
* @test
|
||||||
* @bug 8246077
|
* @bug 8246077 8300416
|
||||||
* @summary Make sure that digest spi and the resulting digest impl are
|
* @summary Make sure that digest spi and the resulting digest impl are
|
||||||
* consistent in the impl of Cloneable interface
|
* consistent in the impl of Cloneable interface, and that clones do not
|
||||||
|
* share memory.
|
||||||
* @run testng TestCloneable
|
* @run testng TestCloneable
|
||||||
*/
|
*/
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
import java.security.*;
|
import java.security.*;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Random;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import org.testng.annotations.DataProvider;
|
import org.testng.annotations.DataProvider;
|
||||||
import org.testng.annotations.Test;
|
import org.testng.annotations.Test;
|
||||||
|
@ -53,7 +57,7 @@ public class TestCloneable {
|
||||||
@Test(dataProvider = "testData")
|
@Test(dataProvider = "testData")
|
||||||
public void test(String algo, String provName)
|
public void test(String algo, String provName)
|
||||||
throws NoSuchProviderException, NoSuchAlgorithmException,
|
throws NoSuchProviderException, NoSuchAlgorithmException,
|
||||||
CloneNotSupportedException {
|
CloneNotSupportedException, InterruptedException {
|
||||||
System.out.print("Testing " + algo + " impl from " + provName);
|
System.out.print("Testing " + algo + " impl from " + provName);
|
||||||
Provider p = Security.getProvider(provName);
|
Provider p = Security.getProvider(provName);
|
||||||
Provider.Service s = p.getService("MessageDigest", algo);
|
Provider.Service s = p.getService("MessageDigest", algo);
|
||||||
|
@ -71,6 +75,52 @@ public class TestCloneable {
|
||||||
System.out.println(": NOT Cloneable");
|
System.out.println(": NOT Cloneable");
|
||||||
Assert.assertThrows(CNSE, ()->md.clone());
|
Assert.assertThrows(CNSE, ()->md.clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
System.out.print("Testing " + algo + " impl from " + provName);
|
||||||
|
final var d1 = MessageDigest.getInstance(algo, provName);
|
||||||
|
final var buffer = ByteBuffer.allocateDirect(1024);
|
||||||
|
final var r = new Random(1024);
|
||||||
|
|
||||||
|
fillBuffer(r, buffer);
|
||||||
|
d1.update(buffer); // this statement triggers tempArray allocation
|
||||||
|
final var d2 = (MessageDigest) d1.clone();
|
||||||
|
assert Arrays.equals(d1.digest(), d2.digest());
|
||||||
|
|
||||||
|
final var t1 = updateThread(d1);
|
||||||
|
final var t2 = updateThread(d2);
|
||||||
|
t1.join();
|
||||||
|
t2.join();
|
||||||
|
|
||||||
|
System.out.println(": Shared data check");
|
||||||
|
// Random is producing the same sequence of bytes for each thread,
|
||||||
|
// and thus each MessageDigest should be equal. When the memory is
|
||||||
|
// shared, they inevitably overwrite each other's tempArray and
|
||||||
|
// you get different results.
|
||||||
|
if (!Arrays.equals(d1.digest(), d2.digest())) {
|
||||||
|
throw new AssertionError("digests differ");
|
||||||
|
}
|
||||||
|
|
||||||
System.out.println("Test Passed");
|
System.out.println("Test Passed");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void fillBuffer(final Random r, final ByteBuffer buffer) {
|
||||||
|
final byte[] bytes = new byte[buffer.capacity()];
|
||||||
|
r.nextBytes(bytes);
|
||||||
|
buffer.clear();
|
||||||
|
buffer.put(bytes);
|
||||||
|
buffer.flip();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Thread updateThread(final MessageDigest d) {
|
||||||
|
final var t = new Thread(() -> {
|
||||||
|
final var r = new Random(1024);
|
||||||
|
final ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
|
||||||
|
for (int i = 0; i < 1024; i++) {
|
||||||
|
fillBuffer(r, buffer);
|
||||||
|
d.update(buffer);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
t.start();
|
||||||
|
return t;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue