shithub: mc

Download patch

ref: 447bbbd25bbb42f2c3da11ac298f52a31bfb573d
parent: a32ef8d8bb4254a5e376fcfb14cf5355f778a29a
author: Ori Bernstein <[email protected]>
date: Mon May 16 17:32:49 EDT 2016

Add better regex debugging.

--- a/lib/regex/compile.myr
+++ b/lib/regex/compile.myr
@@ -6,8 +6,7 @@
 pkg regex =
 	const parse	: (re : byte[:]	-> std.result(ast#, status))
 	const compile	: (re : byte[:] -> std.result(regex#, status))
-	const compileast	: (re : ast# -> regex#)
-	const dbgcompile	: (re : byte[:] -> std.result(regex#, status))
+	const dbgcompile	: (re : byte[:], trace : bool -> std.result(regex#, status))
 	const free	: (re : regex# -> void)
 ;;
 
@@ -30,7 +29,7 @@
 	| `None:	-> `std.Fail `Incomplete
 	| `Fail f:	-> `std.Fail f
 	| `Some t:
-		if re.pat.len > 0
+		if re.pat.len != re.idx
 			-> `std.Fail `Incomplete
 		else
 			-> `std.Ok t
@@ -39,11 +38,18 @@
 }
 
 /* Compiles a pattern into a debug regex. This can be verbose. */
-const dbgcompile = {pat
-	var re
+const dbgcompile = {pat, trace
+	var re, res
 
-	re = std.mk([.pat = pat, .nmatch = 1, .debug = true])
-	-> regexcompile(re, 0)
+	re = std.mk([
+		.pat = pat,
+		.nmatch = 1,
+		.debug = true,
+		.trace = trace,
+		.astloc = std.mkht(std.ptrhash, std.ptreq),
+	])
+	res = regexcompile(re, 0)
+	-> res
 }
 
 /* compiles a pattern into an allocated regex */
@@ -53,18 +59,19 @@
 	| `Fail f:	-> `std.Fail f
 	| `Some t:
 		/*
-		we can stop early if we get 
+		we can bail out of parsing early if we get 
 		an incorrectly encoded char
 		*/
-		if re.pat.len > 0
+		if re.pat.len != re.idx
 			astfree(t)
 			-> `std.Fail (`Incomplete)
 		;;
 		dump(re, t, 0)
-		append(re, `Ilbra 0)
+		append(re, `Ilbra 0, t)
 		gen(re, t)
-		append(re, `Irbra 0)
-		append(re, `Imatch id)
+		std.htput(re.astloc, t, re.pat.len)
+		append(re, `Irbra 0, t)
+		append(re, `Imatch id, t)
 		idump(re)
 		astfree(t)
 		-> `std.Ok re
@@ -72,23 +79,14 @@
 	-> `std.Fail (`Noimpl)
 }
 
-const compileast = {ast
-	var re
-
-	re = std.mk([.pat = "", .nmatch = 1])
-	dump(re, ast, 0)
-	append(re, `Ilbra 0)
-	gen(re, ast)
-	append(re, `Irbra 0)
-	append(re, `Imatch 0)
-	idump(re)
-	-> re
-}
-
 const free = {re
 	/* all the threads should be dead,
 	 so we shouldn't have to free any*/
 	std.slfree(re.prog)
+	if re.debug
+		std.htfree(re.astloc)
+		std.slfree(re.pcidx)
+	;;
 	std.free(re)
 }
 
@@ -106,23 +104,23 @@
 	|`Quest	a:	genquest(re, a)
 
 	/* end matches */
-	|`Chr	c:	genchar(re, c)
-	|`Ranges  sl:	genranges(re, sl)
+	|`Chr	c:	genchar(re, c, t)
+	|`Ranges  sl:	genranges(re, sl, t)
 
 	/* meta */
-	|`Bol:	append(re, `Ibol)
-	|`Eol:	append(re, `Ibol)
-	|`Bow:	append(re, `Ibow)
-	|`Eow:	append(re, `Ieow)
+	|`Bol:	append(re, `Ibol, t)
+	|`Eol:	append(re, `Ibol, t)
+	|`Bow:	append(re, `Ibow, t)
+	|`Eow:	append(re, `Ieow, t)
 	|`Cap	(m, a):
-		append(re, `Ilbra m)
+		append(re, `Ilbra m, t)
 		gen(re, a)
-		append(re, `Irbra m)
+		append(re, `Irbra m, t)
 	;;
 	-> re.proglen
 }
 
-const genranges = {re, sl
+const genranges = {re, sl, ast
 	var lbuf : byte[4], hbuf : byte[4], boundbuf : byte[4]
 	var lsz, hsz, bsz, i
 	var rt : rangetrie#
@@ -146,10 +144,10 @@
 		;;
 		rtinsert(rt, lbuf[:lsz], hbuf[:hsz])
 	;;
-	if re.debug
+	if re.trace
 		rtdump(rt, 0)
 	;;
-	rangegen(re, rt, rt.ranges, rt.link, rangeprogsize(rt) + re.proglen)
+	rangegen(re, ast, rt, rt.ranges, rt.link, rangeprogsize(rt) + re.proglen)
 	rtfree(rt)
 	-> re.proglen
 }
@@ -235,7 +233,7 @@
 	std.free(rt)
 }
 
-const rangegen = {re, rt, ranges, links, end
+const rangegen = {re, ast, rt, ranges, links, end
 	var alt, l0, l1, l2
 	var a, b
 	var n
@@ -245,20 +243,20 @@
 		-> re.proglen
 	elif n == 1
 		(a, b) = ranges[0]
-		append(re, `Irange (a, b))
+		append(re, `Irange (a, b), ast)
 		if links[0].end
 			if links[0].ranges.len > 0
-				append(re, `Ifork (re.prog.len + 1, end))
+				append(re, `Ifork (re.prog.len + 1, end), ast)
 			else
-				append(re, `Ijmp end)
+				append(re, `Ijmp end, ast)
 			;;
 		;;
-		rangegen(re, links[0], links[0].ranges, links[0].link, end)
+		rangegen(re, ast, links[0], links[0].ranges, links[0].link, end)
 	else
 		alt = re.proglen
-		l0 = append(re, `Ifork (-1, -1))
-		l1 = rangegen(re, rt, ranges[0:n/2], links[0:n/2], end)
-		l2 = rangegen(re, rt, ranges[n/2:n], links[n/2:n], end)
+		l0 = append(re, `Ifork (-1, -1), ast)
+		l1 = rangegen(re, ast, rt, ranges[0:n/2], links[0:n/2], end)
+		l2 = rangegen(re, ast, rt, ranges[n/2:n], links[n/2:n], end)
 		re.prog[alt] = `Ifork (l0, l1)
 	;;
 	-> re.proglen
@@ -301,10 +299,10 @@
 	var l2
 
 	alt 	= re.proglen
-	l0	= append(re, `Ifork (-1, -1)) /* needs to be replaced */
+	l0	= append(re, `Ifork (-1, -1), l) /* needs to be replaced */
 		  gen(re, l)
 	jmp	= re.proglen
-	l1 	= append(re, `Ijmp -1) /* needs to be replaced */
+	l1 	= append(re, `Ijmp -1, r) /* needs to be replaced */
 	l2	= gen(re, r)
 
 	re.prog[alt] = `Ifork(l0, l1)
@@ -322,9 +320,9 @@
 
 	l0 	= re.proglen
 	alt	= re.proglen
-	l1 	= append(re, `Ifork (-1, -1)) /* needs to be replaced */
+	l1 	= append(re, `Ifork (-1, -1), rep) /* needs to be replaced */
 	jmp	= gen(re, rep)
-	l2	= append(re, `Ijmp -1)
+	l2	= append(re, `Ijmp -1, rep)
 
 
 	/* reluctant matches should prefer jumping to the end. */
@@ -344,7 +342,7 @@
 	var l1
 
 	alt	= re.proglen
-	l0	= append(re, `Ifork (-1, -1)) /* needs to be replaced */
+	l0	= append(re, `Ifork (-1, -1), q) /* needs to be replaced */
 	l1	= gen(re, q)
 	re.prog[alt] = `Ifork (l0, l1)
 	-> re.proglen
@@ -351,7 +349,7 @@
 }
 
 /* generates a single char match */
-const genchar = {re, c
+const genchar = {re, c, ast
 	var b : byte[4]
 	var n
 
@@ -358,16 +356,25 @@
 	n = std.encode(b[:], c)
 	std.assert(n > 0 && n < 4, "non-utf character in regex\n")
 	for var i = 0; i < n; i++
-		append(re, `Ibyte b[i])
+		append(re, `Ibyte b[i], ast)
 	;;
 	-> re.proglen
 }
 
 /* appends an instructon to an re program */
-const append = {re, insn
+const append = {re, insn, ast
+	var sz
+
 	if re.proglen == re.prog.len
-		std.slgrow(&re.prog, std.max(1, 2*re.proglen))
+		sz = std.max(1, 2*re.proglen)
+		std.slgrow(&re.prog, sz)
+		if re.debug
+			std.slgrow(&re.pcidx, sz)
+		;;
 	;;
+	if re.debug
+		re.pcidx[re.proglen] = std.htgetv(re.astloc, ast, -1)
+	;;
 	re.prog[re.proglen] = insn
 	re.proglen++
 	-> re.proglen
@@ -375,11 +382,11 @@
 
 /* instruction dump */
 const idump = {re
-	if !re.debug
+	if !re.trace
 		-> void
 	;;
 	for var i = 0; i < re.proglen; i++
-		std.put("{}:\t", i)
+		std.put("{}@{}:\t", i, re.pcidx[i])
 		match re.prog[i]
 		/* Char matching. Consume exactly one byte from the string. */
 		| `Ibyte b:		std.put("`Ibyte {} ({})\n", b, b castto(char)) 
@@ -407,7 +414,7 @@
 
 /* AST dump */
 const dump = {re, t, indent
-	if !re.debug
+	if !re.trace
 		-> void
 	;;
 	for var i = 0; i < indent; i++
@@ -469,7 +476,7 @@
 const regexparse = {re
 	match altexpr(re)
 	| `Some t:
-		if re.pat.len == 0
+		if re.pat.len == re.idx
 			-> `Some t
 		else
 			astfree(t)
@@ -484,14 +491,16 @@
 
 const altexpr = {re
 	var ret
+	var idx
 
 	match catexpr(re)
 	| `Some t:
 		ret = t
 		if matchc(re, '|')
+			idx = re.idx
 			match altexpr(re)
 			| `Some rhs:
-				ret = std.mk(`Alt (ret, rhs))
+				ret = mk(re, `Alt (ret, rhs), idx)
 			| `None:
 				astfree(ret)
 				-> `Fail (`Incomplete)
@@ -507,13 +516,15 @@
 
 const catexpr = {re
 	var ret
+	var idx
 
 	match repexpr(re)
 	| `Some t: 
+		idx = re.idx
 		ret = t
 		match catexpr(re)
 		| `Some rhs:
-			ret = std.mk(`Cat (t, rhs))
+			ret = mk(re, `Cat (t, rhs), idx)
 		| `Fail f:	-> `Fail f
 		| `None:	/* nothing */
 		;;
@@ -525,23 +536,25 @@
 
 const repexpr = {re
 	var ret
+	var idx
 
 	match baseexpr(re)
 	| `Some t:
+		idx = re.idx
 		if matchc(re, '*')
                         if matchc(re, '?')
-                                ret = std.mk(`Rstar t)
+				ret = mk(re, `Rstar t, idx)
                         else
-				ret = std.mk(`Star t)
+				ret = mk(re, `Star t, idx)
 			;;
 		elif matchc(re, '+')
                         if matchc(re, '?')
-				ret = std.mk(`Rplus t)
+				ret = mk(re, `Rplus t, idx)
 			else
-				ret = std.mk(`Plus t)
+				ret = mk(re, `Plus t, idx)
 			;;
 		elif matchc(re, '?')
-			ret = std.mk(`Quest t)
+			ret = mk(re, `Quest t, idx)
 		else
 			ret = t
 		;;
@@ -552,9 +565,9 @@
 }
 
 const baseexpr = {re
-	var ret, m
+	var ret, m, idx
 
-	if re.pat.len == 0
+	if re.pat.len == re.idx
 		-> `None
 	;;
 	match peekc(re)
@@ -565,16 +578,19 @@
 	| '+':	-> `Fail `Badrep '+'
 	| '?':	-> `Fail `Badrep '?'
 	| '[':	-> chrclass(re)
-	| '.':	getc(re); ret = std.mk(`Ranges std.sldup([[0, std.Maxcharval]][:]))
-	| '^':	getc(re); ret = std.mk(`Bol)
-	| '$':	getc(re); ret = std.mk(`Eol)
+	| '^':	getc(re); ret = mk(re, `Bol, re.idx)
+	| '$':	getc(re); ret = mk(re, `Eol, re.idx)
+	| '.':	
+		getc(re);
+		ret = mk(re, `Ranges std.sldup([[0, std.Maxcharval]][:]), re.idx)
 	| '(':	
 		m = re.nmatch++
+		idx = re.idx
 		getc(re)
 		match altexpr(re)
 		| `Some s:
 			if matchc(re, ')')
-				-> `Some std.mk(`Cap (m, s))
+				-> `Some mk(re, `Cap (m, s), idx)
 			else
 				-> `Fail `Unbalanced '('
 			;;
@@ -583,13 +599,13 @@
 		;;
 	| '\\':
 		getc(re) /* consume the slash */
-		if re.pat.len == 0
+		if re.pat.len == re.idx
 			-> `Fail `Incomplete
 		;;
 		-> escaped(re)
 	| c:
 		getc(re)
-		ret = std.mk(`Chr c)
+		ret = mk(re, `Chr c, re.idx)
 	;;
 	-> `Some ret
 }
@@ -599,18 +615,18 @@
 
 	match getc(re)
 	/* character classes */
-	| 'd': ret = `Some std.mk(`Ranges std.sldup(_ranges.tabasciidigit[:]))
-	| 'x': ret = `Some std.mk(`Ranges std.sldup(_ranges.tabasciixdigit[:]))
-	| 's': ret = `Some std.mk(`Ranges std.sldup(_ranges.tabasciispace[:]))
-	| 'w': ret = `Some std.mk(`Ranges std.sldup(_ranges.tabasciiword[:]))
-	| 'h': ret = `Some std.mk(`Ranges std.sldup(_ranges.tabasciiblank[:]))
+	| 'd': ret = `Some mk(re, `Ranges std.sldup(_ranges.tabasciidigit[:]), re.idx)
+	| 'x': ret = `Some mk(re, `Ranges std.sldup(_ranges.tabasciixdigit[:]), re.idx)
+	| 's': ret = `Some mk(re, `Ranges std.sldup(_ranges.tabasciispace[:]), re.idx)
+	| 'w': ret = `Some mk(re, `Ranges std.sldup(_ranges.tabasciiword[:]), re.idx)
+	| 'h': ret = `Some mk(re, `Ranges std.sldup(_ranges.tabasciiblank[:]), re.idx)
 
 	/* negated character classes */
-	| 'W': ret = `Some std.mk(`Ranges negate(_ranges.tabasciiword[:]))
-	| 'S': ret = `Some std.mk(`Ranges negate(_ranges.tabasciispace[:]))
-	| 'D': ret = `Some std.mk(`Ranges negate(_ranges.tabasciidigit[:]))
-	| 'X': ret = `Some std.mk(`Ranges negate(_ranges.tabasciixdigit[:]))
-	| 'H': ret = `Some std.mk(`Ranges negate(_ranges.tabasciiblank[:]))
+	| 'W': ret = `Some mk(re, `Ranges negate(_ranges.tabasciiword[:]), re.idx)
+	| 'S': ret = `Some mk(re, `Ranges negate(_ranges.tabasciispace[:]), re.idx)
+	| 'D': ret = `Some mk(re, `Ranges negate(_ranges.tabasciidigit[:]), re.idx)
+	| 'X': ret = `Some mk(re, `Ranges negate(_ranges.tabasciixdigit[:]), re.idx)
+	| 'H': ret = `Some mk(re, `Ranges negate(_ranges.tabasciiblank[:]), re.idx)
 
 	/* unicode character classes */
 	| 'p':	ret = unicodeclass(re, false)
@@ -617,16 +633,16 @@
 	| 'P':  ret = unicodeclass(re, true)
 
 	/* operators that need an escape */
-	| '<': ret = `Some std.mk(`Bow)
-	| '>': ret = `Some std.mk(`Eow)
+	| '<': ret = `Some mk(re, `Bow, re.idx)
+	| '>': ret = `Some mk(re, `Eow, re.idx)
 
 	/* escaped metachars */
-	| '^': ret = `Some std.mk(`Chr '^')
-	| '$': ret = `Some std.mk(`Chr '$')
-	| '.': ret = `Some std.mk(`Chr '.')
-	| '+': ret = `Some std.mk(`Chr '+')
-	| '?': ret = `Some std.mk(`Chr '?')
-	| '*': ret = `Some std.mk(`Chr '*')
+	| '^': ret = `Some mk(re, `Chr '^', re.idx)
+	| '$': ret = `Some mk(re, `Chr '$', re.idx)
+	| '.': ret = `Some mk(re, `Chr '.', re.idx)
+	| '+': ret = `Some mk(re, `Chr '+', re.idx)
+	| '?': ret = `Some mk(re, `Chr '?', re.idx)
+	| '*': ret = `Some mk(re, `Chr '*', re.idx)
 	| chr: ret = `Fail `Badescape chr
 	;;
 	-> ret
@@ -637,17 +653,19 @@
 	var tab
 	var t
 	var n
+	var idx
 
-	if re.pat.len == 0
+	if re.pat.len == re.idx
 		-> `Fail (`Incomplete)
 	;;
 	n = 0
-	s = re.pat
+	idx = re.idx
+	s = re.pat[idx:]
 	/* either a single char pattern, or {pat} */
 	match getc(re)
 	| '{':
 		s = s[1:]
-		while re.pat.len > 0
+		while re.pat.len > re.idx
 			c = getc(re)
 			if c == '}'
 				break
@@ -678,9 +696,9 @@
 		-> `Fail (`Badrange s)
 	;;
 	if !neg
-		t = std.mk(`Ranges std.sldup(tab))
+		t = mk(re, `Ranges std.sldup(tab), idx)
 	else
-		t = std.mk(`Ranges negate(tab))
+		t = mk(re, `Ranges negate(tab), idx)
 	;;
 	-> `Some t
 }
@@ -688,16 +706,18 @@
 const chrclass = {re
 	var rl, m, n
 	var neg
+	var idx
 	var t
 
 	/* we know we saw '[' on entry */
 	matchc(re, '[')
+	idx = re.idx
 	neg = false
 	if matchc(re, '^')
 		neg = true
 	;;
 	rl = rangematch(re, [][:])
-	while peekc(re) != ']' && re.pat.len > 0
+	while peekc(re) != ']' && re.pat.len > re.idx
 		rl = rangematch(re, rl)
 	;;
 	if !matchc(re, ']')
@@ -718,9 +738,9 @@
 	if neg
 		n = negate(m)
 		std.slfree(m)
-		t = std.mk(`Ranges n)
+		t = mk(re, `Ranges n, idx)
 	else
-		t = std.mk(`Ranges m)
+		t = mk(re, `Ranges m, idx)
 	;;
 	-> `Some t
 }
