Skip to content

Commit 65393a0

Browse files
committedOct 13, 2020
8229867: Re-examine synchronization usages in http and https protocol handlers
Reviewed-by: chegar, alanb, michaelm
1 parent 6fe209b commit 65393a0

18 files changed

+970
-568
lines changed
 

‎src/java.base/share/classes/sun/net/www/MessageHeader.java

+14-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1995, 2013, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1995, 2020, 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
@@ -294,7 +294,7 @@ public synchronized Map<String, List<String>> filterAndAddHeaders(
294294
* @param line the line to check.
295295
* @return true if the line might be a request line.
296296
*/
297-
private boolean isRequestline(String line) {
297+
private static boolean isRequestline(String line) {
298298
String k = line.trim();
299299
int i = k.lastIndexOf(' ');
300300
if (i <= 0) return false;
@@ -311,12 +311,23 @@ private boolean isRequestline(String line) {
311311
return (k.substring(i+1, len-3).equalsIgnoreCase("HTTP/"));
312312
}
313313

314+
/** Prints the key-value pairs represented by this
315+
header. Also prints the RFC required blank line
316+
at the end. Omits pairs with a null key. Omits
317+
colon if key-value pair is the requestline. */
318+
public void print(PrintStream p) {
319+
// no synchronization: use cloned arrays instead.
320+
String[] k; String[] v; int n;
321+
synchronized (this) { n = nkeys; k = keys.clone(); v = values.clone(); }
322+
print(n, k, v, p);
323+
}
324+
314325

315326
/** Prints the key-value pairs represented by this
316327
header. Also prints the RFC required blank line
317328
at the end. Omits pairs with a null key. Omits
318329
colon if key-value pair is the requestline. */
319-
public synchronized void print(PrintStream p) {
330+
private static void print(int nkeys, String[] keys, String[] values, PrintStream p) {
320331
for (int i = 0; i < nkeys; i++)
321332
if (keys[i] != null) {
322333
StringBuilder sb = new StringBuilder(keys[i]);

‎src/java.base/share/classes/sun/net/www/MeteredStream.java

+106-67
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1994, 2017, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1994, 2020, 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
@@ -25,9 +25,8 @@
2525

2626
package sun.net.www;
2727

28-
import java.net.URL;
29-
import java.util.*;
3028
import java.io.*;
29+
import java.util.concurrent.locks.ReentrantLock;
3130
import sun.net.ProgressSource;
3231
import sun.net.www.http.ChunkedInputStream;
3332

@@ -44,6 +43,7 @@ public class MeteredStream extends FilterInputStream {
4443
protected long markedCount = 0;
4544
protected int markLimit = -1;
4645
protected ProgressSource pi;
46+
private final ReentrantLock readLock = new ReentrantLock();
4747

4848
public MeteredStream(InputStream is, ProgressSource pi, long expected)
4949
{
@@ -57,7 +57,9 @@ public MeteredStream(InputStream is, ProgressSource pi, long expected)
5757
}
5858
}
5959

60-
private final void justRead(long n) throws IOException {
60+
private final void justRead(long n) throws IOException {
61+
assert isLockHeldByCurrentThread();
62+
6163
if (n == -1) {
6264

6365
/*
@@ -99,6 +101,7 @@ private final void justRead(long n) throws IOException {
99101
* Returns true if the mark is valid, false otherwise
100102
*/
101103
private boolean isMarked() {
104+
assert isLockHeldByCurrentThread();
102105

103106
if (markLimit < 0) {
104107
return false;
@@ -113,94 +116,130 @@ private boolean isMarked() {
113116
return true;
114117
}
115118

116-
public synchronized int read() throws java.io.IOException {
117-
if (closed) {
118-
return -1;
119-
}
120-
int c = in.read();
121-
if (c != -1) {
122-
justRead(1);
123-
} else {
124-
justRead(c);
119+
public int read() throws java.io.IOException {
120+
lock();
121+
try {
122+
if (closed) return -1;
123+
int c = in.read();
124+
if (c != -1) {
125+
justRead(1);
126+
} else {
127+
justRead(c);
128+
}
129+
return c;
130+
} finally {
131+
unlock();
125132
}
126-
return c;
127133
}
128134

129-
public synchronized int read(byte b[], int off, int len)
135+
public int read(byte b[], int off, int len)
130136
throws java.io.IOException {
131-
if (closed) {
132-
return -1;
133-
}
134-
int n = in.read(b, off, len);
135-
justRead(n);
136-
return n;
137-
}
138-
139-
public synchronized long skip(long n) throws IOException {
137+
lock();
138+
try {
139+
if (closed) return -1;
140140

141-
// REMIND: what does skip do on EOF????
142-
if (closed) {
143-
return 0;
141+
int n = in.read(b, off, len);
142+
justRead(n);
143+
return n;
144+
} finally {
145+
unlock();
144146
}
147+
}
145148

146-
if (in instanceof ChunkedInputStream) {
147-
n = in.skip(n);
148-
}
149-
else {
150-
// just skip min(n, num_bytes_left)
151-
long min = (n > expected - count) ? expected - count: n;
152-
n = in.skip(min);
149+
public long skip(long n) throws IOException {
150+
lock();
151+
try {
152+
// REMIND: what does skip do on EOF????
153+
if (closed) return 0;
154+
155+
if (in instanceof ChunkedInputStream) {
156+
n = in.skip(n);
157+
} else {
158+
// just skip min(n, num_bytes_left)
159+
long min = (n > expected - count) ? expected - count : n;
160+
n = in.skip(min);
161+
}
162+
justRead(n);
163+
return n;
164+
} finally {
165+
unlock();
153166
}
154-
justRead(n);
155-
return n;
156167
}
157168

158169
public void close() throws IOException {
159-
if (closed) {
160-
return;
161-
}
162-
if (pi != null)
163-
pi.finishTracking();
170+
lock();
171+
try {
172+
if (closed) return;
173+
if (pi != null)
174+
pi.finishTracking();
164175

165-
closed = true;
166-
in.close();
176+
closed = true;
177+
in.close();
178+
} finally {
179+
unlock();
180+
}
167181
}
168182

169-
public synchronized int available() throws IOException {
170-
return closed ? 0: in.available();
183+
public int available() throws IOException {
184+
lock();
185+
try {
186+
return closed ? 0 : in.available();
187+
} finally {
188+
unlock();
189+
}
171190
}
172191

173-
public synchronized void mark(int readLimit) {
174-
if (closed) {
175-
return;
176-
}
177-
super.mark(readLimit);
192+
public void mark(int readLimit) {
193+
lock();
194+
try {
195+
if (closed) return;
196+
super.mark(readLimit);
178197

179-
/*
180-
* mark the count to restore upon reset
181-
*/
182-
markedCount = count;
183-
markLimit = readLimit;
198+
/*
199+
* mark the count to restore upon reset
200+
*/
201+
markedCount = count;
202+
markLimit = readLimit;
203+
} finally {
204+
unlock();
205+
}
184206
}
185207

186-
public synchronized void reset() throws IOException {
187-
if (closed) {
188-
return;
189-
}
208+
public void reset() throws IOException {
209+
lock();
210+
try {
211+
if (closed) return;
212+
if (!isMarked()) {
213+
throw new IOException("Resetting to an invalid mark");
214+
}
190215

191-
if (!isMarked()) {
192-
throw new IOException ("Resetting to an invalid mark");
216+
count = markedCount;
217+
super.reset();
218+
} finally {
219+
unlock();
193220
}
194-
195-
count = markedCount;
196-
super.reset();
197221
}
198222

199223
public boolean markSupported() {
200-
if (closed) {
201-
return false;
224+
lock();
225+
try {
226+
if (closed) return false;
227+
return super.markSupported();
228+
} finally {
229+
unlock();
202230
}
203-
return super.markSupported();
231+
}
232+
233+
public final void lock() {
234+
readLock.lock();
235+
}
236+
237+
public final void unlock() {
238+
readLock.unlock();
239+
}
240+
241+
public final boolean isLockHeldByCurrentThread() {
242+
return readLock.isHeldByCurrentThread();
204243
}
205244

206245
@SuppressWarnings("deprecation")

0 commit comments

Comments
 (0)
Please sign in to comment.