8241741: Implement Text Blocks as a standard feature in javac

Reviewed-by: jlahoda
This commit is contained in:
Jim Laskey 2020-04-09 10:55:01 -03:00
parent d9bf934831
commit ef8537ec1a
11 changed files with 69 additions and 44 deletions

View File

@ -55,7 +55,7 @@ public @interface PreviewFeature {
public enum Feature { public enum Feature {
PATTERN_MATCHING_IN_INSTANCEOF, PATTERN_MATCHING_IN_INSTANCEOF,
TEXT_BLOCKS, TEXT_BLOCKS, // 8242284: needs to be removed after JDK 15
RECORDS, RECORDS,
; ;
} }

View File

@ -167,7 +167,6 @@ public class Preview {
public boolean isPreview(Feature feature) { public boolean isPreview(Feature feature) {
if (feature == Feature.PATTERN_MATCHING_IN_INSTANCEOF || if (feature == Feature.PATTERN_MATCHING_IN_INSTANCEOF ||
feature == Feature.REIFIABLE_TYPES_INSTANCEOF || feature == Feature.REIFIABLE_TYPES_INSTANCEOF ||
feature == Feature.TEXT_BLOCKS ||
feature == Feature.RECORDS) feature == Feature.RECORDS)
return true; return true;
//Note: this is a backdoor which allows to optionally treat all features as 'preview' (for testing). //Note: this is a backdoor which allows to optionally treat all features as 'preview' (for testing).

View File

@ -233,8 +233,8 @@ public class JavaTokenizer {
if (!multiline) { if (!multiline) {
lexError(reader.bp, Errors.IllegalEscChar); lexError(reader.bp, Errors.IllegalEscChar);
} else { } else {
int start = reader.bp;
checkSourceLevel(reader.bp, Feature.TEXT_BLOCKS); checkSourceLevel(reader.bp, Feature.TEXT_BLOCKS);
int start = reader.bp;
if (reader.ch == '\r' && reader.peekChar() == '\n') { if (reader.ch == '\r' && reader.peekChar() == '\n') {
reader.nextChar(translateEscapesNow); reader.nextChar(translateEscapesNow);
} }
@ -402,6 +402,17 @@ public class JavaTokenizer {
return count; return count;
} }
/** Skip and process a line terminator.
*/
private void skipLineTerminator() {
int start = reader.bp;
if (isCRLF()) {
reader.scanChar();
}
reader.scanChar();
processLineTerminator(start, reader.bp);
}
/** Scan a string literal or text block. /** Scan a string literal or text block.
*/ */
private void scanString(int pos) { private void scanString(int pos) {
@ -425,26 +436,21 @@ public class JavaTokenizer {
checkSourceLevel(pos, Feature.TEXT_BLOCKS); checkSourceLevel(pos, Feature.TEXT_BLOCKS);
isTextBlock = true; isTextBlock = true;
// Verify the open delimiter sequence. // Verify the open delimiter sequence.
boolean hasOpenEOLN = false; while (reader.bp < reader.buflen) {
while (reader.bp < reader.buflen && Character.isWhitespace(reader.ch)) { char ch = reader.ch;
hasOpenEOLN = isEOLN(); if (ch != ' ' && ch != '\t' && ch != FF) {
if (hasOpenEOLN) {
break; break;
} }
reader.scanChar(); reader.scanChar();
} }
// Error if the open delimiter sequence not is """<Whitespace>*<LineTerminator>. if (isEOLN()) {
if (!hasOpenEOLN) { skipLineTerminator();
} else {
// Error if the open delimiter sequence is not
// """<white space>*<LineTerminator>.
lexError(reader.bp, Errors.IllegalTextBlockOpen); lexError(reader.bp, Errors.IllegalTextBlockOpen);
return; return;
} }
// Skip line terminator.
int start = reader.bp;
if (isCRLF()) {
reader.scanChar();
}
reader.scanChar();
processLineTerminator(start, reader.bp);
break; break;
} }
// While characters are available. // While characters are available.
@ -466,13 +472,9 @@ public class JavaTokenizer {
if (openCount == 1) { if (openCount == 1) {
break; break;
} }
skipLineTerminator();
// Add line terminator to string buffer. // Add line terminator to string buffer.
int start = reader.bp; reader.putChar('\n', false);
if (isCRLF()) {
reader.scanChar();
}
reader.putChar('\n', true);
processLineTerminator(start, reader.bp);
// Record first line terminator for error recovery. // Record first line terminator for error recovery.
if (firstEOLN == -1) { if (firstEOLN == -1) {
firstEOLN = reader.bp; firstEOLN = reader.bp;

View File

@ -58,7 +58,7 @@ public class TextBlockAPI {
*/ */
static void test1() { static void test1() {
for (String lineterminators : new String[] { "\n", "\r", "\r\n" }) for (String lineterminators : new String[] { "\n", "\r", "\r\n" })
for (String whitespace : new String[] { "", " ", "\t", "\u2002" }) for (String whitespace : new String[] { "", " ", "\t", "\f" })
for (String content : new String[] { "a", "ab", "abc", "\u2022", "*".repeat(1000), "*".repeat(10000) }) { for (String content : new String[] { "a", "ab", "abc", "\u2022", "*".repeat(1000), "*".repeat(10000) }) {
String code = String code =
"public class CorrectTest {\n" + "public class CorrectTest {\n" +
@ -126,10 +126,9 @@ public class TextBlockAPI {
new JavacTask(TOOLBOX) new JavacTask(TOOLBOX)
.sources(code) .sources(code)
.classpath(".") .classpath(".")
.options("--enable-preview", "-source", JDK_VERSION, "-encoding", "utf8") .options("-encoding", "utf8")
.run(); .run();
String output = new JavaTask(TOOLBOX) String output = new JavaTask(TOOLBOX)
.vmOptions("--enable-preview")
.classpath(".") .classpath(".")
.classArgs("LineTerminatorTest") .classArgs("LineTerminatorTest")
.run() .run()
@ -210,7 +209,7 @@ public class TextBlockAPI {
new JavacTask(TOOLBOX) new JavacTask(TOOLBOX)
.sources(code) .sources(code)
.classpath(".") .classpath(".")
.options("--enable-preview", "-source", JDK_VERSION, "-encoding", "utf8", "-Xlint") .options("-encoding", "utf8", "-Xlint")
.run(); .run();
} }
@ -221,7 +220,7 @@ public class TextBlockAPI {
String output = new JavacTask(TOOLBOX) String output = new JavacTask(TOOLBOX)
.sources(source) .sources(source)
.classpath(".") .classpath(".")
.options("--enable-preview", "-source", JDK_VERSION, "-encoding", "utf8") .options("-encoding", "utf8")
.run() .run()
.writeAll() .writeAll()
.getOutput(Task.OutputKind.DIRECT); .getOutput(Task.OutputKind.DIRECT);
@ -242,7 +241,7 @@ public class TextBlockAPI {
String errors = new JavacTask(TOOLBOX) String errors = new JavacTask(TOOLBOX)
.sources(source) .sources(source)
.classpath(".") .classpath(".")
.options("-XDrawDiagnostics", "--enable-preview", "-source", JDK_VERSION, "-encoding", "utf8") .options("-XDrawDiagnostics", "-encoding", "utf8")
.run(Task.Expect.FAIL) .run(Task.Expect.FAIL)
.writeAll() .writeAll()
.getOutput(Task.OutputKind.DIRECT); .getOutput(Task.OutputKind.DIRECT);

View File

@ -2,7 +2,7 @@
* @test /nodynamiccopyright/ * @test /nodynamiccopyright/
* @bug 8227640 * @bug 8227640
* @summary Verify that illegal escapes in text blocks do not crash the javac. * @summary Verify that illegal escapes in text blocks do not crash the javac.
* @compile/fail/ref=TextBlockIllegalEscape.out --enable-preview -source ${jdk.version} -XDrawDiagnostics TextBlockIllegalEscape.java * @compile/fail/ref=TextBlockIllegalEscape.out -XDrawDiagnostics TextBlockIllegalEscape.java
*/ */
public class TextBlockIllegalEscape { public class TextBlockIllegalEscape {

View File

@ -1,4 +1,2 @@
TextBlockIllegalEscape.java:11:13: compiler.err.illegal.esc.char TextBlockIllegalEscape.java:11:13: compiler.err.illegal.esc.char
- compiler.note.preview.filename: TextBlockIllegalEscape.java
- compiler.note.preview.recompile
1 error 1 error

View File

@ -25,8 +25,8 @@
* @test * @test
* @bug 8223967 * @bug 8223967
* @summary Unit tests for Text Block language changes * @summary Unit tests for Text Block language changes
* @compile --enable-preview -source ${jdk.version} -encoding utf8 TextBlockLang.java * @compile -encoding utf8 TextBlockLang.java
* @run main/othervm --enable-preview TextBlockLang * @run main TextBlockLang
*/ */
public class TextBlockLang { public class TextBlockLang {

View File

@ -21,10 +21,7 @@
* questions. * questions.
*/ */
// key: compiler.warn.preview.feature.use.plural
// key: compiler.misc.feature.text.blocks
// key: compiler.err.unclosed.text.block // key: compiler.err.unclosed.text.block
// options: --enable-preview -source ${jdk.version} -Xlint:preview
class TextBlockCloseDelimiter { class TextBlockCloseDelimiter {
String m() { String m() {

View File

@ -21,10 +21,7 @@
* questions. * questions.
*/ */
// key: compiler.warn.preview.feature.use.plural
// key: compiler.misc.feature.text.blocks
// key: compiler.err.illegal.text.block.open // key: compiler.err.illegal.text.block.open
// options: --enable-preview -source ${jdk.version} -Xlint:preview
class TextBlockOpenDelimiter { class TextBlockOpenDelimiter {
String m() { String m() {

View File

@ -0,0 +1,35 @@
/*
* Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
// key: compiler.misc.feature.text.blocks
// key: compiler.err.feature.not.supported.in.source.plural
// key: compiler.warn.source.no.system.modules.path
// options: -source 14
class TextBlockSource {
String m() {
return """
abc
""";
}
}

View File

@ -21,11 +21,9 @@
* questions. * questions.
*/ */
// key: compiler.warn.preview.feature.use.plural
// key: compiler.misc.feature.text.blocks
// key: compiler.warn.inconsistent.white.space.indentation // key: compiler.warn.inconsistent.white.space.indentation
// key: compiler.warn.trailing.white.space.will.be.removed // key: compiler.warn.trailing.white.space.will.be.removed
// options: --enable-preview -source ${jdk.version} -Xlint:preview,text-blocks // options: -Xlint:text-blocks
class TextBlockWhitespace { class TextBlockWhitespace {
String m() { String m() {