corda/classpath/java/util/regex/Compiler.java
Johannes Schindelin 6626b477ad Replace java.util.regex.* with the new regular expression engine
Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de>
2013-12-04 12:52:03 -06:00

534 lines
15 KiB
Java

/* Copyright (c) 2008-2013, Avian Contributors
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
There is NO WARRANTY for this software. See license.txt for
details. */
package java.util.regex;
import java.util.ArrayList;
import java.util.Stack;
/**
* Compiles regular expressions into {@link PikeVM}s.
*
* @author Johannes Schindelin
*/
class Compiler implements PikeVMOpcodes {
private final static CharacterMatcher regularCharacter =
CharacterMatcher.parse("[^\\\\.*+?|\\[\\]{}()^$]");
private static class Output {
private int[] program;
private int offset;
private int groupCount = -1;
private int findPreambleSize;
private ArrayList<CharacterMatcher> classes;
private ArrayList<PikeVM> lookarounds;
public Output(Expression expr) {
// try-run to determine the code size
expr.writeCode(this);
program = new int[offset];
offset = 0;
groupCount = -1;
classes = new ArrayList<CharacterMatcher>();
lookarounds = new ArrayList<PikeVM>();
// write it out!
expr.writeCode(this);
}
public void add(int opcode) {
if (program != null) {
program[offset] = opcode;
}
offset++;
}
public int markJump() {
return offset++;
}
public void setJump(int mark) {
if (program != null) {
program[mark] = offset;
}
}
public void markFindPreambleEnd() {
findPreambleSize = offset;
}
public PikeVM toVM() {
CharacterMatcher[] classes = new CharacterMatcher[this.classes.size()];
this.classes.toArray(classes);
PikeVM[] lookarounds = new PikeVM[this.lookarounds.size()];
this.lookarounds.toArray(lookarounds);
return new PikeVM(program, findPreambleSize, groupCount, classes,
lookarounds);
}
public int addClass(CharacterMatcher characterClass) {
if (program == null) {
return -1;
}
int result = classes.size();
classes.add(characterClass);
return result;
}
public int addLookaround(PikeVM lookaround) {
if (program == null) {
return -1;
}
int result = lookarounds.size();
lookarounds.add(lookaround);
return result;
}
}
private abstract class Expression {
protected abstract void writeCode(Output output);
}
private class CharacterRange extends Expression {
private final CharacterMatcher characterClass;
public CharacterRange(CharacterMatcher characterClass) {
this.characterClass = characterClass;
}
protected void writeCode(Output output) {
output.add(CHARACTER_CLASS);
output.add(output.addClass(characterClass));
}
public String toString() {
return characterClass.toString();
}
}
private class Repeat extends Expression {
private Expression expr;
private int minCount, maxCount;
private boolean greedy;
public Repeat(Expression expr, int minCount, int maxCount, boolean greedy) {
if (minCount < 0) {
throw new RuntimeException("Unexpected min count: " + minCount);
}
if (maxCount != -1) {
if (maxCount == 0) {
throw new RuntimeException("Unexpected max count: " + maxCount);
}
if (minCount > maxCount) {
throw new RuntimeException("Unexpected range: " + minCount + ", " + maxCount);
}
}
this.expr = expr;
this.minCount = minCount;
this.maxCount = maxCount;
this.greedy = greedy;
}
protected void writeCode(Output output) {
int start = output.offset;
int splitJmp = greedy ? SPLIT_JMP : SPLIT;
int split = greedy ? SPLIT : SPLIT_JMP;
for (int i = 1; i < minCount; ++ i) {
expr.writeCode(output);
}
if (maxCount == -1) {
if (minCount > 0) {
int jump = output.offset;
expr.writeCode(output);
output.add(splitJmp);
output.add(jump);
} else {
output.add(split);
int jump = output.markJump();
expr.writeCode(output);
output.add(splitJmp);
output.add(start + 2);
output.setJump(jump);
}
} else {
if (minCount > 0) {
expr.writeCode(output);
}
if (maxCount > minCount) {
int[] jumps = new int[maxCount - minCount];
for (int i = 0; i < jumps.length; ++ i) {
output.add(split);
jumps[i] = output.markJump();
expr.writeCode(output);
}
for (int jump : jumps) {
output.setJump(jump);
}
}
}
}
public String toString() {
String qualifier = greedy ? "" : "?";
if (minCount == 0 && maxCount < 2) {
return expr.toString() + (minCount < 0 ? "*" : "?") + qualifier;
}
if (minCount == 1 && maxCount < 0) {
return expr.toString() + "+" + qualifier;
}
return expr.toString() + "{" + minCount + ","
+ (maxCount < 0 ? "" : "" + maxCount) + "}" + qualifier;
}
}
private class Group extends Expression {
private final boolean capturing;
private ArrayList<Expression> list = new ArrayList<Expression>();
private ArrayList<Group> alternatives;
public Group(boolean capturing, ArrayList<Expression> initialList) {
this.capturing = capturing;
if (initialList != null) {
list.addAll(initialList);
}
}
public void push(Expression expr) {
list.add(expr);
}
public void push(final int c) {
push(new Expression() {
public void writeCode(Output output) {
output.add(c);
}
public String toString() {
if (c >= 0) {
return "" + (char)c;
}
switch (c) {
case DOT:
return ".";
case WORD_BOUNDARY:
return "\\b";
case NON_WORD_BOUNDARY:
return "\\B";
case LINE_START:
return "^";
case LINE_END:
return "$";
default:
throw new RuntimeException("Unhandled opcode: " + c);
}
}
});
}
public void startAlternative() {
if (alternatives == null) {
alternatives = new ArrayList<Group>();
}
alternatives.add(new Group(false, list));
list.clear();
}
public Expression pop() {
Expression result = list.remove(list.size() - 1);
return result;
}
protected void writeCode(Output output) {
int groupIndex = -1;
if (capturing) {
groupIndex = ++ output.groupCount;
output.add(SAVE_OFFSET);
output.add(2 * groupIndex);
}
int[] jumps = null;
if (alternatives != null) {
jumps = new int[alternatives.size()];
int i = 0;
for (Group alternative : alternatives) {
output.add(SPLIT);
int jump = output.markJump();
alternative.writeCode(output);
output.add(JMP);
jumps[i++] = output.markJump();
output.setJump(jump);
}
}
for (Expression expr : list) {
expr.writeCode(output);
}
if (jumps != null) {
for (int jump : jumps) {
output.setJump(jump);
}
}
if (capturing) {
output.add(SAVE_OFFSET);
output.add(2 * groupIndex + 1);
}
}
public String toString() {
StringBuilder builder = new StringBuilder();
if (alternatives != null || list.size() > 1) {
builder.append('(');
if (!capturing) {
builder.append("?:");
}
}
if (alternatives != null) {
for (Group alternative : alternatives) {
builder.append(alternative).append('|');
}
}
for (Expression expr : list) {
builder.append(expr);
}
if (alternatives != null || list.size() > 1) {
builder.append(')');
}
return builder.toString();
}
}
private class Lookaround extends Expression {
private final Group group = new Group(false, null);
private final boolean forward, negative;
public Lookaround(boolean forward, boolean negative) {
this.forward = forward;
this.negative = negative;
}
@Override
protected void writeCode(Output output) {
PikeVM vm = new Output(group).toVM();
if (!forward) {
vm.reverse();
}
output.add(forward ?
(negative ? NEGATIVE_LOOKAHEAD : LOOKAHEAD) :
(negative ? NEGATIVE_LOOKAHEAD : LOOKBEHIND));
output.add(output.addLookaround(vm));
}
public String toString() {
String inner = group.toString();
if (inner.startsWith("(?:")) {
inner = inner.substring(3);
} else {
inner += ")";
}
return "(?=" + inner;
}
}
private class Group0 extends Expression {
private final Group group;
public Group0() {
group = new Group(true, null);
}
public void writeCode(Output output) {
// find() preamble
int start = output.offset;
output.add(SPLIT_JMP);
output.add(start + 5);
output.add(DOTALL);
output.add(SPLIT);
output.add(start + 2);
output.markFindPreambleEnd();
group.writeCode(output);
}
public String toString() {
String inner = group.toString();
return inner.startsWith("(?:") && inner.endsWith(")") ?
inner.substring(1, inner.length() - 1) : inner;
}
}
private Group0 root;
private Stack<Group> groups;
public Compiler() {
root = new Group0();
groups = new Stack<Group>();
groups.add(root.group);
}
public Pattern compile(String regex) {
char[] array = regex.toCharArray();
CharacterMatcher.Parser characterClassParser =
new CharacterMatcher.Parser(array);
for (int index = 0; index < array.length; ++ index) {
char c = array[index];
Group current = groups.peek();
if (regularCharacter.matches(c)) {
current.push(c);
continue;
}
switch (c) {
case '.':
current.push(DOT);
continue;
case '\\':
int unescaped = characterClassParser.parseEscapedCharacter(index + 1);
if (unescaped >= 0) {
index = characterClassParser.getEndOffset() - 1;
current.push((char)unescaped);
continue;
}
CharacterMatcher characterClass = characterClassParser.parseClass(index);
if (characterClass != null) {
index = characterClassParser.getEndOffset() - 1;
current.push(new CharacterRange(characterClass));
continue;
}
switch (array[index + 1]) {
case 'b':
index++;
current.push(WORD_BOUNDARY);
continue;
case 'B':
index++;
current.push(NON_WORD_BOUNDARY);
continue;
}
throw new RuntimeException("Parse error @" + index + ": " + regex);
case '?':
case '*':
case '+': {
boolean greedy = true;
if (index + 1 < array.length && array[index + 1] == '?') {
greedy = false;
++ index;
}
current.push(new Repeat(current.pop(),
c == '+' ? 1 : 0, c == '?' ? 1 : -1, greedy));
continue;
}
case '{': {
++ index;
int length = characterClassParser.digits(index, 8, 10);
int min = Integer.parseInt(regex.substring(index, index + length));
int max = min;
index += length - 1;
c = index + 1 < array.length ? array[index + 1] : 0;
if (c == ',') {
++ index;
length = characterClassParser.digits(index + 1, 8, 10);
max = length == 0 ? -1 :
Integer.parseInt(regex.substring(index + 1, index + 1 + length));
index += length;
c = index + 1< array.length ? array[index + 1] : 0;
}
if (c != '}') {
throw new RuntimeException("Invalid quantifier @" + index + ": "
+ regex);
}
++ index;
boolean greedy = true;
if (index + 1 < array.length && array[index + 1] == '?') {
++ index;
greedy = false;
}
current.push(new Repeat(current.pop(), min, max, greedy));
continue;
}
case '(': {
boolean capturing = true;
if (index + 1 < array.length && array[index + 1] == '?') {
index += 2;
if (index >= array.length) {
throw new RuntimeException("Short pattern @" + index + ": "
+ regex);
}
c = array[index];
boolean lookAhead = true;
if (c == '<') {
if (++ index >= array.length) {
throw new RuntimeException("Short pattern @" + index + ": "
+ regex);
}
lookAhead = false;
c = array[index];
if (c != '=' && c != '!') {
throw new IllegalArgumentException("Named groups not supported @"
+ index + ": " + regex);
}
}
switch (c) {
case ':':
capturing = false;
break;
case '!':
case '=': {
capturing = false;
Lookaround lookaround = new Lookaround(lookAhead, c == '!');
current.push(lookaround);
groups.push(lookaround.group);
continue;
}
default:
throw new UnsupportedOperationException("Not yet supported: "
+ regex.substring(index));
}
}
current.push(groups.push(new Group(capturing, null)));
continue;
}
case ')':
if (groups.size() < 2) {
throw new RuntimeException("Invalid group close @" + index + ": "
+ regex);
}
groups.pop();
continue;
case '[': {
CharacterMatcher matcher = characterClassParser.parseClass(index);
if (matcher == null) {
throw new RuntimeException("Invalid range @" + index + ": " + regex);
}
current.push(new CharacterRange(matcher));
index = characterClassParser.getEndOffset() - 1;
continue;
}
case '|':
current.startAlternative();
continue;
case '^':
current.push(LINE_START);
continue;
case '$':
current.push(LINE_END);
continue;
default:
throw new RuntimeException("Parse error @" + index + ": " + regex);
}
}
if (groups.size() != 1) {
throw new IllegalArgumentException("Unclosed groups: ("
+ (groups.size() - 1) + "): " + regex);
}
PikeVM vm = new Output(root).toVM();
String plain = vm.isPlainString();
if (plain != null) {
return new TrivialPattern(regex, plain, 0);
}
return new RegexPattern(regex, 0, vm);
}
}