Commit 20101894 authored by Austin Clements's avatar Austin Clements

runtime: remove legacy eager write barrier

Now that the buffered write barrier is implemented for all
architectures, we can remove the old eager write barrier
implementation. This CL removes the implementation from the runtime,
support in the compiler for calling it, and updates some compiler
tests that relied on the old eager barrier support. It also makes sure
that all of the useful comments from the old write barrier
implementation still have a place to live.

Fixes #22460.

Updates #21640 since this fixes the layering concerns of the write
barrier (but not the other things in that issue).

Change-Id: I580f93c152e89607e0a72fe43370237ba97bae74
Reviewed-on: https://go-review.googlesource.com/92705
Run-TryBot: Austin Clements <austin@google.com>
Reviewed-by: default avatarRick Hudson <rlh@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent 24531088
...@@ -468,7 +468,7 @@ var linuxAMD64Tests = []*asmTest{ ...@@ -468,7 +468,7 @@ var linuxAMD64Tests = []*asmTest{
*t = T2{} *t = T2{}
} }
`, `,
pos: []string{"\tXORPS\tX., X", "\tMOVUPS\tX., \\(.*\\)", "\tMOVQ\t\\$0, 16\\(.*\\)", "\tCALL\truntime\\.(writebarrierptr|gcWriteBarrier)\\(SB\\)"}, pos: []string{"\tXORPS\tX., X", "\tMOVUPS\tX., \\(.*\\)", "\tMOVQ\t\\$0, 16\\(.*\\)", "\tCALL\truntime\\.gcWriteBarrier\\(SB\\)"},
}, },
// Rotate tests // Rotate tests
{ {
......
...@@ -105,56 +105,55 @@ var runtimeDecls = [...]struct { ...@@ -105,56 +105,55 @@ var runtimeDecls = [...]struct {
{"chansend1", funcTag, 81}, {"chansend1", funcTag, 81},
{"closechan", funcTag, 23}, {"closechan", funcTag, 23},
{"writeBarrier", varTag, 83}, {"writeBarrier", varTag, 83},
{"writebarrierptr", funcTag, 84}, {"typedmemmove", funcTag, 84},
{"typedmemmove", funcTag, 85}, {"typedmemclr", funcTag, 85},
{"typedmemclr", funcTag, 86}, {"typedslicecopy", funcTag, 86},
{"typedslicecopy", funcTag, 87}, {"selectnbsend", funcTag, 87},
{"selectnbsend", funcTag, 88}, {"selectnbrecv", funcTag, 88},
{"selectnbrecv", funcTag, 89}, {"selectnbrecv2", funcTag, 90},
{"selectnbrecv2", funcTag, 91}, {"newselect", funcTag, 91},
{"newselect", funcTag, 92}, {"selectsend", funcTag, 92},
{"selectsend", funcTag, 93}, {"selectrecv", funcTag, 93},
{"selectrecv", funcTag, 94},
{"selectdefault", funcTag, 55}, {"selectdefault", funcTag, 55},
{"selectgo", funcTag, 95}, {"selectgo", funcTag, 94},
{"block", funcTag, 5}, {"block", funcTag, 5},
{"makeslice", funcTag, 97}, {"makeslice", funcTag, 96},
{"makeslice64", funcTag, 98}, {"makeslice64", funcTag, 97},
{"growslice", funcTag, 99}, {"growslice", funcTag, 98},
{"memmove", funcTag, 100}, {"memmove", funcTag, 99},
{"memclrNoHeapPointers", funcTag, 101}, {"memclrNoHeapPointers", funcTag, 100},
{"memclrHasPointers", funcTag, 101}, {"memclrHasPointers", funcTag, 100},
{"memequal", funcTag, 102}, {"memequal", funcTag, 101},
{"memequal8", funcTag, 103}, {"memequal8", funcTag, 102},
{"memequal16", funcTag, 103}, {"memequal16", funcTag, 102},
{"memequal32", funcTag, 103}, {"memequal32", funcTag, 102},
{"memequal64", funcTag, 103}, {"memequal64", funcTag, 102},
{"memequal128", funcTag, 103}, {"memequal128", funcTag, 102},
{"int64div", funcTag, 104}, {"int64div", funcTag, 103},
{"uint64div", funcTag, 105}, {"uint64div", funcTag, 104},
{"int64mod", funcTag, 104}, {"int64mod", funcTag, 103},
{"uint64mod", funcTag, 105}, {"uint64mod", funcTag, 104},
{"float64toint64", funcTag, 106}, {"float64toint64", funcTag, 105},
{"float64touint64", funcTag, 107}, {"float64touint64", funcTag, 106},
{"float64touint32", funcTag, 108}, {"float64touint32", funcTag, 107},
{"int64tofloat64", funcTag, 109}, {"int64tofloat64", funcTag, 108},
{"uint64tofloat64", funcTag, 110}, {"uint64tofloat64", funcTag, 109},
{"uint32tofloat64", funcTag, 111}, {"uint32tofloat64", funcTag, 110},
{"complex128div", funcTag, 112}, {"complex128div", funcTag, 111},
{"racefuncenter", funcTag, 113}, {"racefuncenter", funcTag, 112},
{"racefuncexit", funcTag, 5}, {"racefuncexit", funcTag, 5},
{"raceread", funcTag, 113}, {"raceread", funcTag, 112},
{"racewrite", funcTag, 113}, {"racewrite", funcTag, 112},
{"racereadrange", funcTag, 114}, {"racereadrange", funcTag, 113},
{"racewriterange", funcTag, 114}, {"racewriterange", funcTag, 113},
{"msanread", funcTag, 114}, {"msanread", funcTag, 113},
{"msanwrite", funcTag, 114}, {"msanwrite", funcTag, 113},
{"support_popcnt", varTag, 11}, {"support_popcnt", varTag, 11},
{"support_sse41", varTag, 11}, {"support_sse41", varTag, 11},
} }
func runtimeTypes() []*types.Type { func runtimeTypes() []*types.Type {
var typs [115]*types.Type var typs [114]*types.Type
typs[0] = types.Bytetype typs[0] = types.Bytetype
typs[1] = types.NewPtr(typs[0]) typs[1] = types.NewPtr(typs[0])
typs[2] = types.Types[TANY] typs[2] = types.Types[TANY]
...@@ -239,36 +238,35 @@ func runtimeTypes() []*types.Type { ...@@ -239,36 +238,35 @@ func runtimeTypes() []*types.Type {
typs[81] = functype(nil, []*Node{anonfield(typs[80]), anonfield(typs[3])}, nil) typs[81] = functype(nil, []*Node{anonfield(typs[80]), anonfield(typs[3])}, nil)
typs[82] = types.NewArray(typs[0], 3) typs[82] = types.NewArray(typs[0], 3)
typs[83] = tostruct([]*Node{namedfield("enabled", typs[11]), namedfield("pad", typs[82]), namedfield("needed", typs[11]), namedfield("cgo", typs[11]), namedfield("alignme", typs[17])}) typs[83] = tostruct([]*Node{namedfield("enabled", typs[11]), namedfield("pad", typs[82]), namedfield("needed", typs[11]), namedfield("cgo", typs[11]), namedfield("alignme", typs[17])})
typs[84] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[2])}, nil) typs[84] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3]), anonfield(typs[3])}, nil)
typs[85] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3]), anonfield(typs[3])}, nil) typs[85] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3])}, nil)
typs[86] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[3])}, nil) typs[86] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[2]), anonfield(typs[2])}, []*Node{anonfield(typs[32])})
typs[87] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[2]), anonfield(typs[2])}, []*Node{anonfield(typs[32])}) typs[87] = functype(nil, []*Node{anonfield(typs[80]), anonfield(typs[3])}, []*Node{anonfield(typs[11])})
typs[88] = functype(nil, []*Node{anonfield(typs[80]), anonfield(typs[3])}, []*Node{anonfield(typs[11])}) typs[88] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[77])}, []*Node{anonfield(typs[11])})
typs[89] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[77])}, []*Node{anonfield(typs[11])}) typs[89] = types.NewPtr(typs[11])
typs[90] = types.NewPtr(typs[11]) typs[90] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[89]), anonfield(typs[77])}, []*Node{anonfield(typs[11])})
typs[91] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[90]), anonfield(typs[77])}, []*Node{anonfield(typs[11])}) typs[91] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15]), anonfield(typs[8])}, nil)
typs[92] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15]), anonfield(typs[8])}, nil) typs[92] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[80]), anonfield(typs[3])}, nil)
typs[93] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[80]), anonfield(typs[3])}, nil) typs[93] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[77]), anonfield(typs[3]), anonfield(typs[89])}, nil)
typs[94] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[77]), anonfield(typs[3]), anonfield(typs[90])}, nil) typs[94] = functype(nil, []*Node{anonfield(typs[1])}, []*Node{anonfield(typs[32])})
typs[95] = functype(nil, []*Node{anonfield(typs[1])}, []*Node{anonfield(typs[32])}) typs[95] = types.NewSlice(typs[2])
typs[96] = types.NewSlice(typs[2]) typs[96] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[32]), anonfield(typs[32])}, []*Node{anonfield(typs[95])})
typs[97] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[32]), anonfield(typs[32])}, []*Node{anonfield(typs[96])}) typs[97] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15]), anonfield(typs[15])}, []*Node{anonfield(typs[95])})
typs[98] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[15]), anonfield(typs[15])}, []*Node{anonfield(typs[96])}) typs[98] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[95]), anonfield(typs[32])}, []*Node{anonfield(typs[95])})
typs[99] = functype(nil, []*Node{anonfield(typs[1]), anonfield(typs[96]), anonfield(typs[32])}, []*Node{anonfield(typs[96])}) typs[99] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3]), anonfield(typs[48])}, nil)
typs[100] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3]), anonfield(typs[48])}, nil) typs[100] = functype(nil, []*Node{anonfield(typs[57]), anonfield(typs[48])}, nil)
typs[101] = functype(nil, []*Node{anonfield(typs[57]), anonfield(typs[48])}, nil) typs[101] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3]), anonfield(typs[48])}, []*Node{anonfield(typs[11])})
typs[102] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3]), anonfield(typs[48])}, []*Node{anonfield(typs[11])}) typs[102] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3])}, []*Node{anonfield(typs[11])})
typs[103] = functype(nil, []*Node{anonfield(typs[3]), anonfield(typs[3])}, []*Node{anonfield(typs[11])}) typs[103] = functype(nil, []*Node{anonfield(typs[15]), anonfield(typs[15])}, []*Node{anonfield(typs[15])})
typs[104] = functype(nil, []*Node{anonfield(typs[15]), anonfield(typs[15])}, []*Node{anonfield(typs[15])}) typs[104] = functype(nil, []*Node{anonfield(typs[17]), anonfield(typs[17])}, []*Node{anonfield(typs[17])})
typs[105] = functype(nil, []*Node{anonfield(typs[17]), anonfield(typs[17])}, []*Node{anonfield(typs[17])}) typs[105] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[15])})
typs[106] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[15])}) typs[106] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[17])})
typs[107] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[17])}) typs[107] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[59])})
typs[108] = functype(nil, []*Node{anonfield(typs[13])}, []*Node{anonfield(typs[59])}) typs[108] = functype(nil, []*Node{anonfield(typs[15])}, []*Node{anonfield(typs[13])})
typs[109] = functype(nil, []*Node{anonfield(typs[15])}, []*Node{anonfield(typs[13])}) typs[109] = functype(nil, []*Node{anonfield(typs[17])}, []*Node{anonfield(typs[13])})
typs[110] = functype(nil, []*Node{anonfield(typs[17])}, []*Node{anonfield(typs[13])}) typs[110] = functype(nil, []*Node{anonfield(typs[59])}, []*Node{anonfield(typs[13])})
typs[111] = functype(nil, []*Node{anonfield(typs[59])}, []*Node{anonfield(typs[13])}) typs[111] = functype(nil, []*Node{anonfield(typs[19]), anonfield(typs[19])}, []*Node{anonfield(typs[19])})
typs[112] = functype(nil, []*Node{anonfield(typs[19]), anonfield(typs[19])}, []*Node{anonfield(typs[19])}) typs[112] = functype(nil, []*Node{anonfield(typs[48])}, nil)
typs[113] = functype(nil, []*Node{anonfield(typs[48])}, nil) typs[113] = functype(nil, []*Node{anonfield(typs[48]), anonfield(typs[48])}, nil)
typs[114] = functype(nil, []*Node{anonfield(typs[48]), anonfield(typs[48])}, nil)
return typs[:] return typs[:]
} }
...@@ -136,8 +136,6 @@ var writeBarrier struct { ...@@ -136,8 +136,6 @@ var writeBarrier struct {
alignme uint64 alignme uint64
} }
func writebarrierptr(dst *any, src any)
// *byte is really *runtime.Type // *byte is really *runtime.Type
func typedmemmove(typ *byte, dst *any, src *any) func typedmemmove(typ *byte, dst *any, src *any)
func typedmemclr(typ *byte, dst *any) func typedmemclr(typ *byte, dst *any)
......
...@@ -292,7 +292,6 @@ var ( ...@@ -292,7 +292,6 @@ var (
assertI2I2, assertI2I2,
goschedguarded, goschedguarded,
writeBarrier, writeBarrier,
writebarrierptr,
gcWriteBarrier, gcWriteBarrier,
typedmemmove, typedmemmove,
typedmemclr, typedmemclr,
......
...@@ -44,7 +44,6 @@ var ( ...@@ -44,7 +44,6 @@ var (
Debug_slice int Debug_slice int
Debug_vlog bool Debug_vlog bool
Debug_wb int Debug_wb int
Debug_eagerwb int
Debug_pctab string Debug_pctab string
Debug_locationlist int Debug_locationlist int
Debug_typecheckinl int Debug_typecheckinl int
...@@ -73,7 +72,6 @@ var debugtab = []struct { ...@@ -73,7 +72,6 @@ var debugtab = []struct {
{"slice", "print information about slice compilation", &Debug_slice}, {"slice", "print information about slice compilation", &Debug_slice},
{"typeassert", "print information about type assertion inlining", &Debug_typeassert}, {"typeassert", "print information about type assertion inlining", &Debug_typeassert},
{"wb", "print information about write barriers", &Debug_wb}, {"wb", "print information about write barriers", &Debug_wb},
{"eagerwb", "use unbuffered write barrier", &Debug_eagerwb},
{"export", "print export data", &Debug_export}, {"export", "print export data", &Debug_export},
{"pctab", "print named pc-value table", &Debug_pctab}, {"pctab", "print named pc-value table", &Debug_pctab},
{"locationlists", "print information about DWARF location list creation", &Debug_locationlist}, {"locationlists", "print information about DWARF location list creation", &Debug_locationlist},
...@@ -407,14 +405,6 @@ func Main(archInit func(*Arch)) { ...@@ -407,14 +405,6 @@ func Main(archInit func(*Arch)) {
Debug['l'] = 1 - Debug['l'] Debug['l'] = 1 - Debug['l']
} }
switch objabi.GOARCH {
case "amd64", "amd64p32", "386", "arm", "arm64", "ppc64", "ppc64le", "mips64", "mips64le", "mips", "mipsle", "s390x":
default:
// Other architectures don't support the buffered
// write barrier yet.
Debug_eagerwb = 1
}
trackScopes = flagDWARF && ((Debug['l'] == 0 && Debug['N'] != 0) || Ctxt.Flag_locationlists) trackScopes = flagDWARF && ((Debug['l'] == 0 && Debug['N'] != 0) || Ctxt.Flag_locationlists)
Widthptr = thearch.LinkArch.PtrSize Widthptr = thearch.LinkArch.PtrSize
......
...@@ -95,7 +95,6 @@ func initssaconfig() { ...@@ -95,7 +95,6 @@ func initssaconfig() {
assertI2I2 = sysfunc("assertI2I2") assertI2I2 = sysfunc("assertI2I2")
goschedguarded = sysfunc("goschedguarded") goschedguarded = sysfunc("goschedguarded")
writeBarrier = sysfunc("writeBarrier") writeBarrier = sysfunc("writeBarrier")
writebarrierptr = sysfunc("writebarrierptr")
gcWriteBarrier = sysfunc("gcWriteBarrier") gcWriteBarrier = sysfunc("gcWriteBarrier")
typedmemmove = sysfunc("typedmemmove") typedmemmove = sysfunc("typedmemmove")
typedmemclr = sysfunc("typedmemclr") typedmemclr = sysfunc("typedmemclr")
...@@ -5380,10 +5379,6 @@ func (e *ssafn) Debug_checknil() bool { ...@@ -5380,10 +5379,6 @@ func (e *ssafn) Debug_checknil() bool {
return Debug_checknil != 0 return Debug_checknil != 0
} }
func (e *ssafn) Debug_eagerwb() bool {
return Debug_eagerwb != 0
}
func (e *ssafn) UseWriteBarrier() bool { func (e *ssafn) UseWriteBarrier() bool {
return use_writebarrier return use_writebarrier
} }
...@@ -5394,8 +5389,6 @@ func (e *ssafn) Syslook(name string) *obj.LSym { ...@@ -5394,8 +5389,6 @@ func (e *ssafn) Syslook(name string) *obj.LSym {
return goschedguarded return goschedguarded
case "writeBarrier": case "writeBarrier":
return writeBarrier return writeBarrier
case "writebarrierptr":
return writebarrierptr
case "gcWriteBarrier": case "gcWriteBarrier":
return gcWriteBarrier return gcWriteBarrier
case "typedmemmove": case "typedmemmove":
......
...@@ -89,7 +89,6 @@ type Logger interface { ...@@ -89,7 +89,6 @@ type Logger interface {
// Forwards the Debug flags from gc // Forwards the Debug flags from gc
Debug_checknil() bool Debug_checknil() bool
Debug_eagerwb() bool
} }
type Frontend interface { type Frontend interface {
......
...@@ -134,7 +134,6 @@ func (d DummyFrontend) Log() bool { return true } ...@@ -134,7 +134,6 @@ func (d DummyFrontend) Log() bool { return true }
func (d DummyFrontend) Fatalf(_ src.XPos, msg string, args ...interface{}) { d.t.Fatalf(msg, args...) } func (d DummyFrontend) Fatalf(_ src.XPos, msg string, args ...interface{}) { d.t.Fatalf(msg, args...) }
func (d DummyFrontend) Warnl(_ src.XPos, msg string, args ...interface{}) { d.t.Logf(msg, args...) } func (d DummyFrontend) Warnl(_ src.XPos, msg string, args ...interface{}) { d.t.Logf(msg, args...) }
func (d DummyFrontend) Debug_checknil() bool { return false } func (d DummyFrontend) Debug_checknil() bool { return false }
func (d DummyFrontend) Debug_eagerwb() bool { return false }
var dummyTypes Types var dummyTypes Types
......
...@@ -31,7 +31,7 @@ func needwb(v *Value) bool { ...@@ -31,7 +31,7 @@ func needwb(v *Value) bool {
// and runtime calls, like // and runtime calls, like
// //
// if writeBarrier.enabled { // if writeBarrier.enabled {
// writebarrierptr(ptr, val) // gcWriteBarrier(ptr, val) // Not a regular Go call
// } else { // } else {
// *ptr = val // *ptr = val
// } // }
...@@ -44,7 +44,7 @@ func writebarrier(f *Func) { ...@@ -44,7 +44,7 @@ func writebarrier(f *Func) {
} }
var sb, sp, wbaddr, const0 *Value var sb, sp, wbaddr, const0 *Value
var writebarrierptr, typedmemmove, typedmemclr, gcWriteBarrier *obj.LSym var typedmemmove, typedmemclr, gcWriteBarrier *obj.LSym
var stores, after []*Value var stores, after []*Value
var sset *sparseSet var sset *sparseSet
var storeNumber []int32 var storeNumber []int32
...@@ -96,10 +96,7 @@ func writebarrier(f *Func) { ...@@ -96,10 +96,7 @@ func writebarrier(f *Func) {
} }
wbsym := f.fe.Syslook("writeBarrier") wbsym := f.fe.Syslook("writeBarrier")
wbaddr = f.Entry.NewValue1A(initpos, OpAddr, f.Config.Types.UInt32Ptr, wbsym, sb) wbaddr = f.Entry.NewValue1A(initpos, OpAddr, f.Config.Types.UInt32Ptr, wbsym, sb)
writebarrierptr = f.fe.Syslook("writebarrierptr") gcWriteBarrier = f.fe.Syslook("gcWriteBarrier")
if !f.fe.Debug_eagerwb() {
gcWriteBarrier = f.fe.Syslook("gcWriteBarrier")
}
typedmemmove = f.fe.Syslook("typedmemmove") typedmemmove = f.fe.Syslook("typedmemmove")
typedmemclr = f.fe.Syslook("typedmemclr") typedmemclr = f.fe.Syslook("typedmemclr")
const0 = f.ConstInt32(initpos, f.Config.Types.UInt32, 0) const0 = f.ConstInt32(initpos, f.Config.Types.UInt32, 0)
...@@ -198,7 +195,6 @@ func writebarrier(f *Func) { ...@@ -198,7 +195,6 @@ func writebarrier(f *Func) {
var val *Value var val *Value
switch w.Op { switch w.Op {
case OpStoreWB: case OpStoreWB:
fn = writebarrierptr
val = w.Args[1] val = w.Args[1]
nWBops-- nWBops--
case OpMoveWB: case OpMoveWB:
...@@ -217,11 +213,13 @@ func writebarrier(f *Func) { ...@@ -217,11 +213,13 @@ func writebarrier(f *Func) {
switch w.Op { switch w.Op {
case OpStoreWB, OpMoveWB, OpZeroWB: case OpStoreWB, OpMoveWB, OpZeroWB:
volatile := w.Op == OpMoveWB && isVolatile(val) volatile := w.Op == OpMoveWB && isVolatile(val)
if w.Op == OpStoreWB && !f.fe.Debug_eagerwb() { if w.Op == OpStoreWB {
memThen = bThen.NewValue3A(pos, OpWB, types.TypeMem, gcWriteBarrier, ptr, val, memThen) memThen = bThen.NewValue3A(pos, OpWB, types.TypeMem, gcWriteBarrier, ptr, val, memThen)
} else { } else {
memThen = wbcall(pos, bThen, fn, typ, ptr, val, memThen, sp, sb, volatile) memThen = wbcall(pos, bThen, fn, typ, ptr, val, memThen, sp, sb, volatile)
} }
// Note that we set up a writebarrier function call.
f.fe.SetWBPos(pos)
case OpVarDef, OpVarLive, OpVarKill: case OpVarDef, OpVarLive, OpVarKill:
memThen = bThen.NewValue1A(pos, w.Op, types.TypeMem, w.Aux, memThen) memThen = bThen.NewValue1A(pos, w.Op, types.TypeMem, w.Aux, memThen)
} }
...@@ -239,11 +237,6 @@ func writebarrier(f *Func) { ...@@ -239,11 +237,6 @@ func writebarrier(f *Func) {
case OpVarDef, OpVarLive, OpVarKill: case OpVarDef, OpVarLive, OpVarKill:
memElse = bElse.NewValue1A(pos, w.Op, types.TypeMem, w.Aux, memElse) memElse = bElse.NewValue1A(pos, w.Op, types.TypeMem, w.Aux, memElse)
} }
if fn != nil {
// Note that we set up a writebarrier function call.
f.fe.SetWBPos(pos)
}
} }
// merge memory // merge memory
......
...@@ -25,5 +25,3 @@ runtime/asm_386.s: [386] uint32tofloat64: function uint32tofloat64 missing Go de ...@@ -25,5 +25,3 @@ runtime/asm_386.s: [386] uint32tofloat64: function uint32tofloat64 missing Go de
runtime/asm_386.s: [386] float64touint32: function float64touint32 missing Go declaration runtime/asm_386.s: [386] float64touint32: function float64touint32 missing Go declaration
runtime/asm_386.s: [386] stackcheck: function stackcheck missing Go declaration runtime/asm_386.s: [386] stackcheck: function stackcheck missing Go declaration
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -15,6 +15,9 @@ runtime/asm_ARCHSUFF.s: [GOARCH] cannot check cross-package assembly function: E ...@@ -15,6 +15,9 @@ runtime/asm_ARCHSUFF.s: [GOARCH] cannot check cross-package assembly function: E
runtime/asm_ARCHSUFF.s: [GOARCH] cannot check cross-package assembly function: IndexByte is in package bytes runtime/asm_ARCHSUFF.s: [GOARCH] cannot check cross-package assembly function: IndexByte is in package bytes
runtime/asm_ARCHSUFF.s: [GOARCH] cannot check cross-package assembly function: IndexByte is in package strings runtime/asm_ARCHSUFF.s: [GOARCH] cannot check cross-package assembly function: IndexByte is in package strings
// The write barrier is called directly by the compiler, so no Go def
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
// Legitimate vet complaints in which we are testing for correct runtime behavior // Legitimate vet complaints in which we are testing for correct runtime behavior
// in bad situations that vet can also detect statically. // in bad situations that vet can also detect statically.
encoding/json/decode_test.go: struct field m has json tag but is not exported encoding/json/decode_test.go: struct field m has json tag but is not exported
......
...@@ -31,4 +31,3 @@ runtime/duff_amd64.s: [amd64] duffcopy: function duffcopy missing Go declaration ...@@ -31,4 +31,3 @@ runtime/duff_amd64.s: [amd64] duffcopy: function duffcopy missing Go declaration
runtime/asm_amd64.s: [amd64] stackcheck: function stackcheck missing Go declaration runtime/asm_amd64.s: [amd64] stackcheck: function stackcheck missing Go declaration
runtime/asm_amd64.s: [amd64] indexShortStr: function indexShortStr missing Go declaration runtime/asm_amd64.s: [amd64] indexShortStr: function indexShortStr missing Go declaration
runtime/asm_amd64.s: [amd64] countByte: function countByte missing Go declaration runtime/asm_amd64.s: [amd64] countByte: function countByte missing Go declaration
runtime/asm_amd64.s: [amd64] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -16,5 +16,3 @@ runtime/tls_arm.s: [arm] load_g: function load_g missing Go declaration ...@@ -16,5 +16,3 @@ runtime/tls_arm.s: [arm] load_g: function load_g missing Go declaration
runtime/tls_arm.s: [arm] _initcgo: function _initcgo missing Go declaration runtime/tls_arm.s: [arm] _initcgo: function _initcgo missing Go declaration
runtime/internal/atomic/asm_arm.s: [arm] cas: function cas missing Go declaration runtime/internal/atomic/asm_arm.s: [arm] cas: function cas missing Go declaration
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -9,5 +9,3 @@ runtime/duff_arm64.s: [arm64] duffzero: function duffzero missing Go declaration ...@@ -9,5 +9,3 @@ runtime/duff_arm64.s: [arm64] duffzero: function duffzero missing Go declaration
runtime/duff_arm64.s: [arm64] duffcopy: function duffcopy missing Go declaration runtime/duff_arm64.s: [arm64] duffcopy: function duffcopy missing Go declaration
runtime/tls_arm64.s: [arm64] load_g: function load_g missing Go declaration runtime/tls_arm64.s: [arm64] load_g: function load_g missing Go declaration
runtime/tls_arm64.s: [arm64] save_g: function save_g missing Go declaration runtime/tls_arm64.s: [arm64] save_g: function save_g missing Go declaration
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -4,5 +4,3 @@ runtime/asm_mips64x.s: [GOARCH] abort: function abort missing Go declaration ...@@ -4,5 +4,3 @@ runtime/asm_mips64x.s: [GOARCH] abort: function abort missing Go declaration
runtime/duff_mips64x.s: [GOARCH] duffzero: function duffzero missing Go declaration runtime/duff_mips64x.s: [GOARCH] duffzero: function duffzero missing Go declaration
runtime/tls_mips64x.s: [GOARCH] save_g: function save_g missing Go declaration runtime/tls_mips64x.s: [GOARCH] save_g: function save_g missing Go declaration
runtime/tls_mips64x.s: [GOARCH] load_g: function load_g missing Go declaration runtime/tls_mips64x.s: [GOARCH] load_g: function load_g missing Go declaration
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -7,5 +7,3 @@ runtime/asm_mipsx.s: [GOARCH] cannot check cross-package assembly function: Comp ...@@ -7,5 +7,3 @@ runtime/asm_mipsx.s: [GOARCH] cannot check cross-package assembly function: Comp
runtime/sys_linux_mipsx.s: [GOARCH] clone: 12(R29) should be mp+8(FP) runtime/sys_linux_mipsx.s: [GOARCH] clone: 12(R29) should be mp+8(FP)
runtime/sys_linux_mipsx.s: [GOARCH] clone: 4(R29) should be flags+0(FP) runtime/sys_linux_mipsx.s: [GOARCH] clone: 4(R29) should be flags+0(FP)
runtime/sys_linux_mipsx.s: [GOARCH] clone: 8(R29) should be stk+4(FP) runtime/sys_linux_mipsx.s: [GOARCH] clone: 8(R29) should be stk+4(FP)
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -27,5 +27,3 @@ runtime/asm_amd64p32.s: [amd64p32] indexbytebody: function indexbytebody missing ...@@ -27,5 +27,3 @@ runtime/asm_amd64p32.s: [amd64p32] indexbytebody: function indexbytebody missing
runtime/asm_amd64p32.s: [amd64p32] asmcgocall: RET without writing to 4-byte ret+8(FP) runtime/asm_amd64p32.s: [amd64p32] asmcgocall: RET without writing to 4-byte ret+8(FP)
runtime/asm_amd64p32.s: [amd64p32] stackcheck: function stackcheck missing Go declaration runtime/asm_amd64p32.s: [amd64p32] stackcheck: function stackcheck missing Go declaration
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -10,5 +10,3 @@ runtime/asm_ppc64x.s: [GOARCH] addmoduledata: function addmoduledata missing Go ...@@ -10,5 +10,3 @@ runtime/asm_ppc64x.s: [GOARCH] addmoduledata: function addmoduledata missing Go
runtime/duff_ppc64x.s: [GOARCH] duffzero: function duffzero missing Go declaration runtime/duff_ppc64x.s: [GOARCH] duffzero: function duffzero missing Go declaration
runtime/tls_ppc64x.s: [GOARCH] save_g: function save_g missing Go declaration runtime/tls_ppc64x.s: [GOARCH] save_g: function save_g missing Go declaration
runtime/tls_ppc64x.s: [GOARCH] load_g: function load_g missing Go declaration runtime/tls_ppc64x.s: [GOARCH] load_g: function load_g missing Go declaration
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -15,5 +15,3 @@ runtime/memclr_s390x.s: [s390x] memclr_s390x_exrl_xc: function memclr_s390x_exrl ...@@ -15,5 +15,3 @@ runtime/memclr_s390x.s: [s390x] memclr_s390x_exrl_xc: function memclr_s390x_exrl
runtime/memmove_s390x.s: [s390x] memmove_s390x_exrl_mvc: function memmove_s390x_exrl_mvc missing Go declaration runtime/memmove_s390x.s: [s390x] memmove_s390x_exrl_mvc: function memmove_s390x_exrl_mvc missing Go declaration
runtime/tls_s390x.s: [s390x] save_g: function save_g missing Go declaration runtime/tls_s390x.s: [s390x] save_g: function save_g missing Go declaration
runtime/tls_s390x.s: [s390x] load_g: function load_g missing Go declaration runtime/tls_s390x.s: [s390x] load_g: function load_g missing Go declaration
runtime/asm_ARCHSUFF.s: [GOARCH] gcWriteBarrier: function gcWriteBarrier missing Go declaration
...@@ -2397,7 +2397,13 @@ TEXT runtime·gcWriteBarrier(SB),NOSPLIT,$120 ...@@ -2397,7 +2397,13 @@ TEXT runtime·gcWriteBarrier(SB),NOSPLIT,$120
CMPQ R14, (p_wbBuf+wbBuf_end)(R13) CMPQ R14, (p_wbBuf+wbBuf_end)(R13)
// Record the write. // Record the write.
MOVQ AX, -16(R14) // Record value MOVQ AX, -16(R14) // Record value
MOVQ (DI), R13 // TODO: This turns bad writes into bad reads. // Note: This turns bad pointer writes into bad
// pointer reads, which could be confusing. We could avoid
// reading from obviously bad pointers, which would
// take care of the vast majority of these. We could
// patch this up in the signal handler, or use XCHG to
// combine the read and the write.
MOVQ (DI), R13
MOVQ R13, -8(R14) // Record *slot MOVQ R13, -8(R14) // Record *slot
// Is the buffer full? (flags set in CMPQ above) // Is the buffer full? (flags set in CMPQ above)
JEQ flush JEQ flush
......
...@@ -6,10 +6,10 @@ ...@@ -6,10 +6,10 @@
// //
// For the concurrent garbage collector, the Go compiler implements // For the concurrent garbage collector, the Go compiler implements
// updates to pointer-valued fields that may be in heap objects by // updates to pointer-valued fields that may be in heap objects by
// emitting calls to write barriers. This file contains the actual write barrier // emitting calls to write barriers. The main write barrier for
// implementation, gcmarkwb_m, and the various wrappers called by the // individual pointer writes is gcWriteBarrier and is implemented in
// compiler to implement pointer assignment, slice assignment, // assembly. This file contains write barrier entry points for bulk
// typed memmove, and so on. // operations. See also mwbbuf.go.
package runtime package runtime
...@@ -18,10 +18,7 @@ import ( ...@@ -18,10 +18,7 @@ import (
"unsafe" "unsafe"
) )
// gcmarkwb_m is the mark-phase write barrier, the only barrier we have. // Go uses a hybrid barrier that combines a Yuasa-style deletion
// The rest of this file exists only to make calls to this function.
//
// This is a hybrid barrier that combines a Yuasa-style deletion
// barrier—which shades the object whose reference is being // barrier—which shades the object whose reference is being
// overwritten—with Dijkstra insertion barrier—which shades the object // overwritten—with Dijkstra insertion barrier—which shades the object
// whose reference is being written. The insertion part of the barrier // whose reference is being written. The insertion part of the barrier
...@@ -137,105 +134,17 @@ import ( ...@@ -137,105 +134,17 @@ import (
// reachable by some goroutine that currently cannot reach it. // reachable by some goroutine that currently cannot reach it.
// //
// //
//go:nowritebarrierrec // Signal handler pointer writes:
//go:systemstack
func gcmarkwb_m(slot *uintptr, ptr uintptr) {
if writeBarrier.needed {
// Note: This turns bad pointer writes into bad
// pointer reads, which could be confusing. We avoid
// reading from obviously bad pointers, which should
// take care of the vast majority of these. We could
// patch this up in the signal handler, or use XCHG to
// combine the read and the write. Checking inheap is
// insufficient since we need to track changes to
// roots outside the heap.
//
// Note: profbuf.go omits a barrier during signal handler
// profile logging; that's safe only because this deletion barrier exists.
// If we remove the deletion barrier, we'll have to work out
// a new way to handle the profile logging.
if slot1 := uintptr(unsafe.Pointer(slot)); slot1 >= minPhysPageSize {
if optr := *slot; optr != 0 {
shade(optr)
}
}
// TODO: Make this conditional on the caller's stack color.
if ptr != 0 && inheap(ptr) {
shade(ptr)
}
}
}
// writebarrierptr_prewrite1 invokes a write barrier for *dst = src
// prior to the write happening.
//
// Write barrier calls must not happen during critical GC and scheduler
// related operations. In particular there are times when the GC assumes
// that the world is stopped but scheduler related code is still being
// executed, dealing with syscalls, dealing with putting gs on runnable
// queues and so forth. This code cannot execute write barriers because
// the GC might drop them on the floor. Stopping the world involves removing
// the p associated with an m. We use the fact that m.p == nil to indicate
// that we are in one these critical section and throw if the write is of
// a pointer to a heap object.
//go:nosplit
func writebarrierptr_prewrite1(dst *uintptr, src uintptr) {
mp := acquirem()
if mp.inwb || mp.dying > 0 {
// We explicitly allow write barriers in startpanic_m,
// since we're going down anyway. Ignore them here.
releasem(mp)
return
}
systemstack(func() {
if mp.p == 0 && memstats.enablegc && !mp.inwb && inheap(src) {
throw("writebarrierptr_prewrite1 called with mp.p == nil")
}
mp.inwb = true
gcmarkwb_m(dst, src)
})
mp.inwb = false
releasem(mp)
}
// NOTE: Really dst *unsafe.Pointer, src unsafe.Pointer,
// but if we do that, Go inserts a write barrier on *dst = src.
//go:nosplit
func writebarrierptr(dst *uintptr, src uintptr) {
if writeBarrier.cgo {
cgoCheckWriteBarrier(dst, src)
}
if !writeBarrier.needed {
*dst = src
return
}
if src != 0 && src < minPhysPageSize {
systemstack(func() {
print("runtime: writebarrierptr *", dst, " = ", hex(src), "\n")
throw("bad pointer in write barrier")
})
}
writebarrierptr_prewrite1(dst, src)
*dst = src
}
// writebarrierptr_prewrite is like writebarrierptr, but the store
// will be performed by the caller after this call. The caller must
// not allow preemption between this call and the write.
// //
//go:nosplit // In general, the signal handler cannot safely invoke the write
func writebarrierptr_prewrite(dst *uintptr, src uintptr) { // barrier because it may run without a P or even during the write
if writeBarrier.cgo { // barrier.
cgoCheckWriteBarrier(dst, src) //
} // There is exactly one exception: profbuf.go omits a barrier during
if !writeBarrier.needed { // signal handler profile logging. That's safe only because of the
return // deletion barrier. See profbuf.go for a detailed argument. If we
} // remove the deletion barrier, we'll have to work out a new way to
if src != 0 && src < minPhysPageSize { // handle the profile logging.
systemstack(func() { throw("bad pointer in write barrier") })
}
writebarrierptr_prewrite1(dst, src)
}
// typedmemmove copies a value of type t to dst from src. // typedmemmove copies a value of type t to dst from src.
// Must be nosplit, see #16026. // Must be nosplit, see #16026.
......
...@@ -5,6 +5,9 @@ ...@@ -5,6 +5,9 @@
// This implements the write barrier buffer. The write barrier itself // This implements the write barrier buffer. The write barrier itself
// is gcWriteBarrier and is implemented in assembly. // is gcWriteBarrier and is implemented in assembly.
// //
// See mbarrier.go for algorithmic details on the write barrier. This
// file deals only with the buffer.
//
// The write barrier has a fast path and a slow path. The fast path // The write barrier has a fast path and a slow path. The fast path
// simply enqueues to a per-P write barrier buffer. It's written in // simply enqueues to a per-P write barrier buffer. It's written in
// assembly and doesn't clobber any general purpose registers, so it // assembly and doesn't clobber any general purpose registers, so it
...@@ -111,16 +114,21 @@ func (b *wbBuf) discard() { ...@@ -111,16 +114,21 @@ func (b *wbBuf) discard() {
// if !buf.putFast(old, new) { // if !buf.putFast(old, new) {
// wbBufFlush(...) // wbBufFlush(...)
// } // }
// ... actual memory write ...
// //
// The arguments to wbBufFlush depend on whether the caller is doing // The arguments to wbBufFlush depend on whether the caller is doing
// its own cgo pointer checks. If it is, then this can be // its own cgo pointer checks. If it is, then this can be
// wbBufFlush(nil, 0). Otherwise, it must pass the slot address and // wbBufFlush(nil, 0). Otherwise, it must pass the slot address and
// new. // new.
// //
// Since buf is a per-P resource, the caller must ensure there are no // The caller must ensure there are no preemption points during the
// preemption points while buf is in use. // above sequence. There must be no preemption points while buf is in
// use because it is a per-P resource. There must be no preemption
// points between the buffer put and the write to memory because this
// could allow a GC phase change, which could result in missed write
// barriers.
// //
// It must be nowritebarrierrec to because write barriers here would // putFast must be nowritebarrierrec to because write barriers here would
// corrupt the write barrier buffer. It (and everything it calls, if // corrupt the write barrier buffer. It (and everything it calls, if
// it called anything) has to be nosplit to avoid scheduling on to a // it called anything) has to be nosplit to avoid scheduling on to a
// different P and a different buffer. // different P and a different buffer.
...@@ -214,6 +222,14 @@ func wbBufFlush1(_p_ *p) { ...@@ -214,6 +222,14 @@ func wbBufFlush1(_p_ *p) {
// //
// TODO: Should scanobject/scanblock just stuff pointers into // TODO: Should scanobject/scanblock just stuff pointers into
// the wbBuf? Then this would become the sole greying path. // the wbBuf? Then this would become the sole greying path.
//
// TODO: We could avoid shading any of the "new" pointers in
// the buffer if the stack has been shaded, or even avoid
// putting them in the buffer at all (which would double its
// capacity). This is slightly complicated with the buffer; we
// could track whether any un-shaded goroutine has used the
// buffer, or just track globally whether there are any
// un-shaded stacks and flush after each stack scan.
gcw := &_p_.gcw gcw := &_p_.gcw
pos := 0 pos := 0
arenaStart := mheap_.arena_start arenaStart := mheap_.arena_start
......
// errorcheck -0 -live -d=eagerwb // errorcheck -0 -live
// Copyright 2016 The Go Authors. All rights reserved. // Copyright 2016 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style // Use of this source code is governed by a BSD-style
...@@ -7,10 +7,6 @@ ...@@ -7,10 +7,6 @@
// Issue 15747: liveness analysis was marking heap-escaped params live too much, // Issue 15747: liveness analysis was marking heap-escaped params live too much,
// and worse was using the wrong bitmap bits to do so. // and worse was using the wrong bitmap bits to do so.
// TODO(austin): This expects function calls to the write barrier, so
// we enable the legacy eager write barrier. Fix this once the
// buffered write barrier works on all arches.
package p package p
var global *[]byte var global *[]byte
...@@ -21,14 +17,14 @@ type T struct{ M string } ...@@ -21,14 +17,14 @@ type T struct{ M string }
var b bool var b bool
func f1(q *Q, xx []byte) interface{} { // ERROR "live at call to newobject: xx$" "live at call to writebarrierptr: &xx$" "live at entry to f1: xx$" func f1(q *Q, xx []byte) interface{} { // ERROR "live at call to newobject: xx$" "live at entry to f1: xx$"
// xx was copied from the stack to the heap on the previous line: // xx was copied from the stack to the heap on the previous line:
// xx was live for the first two prints but then it switched to &xx // xx was live for the first two prints but then it switched to &xx
// being live. We should not see plain xx again. // being live. We should not see plain xx again.
if b { if b {
global = &xx // ERROR "live at call to writebarrierptr: &xx$" global = &xx
} }
xx, _, err := f2(xx, 5) // ERROR "live at call to f2: &xx$" "live at call to writebarrierptr: err.data err.type$" xx, _, err := f2(xx, 5) // ERROR "live at call to f2: &xx$"
if err != nil { if err != nil {
return err return err
} }
...@@ -38,7 +34,7 @@ func f1(q *Q, xx []byte) interface{} { // ERROR "live at call to newobject: xx$" ...@@ -38,7 +34,7 @@ func f1(q *Q, xx []byte) interface{} { // ERROR "live at call to newobject: xx$"
//go:noinline //go:noinline
func f2(d []byte, n int) (odata, res []byte, e interface{}) { // ERROR "live at entry to f2: d$" func f2(d []byte, n int) (odata, res []byte, e interface{}) { // ERROR "live at entry to f2: d$"
if n > len(d) { if n > len(d) {
return d, nil, &T{M: "hello"} // ERROR "live at call to newobject: d" "live at call to writebarrierptr: d" return d, nil, &T{M: "hello"} // ERROR "live at call to newobject: d"
} }
res = d[:n] res = d[:n]
odata = d[n:] odata = d[n:]
......
// errorcheck -0 -live -l -d=compilelater,eagerwb // errorcheck -0 -live -l -d=compilelater
// Copyright 2017 The Go Authors. All rights reserved. // Copyright 2017 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style // Use of this source code is governed by a BSD-style
...@@ -8,10 +8,6 @@ ...@@ -8,10 +8,6 @@
// due to propagation of addrtaken to outer variables for // due to propagation of addrtaken to outer variables for
// closure variables. // closure variables.
// TODO(austin): This expects function calls to the write barrier, so
// we enable the legacy eager write barrier. Fix this once the
// buffered write barrier works on all arches.
package p package p
type T struct { type T struct {
...@@ -21,7 +17,7 @@ type T struct { ...@@ -21,7 +17,7 @@ type T struct {
func f(a T) { // ERROR "live at entry to f: a" func f(a T) { // ERROR "live at entry to f: a"
var e interface{} var e interface{}
func() { // ERROR "live at entry to f.func1: a &e" func() { // ERROR "live at entry to f.func1: a &e"
e = a.s // ERROR "live at call to convT2Estring: a &e" "live at call to writebarrierptr: a" e = a.s // ERROR "live at call to convT2Estring: a &e"
}() // ERROR "live at call to f.func1: e$" }() // ERROR "live at call to f.func1: e$"
// Before the fix, both a and e were live at the previous line. // Before the fix, both a and e were live at the previous line.
_ = e _ = e
......
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