// 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 file implements printing of syntax tree structures. package syntax import ( "fmt" "io" "reflect" "unicode" "unicode/utf8" ) // Fdump dumps the structure of the syntax tree rooted at n to w. // It is intended for debugging purposes; no specific output format // is guaranteed. func Fdump(w io.Writer, n Node) (err error) { p := dumper{ output: w, ptrmap: make(map[Node]int), last: '\n', // force printing of line number on first line } defer func() { if e := recover(); e != nil { err = e.(localError).err // re-panics if it's not a localError } }() if n == nil { p.printf("nil\n") return } p.dump(reflect.ValueOf(n), n) p.printf("\n") return } type dumper struct { output io.Writer ptrmap map[Node]int // node -> dump line number indent int // current indentation level last byte // last byte processed by Write line int // current line number } var indentBytes = []byte(". ") func (p *dumper) Write(data []byte) (n int, err error) { var m int for i, b := range data { // invariant: data[0:n] has been written if b == '\n' { m, err = p.output.Write(data[n : i+1]) n += m if err != nil { return } } else if p.last == '\n' { p.line++ _, err = fmt.Fprintf(p.output, "%6d ", p.line) if err != nil { return } for j := p.indent; j > 0; j-- { _, err = p.output.Write(indentBytes) if err != nil { return } } } p.last = b } if len(data) > n { m, err = p.output.Write(data[n:]) n += m } return } // localError wraps locally caught errors so we can distinguish // them from genuine panics which we don't want to return as errors. type localError struct { err error } // printf is a convenience wrapper that takes care of print errors. func (p *dumper) printf(format string, args ...interface{}) { if _, err := fmt.Fprintf(p, format, args...); err != nil { panic(localError{err}) } } // dump prints the contents of x. // If x is the reflect.Value of a struct s, where &s // implements Node, then &s should be passed for n - // this permits printing of the unexported span and // comments fields of the embedded isNode field by // calling the Span() and Comment() instead of using // reflection. func (p *dumper) dump(x reflect.Value, n Node) { switch x.Kind() { case reflect.Interface: if x.IsNil() { p.printf("nil") return } p.dump(x.Elem(), nil) case reflect.Ptr: if x.IsNil() { p.printf("nil") return } // special cases for identifiers w/o attached comments (common case) if x, ok := x.Interface().(*Name); ok { p.printf("%s @ %v", x.Value, x.Pos()) return } p.printf("*") // Fields may share type expressions, and declarations // may share the same group - use ptrmap to keep track // of nodes that have been printed already. if ptr, ok := x.Interface().(Node); ok { if line, exists := p.ptrmap[ptr]; exists { p.printf("(Node @ %d)", line) return } p.ptrmap[ptr] = p.line n = ptr } p.dump(x.Elem(), n) case reflect.Slice: if x.IsNil() { p.printf("nil") return } p.printf("%s (%d entries) {", x.Type(), x.Len()) if x.Len() > 0 { p.indent++ p.printf("\n") for i, n := 0, x.Len(); i < n; i++ { p.printf("%d: ", i) p.dump(x.Index(i), nil) p.printf("\n") } p.indent-- } p.printf("}") case reflect.Struct: typ := x.Type() // if span, ok := x.Interface().(lexical.Span); ok { // p.printf("%s", &span) // return // } p.printf("%s {", typ) p.indent++ first := true if n != nil { p.printf("\n") first = false // p.printf("Span: %s\n", n.Span()) // if c := *n.Comments(); c != nil { // p.printf("Comments: ") // p.dump(reflect.ValueOf(c), nil) // a Comment is not a Node // p.printf("\n") // } } for i, n := 0, typ.NumField(); i < n; i++ { // Exclude non-exported fields because their // values cannot be accessed via reflection. if name := typ.Field(i).Name; isExported(name) { if first { p.printf("\n") first = false } p.printf("%s: ", name) p.dump(x.Field(i), nil) p.printf("\n") } } p.indent-- p.printf("}") default: switch x := x.Interface().(type) { case string: // print strings in quotes p.printf("%q", x) default: p.printf("%v", x) } } } func isExported(name string) bool { ch, _ := utf8.DecodeRuneInString(name) return unicode.IsUpper(ch) }