]> Cypherpunks repositories - gostls13.git/commitdiff
cmd/compile: don't combine loads in generated equality functions
authorkhr@golang.org <khr@golang.org>
Fri, 3 May 2024 19:55:34 +0000 (12:55 -0700)
committerGopher Robot <gobot@golang.org>
Mon, 6 May 2024 15:34:04 +0000 (15:34 +0000)
... if the architecture can't do unaligned loads.
We already handle this in a few places, but this particular place
was added in CL 399542 and missed this additional restriction.

Fixes #67160

Change-Id: I45988f11ff3ed45df1c4da3f0931ab1fdb22dbfe
Reviewed-on: https://go-review.googlesource.com/c/go/+/583175
Reviewed-by: Cuong Manh Le <cuong.manhle.vn@gmail.com>
Auto-Submit: Keith Randall <khr@google.com>
Reviewed-by: Keith Randall <khr@google.com>
LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
Reviewed-by: Derek Parker <parkerderek86@gmail.com>
Reviewed-by: Cherry Mui <cherryyz@google.com>
src/cmd/compile/internal/compare/compare.go
test/fixedbugs/issue67160.go [new file with mode: 0644]

index 2f137daf822a91466ef3ff0c221bafcc68d1e3d0..638eb37a6221098ef8f999f349a1cb80e36134c8 100644 (file)
@@ -147,7 +147,7 @@ func calculateCostForType(t *types.Type) int64 {
                return EqStructCost(t)
        case types.TSLICE:
                // Slices are not comparable.
-               base.Fatalf("eqStructFieldCost: unexpected slice type")
+               base.Fatalf("calculateCostForType: unexpected slice type")
        case types.TARRAY:
                elemCost := calculateCostForType(t.Elem())
                cost = t.NumElem() * elemCost
@@ -370,6 +370,11 @@ func eqmem(p, q ir.Node, field int, size int64) ir.Node {
 }
 
 func eqmemfunc(size int64, t *types.Type) (fn *ir.Name, needsize bool) {
+       if !base.Ctxt.Arch.CanMergeLoads && t.Alignment() < int64(base.Ctxt.Arch.Alignment) && t.Alignment() < t.Size() {
+               // We can't use larger comparisons if the value might not be aligned
+               // enough for the larger comparison. See issues 46283 and 67160.
+               size = 0
+       }
        switch size {
        case 1, 2, 4, 8, 16:
                buf := fmt.Sprintf("memequal%d", int(size)*8)
diff --git a/test/fixedbugs/issue67160.go b/test/fixedbugs/issue67160.go
new file mode 100644 (file)
index 0000000..be45a61
--- /dev/null
@@ -0,0 +1,32 @@
+// run
+
+// Copyright 2024 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.
+
+// Test to make sure that we don't try using larger loads for
+// generated equality functions on architectures that can't do
+// unaligned loads.
+
+package main
+
+// T has a big field that wants to be compared with larger loads/stores.
+// T is "special" because of the unnamed field, so it needs a generated equality function.
+// T is an odd number of bytes in size and has alignment 1.
+type T struct {
+       src [8]byte
+       _   byte
+}
+
+// U contains 8 copies of T, each at a different %8 alignment.
+type U [8]T
+
+//go:noinline
+func f(x, y *U) bool {
+       return *x == *y
+}
+
+func main() {
+       var a U
+       _ = f(&a, &a)
+}