Commit 48d3c32b authored by Josh Bleecher Snyder's avatar Josh Bleecher Snyder

cmd/compile: teach rulegen to |-expand multiple |s in a single op

I want to be able to write

MOV(Q|Q|L|L|L|W|W|B)loadidx(1|8|1|4|8|1|2|1)

instead of

MOV(Qloadidx1|Qloadidx8|Lloadidx1|Lloadidx4|Lloadidx8|Wloadidx1|Wloadidx2|Bloadidx1)

in rewrite rules.

Both are fairly cryptic and hard to review, but the former
is at least compact, which helps to not obscure the structure
of the rest of the rule.

Support that by adjusting rulegen's expansion.

Instead of looking for an op that begins with "(", ends with " ",
and has exactly one set of parens in it, look for everything of the
form "(...|...)".

That has false positives: Go code in the && conditions and AuxInt expressions.
Those are easily checked for syntactically: && conditions are between && and ->,
and AuxInt expressions are inside square brackets.
After ruling out those false positives, we can keep everything else,
regardless of where it is.

No change to the generated code for existing rules.

Change-Id: I5b70a190e268989504f53cb2cce2f9a50170d8a2
Reviewed-on: https://go-review.googlesource.com/c/go/+/166737
Run-TryBot: Josh Bleecher Snyder <josharian@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarCherry Zhang <cherryyz@google.com>
parent 07b4b4a1
...@@ -815,20 +815,39 @@ func isVariable(s string) bool { ...@@ -815,20 +815,39 @@ func isVariable(s string) bool {
} }
// opRegexp is a regular expression to find the opcode portion of s-expressions. // opRegexp is a regular expression to find the opcode portion of s-expressions.
var opRegexp = regexp.MustCompile(`[(]\w*[(](\w+[|])+\w+[)]\w* `) var opRegexp = regexp.MustCompile(`[(](\w+[|])+\w+[)]`)
// excludeFromExpansion reports whether the substring s[idx[0]:idx[1]] in a rule
// should be disregarded as a candidate for | expansion.
// It uses simple syntactic checks to see whether the substring
// is inside an AuxInt expression or inside the && conditions.
func excludeFromExpansion(s string, idx []int) bool {
left := s[:idx[0]]
if strings.LastIndexByte(left, '[') > strings.LastIndexByte(left, ']') {
// Inside an AuxInt expression.
return true
}
right := s[idx[1]:]
if strings.Contains(left, "&&") && strings.Contains(right, "->") {
// Inside && conditions.
return true
}
return false
}
// expandOr converts a rule into multiple rules by expanding | ops. // expandOr converts a rule into multiple rules by expanding | ops.
func expandOr(r string) []string { func expandOr(r string) []string {
// Find every occurrence of |-separated things at the opcode position. // Find every occurrence of |-separated things.
// They look like (MOV(B|W|L|Q|SS|SD)load // They look like MOV(B|W|L|Q|SS|SD)load or MOV(Q|L)loadidx(1|8).
// Note: there might be false positives in parts of rules that are Go code
// (e.g. && conditions, AuxInt expressions, etc.). There are currently no
// such false positives, so I'm not too worried about it.
// Generate rules selecting one case from each |-form. // Generate rules selecting one case from each |-form.
// Count width of |-forms. They must match. // Count width of |-forms. They must match.
n := 1 n := 1
for _, s := range opRegexp.FindAllString(r, -1) { for _, idx := range opRegexp.FindAllStringIndex(r, -1) {
if excludeFromExpansion(r, idx) {
continue
}
s := r[idx[0]:idx[1]]
c := strings.Count(s, "|") + 1 c := strings.Count(s, "|") + 1
if c == 1 { if c == 1 {
continue continue
...@@ -842,16 +861,22 @@ func expandOr(r string) []string { ...@@ -842,16 +861,22 @@ func expandOr(r string) []string {
// No |-form in this rule. // No |-form in this rule.
return []string{r} return []string{r}
} }
// Build each new rule.
res := make([]string, n) res := make([]string, n)
for i := 0; i < n; i++ { for i := 0; i < n; i++ {
res[i] = opRegexp.ReplaceAllStringFunc(r, func(s string) string { buf := new(strings.Builder)
if strings.Count(s, "|") == 0 { x := 0
return s for _, idx := range opRegexp.FindAllStringIndex(r, -1) {
} if excludeFromExpansion(r, idx) {
s = s[1 : len(s)-1] // remove leading "(" and trailing " " continue
x, y := strings.Index(s, "("), strings.Index(s, ")") }
return "(" + s[:x] + strings.Split(s[x+1:y], "|")[i] + s[y+1:] + " " buf.WriteString(r[x:idx[0]]) // write bytes we've skipped over so far
}) s := r[idx[0]+1 : idx[1]-1] // remove leading "(" and trailing ")"
buf.WriteString(strings.Split(s, "|")[i]) // write the op component for this rule
x = idx[1] // note that we've written more bytes
}
buf.WriteString(r[x:])
res[i] = buf.String()
} }
return res return res
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment