Skip to content

Commit d42f541

Browse files
author
Andrew Leonard
committedJul 8, 2020
8248231: deserializeLambda created with wrong encoding if platform encoding not UTF-8
Reviewed-by: vromero
1 parent 1d5ec8f commit d42f541

File tree

3 files changed

+180
-2
lines changed

3 files changed

+180
-2
lines changed
 

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/LambdaToMethod.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2010, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2010, 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
@@ -2425,7 +2425,8 @@ protected void append(char ch) {
24252425

24262426
@Override
24272427
protected void append(byte[] ba) {
2428-
sb.append(new String(ba));
2428+
Name name = names.fromUtf(ba);
2429+
sb.append(name.toString());
24292430
}
24302431

24312432
@Override
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
/*
2+
* Copyright (c) 2020, Oracle and/or its affiliates. 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+
/**
25+
* @test
26+
* @bug 8248231
27+
* @summary Test to verify lambda serialization uses the correct UTF-8 encoding
28+
* @library /test/lib
29+
* @build jdk.test.lib.JDKToolFinder
30+
* jdk.test.lib.process.ProcessTools
31+
* @run testng LambdaFileEncodingSerialization
32+
*/
33+
import java.io.File;
34+
import java.io.BufferedReader;
35+
import java.io.InputStreamReader;
36+
import java.io.IOException;
37+
import java.util.List;
38+
import java.util.Map;
39+
import java.util.Arrays;
40+
import java.util.ArrayList;
41+
42+
import org.testng.annotations.Test;
43+
import static org.testng.Assert.assertTrue;
44+
45+
import jdk.test.lib.JDKToolFinder;
46+
import jdk.test.lib.process.ProcessTools;
47+
48+
public class LambdaFileEncodingSerialization {
49+
50+
private static final String TEST_NAME = "TestLambdaFileEncodingSerialization";
51+
52+
@Test
53+
public static void testDeserializeLambdaEncoding() throws Throwable {
54+
55+
String javac = JDKToolFinder.getTestJDKTool("javac");
56+
String java = JDKToolFinder.getTestJDKTool("java");
57+
58+
String srcDir = System.getProperty("test.src");
59+
60+
// Compile <TEST_NAME>.java using ISO-8859-1 encoding
61+
String opts = "-J-Dfile.encoding=ISO-8859-1 -cp . -d .";
62+
String file = srcDir + File.separator + TEST_NAME + ".java";
63+
int exitCode = runCmd(javac, opts, file);
64+
assertTrue(exitCode == 0, "Command " + javac + " " + opts + " " + file + " , failed exitCode = "+exitCode);
65+
66+
// Execute TEST_NAME containing the re-serialized lambda
67+
opts = "-cp .";
68+
file = TEST_NAME;
69+
exitCode = runCmd(java, opts, file);
70+
assertTrue(exitCode == 0, "Command " + java + " " + opts + " " + file + " , failed exitCode = "+exitCode);
71+
}
72+
73+
// Run a command
74+
private static int runCmd(String prog, String options, String file) throws Throwable {
75+
76+
List<String> argList = new ArrayList<String>();
77+
argList.add(prog);
78+
argList.addAll(Arrays.asList(options.split(" ")));
79+
argList.add(file);
80+
81+
ProcessBuilder pb = new ProcessBuilder(argList);
82+
Map<String, String> env = pb.environment();
83+
env.put("LC_ALL", "en_US.UTF-8"); // Ensure locale supports the test requirements, lambda with a UTF char
84+
85+
int exitCode = ProcessTools.executeCommand(pb).outputTo(System.out)
86+
.errorTo(System.err)
87+
.getExitValue();
88+
return exitCode;
89+
}
90+
91+
}
92+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
/*
2+
* Copyright (c) 2020, Oracle and/or its affiliates. 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+
import java.io.ByteArrayInputStream;
25+
import java.io.ByteArrayOutputStream;
26+
import java.io.IOException;
27+
import java.io.ObjectInputStream;
28+
import java.io.ObjectOutputStream;
29+
import java.io.Serializable;
30+
import java.nio.charset.*;
31+
import java.util.concurrent.Callable;
32+
33+
/*
34+
* Testcase for verifying deserializeLambda containing a non-ASCII mappable char
35+
* is correctly handled as UTF-8
36+
*/
37+
public class TestLambdaFileEncodingSerialization {
38+
public static class ABCâ implements Serializable {
39+
public String msg;
40+
public ABCâ() {
41+
msg = "Hello world";
42+
}
43+
public static Callable<ABCâ> getHello() {
44+
return (Callable<ABCâ> & Serializable) () -> {
45+
return new ABCâ();
46+
};
47+
}
48+
}
49+
50+
// Re-serialize the object containing the lambda
51+
private static <T> T reserialize(T o) throws IOException {
52+
ByteArrayOutputStream baos = new ByteArrayOutputStream();
53+
ObjectOutputStream oos = new ObjectOutputStream(baos);
54+
55+
oos.writeObject(o);
56+
57+
oos.close();
58+
59+
ObjectInputStream iis = new ObjectInputStream(new ByteArrayInputStream(baos.toByteArray()));
60+
61+
try {
62+
o = (T)iis.readObject();
63+
} catch (ClassNotFoundException e) {
64+
throw new IOException(e);
65+
}
66+
iis.close();
67+
return o;
68+
}
69+
70+
public static void main(String args[]) throws Exception{
71+
System.out.println("Default charset = "+Charset.defaultCharset());
72+
73+
// Construct class containing suitable UTF-8 char
74+
Callable<ABCâ> foo = ABCâ.getHello();
75+
ABCâ hello = new ABCâ();
76+
77+
// re-serialize hello
78+
ABCâ rh = reserialize(hello);
79+
System.out.println(rh.msg);
80+
81+
// re-serialize foo and call()
82+
reserialize(foo).call();
83+
}
84+
}
85+

0 commit comments

Comments
 (0)
Please sign in to comment.