Skip to content

Commit 18fc350

Browse files
djelinskiXueleiFan
authored andcommittedMar 7, 2021
8259886: Improve SSL session cache performance and scalability
Reviewed-by: erikj, xuelei
1 parent 3844ce4 commit 18fc350

File tree

3 files changed

+150
-1
lines changed

3 files changed

+150
-1
lines changed
 

‎make/test/BuildMicrobenchmark.gmk

+2
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,7 @@ $(eval $(call SetupJavaCompilation, BUILD_INDIFY, \
8484
#### Compile Targets
8585

8686
# Building microbenchmark requires the jdk.unsupported and java.management modules.
87+
# sun.security.util is required to compile Cache benchmark
8788

8889
# Build microbenchmark suite for the current JDK
8990
$(eval $(call SetupJavaCompilation, BUILD_JDK_MICROBENCHMARK, \
@@ -93,6 +94,7 @@ $(eval $(call SetupJavaCompilation, BUILD_JDK_MICROBENCHMARK, \
9394
DISABLED_WARNINGS := processing rawtypes cast serial, \
9495
SRC := $(MICROBENCHMARK_SRC), \
9596
BIN := $(MICROBENCHMARK_CLASSES), \
97+
JAVAC_FLAGS := --add-exports java.base/sun.security.util=ALL-UNNAMED, \
9698
JAVA_FLAGS := --add-modules jdk.unsupported --limit-modules java.management, \
9799
))
98100

‎src/java.base/share/classes/sun/security/util/Cache.java

+20-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2002, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2002, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -248,6 +248,7 @@ class MemoryCache<K,V> extends Cache<K,V> {
248248
private final Map<K, CacheEntry<K,V>> cacheMap;
249249
private int maxSize;
250250
private long lifetime;
251+
private long nextExpirationTime = Long.MAX_VALUE;
251252

252253
// ReferenceQueue is of type V instead of Cache<K,V>
253254
// to allow SoftCacheEntry to extend SoftReference<V>
@@ -317,12 +318,18 @@ private void expungeExpiredEntries() {
317318
}
318319
int cnt = 0;
319320
long time = System.currentTimeMillis();
321+
if (nextExpirationTime > time) {
322+
return;
323+
}
324+
nextExpirationTime = Long.MAX_VALUE;
320325
for (Iterator<CacheEntry<K,V>> t = cacheMap.values().iterator();
321326
t.hasNext(); ) {
322327
CacheEntry<K,V> entry = t.next();
323328
if (entry.isValid(time) == false) {
324329
t.remove();
325330
cnt++;
331+
} else if (nextExpirationTime > entry.getExpirationTime()) {
332+
nextExpirationTime = entry.getExpirationTime();
326333
}
327334
}
328335
if (DEBUG) {
@@ -356,6 +363,9 @@ public synchronized void put(K key, V value) {
356363
emptyQueue();
357364
long expirationTime = (lifetime == 0) ? 0 :
358365
System.currentTimeMillis() + lifetime;
366+
if (expirationTime < nextExpirationTime) {
367+
nextExpirationTime = expirationTime;
368+
}
359369
CacheEntry<K,V> newEntry = newEntry(key, value, expirationTime, queue);
360370
CacheEntry<K,V> oldEntry = cacheMap.put(key, newEntry);
361371
if (oldEntry != null) {
@@ -470,6 +480,7 @@ private static interface CacheEntry<K,V> {
470480

471481
V getValue();
472482

483+
long getExpirationTime();
473484
}
474485

475486
private static class HardCacheEntry<K,V> implements CacheEntry<K,V> {
@@ -492,6 +503,10 @@ public V getValue() {
492503
return value;
493504
}
494505

506+
public long getExpirationTime() {
507+
return expirationTime;
508+
}
509+
495510
public boolean isValid(long currentTime) {
496511
boolean valid = (currentTime <= expirationTime);
497512
if (valid == false) {
@@ -529,6 +544,10 @@ public V getValue() {
529544
return get();
530545
}
531546

547+
public long getExpirationTime() {
548+
return expirationTime;
549+
}
550+
532551
public boolean isValid(long currentTime) {
533552
boolean valid = (currentTime <= expirationTime) && (get() != null);
534553
if (valid == false) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
/*
2+
* Copyright (c) 2021, Dynatrace LLC. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
package org.openjdk.bench.java.security;
25+
26+
import org.openjdk.jmh.annotations.Benchmark;
27+
import org.openjdk.jmh.annotations.BenchmarkMode;
28+
import org.openjdk.jmh.annotations.Fork;
29+
import org.openjdk.jmh.annotations.Level;
30+
import org.openjdk.jmh.annotations.Mode;
31+
import org.openjdk.jmh.annotations.OutputTimeUnit;
32+
import org.openjdk.jmh.annotations.Param;
33+
import org.openjdk.jmh.annotations.Scope;
34+
import org.openjdk.jmh.annotations.Setup;
35+
import org.openjdk.jmh.annotations.State;
36+
import org.openjdk.jmh.annotations.TearDown;
37+
38+
import java.util.concurrent.TimeUnit;
39+
import java.util.stream.IntStream;
40+
41+
import sun.security.util.Cache;
42+
43+
@BenchmarkMode(Mode.AverageTime)
44+
@OutputTimeUnit(TimeUnit.NANOSECONDS)
45+
@Fork(jvmArgsAppend = {"--add-exports", "java.base/sun.security.util=ALL-UNNAMED", "-Xmx1g"})
46+
public class CacheBench {
47+
48+
@State(Scope.Benchmark)
49+
public static class SharedState {
50+
Cache<Integer, Integer> cache;
51+
52+
@Param({"20480", "204800", "5120000"})
53+
int size;
54+
55+
@Param({"86400", "0"})
56+
int timeout;
57+
58+
@Setup
59+
public void setup() {
60+
cache = Cache.newSoftMemoryCache(size, timeout);
61+
IntStream.range(0, size).boxed().forEach(i -> cache.put(i, i));
62+
}
63+
}
64+
65+
@State(Scope.Thread)
66+
public static class GetPutState {
67+
Integer[] intArray;
68+
int index;
69+
70+
@Setup
71+
public void setup(SharedState benchState) {
72+
intArray = IntStream.range(0, benchState.size + 1).boxed().toArray(Integer[]::new);
73+
index = 0;
74+
}
75+
76+
@TearDown(Level.Invocation)
77+
public void tearDown() {
78+
index++;
79+
if (index >= intArray.length) {
80+
index = 0;
81+
}
82+
}
83+
}
84+
85+
@Benchmark
86+
public void put(SharedState benchState, GetPutState state) {
87+
Integer i = state.intArray[state.index];
88+
benchState.cache.put(i, i);
89+
}
90+
91+
@Benchmark
92+
public Integer get(SharedState benchState, GetPutState state) {
93+
Integer i = state.intArray[state.index];
94+
return benchState.cache.get(i);
95+
}
96+
97+
@State(Scope.Thread)
98+
public static class RemoveState {
99+
Integer[] intArray;
100+
int index;
101+
SharedState benchState;
102+
103+
@Setup
104+
public void setup(SharedState benchState) {
105+
this.benchState = benchState;
106+
intArray = IntStream.range(0, benchState.size).boxed().toArray(Integer[]::new);
107+
index = 0;
108+
}
109+
110+
@TearDown(Level.Invocation)
111+
public void tearDown() {
112+
// add back removed item
113+
Integer i = intArray[index];
114+
benchState.cache.put(i, i);
115+
116+
index++;
117+
if (index >= intArray.length) {
118+
index = 0;
119+
}
120+
}
121+
}
122+
123+
@Benchmark
124+
public void remove(SharedState benchState, RemoveState state) {
125+
Integer i = state.intArray[state.index];
126+
benchState.cache.remove(i);
127+
}
128+
}

0 commit comments

Comments
 (0)
Please sign in to comment.