| // Copyright 2011 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package syntax |
| |
| import "unicode" |
| |
| // A patchList is a list of instruction pointers that need to be filled in (patched). |
| // Because the pointers haven't been filled in yet, we can reuse their storage |
| // to hold the list. It's kind of sleazy, but works well in practice. |
| // See https://swtch.com/~rsc/regexp/regexp1.html for inspiration. |
| // |
| // These aren't really pointers: they're integers, so we can reinterpret them |
| // this way without using package unsafe. A value l.head denotes |
| // p.inst[l.head>>1].Out (l.head&1==0) or .Arg (l.head&1==1). |
| // head == 0 denotes the empty list, okay because we start every program |
| // with a fail instruction, so we'll never want to point at its output link. |
| type patchList struct { |
| head, tail uint32 |
| } |
| |
| func makePatchList(n uint32) patchList { |
| return patchList{n, n} |
| } |
| |
| func (l patchList) patch(p *Prog, val uint32) { |
| head := l.head |
| for head != 0 { |
| i := &p.Inst[head>>1] |
| if head&1 == 0 { |
| head = i.Out |
| i.Out = val |
| } else { |
| head = i.Arg |
| i.Arg = val |
| } |
| } |
| } |
| |
| func (l1 patchList) append(p *Prog, l2 patchList) patchList { |
| if l1.head == 0 { |
| return l2 |
| } |
| if l2.head == 0 { |
| return l1 |
| } |
| |
| i := &p.Inst[l1.tail>>1] |
| if l1.tail&1 == 0 { |
| i.Out = l2.head |
| } else { |
| i.Arg = l2.head |
| } |
| return patchList{l1.head, l2.tail} |
| } |
| |
| // A frag represents a compiled program fragment. |
| type frag struct { |
| i uint32 // index of first instruction |
| out patchList // where to record end instruction |
| nullable bool // whether fragment can match empty string |
| } |
| |
| type compiler struct { |
| p *Prog |
| } |
| |
| // Compile compiles the regexp into a program to be executed. |
| // The regexp should have been simplified already (returned from re.Simplify). |
| func Compile(re *Regexp) (*Prog, error) { |
| var c compiler |
| c.init() |
| f := c.compile(re) |
| f.out.patch(c.p, c.inst(InstMatch).i) |
| c.p.Start = int(f.i) |
| return c.p, nil |
| } |
| |
| func (c *compiler) init() { |
| c.p = new(Prog) |
| c.p.NumCap = 2 // implicit ( and ) for whole match $0 |
| c.inst(InstFail) |
| } |
| |
| var anyRuneNotNL = []rune{0, '\n' - 1, '\n' + 1, unicode.MaxRune} |
| var anyRune = []rune{0, unicode.MaxRune} |
| |
| func (c *compiler) compile(re *Regexp) frag { |
| switch re.Op { |
| case OpNoMatch: |
| return c.fail() |
| case OpEmptyMatch: |
| return c.nop() |
| case OpLiteral: |
| if len(re.Rune) == 0 { |
| return c.nop() |
| } |
| var f frag |
| for j := range re.Rune { |
| f1 := c.rune(re.Rune[j:j+1], re.Flags) |
| if j == 0 { |
| f = f1 |
| } else { |
| f = c.cat(f, f1) |
| } |
| } |
| return f |
| case OpCharClass: |
| return c.rune(re.Rune, re.Flags) |
| case OpAnyCharNotNL: |
| return c.rune(anyRuneNotNL, 0) |
| case OpAnyChar: |
| return c.rune(anyRune, 0) |
| case OpBeginLine: |
| return c.empty(EmptyBeginLine) |
| case OpEndLine: |
| return c.empty(EmptyEndLine) |
| case OpBeginText: |
| return c.empty(EmptyBeginText) |
| case OpEndText: |
| return c.empty(EmptyEndText) |
| case OpWordBoundary: |
| return c.empty(EmptyWordBoundary) |
| case OpNoWordBoundary: |
| return c.empty(EmptyNoWordBoundary) |
| case OpCapture: |
| bra := c.cap(uint32(re.Cap << 1)) |
| sub := c.compile(re.Sub[0]) |
| ket := c.cap(uint32(re.Cap<<1 | 1)) |
| return c.cat(c.cat(bra, sub), ket) |
| case OpStar: |
| return c.star(c.compile(re.Sub[0]), re.Flags&NonGreedy != 0) |
| case OpPlus: |
| return c.plus(c.compile(re.Sub[0]), re.Flags&NonGreedy != 0) |
| case OpQuest: |
| return c.quest(c.compile(re.Sub[0]), re.Flags&NonGreedy != 0) |
| case OpConcat: |
| if len(re.Sub) == 0 { |
| return c.nop() |
| } |
| var f frag |
| for i, sub := range re.Sub { |
| if i == 0 { |
| f = c.compile(sub) |
| } else { |
| f = c.cat(f, c.compile(sub)) |
| } |
| } |
| return f |
| case OpAlternate: |
| var f frag |
| for _, sub := range re.Sub { |
| f = c.alt(f, c.compile(sub)) |
| } |
| return f |
| } |
| panic("regexp: unhandled case in compile") |
| } |
| |
| func (c *compiler) inst(op InstOp) frag { |
| // TODO: impose length limit |
| f := frag{i: uint32(len(c.p.Inst)), nullable: true} |
| c.p.Inst = append(c.p.Inst, Inst{Op: op}) |
| return f |
| } |
| |
| func (c *compiler) nop() frag { |
| f := c.inst(InstNop) |
| f.out = makePatchList(f.i << 1) |
| return f |
| } |
| |
| func (c *compiler) fail() frag { |
| return frag{} |
| } |
| |
| func (c *compiler) cap(arg uint32) frag { |
| f := c.inst(InstCapture) |
| f.out = makePatchList(f.i << 1) |
| c.p.Inst[f.i].Arg = arg |
| |
| if c.p.NumCap < int(arg)+1 { |
| c.p.NumCap = int(arg) + 1 |
| } |
| return f |
| } |
| |
| func (c *compiler) cat(f1, f2 frag) frag { |
| // concat of failure is failure |
| if f1.i == 0 || f2.i == 0 { |
| return frag{} |
| } |
| |
| // TODO: elide nop |
| |
| f1.out.patch(c.p, f2.i) |
| return frag{f1.i, f2.out, f1.nullable && f2.nullable} |
| } |
| |
| func (c *compiler) alt(f1, f2 frag) frag { |
| // alt of failure is other |
| if f1.i == 0 { |
| return f2 |
| } |
| if f2.i == 0 { |
| return f1 |
| } |
| |
| f := c.inst(InstAlt) |
| i := &c.p.Inst[f.i] |
| i.Out = f1.i |
| i.Arg = f2.i |
| f.out = f1.out.append(c.p, f2.out) |
| f.nullable = f1.nullable || f2.nullable |
| return f |
| } |
| |
| func (c *compiler) quest(f1 frag, nongreedy bool) frag { |
| f := c.inst(InstAlt) |
| i := &c.p.Inst[f.i] |
| if nongreedy { |
| i.Arg = f1.i |
| f.out = makePatchList(f.i << 1) |
| } else { |
| i.Out = f1.i |
| f.out = makePatchList(f.i<<1 | 1) |
| } |
| f.out = f.out.append(c.p, f1.out) |
| return f |
| } |
| |
| // loop returns the fragment for the main loop of a plus or star. |
| // For plus, it can be used after changing the entry to f1.i. |
| // For star, it can be used directly when f1 can't match an empty string. |
| // (When f1 can match an empty string, f1* must be implemented as (f1+)? |
| // to get the priority match order correct.) |
| func (c *compiler) loop(f1 frag, nongreedy bool) frag { |
| f := c.inst(InstAlt) |
| i := &c.p.Inst[f.i] |
| if nongreedy { |
| i.Arg = f1.i |
| f.out = makePatchList(f.i << 1) |
| } else { |
| i.Out = f1.i |
| f.out = makePatchList(f.i<<1 | 1) |
| } |
| f1.out.patch(c.p, f.i) |
| return f |
| } |
| |
| func (c *compiler) star(f1 frag, nongreedy bool) frag { |
| if f1.nullable { |
| // Use (f1+)? to get priority match order correct. |
| // See golang.org/issue/46123. |
| return c.quest(c.plus(f1, nongreedy), nongreedy) |
| } |
| return c.loop(f1, nongreedy) |
| } |
| |
| func (c *compiler) plus(f1 frag, nongreedy bool) frag { |
| return frag{f1.i, c.loop(f1, nongreedy).out, f1.nullable} |
| } |
| |
| func (c *compiler) empty(op EmptyOp) frag { |
| f := c.inst(InstEmptyWidth) |
| c.p.Inst[f.i].Arg = uint32(op) |
| f.out = makePatchList(f.i << 1) |
| return f |
| } |
| |
| func (c *compiler) rune(r []rune, flags Flags) frag { |
| f := c.inst(InstRune) |
| f.nullable = false |
| i := &c.p.Inst[f.i] |
| i.Rune = r |
| flags &= FoldCase // only relevant flag is FoldCase |
| if len(r) != 1 || unicode.SimpleFold(r[0]) == r[0] { |
| // and sometimes not even that |
| flags &^= FoldCase |
| } |
| i.Arg = uint32(flags) |
| f.out = makePatchList(f.i << 1) |
| |
| // Special cases for exec machine. |
| switch { |
| case flags&FoldCase == 0 && (len(r) == 1 || len(r) == 2 && r[0] == r[1]): |
| i.Op = InstRune1 |
| case len(r) == 2 && r[0] == 0 && r[1] == unicode.MaxRune: |
| i.Op = InstRuneAny |
| case len(r) == 4 && r[0] == 0 && r[1] == '\n'-1 && r[2] == '\n'+1 && r[3] == unicode.MaxRune: |
| i.Op = InstRuneAnyNotNL |
| } |
| |
| return f |
| } |