diff --git a/test/regex/PikeVM.java b/test/regex/PikeVM.java index 9219280c68..db8ad44ef7 100644 --- a/test/regex/PikeVM.java +++ b/test/regex/PikeVM.java @@ -70,6 +70,34 @@ class PikeVM implements PikeVMOpcodes { offsets = new int[program.length + 1][]; } + public ThreadQueue(int startPC) { + head = tail = startPC; + next = new int[program.length + 1]; + offsets = new int[program.length + 1][]; + offsets[head] = new int[offsetsCount]; + } + + public int queueOneImmediately(ThreadQueue into) { + for (;;) { + if (head < 0) { + return -1; + } + boolean wasQueued = queueNext(head, head, into); + int pc = head; + if (head == tail) { + head = tail = -1; + } else { + head = next[pc] - 1; + next[pc] = 0; + } + offsets[pc] = null; + if (wasQueued) { + into.tail = pc; + return pc; + } + } + } + /** * Schedules the instruction at {@code nextPC} to be executed immediately. *
@@ -141,8 +169,7 @@ class PikeVM implements PikeVMOpcodes { } else { next.next[next.tail] = nextPC + 1; } - next.offsets[nextPC] = - currentPC < 0 ? new int[offsetsCount] : offsets[currentPC]; + next.offsets[nextPC] = offsets[currentPC]; next.tail = nextPC; return true; } @@ -223,7 +250,7 @@ class PikeVM implements PikeVMOpcodes { ThreadQueue next = new ThreadQueue(); // initialize the first thread - current.queueNext(-1, 0, current); + ThreadQueue queued = new ThreadQueue(0); if (!anchorStart) { // this requires non-greedy matching throw new UnsupportedOperationException(); @@ -231,7 +258,7 @@ class PikeVM implements PikeVMOpcodes { boolean foundMatch = false; for (int i = start; i <= end; ++i) { - if (current.isEmpty()) { + if (queued.isEmpty()) { // no threads left return foundMatch; } @@ -240,6 +267,9 @@ class PikeVM implements PikeVMOpcodes { int pc = -1; for (;;) { pc = current.next(pc); + if (pc < 0) { + pc = queued.queueOneImmediately(current); + } if (pc < 0) { break; } @@ -274,6 +304,10 @@ class PikeVM implements PikeVMOpcodes { current.queueImmediately(pc, program[pc + 1], true); current.queueImmediately(pc, pc + 2, false); break; + case SPLIT_JMP: + current.queueImmediately(pc, pc + 2, true); + current.queueImmediately(pc, program[pc + 1], false); + break; case JMP: current.queueImmediately(pc, program[pc + 1], false); break; @@ -292,8 +326,8 @@ class PikeVM implements PikeVMOpcodes { current.clean(); // prepare for next step - ThreadQueue swap = current; - current = next; + ThreadQueue swap = queued; + queued = next; next = swap; } return foundMatch; diff --git a/test/regex/PikeVMOpcodes.java b/test/regex/PikeVMOpcodes.java index c12ad99427..e281aa25ab 100644 --- a/test/regex/PikeVMOpcodes.java +++ b/test/regex/PikeVMOpcodes.java @@ -25,5 +25,6 @@ interface PikeVMOpcodes { final static int SAVE_OFFSET = -40; final static int SPLIT = -50; - final static int JMP = -51; + final static int SPLIT_JMP = -51; // this split prefers to jump + final static int JMP = -52; }