Commit 135389d0 authored by Keith Randall's avatar Keith Randall

cmd/internal/gc: Use shifts for powers-of-two indexing

Fixes #10638

Change-Id: I7bbaad7e5a599aa94d1d158e903596231c7e9897
Reviewed-on: https://go-review.googlesource.com/9535Reviewed-by: default avatarJosh Bleecher Snyder <josharian@gmail.com>
parent 71274e48
...@@ -1132,12 +1132,18 @@ func Agenr(n *Node, a *Node, res *Node) { ...@@ -1132,12 +1132,18 @@ func Agenr(n *Node, a *Node, res *Node) {
} else if w == 1 { } else if w == 1 {
Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3) Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3)
} else { } else {
Regalloc(&n4, Types[TUINT32], nil) if w&(w-1) == 0 {
Nodconst(&n1, Types[TUINT32], int64(w)) // Power of 2. Use shift.
Thearch.Gmove(&n1, &n4) Thearch.Ginscon(Thearch.Optoas(OLSH, Types[TUINT32]), int64(log2(uint64(w))), &n2)
Thearch.Gins(Thearch.Optoas(OMUL, Types[TUINT32]), &n4, &n2) } else {
// Not a power of 2. Use multiply.
Regalloc(&n4, Types[TUINT32], nil)
Nodconst(&n1, Types[TUINT32], int64(w))
Thearch.Gmove(&n1, &n4)
Thearch.Gins(Thearch.Optoas(OMUL, Types[TUINT32]), &n4, &n2)
Regfree(&n4)
}
Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3) Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3)
Regfree(&n4)
} }
*a = n3 *a = n3
Regfree(&n2) Regfree(&n2)
...@@ -1292,8 +1298,13 @@ func Agenr(n *Node, a *Node, res *Node) { ...@@ -1292,8 +1298,13 @@ func Agenr(n *Node, a *Node, res *Node) {
} else if w == 1 { } else if w == 1 {
Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3) Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3)
} else { } else {
Nodconst(&tmp, Types[TUINT32], int64(w)) if w&(w-1) == 0 {
Thearch.Gins(Thearch.Optoas(OMUL, Types[TUINT32]), &tmp, &n2) // Power of 2. Use shift.
Thearch.Ginscon(Thearch.Optoas(OLSH, Types[TUINT32]), int64(log2(uint64(w))), &n2)
} else {
// Not a power of 2. Use multiply.
Thearch.Ginscon(Thearch.Optoas(OMUL, Types[TUINT32]), int64(w), &n2)
}
Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3) Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3)
} }
...@@ -1485,7 +1496,13 @@ func Agenr(n *Node, a *Node, res *Node) { ...@@ -1485,7 +1496,13 @@ func Agenr(n *Node, a *Node, res *Node) {
} else if w == 1 { } else if w == 1 {
Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3) Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3)
} else { } else {
Thearch.Ginscon(Thearch.Optoas(OMUL, t), int64(w), &n2) if w&(w-1) == 0 {
// Power of 2. Use shift.
Thearch.Ginscon(Thearch.Optoas(OLSH, t), int64(log2(w)), &n2)
} else {
// Not a power of 2. Use multiply.
Thearch.Ginscon(Thearch.Optoas(OMUL, t), int64(w), &n2)
}
Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3) Thearch.Gins(Thearch.Optoas(OADD, Types[Tptr]), &n2, &n3)
} }
...@@ -1502,6 +1519,15 @@ func Agenr(n *Node, a *Node, res *Node) { ...@@ -1502,6 +1519,15 @@ func Agenr(n *Node, a *Node, res *Node) {
} }
} }
// log2 returns the logarithm base 2 of n. n must be a power of 2.
func log2(n uint64) int {
x := 0
for n>>uint(x) != 1 {
x++
}
return x
}
/* /*
* generate: * generate:
* res = &n; * res = &n;
......
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