Skip to content

Commit f4756fd

Browse files
author
Valerie Peng
committedJul 7, 2020
8248505: Unexpected NoSuchAlgorithmException when using secure random impl from BCFIPS provider
Use getService(...) call for Provider.getDefaultSecureRandomService() Reviewed-by: weijun, coffeys, mullan
1 parent ca91da0 commit f4756fd

File tree

2 files changed

+65
-23
lines changed

2 files changed

+65
-23
lines changed
 

‎src/java.base/share/classes/java/security/Provider.java

+18-15
Original file line numberDiff line numberDiff line change
@@ -868,7 +868,7 @@ private void check(String directive) {
868868
// For backward compatibility, the registration ordering of
869869
// SecureRandom (RNG) algorithms needs to be preserved for
870870
// "new SecureRandom()" calls when this provider is used
871-
private transient Set<Service> prngServices;
871+
private transient Set<String> prngAlgos;
872872

873873
// Map<ServiceKey,Service>
874874
// used for services added via legacy methods, init on demand
@@ -1089,7 +1089,7 @@ private void implClear() {
10891089
legacyChanged = false;
10901090
servicesChanged = false;
10911091
serviceSet = null;
1092-
prngServices = null;
1092+
prngAlgos = null;
10931093
super.clear();
10941094
putId();
10951095
}
@@ -1221,7 +1221,7 @@ private void parseLegacyPut(String name, String value) {
12211221
s.className = className;
12221222

12231223
if (type.equals("SecureRandom")) {
1224-
updateSecureRandomEntries(true, s);
1224+
updateSecureRandomEntries(true, s.algorithm);
12251225
}
12261226
} else { // attribute
12271227
// e.g. put("MessageDigest.SHA-1 ImplementedIn", "Software");
@@ -1383,25 +1383,25 @@ protected void putService(Service s) {
13831383
synchronized (this) {
13841384
putPropertyStrings(s);
13851385
if (type.equals("SecureRandom")) {
1386-
updateSecureRandomEntries(true, s);
1386+
updateSecureRandomEntries(true, s.algorithm);
13871387
}
13881388
}
13891389
}
13901390

1391-
private void updateSecureRandomEntries(boolean doAdd, Service s) {
1391+
// keep tracks of the registered secure random algos and store them in order
1392+
private void updateSecureRandomEntries(boolean doAdd, String s) {
13921393
Objects.requireNonNull(s);
13931394
if (doAdd) {
1394-
if (prngServices == null) {
1395-
prngServices = new LinkedHashSet<Service>();
1395+
if (prngAlgos == null) {
1396+
prngAlgos = new LinkedHashSet<String>();
13961397
}
1397-
prngServices.add(s);
1398+
prngAlgos.add(s);
13981399
} else {
1399-
prngServices.remove(s);
1400+
prngAlgos.remove(s);
14001401
}
14011402

14021403
if (debug != null) {
1403-
debug.println((doAdd? "Add":"Remove") + " SecureRandom algo " +
1404-
s.getAlgorithm());
1404+
debug.println((doAdd? "Add":"Remove") + " SecureRandom algo " + s);
14051405
}
14061406
}
14071407

@@ -1411,12 +1411,15 @@ synchronized Service getDefaultSecureRandomService() {
14111411
checkInitialized();
14121412

14131413
if (legacyChanged) {
1414-
prngServices = null;
1414+
prngAlgos = null;
14151415
ensureLegacyParsed();
14161416
}
14171417

1418-
if (prngServices != null && !prngServices.isEmpty()) {
1419-
return prngServices.iterator().next();
1418+
if (prngAlgos != null && !prngAlgos.isEmpty()) {
1419+
// IMPORTANT: use the Service obj returned by getService(...) call
1420+
// as providers may override putService(...)/getService(...) and
1421+
// return their own Service objects
1422+
return getService("SecureRandom", prngAlgos.iterator().next());
14201423
}
14211424

14221425
return null;
@@ -1516,7 +1519,7 @@ private void implRemoveService(Service s) {
15161519
synchronized (this) {
15171520
removePropertyStrings(s);
15181521
if (type.equals("SecureRandom")) {
1519-
updateSecureRandomEntries(false, s);
1522+
updateSecureRandomEntries(false, s.algorithm);
15201523
}
15211524
}
15221525
}

‎test/jdk/java/security/SecureRandom/DefaultAlgo.java

+47-8
Original file line numberDiff line numberDiff line change
@@ -26,20 +26,22 @@
2626
import java.security.Security;
2727
import java.security.SecureRandom;
2828
import java.security.Provider.Service;
29-
import java.util.Objects;
3029
import java.util.Arrays;
3130
import sun.security.provider.SunEntries;
3231

3332
/**
3433
* @test
35-
* @bug 8228613 8246613
34+
* @bug 8228613 8246613 8248505
3635
* @summary Ensure that the default SecureRandom algo used is based
3736
* on the registration ordering, and falls to next provider
3837
* if none are found
3938
* @modules java.base/sun.security.provider
4039
*/
4140
public class DefaultAlgo {
4241

42+
private static final String SR_IMPLCLASS =
43+
"sun.security.provider.SecureRandom";
44+
4345
public static void main(String[] args) throws Exception {
4446
String[] algos = { "A", "B", "C" };
4547
test3rdParty(algos);
@@ -51,7 +53,8 @@ public static void main(String[] args) throws Exception {
5153
private static void test3rdParty(String[] algos) {
5254
Provider[] provs = {
5355
new SampleLegacyProvider(algos),
54-
new SampleServiceProvider(algos)
56+
new SampleServiceProvider(algos),
57+
new CustomProvider(algos)
5558
};
5659
for (Provider p : provs) {
5760
checkDefault(p, algos);
@@ -72,9 +75,10 @@ private static void validate(SecureRandom sr, String pName, String algo) {
7275
}
7376

7477
private static void checkDefault(Provider p, String ... algos) {
75-
out.println(p.getName() + " with " + Arrays.toString(algos));
76-
int pos = Security.insertProviderAt(p, 1);
7778
String pName = p.getName();
79+
out.println(pName + " with " + Arrays.toString(algos));
80+
int pos = Security.insertProviderAt(p, 1);
81+
7882
boolean isLegacy = pName.equals("SampleLegacy");
7983
try {
8084
if (isLegacy) {
@@ -91,7 +95,13 @@ private static void checkDefault(Provider p, String ... algos) {
9195
out.println("=> Test Passed");
9296
} finally {
9397
if (pos != -1) {
94-
Security.removeProvider(p.getName());
98+
Security.removeProvider(pName);
99+
}
100+
if (isLegacy) {
101+
// add back the removed algos
102+
for (String s : algos) {
103+
p.put("SecureRandom." + s, SR_IMPLCLASS);
104+
}
95105
}
96106
}
97107
}
@@ -100,7 +110,7 @@ private static class SampleLegacyProvider extends Provider {
100110
SampleLegacyProvider(String[] listOfSupportedRNGs) {
101111
super("SampleLegacy", "1.0", "test provider using legacy put");
102112
for (String s : listOfSupportedRNGs) {
103-
put("SecureRandom." + s, "sun.security.provider.SecureRandom");
113+
put("SecureRandom." + s, SR_IMPLCLASS);
104114
}
105115
}
106116
}
@@ -110,8 +120,37 @@ private static class SampleServiceProvider extends Provider {
110120
super("SampleService", "1.0", "test provider using putService");
111121
for (String s : listOfSupportedRNGs) {
112122
putService(new Provider.Service(this, "SecureRandom", s,
113-
"sun.security.provider.SecureRandom", null, null));
123+
SR_IMPLCLASS, null, null));
124+
}
125+
}
126+
}
127+
128+
// custom provider which overrides putService(...)/getService() and uses
129+
// its own Service impl
130+
private static class CustomProvider extends Provider {
131+
private static class CustomService extends Provider.Service {
132+
CustomService(Provider p, String type, String algo, String cName) {
133+
super(p, type, algo, cName, null, null);
114134
}
115135
}
136+
137+
CustomProvider(String[] listOfSupportedRNGs) {
138+
super("Custom", "1.0", "test provider overrides putService with " +
139+
" custom service with legacy registration");
140+
for (String s : listOfSupportedRNGs) {
141+
putService(new CustomService(this, "SecureRandom", s ,
142+
SR_IMPLCLASS));
143+
}
144+
}
145+
@Override
146+
protected void putService(Provider.Service s) {
147+
// convert to legacy puts
148+
put(s.getType() + "." + s.getAlgorithm(), s.getClassName());
149+
put(s.getType() + ":" + s.getAlgorithm(), s);
150+
}
151+
@Override
152+
public Provider.Service getService(String type, String algo) {
153+
return (Provider.Service) get(type + ":" + algo);
154+
}
116155
}
117156
}

0 commit comments

Comments
 (0)
Please sign in to comment.