8241741: Implement Text Blocks as a standard feature in javac
Reviewed-by: jlahoda
This commit is contained in:
parent
d9bf934831
commit
ef8537ec1a
@ -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,
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
@ -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).
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
// Add line terminator to string buffer.
|
skipLineTerminator();
|
||||||
int start = reader.bp;
|
// Add line terminator to string buffer.
|
||||||
if (isCRLF()) {
|
reader.putChar('\n', false);
|
||||||
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;
|
||||||
|
@ -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);
|
||||||
|
@ -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 {
|
||||||
|
@ -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
|
||||||
|
@ -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 {
|
||||||
|
@ -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() {
|
||||||
|
@ -21,10 +21,7 @@
|
|||||||
* questions.
|
* questions.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// key: compiler.warn.preview.feature.use.plural
|
// key: compiler.err.illegal.text.block.open
|
||||||
// key: compiler.misc.feature.text.blocks
|
|
||||||
// key: compiler.err.illegal.text.block.open
|
|
||||||
// options: --enable-preview -source ${jdk.version} -Xlint:preview
|
|
||||||
|
|
||||||
class TextBlockOpenDelimiter {
|
class TextBlockOpenDelimiter {
|
||||||
String m() {
|
String m() {
|
||||||
|
@ -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
|
||||||
|
""";
|
||||||
|
}
|
||||||
|
}
|
@ -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() {
|
||||||
|
Loading…
Reference in New Issue
Block a user