1
0
mirror of https://github.com/golang/go synced 2024-11-23 08:20:05 -07:00

cmd/compile: for arm, zero unaligned memory 1 byte at a time

If memory might be unaligned, zero it one byte at a time
instead of 4 bytes at a time.

Fixes #15902

Change-Id: I4eff0840e042e2f137c1a4028f08793eb7dfd703
Reviewed-on: https://go-review.googlesource.com/23587
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Minux Ma <minux@golang.org>
This commit is contained in:
Keith Randall 2016-05-31 14:55:12 -07:00
parent e29e0ba19a
commit 52fe472472
2 changed files with 53 additions and 0 deletions

View File

@ -341,6 +341,11 @@ func clearfat(nl *gc.Node) {
c := w % 4 // bytes c := w % 4 // bytes
q := w / 4 // quads q := w / 4 // quads
if nl.Type.Align < 4 {
q = 0
c = w
}
var r0 gc.Node var r0 gc.Node
r0.Op = gc.OREGISTER r0.Op = gc.OREGISTER
@ -395,6 +400,27 @@ func clearfat(nl *gc.Node) {
} }
} }
if c > 4 {
// Loop to zero unaligned memory.
var end gc.Node
gc.Regalloc(&end, gc.Types[gc.Tptr], nil)
p := gins(arm.AMOVW, &dst, &end)
p.From.Type = obj.TYPE_ADDR
p.From.Offset = int64(c)
p = gins(arm.AMOVB, &nz, &dst)
p.To.Type = obj.TYPE_MEM
p.To.Offset = 1
p.Scond |= arm.C_PBIT
pl := p
p = gins(arm.ACMP, &dst, nil)
raddr(&end, p)
gc.Patch(gc.Gbranch(arm.ABNE, nil, 0), pl)
gc.Regfree(&end)
c = 0
}
var p *obj.Prog var p *obj.Prog
for c > 0 { for c > 0 {
p = gins(arm.AMOVB, &nz, &dst) p = gins(arm.AMOVB, &nz, &dst)

View File

@ -0,0 +1,27 @@
// run
// Copyright 2016 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.
// This test makes sure we don't use 4-byte unaligned writes
// to zero memory on architectures that don't support them.
package main
type T struct {
a byte
b [10]byte
}
//go:noinline
func f(t *T) {
// t will be aligned, so &t.b won't be.
t.b = [10]byte{}
}
var t T
func main() {
f(&t)
}