@@ -784,14 +804,13 @@
 
 
 const matchc = {re, c
-	var str
 	var chr
 
-	(chr, str) = std.strstep(re.pat)
+	chr = std.decode(re.pat[re.idx:])
 	if chr != c
 		-> false
 	;;
-	re.pat = str
+	re.idx += std.charlen(chr)
 	-> true
 }
 
@@ -798,15 +817,13 @@
 const getc = {re
 	var c
 
-	(c, re.pat) = std.strstep(re.pat)
+	c = std.decode(re.pat[re.idx:])
+	re.idx += std.charlen(c)
 	-> c
 }
 
 const peekc = {re
-	var c
-
-	(c, _) = std.strstep(re.pat)
-	-> c
+	-> std.decode(re.pat[re.idx:])
 }
 
 const astfree = {t
@@ -842,6 +859,16 @@
 	| `Badrange s:	std.sbfmt(sb, "invalid range name {}", s)
 	| `Badescape c:	std.sbfmt(sb, "invalid escape code {}", c)
 	;;
+}
+
+const mk = {re, ast, loc
+	var n
+
+	n = std.mk(ast)
+	if re.debug
+		std.htput(re.astloc, n, loc)
+	;;
+	-> n
 }
 
 const __init__ = {
--- a/lib/regex/interp.myr
+++ b/lib/regex/interp.myr
@@ -104,7 +104,8 @@
 			thr = re.runq
 			re.runq = thr.next
 
-			trace(re, thr, "\nrunning tid={}, ip={}, s[{}]={}\n", thr.tid, thr.ip, re.strp, std.decode(re.str[re.strp:]))
+			trace(re, thr, "\nrunning tid={}, ip={}, s[{}]={}\n", \
+				thr.tid, thr.ip, re.strp, std.decode(re.str[re.strp:]))
 			ip = thr.ip
 			consumed = step(re, thr, -1)
 			while !consumed
@@ -143,7 +144,7 @@
 			;;
 		;;
 		std.bsclear(states)
-		trace(re, thr, "switch\n")
+		trace(re, Zthr, "switch\n")
 		re.runq = re.expired
 		re.expired = Zthr
 		re.expiredtail = Zthr
@@ -277,7 +278,8 @@
 	trace(re, thr, "\t\tdie {}: {}\n", thr.tid, msg)
         if !thr.dead
 		re.nthr--
-        ;;
+	;;
+	re.lastip = thr.ip
 	thr.dead = true
 }
 
