// Copyright 2009 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. package runtime import ( "internal/strconv" "unsafe" ) // The compiler knows that a print of a value of this type // should use printhex instead of printuint (decimal). type hex uint64 // The compiler knows that a print of a value of this type should use // printquoted instead of printstring. type quoted string func bytes(s string) (ret []byte) { rp := (*slice)(unsafe.Pointer(&ret)) sp := stringStructOf(&s) rp.array = sp.str rp.len = sp.len rp.cap = sp.len return } var ( // printBacklog is a circular buffer of messages written with the builtin // print* functions, for use in postmortem analysis of core dumps. printBacklog [512]byte printBacklogIndex int ) // recordForPanic maintains a circular buffer of messages written by the // runtime leading up to a process crash, allowing the messages to be // extracted from a core dump. // // The text written during a process crash (following "panic" or "fatal // error") is not saved, since the goroutine stacks will generally be readable // from the runtime data structures in the core file. func recordForPanic(b []byte) { printlock() if panicking.Load() == 0 { // Not actively crashing: maintain circular buffer of print output. for i := 0; i < len(b); { n := copy(printBacklog[printBacklogIndex:], b[i:]) i += n printBacklogIndex += n printBacklogIndex %= len(printBacklog) } } printunlock() } var debuglock mutex // The compiler emits calls to printlock and printunlock around // the multiple calls that implement a single Go print or println // statement. Some of the print helpers (printslice, for example) // call print recursively. There is also the problem of a crash // happening during the print routines and needing to acquire // the print lock to print information about the crash. // For both these reasons, let a thread acquire the printlock 'recursively'. func printlock() { mp := getg().m mp.locks++ // do not reschedule between printlock++ and lock(&debuglock). mp.printlock++ if mp.printlock == 1 { lock(&debuglock) } mp.locks-- // now we know debuglock is held and holding up mp.locks for us. } func printunlock() { mp := getg().m mp.printlock-- if mp.printlock == 0 { unlock(&debuglock) } } // write to goroutine-local buffer if diverting output, // or else standard error. func gwrite(b []byte) { if len(b) == 0 { return } recordForPanic(b) gp := getg() // Don't use the writebuf if gp.m is dying. We want anything // written through gwrite to appear in the terminal rather // than be written to in some buffer, if we're in a panicking state. // Note that we can't just clear writebuf in the gp.m.dying case // because a panic isn't allowed to have any write barriers. if gp == nil || gp.writebuf == nil || gp.m.dying > 0 { writeErr(b) return } n := copy(gp.writebuf[len(gp.writebuf):cap(gp.writebuf)], b) gp.writebuf = gp.writebuf[:len(gp.writebuf)+n] } func printsp() { printstring(" ") } func printnl() { printstring("\n") } func printbool(v bool) { if v { printstring("true") } else { printstring("false") } } func printfloat64(v float64) { var buf [20]byte gwrite(strconv.AppendFloat(buf[:0], v, 'g', -1, 64)) } func printfloat32(v float32) { var buf [20]byte gwrite(strconv.AppendFloat(buf[:0], float64(v), 'g', -1, 32)) } func printcomplex128(c complex128) { var buf [44]byte gwrite(strconv.AppendComplex(buf[:0], c, 'g', -1, 128)) } func printcomplex64(c complex64) { var buf [44]byte gwrite(strconv.AppendComplex(buf[:0], complex128(c), 'g', -1, 64)) } func printuint(v uint64) { // Note: Avoiding strconv.AppendUint so that it's clearer // that there are no allocations in this routine. // cmd/link/internal/ld.TestAbstractOriginSanity // sees the append and doesn't realize it doesn't allocate. var buf [20]byte i := strconv.RuntimeFormatBase10(buf[:], v) gwrite(buf[i:]) } func printint(v int64) { // Note: Avoiding strconv.AppendUint so that it's clearer // that there are no allocations in this routine. // cmd/link/internal/ld.TestAbstractOriginSanity // sees the append and doesn't realize it doesn't allocate. neg := v < 0 u := uint64(v) if neg { u = -u } var buf [20]byte i := strconv.RuntimeFormatBase10(buf[:], u) if neg { i-- buf[i] = '-' } gwrite(buf[i:]) } var minhexdigits = 0 // protected by printlock func printhexopts(include0x bool, mindigits int, v uint64) { const dig = "0123456789abcdef" var buf [100]byte i := len(buf) for i--; i > 0; i-- { buf[i] = dig[v%16] if v < 16 && len(buf)-i >= mindigits { break } v /= 16 } if include0x { i-- buf[i] = 'x' i-- buf[i] = '0' } gwrite(buf[i:]) } func printhex(v uint64) { printhexopts(true, minhexdigits, v) } func printquoted(s string) { printlock() gwrite([]byte(`"`)) for _, r := range s { switch r { case '\n': gwrite([]byte(`\n`)) continue case '\r': gwrite([]byte(`\r`)) continue case '\t': gwrite([]byte(`\t`)) print() continue case '\\', '"': gwrite([]byte{byte('\\'), byte(r)}) continue } // For now, only allow basic printable ascii through unescaped if r >= ' ' && r <= '~' { gwrite([]byte{byte(r)}) } else if r < 127 { gwrite(bytes(`\x`)) printhexopts(false, 2, uint64(r)) } else if r < 0x1_0000 { gwrite(bytes(`\u`)) printhexopts(false, 4, uint64(r)) } else { gwrite(bytes(`\U`)) printhexopts(false, 8, uint64(r)) } } gwrite([]byte{byte('"')}) printunlock() } func printpointer(p unsafe.Pointer) { printhex(uint64(uintptr(p))) } func printuintptr(p uintptr) { printhex(uint64(p)) } func printstring(s string) { gwrite(bytes(s)) } func printslice(s []byte) { sp := (*slice)(unsafe.Pointer(&s)) print("[", len(s), "/", cap(s), "]") printpointer(sp.array) } func printeface(e eface) { print("(", e._type, ",", e.data, ")") } func printiface(i iface) { print("(", i.tab, ",", i.data, ")") }