@@ -322,9 +324,15 @@
 const trace : (re : regex#, thr : rethread#, msg : byte[:], args : ... -> void) = {re, thr, msg, args
 	var ap
 
-	if re.debug
+	if re.trace && thr != Zthr
 		ap = std.vastart(&args)
 		std.putv(msg, &ap)
+		std.put("\t{}\n", re.pat)
+		std.put("\t")
+		for var i = 0; i < re.pcidx[thr.ip] - 1; i++
+			std.put(" ")
+		;;
+		std.put("^\n")
 	;;
 }
 
--- a/lib/regex/redump.myr
+++ b/lib/regex/redump.myr
@@ -23,9 +23,9 @@
 		;;
 	;;
 	if verbose
-		comp = regex.dbgcompile(cmd.args[0])
+		comp = regex.dbgcompile(cmd.args[0], true)
 	else
-		comp = regex.compile(cmd.args[0])
+		comp = regex.dbgcompile(cmd.args[0], false)
 	;;
 	match comp
 	| `std.Fail m:	
@@ -70,21 +70,26 @@
 }
 
 const show = {re, ln, mg
-	var i
-
 	match mg
 	| `std.Some rl:
 		std.put("Matched: {}\n", rl[0])
-		for i = 1; i < rl.len; i++
+		for var i = 1; i < rl.len; i++
 			std.put("\tgroup {}: {}\n", i, rl[i])
 		;;
 	| `std.None:
 		std.put("Match failed:\n")
+		std.put("\t{}\n", re.pat)
+		caret(re, re.pcidx[re.lastip] - 1)
 		std.put("\t{}\n", ln)
-		std.put("\t")
-		for i = 0; i < re.strp - 1; i++
-			std.put("~")
-		;;
-		std.put("^\n")
+		caret(re, re.strp - 1)
 	;;
 }
+
+const caret = {re, idx
+	std.put("\t")
+	for var i = 0; i < idx; i++
+		std.put("~")
+	;;
+	std.put("^\n")
+}
+
--- a/lib/regex/types.myr
+++ b/lib/regex/types.myr
@@ -39,7 +39,9 @@
 	type regex = struct
 		/* compile state */
 		debug	: bool
+		trace	: bool
 		pat	: byte[:]
+		idx	: std.size
 		nmatch	: std.size
 
 		/* VM state */
@@ -51,6 +53,11 @@
 		nthr	: std.size
 		str	: byte[:]
 		strp	: std.size
+
+		/* debug state */
+		astloc	: std.htab(ast#, std.size)#
+		pcidx	: std.size[:]
+		lastip	: std.size
 	;;
 
 	pkglocal type rethread = struct