blob: 64f6f6ad42e2502d09a8a9ff739830402bfeda3b [file] [log] [blame]
Rob Pike418b97c2008-10-24 16:33:29 -07001// Copyright 2009 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
Rob Pikeb6ce2a72009-06-22 18:09:40 -07005/*
6 Package fmt implements formatted I/O with functions analogous
7 to C's printf. The format 'verbs' are derived from C's but
8 are simpler.
9
10 The verbs:
11
12 General:
Russ Coxa843b452009-08-31 16:38:30 -070013 %v the value in a default format.
Rob Pikeb6ce2a72009-06-22 18:09:40 -070014 when printing structs, the plus flag (%+v) adds field names
Russ Coxa843b452009-08-31 16:38:30 -070015 %#v a Go-syntax representation of the value
16 %T a Go-syntax representation of the type of the value
17
Rob Pikeb6ce2a72009-06-22 18:09:40 -070018 Boolean:
19 %t the word true or false
20 Integer:
21 %b base 2
22 %c the character represented by the corresponding Unicode code point
23 %d base 10
24 %o base 8
25 %x base 16, with lower-case letters for a-f
26 %X base 16, with upper-case letters for A-F
27 Floating-point:
28 %e scientific notation, e.g. -1234.456e+78
Russ Coxa843b452009-08-31 16:38:30 -070029 %E scientific notation, e.g. -1234.456E+78
Rob Pikeb6ce2a72009-06-22 18:09:40 -070030 %f decimal point but no exponent, e.g. 123.456
31 %g whichever of %e or %f produces more compact output
Russ Coxa843b452009-08-31 16:38:30 -070032 %G whichever of %E or %f produces more compact output
Rob Pikeb6ce2a72009-06-22 18:09:40 -070033 String and slice of bytes:
34 %s the uninterpreted bytes of the string or slice
35 %q a double-quoted string safely escaped with Go syntax
36 %x base 16 notation with two characters per byte
37 Pointer:
38 %p base 16 notation, with leading 0x
Rob Pikeb6ce2a72009-06-22 18:09:40 -070039
40 There is no 'u' flag. Integers are printed unsigned if they have unsigned type.
41 Similarly, there is no need to specify the size of the operand (int8, int64).
42
43 For numeric values, the width and precision flags control
44 formatting; width sets the width of the field, precision the
45 number of places after the decimal, if appropriate. The
46 format %6.2f prints 123.45.
47
48 Other flags:
49 + always print a sign for numeric values
50 - pad with spaces on the right rather than the left (left-justify the field)
51 # alternate format: add leading 0 for octal (%#o), 0x for hex (%#x);
52 suppress 0x for %p (%#p);
53 print a raw (backquoted) string if possible for %q (%#q)
54 ' ' (space) leave a space for elided sign in numbers (% d);
55 put spaces between bytes printing strings or slices in hex (% x)
56 0 pad with leading zeros rather than spaces
57
58 For each Printf-like function, there is also a Print function
59 that takes no format and is equivalent to saying %v for every
60 operand. Another variant Println inserts blanks between
61 operands and appends a newline.
62
Russ Cox0d400a72009-07-07 11:03:31 -070063 Regardless of the verb, if an operand is an interface value,
64 the internal concrete value is used, not the interface itself.
65 Thus:
66 var i interface{} = 23;
67 fmt.Printf("%v\n", i);
68 will print 23.
69
Rob Pike07a497f2009-07-30 16:57:46 -070070 If an operand implements interface Formatter, that interface
Rob Pikeb6ce2a72009-06-22 18:09:40 -070071 can be used for fine control of formatting.
72
73 If an operand implements method String() string that method
74 will be used for %v, %s, or Print etc.
75*/
Rob Pike418b97c2008-10-24 16:33:29 -070076package fmt
77
Rob Pikedb25e782008-10-26 08:27:50 -070078
Rob Pike418b97c2008-10-24 16:33:29 -070079import (
Rob Pike23553952008-11-14 10:42:45 -080080 "io";
Rob Pike418b97c2008-10-24 16:33:29 -070081 "os";
Russ Cox1f6463f2009-04-16 20:52:37 -070082 "reflect";
Russ Cox36096242009-01-16 14:58:14 -080083 "utf8";
Rob Pike418b97c2008-10-24 16:33:29 -070084)
85
Rob Pike28ba9772009-06-23 15:20:30 -070086// State represents the printer state passed to custom formatters.
Rob Pikec8b47c62009-05-08 11:22:57 -070087// It provides access to the io.Writer interface plus information about
Rob Pike85647c92009-03-06 03:35:38 -080088// the flags and options for the operand's format specifier.
Rob Pike28ba9772009-06-23 15:20:30 -070089type State interface {
Rob Pike85647c92009-03-06 03:35:38 -080090 // Write is the function to call to emit formatted output to be printed.
Rob Pikeaaf63f82009-04-17 00:08:24 -070091 Write(b []byte) (ret int, err os.Error);
Rob Pike85647c92009-03-06 03:35:38 -080092 // Width returns the value of the width option and whether it has been set.
Russ Coxecb863a2009-10-06 15:38:57 -070093 Width() (wid int, ok bool);
Rob Pike85647c92009-03-06 03:35:38 -080094 // Precision returns the value of the precision option and whether it has been set.
Russ Coxecb863a2009-10-06 15:38:57 -070095 Precision() (prec int, ok bool);
Russ Coxbf67afc2008-12-11 16:53:33 -080096
Rob Pike85647c92009-03-06 03:35:38 -080097 // Flag returns whether the flag c, a character, has been set.
Russ Coxecb863a2009-10-06 15:38:57 -070098 Flag(int) bool;
Rob Pikef15dfa72008-11-06 10:40:57 -080099}
100
Russ Coxa843b452009-08-31 16:38:30 -0700101// Formatter is the interface implemented by values with a custom formatter.
Rob Pike85647c92009-03-06 03:35:38 -0800102// The implementation of Format may call Sprintf or Fprintf(f) etc.
103// to generate its output.
Rob Pike28ba9772009-06-23 15:20:30 -0700104type Formatter interface {
105 Format(f State, c int);
Rob Pike91212bd2008-11-06 11:38:44 -0800106}
107
Russ Coxa843b452009-08-31 16:38:30 -0700108// Stringer is implemented by any value that has a String method(),
109// which defines the ``native'' format for that value.
110// The String method is used to print values passed as an operand
111// to a %s or %v format or to an unformatted printer such as Print.
Rob Pikec8b47c62009-05-08 11:22:57 -0700112type Stringer interface {
Russ Coxecb863a2009-10-06 15:38:57 -0700113 String() string;
Rob Pikef15dfa72008-11-06 10:40:57 -0800114}
115
Russ Coxa843b452009-08-31 16:38:30 -0700116// GoStringer is implemented by any value that has a GoString() method,
117// which defines the Go syntax for that value.
118// The GoString method is used to print values passed as an operand
119// to a %#v format.
120type GoStringer interface {
Russ Coxecb863a2009-10-06 15:38:57 -0700121 GoString() string;
Russ Coxa843b452009-08-31 16:38:30 -0700122}
123
Rob Pike85647c92009-03-06 03:35:38 -0800124const runeSelf = utf8.RuneSelf
Rob Pike497bb9c2009-01-15 15:40:27 -0800125const allocSize = 32
Rob Pike418b97c2008-10-24 16:33:29 -0700126
Rob Pike497bb9c2009-01-15 15:40:27 -0800127type pp struct {
Rob Pike418b97c2008-10-24 16:33:29 -0700128 n int;
Russ Coxd47d8882008-12-18 22:37:22 -0800129 buf []byte;
Rob Pike418b97c2008-10-24 16:33:29 -0700130 fmt *Fmt;
131}
132
Rob Pikedb1656f2009-01-16 13:29:43 -0800133func newPrinter() *pp {
Rob Pike497bb9c2009-01-15 15:40:27 -0800134 p := new(pp);
Russ Cox04960402009-08-10 22:02:51 -0700135 p.fmt = New();
Rob Pike418b97c2008-10-24 16:33:29 -0700136 return p;
137}
138
Rob Pike497bb9c2009-01-15 15:40:27 -0800139func (p *pp) Width() (wid int, ok bool) {
Russ Coxecb863a2009-10-06 15:38:57 -0700140 return p.fmt.wid, p.fmt.wid_present;
Rob Pikef15dfa72008-11-06 10:40:57 -0800141}
142
Rob Pike497bb9c2009-01-15 15:40:27 -0800143func (p *pp) Precision() (prec int, ok bool) {
Russ Coxecb863a2009-10-06 15:38:57 -0700144 return p.fmt.prec, p.fmt.prec_present;
Russ Coxbf67afc2008-12-11 16:53:33 -0800145}
146
Rob Pike497bb9c2009-01-15 15:40:27 -0800147func (p *pp) Flag(b int) bool {
Russ Coxbf67afc2008-12-11 16:53:33 -0800148 switch b {
149 case '-':
150 return p.fmt.minus;
151 case '+':
152 return p.fmt.plus;
153 case '#':
154 return p.fmt.sharp;
155 case ' ':
156 return p.fmt.space;
157 case '0':
158 return p.fmt.zero;
159 }
Russ Coxecb863a2009-10-06 15:38:57 -0700160 return false;
Rob Pikef15dfa72008-11-06 10:40:57 -0800161}
162
Rob Pike497bb9c2009-01-15 15:40:27 -0800163func (p *pp) ensure(n int) {
Russ Coxd47d8882008-12-18 22:37:22 -0800164 if len(p.buf) < n {
Rob Pike497bb9c2009-01-15 15:40:27 -0800165 newn := allocSize + len(p.buf);
Rob Pike418b97c2008-10-24 16:33:29 -0700166 if newn < n {
Russ Coxecb863a2009-10-06 15:38:57 -0700167 newn = n+allocSize;
Rob Pike418b97c2008-10-24 16:33:29 -0700168 }
Russ Cox55645042009-01-06 15:19:02 -0800169 b := make([]byte, newn);
Rob Pike418b97c2008-10-24 16:33:29 -0700170 for i := 0; i < p.n; i++ {
171 b[i] = p.buf[i];
172 }
173 p.buf = b;
174 }
175}
176
Rob Pike497bb9c2009-01-15 15:40:27 -0800177func (p *pp) addstr(s string) {
Rob Pike418b97c2008-10-24 16:33:29 -0700178 n := len(s);
179 p.ensure(p.n + n);
180 for i := 0; i < n; i++ {
181 p.buf[p.n] = s[i];
182 p.n++;
183 }
184}
185
Rob Pike497bb9c2009-01-15 15:40:27 -0800186func (p *pp) addbytes(b []byte, start, end int) {
Russ Coxecb863a2009-10-06 15:38:57 -0700187 p.ensure(p.n + end - start);
Rob Pike418b97c2008-10-24 16:33:29 -0700188 for i := start; i < end; i++ {
189 p.buf[p.n] = b[i];
190 p.n++;
191 }
192}
193
Rob Pike497bb9c2009-01-15 15:40:27 -0800194func (p *pp) add(c int) {
Rob Pike418b97c2008-10-24 16:33:29 -0700195 p.ensure(p.n + 1);
Rob Pike497bb9c2009-01-15 15:40:27 -0800196 if c < runeSelf {
Rob Pike418b97c2008-10-24 16:33:29 -0700197 p.buf[p.n] = byte(c);
198 p.n++;
199 } else {
200 p.addstr(string(c));
201 }
202}
203
Rob Pike3200b062008-11-04 13:57:21 -0800204// Implement Write so we can call fprintf on a P, for
205// recursive use in custom verbs.
Rob Pikeaaf63f82009-04-17 00:08:24 -0700206func (p *pp) Write(b []byte) (ret int, err os.Error) {
Rob Pike3200b062008-11-04 13:57:21 -0800207 p.addbytes(b, 0, len(b));
208 return len(b), nil;
Rob Pike418b97c2008-10-24 16:33:29 -0700209}
210
Rob Pike418b97c2008-10-24 16:33:29 -0700211// These routines end in 'f' and take a format string.
212
Rob Pike85647c92009-03-06 03:35:38 -0800213// Fprintf formats according to a format specifier and writes to w.
Rob Pikec8b47c62009-05-08 11:22:57 -0700214func Fprintf(w io.Writer, format string, a ...) (n int, error os.Error) {
Russ Cox0d400a72009-07-07 11:03:31 -0700215 v := reflect.NewValue(a).(*reflect.StructValue);
Rob Pikedb1656f2009-01-16 13:29:43 -0800216 p := newPrinter();
Rob Pike418b97c2008-10-24 16:33:29 -0700217 p.doprintf(format, v);
Russ Coxecb863a2009-10-06 15:38:57 -0700218 n, error = w.Write(p.buf[0 : p.n]);
Rob Pike418b97c2008-10-24 16:33:29 -0700219 return n, error;
220}
221
Rob Pike85647c92009-03-06 03:35:38 -0800222// Printf formats according to a format specifier and writes to standard output.
Rob Pikeaaf63f82009-04-17 00:08:24 -0700223func Printf(format string, v ...) (n int, errno os.Error) {
Rob Pike61f33022009-01-15 13:48:11 -0800224 n, errno = Fprintf(os.Stdout, format, v);
Rob Pike418b97c2008-10-24 16:33:29 -0700225 return n, errno;
226}
227
Rob Pike85647c92009-03-06 03:35:38 -0800228// Sprintf formats according to a format specifier and returns the resulting string.
Russ Cox839a6842009-01-20 14:40:40 -0800229func Sprintf(format string, a ...) string {
Russ Cox0d400a72009-07-07 11:03:31 -0700230 v := reflect.NewValue(a).(*reflect.StructValue);
Rob Pikedb1656f2009-01-16 13:29:43 -0800231 p := newPrinter();
Robert Griesemer3a2c0a92008-11-06 11:56:08 -0800232 p.doprintf(format, v);
Rob Pike418b97c2008-10-24 16:33:29 -0700233 s := string(p.buf)[0 : p.n];
Rob Pike418b97c2008-10-24 16:33:29 -0700234 return s;
235}
236
Rob Pike85647c92009-03-06 03:35:38 -0800237// These routines do not take a format string
Rob Pike418b97c2008-10-24 16:33:29 -0700238
Rob Pike85647c92009-03-06 03:35:38 -0800239// Fprint formats using the default formats for its operands and writes to w.
240// Spaces are added between operands when neither is a string.
Rob Pikec8b47c62009-05-08 11:22:57 -0700241func Fprint(w io.Writer, a ...) (n int, error os.Error) {
Russ Cox0d400a72009-07-07 11:03:31 -0700242 v := reflect.NewValue(a).(*reflect.StructValue);
Rob Pikedb1656f2009-01-16 13:29:43 -0800243 p := newPrinter();
Rob Pike2d4f7ba2008-11-02 12:33:02 -0800244 p.doprint(v, false, false);
Russ Coxecb863a2009-10-06 15:38:57 -0700245 n, error = w.Write(p.buf[0 : p.n]);
Rob Pike418b97c2008-10-24 16:33:29 -0700246 return n, error;
247}
248
Rob Pike85647c92009-03-06 03:35:38 -0800249// Print formats using the default formats for its operands and writes to standard output.
250// Spaces are added between operands when neither is a string.
Rob Pikeaaf63f82009-04-17 00:08:24 -0700251func Print(v ...) (n int, errno os.Error) {
Rob Pike61f33022009-01-15 13:48:11 -0800252 n, errno = Fprint(os.Stdout, v);
Rob Pike418b97c2008-10-24 16:33:29 -0700253 return n, errno;
254}
255
Rob Pike85647c92009-03-06 03:35:38 -0800256// Sprint formats using the default formats for its operands and returns the resulting string.
257// Spaces are added between operands when neither is a string.
Russ Cox839a6842009-01-20 14:40:40 -0800258func Sprint(a ...) string {
Russ Cox0d400a72009-07-07 11:03:31 -0700259 v := reflect.NewValue(a).(*reflect.StructValue);
Rob Pikedb1656f2009-01-16 13:29:43 -0800260 p := newPrinter();
Robert Griesemer3a2c0a92008-11-06 11:56:08 -0800261 p.doprint(v, false, false);
Rob Pike418b97c2008-10-24 16:33:29 -0700262 s := string(p.buf)[0 : p.n];
Rob Pike418b97c2008-10-24 16:33:29 -0700263 return s;
264}
265
266// These routines end in 'ln', do not take a format string,
267// always add spaces between operands, and add a newline
268// after the last operand.
269
Rob Pike85647c92009-03-06 03:35:38 -0800270// Fprintln formats using the default formats for its operands and writes to w.
271// Spaces are always added between operands and a newline is appended.
Rob Pikec8b47c62009-05-08 11:22:57 -0700272func Fprintln(w io.Writer, a ...) (n int, error os.Error) {
Russ Cox0d400a72009-07-07 11:03:31 -0700273 v := reflect.NewValue(a).(*reflect.StructValue);
Rob Pikedb1656f2009-01-16 13:29:43 -0800274 p := newPrinter();
Rob Pike2d4f7ba2008-11-02 12:33:02 -0800275 p.doprint(v, true, true);
Russ Coxecb863a2009-10-06 15:38:57 -0700276 n, error = w.Write(p.buf[0 : p.n]);
Rob Pike418b97c2008-10-24 16:33:29 -0700277 return n, error;
278}
279
Rob Pike85647c92009-03-06 03:35:38 -0800280// Println formats using the default formats for its operands and writes to standard output.
281// Spaces are always added between operands and a newline is appended.
Rob Pikeaaf63f82009-04-17 00:08:24 -0700282func Println(v ...) (n int, errno os.Error) {
Rob Pike61f33022009-01-15 13:48:11 -0800283 n, errno = Fprintln(os.Stdout, v);
Rob Pike418b97c2008-10-24 16:33:29 -0700284 return n, errno;
285}
286
Rob Pike85647c92009-03-06 03:35:38 -0800287// Sprintln formats using the default formats for its operands and returns the resulting string.
288// Spaces are always added between operands and a newline is appended.
Russ Cox839a6842009-01-20 14:40:40 -0800289func Sprintln(a ...) string {
Russ Cox0d400a72009-07-07 11:03:31 -0700290 v := reflect.NewValue(a).(*reflect.StructValue);
Rob Pikedb1656f2009-01-16 13:29:43 -0800291 p := newPrinter();
Robert Griesemer3a2c0a92008-11-06 11:56:08 -0800292 p.doprint(v, true, true);
Rob Pike418b97c2008-10-24 16:33:29 -0700293 s := string(p.buf)[0 : p.n];
Rob Pike418b97c2008-10-24 16:33:29 -0700294 return s;
295}
296
Russ Cox387df5e2008-11-24 14:51:33 -0800297
298// Get the i'th arg of the struct value.
299// If the arg itself is an interface, return a value for
300// the thing inside the interface, not the interface itself.
Russ Cox0d400a72009-07-07 11:03:31 -0700301func getField(v *reflect.StructValue, i int) reflect.Value {
Russ Cox387df5e2008-11-24 14:51:33 -0800302 val := v.Field(i);
Russ Cox0d400a72009-07-07 11:03:31 -0700303 if i, ok := val.(*reflect.InterfaceValue); ok {
304 if inter := i.Interface(); inter != nil {
305 return reflect.NewValue(inter);
306 }
Russ Cox387df5e2008-11-24 14:51:33 -0800307 }
308 return val;
309}
310
Rob Pike418b97c2008-10-24 16:33:29 -0700311// Getters for the fields of the argument structure.
312
Rob Pikef15dfa72008-11-06 10:40:57 -0800313func getBool(v reflect.Value) (val bool, ok bool) {
Russ Cox0d400a72009-07-07 11:03:31 -0700314 if b, ok := v.(*reflect.BoolValue); ok {
315 return b.Get(), true;
Rob Pikef15dfa72008-11-06 10:40:57 -0800316 }
Russ Cox0d400a72009-07-07 11:03:31 -0700317 return;
Rob Pikef15dfa72008-11-06 10:40:57 -0800318}
319
Rob Pike418b97c2008-10-24 16:33:29 -0700320func getInt(v reflect.Value) (val int64, signed, ok bool) {
Russ Cox0d400a72009-07-07 11:03:31 -0700321 switch v := v.(type) {
322 case *reflect.IntValue:
323 return int64(v.Get()), true, true;
324 case *reflect.Int8Value:
325 return int64(v.Get()), true, true;
326 case *reflect.Int16Value:
327 return int64(v.Get()), true, true;
328 case *reflect.Int32Value:
329 return int64(v.Get()), true, true;
330 case *reflect.Int64Value:
331 return int64(v.Get()), true, true;
332 case *reflect.UintValue:
333 return int64(v.Get()), false, true;
334 case *reflect.Uint8Value:
335 return int64(v.Get()), false, true;
336 case *reflect.Uint16Value:
337 return int64(v.Get()), false, true;
338 case *reflect.Uint32Value:
339 return int64(v.Get()), false, true;
340 case *reflect.Uint64Value:
341 return int64(v.Get()), false, true;
342 case *reflect.UintptrValue:
343 return int64(v.Get()), false, true;
Rob Pike418b97c2008-10-24 16:33:29 -0700344 }
Russ Cox0d400a72009-07-07 11:03:31 -0700345 return;
Rob Pike418b97c2008-10-24 16:33:29 -0700346}
347
348func getString(v reflect.Value) (val string, ok bool) {
Russ Cox0d400a72009-07-07 11:03:31 -0700349 if v, ok := v.(*reflect.StringValue); ok {
350 return v.Get(), true;
Rob Pike418b97c2008-10-24 16:33:29 -0700351 }
Russ Cox0d400a72009-07-07 11:03:31 -0700352 if bytes, ok := v.Interface().([]byte); ok {
353 return string(bytes), true;
354 }
355 return;
Rob Pike418b97c2008-10-24 16:33:29 -0700356}
357
Russ Cox079c00a2008-11-17 12:34:03 -0800358func getFloat32(v reflect.Value) (val float32, ok bool) {
Russ Cox0d400a72009-07-07 11:03:31 -0700359 switch v := v.(type) {
360 case *reflect.Float32Value:
361 return float32(v.Get()), true;
362 case *reflect.FloatValue:
Russ Coxecb863a2009-10-06 15:38:57 -0700363 if v.Type().Size() * 8 == 32 {
Russ Cox0d400a72009-07-07 11:03:31 -0700364 return float32(v.Get()), true;
Russ Cox079c00a2008-11-17 12:34:03 -0800365 }
366 }
Russ Cox0d400a72009-07-07 11:03:31 -0700367 return;
Russ Cox079c00a2008-11-17 12:34:03 -0800368}
369
370func getFloat64(v reflect.Value) (val float64, ok bool) {
Russ Cox0d400a72009-07-07 11:03:31 -0700371 switch v := v.(type) {
372 case *reflect.FloatValue:
Russ Coxecb863a2009-10-06 15:38:57 -0700373 if v.Type().Size() * 8 == 64 {
Russ Cox0d400a72009-07-07 11:03:31 -0700374 return float64(v.Get()), true;
Russ Cox079c00a2008-11-17 12:34:03 -0800375 }
Russ Cox0d400a72009-07-07 11:03:31 -0700376 case *reflect.Float64Value:
377 return float64(v.Get()), true;
Rob Pike418b97c2008-10-24 16:33:29 -0700378 }
Russ Cox0d400a72009-07-07 11:03:31 -0700379 return;
Rob Pike418b97c2008-10-24 16:33:29 -0700380}
381
Rob Pike50d06952008-12-09 15:41:21 -0800382func getPtr(v reflect.Value) (val uintptr, ok bool) {
Russ Cox0d400a72009-07-07 11:03:31 -0700383 switch v := v.(type) {
384 case *reflect.PtrValue:
385 return uintptr(v.Get()), true;
Rob Pike418b97c2008-10-24 16:33:29 -0700386 }
Russ Cox0d400a72009-07-07 11:03:31 -0700387 return;
Rob Pikeb0d62672008-12-22 11:04:17 -0800388}
389
Rob Pikef15dfa72008-11-06 10:40:57 -0800390// Convert ASCII to integer. n is 0 (and got is false) if no number present.
Rob Pike418b97c2008-10-24 16:33:29 -0700391
392func parsenum(s string, start, end int) (n int, got bool, newi int) {
393 if start >= end {
Russ Coxecb863a2009-10-06 15:38:57 -0700394 return 0, false, end;
Rob Pike418b97c2008-10-24 16:33:29 -0700395 }
Rob Pike418b97c2008-10-24 16:33:29 -0700396 isnum := false;
397 num := 0;
398 for '0' <= s[start] && s[start] <= '9' {
Russ Coxecb863a2009-10-06 15:38:57 -0700399 num = num*10 + int(s[start]-'0');
Rob Pike418b97c2008-10-24 16:33:29 -0700400 start++;
401 isnum = true;
402 }
403 return num, isnum, start;
404}
405
Russ Coxa843b452009-08-31 16:38:30 -0700406type uintptrGetter interface {
407 Get() uintptr;
408}
409
410func (p *pp) printField(field reflect.Value, plus, sharp bool, depth int) (was_string bool) {
Rob Pikeac09eb42008-12-11 12:59:49 -0800411 inter := field.Interface();
412 if inter != nil {
Russ Coxa843b452009-08-31 16:38:30 -0700413 switch {
414 default:
415 if stringer, ok := inter.(Stringer); ok {
416 p.addstr(stringer.String());
417 return false; // this value is not a string
418 }
419 case sharp:
420 if stringer, ok := inter.(GoStringer); ok {
421 p.addstr(stringer.GoString());
422 return false; // this value is not a string
423 }
Rob Pikeac09eb42008-12-11 12:59:49 -0800424 }
Russ Coxecb863a2009-10-06 15:38:57 -0700425 }
426 s := "";
Russ Coxa843b452009-08-31 16:38:30 -0700427BigSwitch:
Russ Cox0d400a72009-07-07 11:03:31 -0700428 switch f := field.(type) {
429 case *reflect.BoolValue:
430 s = p.fmt.Fmt_boolean(f.Get()).Str();
431 case *reflect.Float32Value:
432 s = p.fmt.Fmt_g32(f.Get()).Str();
433 case *reflect.Float64Value:
434 s = p.fmt.Fmt_g64(f.Get()).Str();
435 case *reflect.FloatValue:
Russ Coxecb863a2009-10-06 15:38:57 -0700436 if field.Type().Size() * 8 == 32 {
Russ Cox0d400a72009-07-07 11:03:31 -0700437 s = p.fmt.Fmt_g32(float32(f.Get())).Str();
Russ Cox079c00a2008-11-17 12:34:03 -0800438 } else {
Russ Cox0d400a72009-07-07 11:03:31 -0700439 s = p.fmt.Fmt_g64(float64(f.Get())).Str();
Russ Cox079c00a2008-11-17 12:34:03 -0800440 }
Russ Cox0d400a72009-07-07 11:03:31 -0700441 case *reflect.StringValue:
Russ Coxa843b452009-08-31 16:38:30 -0700442 if sharp {
443 s = p.fmt.Fmt_q(f.Get()).Str();
444 } else {
445 s = p.fmt.Fmt_s(f.Get()).Str();
446 was_string = true;
Rob Pikee2621b82008-11-13 15:20:52 -0800447 }
Rob Pike3928a4e2009-07-09 17:30:07 -0700448 case *reflect.MapValue:
Russ Coxa843b452009-08-31 16:38:30 -0700449 if sharp {
450 p.addstr(field.Type().String());
451 p.addstr("{");
452 } else {
453 p.addstr("map[");
454 }
Rob Pike3928a4e2009-07-09 17:30:07 -0700455 keys := f.Keys();
456 for i, key := range keys {
457 if i > 0 {
Russ Coxa843b452009-08-31 16:38:30 -0700458 if sharp {
459 p.addstr(", ");
460 } else {
461 p.addstr(" ");
462 }
Rob Pike3928a4e2009-07-09 17:30:07 -0700463 }
Russ Coxa843b452009-08-31 16:38:30 -0700464 p.printField(key, plus, sharp, depth+1);
Rob Pike3928a4e2009-07-09 17:30:07 -0700465 p.addstr(":");
Russ Coxa843b452009-08-31 16:38:30 -0700466 p.printField(f.Elem(key), plus, sharp, depth+1);
Rob Pike3928a4e2009-07-09 17:30:07 -0700467 }
Russ Coxa843b452009-08-31 16:38:30 -0700468 if sharp {
469 p.addstr("}");
470 } else {
471 p.addstr("]");
472 }
Russ Cox0d400a72009-07-07 11:03:31 -0700473 case *reflect.StructValue:
Russ Coxa843b452009-08-31 16:38:30 -0700474 if sharp {
475 p.addstr(field.Type().String());
476 }
Rob Pikee2621b82008-11-13 15:20:52 -0800477 p.add('{');
Russ Cox0d400a72009-07-07 11:03:31 -0700478 v := f;
479 t := v.Type().(*reflect.StructType);
Russ Cox67a7aba2008-12-16 14:39:29 -0800480 p.fmt.clearflags(); // clear flags for p.printField
Russ Coxecb863a2009-10-06 15:38:57 -0700481 for i := 0; i < v.NumField(); i++ {
Russ Coxbf67afc2008-12-11 16:53:33 -0800482 if i > 0 {
Russ Coxa843b452009-08-31 16:38:30 -0700483 if sharp {
484 p.addstr(", ");
485 } else {
486 p.addstr(" ");
487 }
Russ Coxbf67afc2008-12-11 16:53:33 -0800488 }
Russ Coxa843b452009-08-31 16:38:30 -0700489 if plus || sharp {
Russ Cox0d400a72009-07-07 11:03:31 -0700490 if f := t.Field(i); f.Name != "" {
491 p.addstr(f.Name);
Rob Pike3928a4e2009-07-09 17:30:07 -0700492 p.add(':');
Russ Coxbf67afc2008-12-11 16:53:33 -0800493 }
494 }
Russ Coxa843b452009-08-31 16:38:30 -0700495 p.printField(getField(v, i), plus, sharp, depth+1);
Russ Coxbf67afc2008-12-11 16:53:33 -0800496 }
Russ Coxa843b452009-08-31 16:38:30 -0700497 p.addstr("}");
Russ Cox0d400a72009-07-07 11:03:31 -0700498 case *reflect.InterfaceValue:
499 value := f.Elem();
Russ Coxac6ebfd2009-04-06 21:28:04 -0700500 if value == nil {
Russ Coxa843b452009-08-31 16:38:30 -0700501 if sharp {
502 p.addstr(field.Type().String());
503 p.addstr("(nil)");
504 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700505 s = "<nil>";
Russ Coxa843b452009-08-31 16:38:30 -0700506 }
Rob Pikeac09eb42008-12-11 12:59:49 -0800507 } else {
Russ Coxa843b452009-08-31 16:38:30 -0700508 return p.printField(value, plus, sharp, depth+1);
Rob Pikeac09eb42008-12-11 12:59:49 -0800509 }
Russ Coxa843b452009-08-31 16:38:30 -0700510 case reflect.ArrayOrSliceValue:
511 if sharp {
512 p.addstr(field.Type().String());
513 p.addstr("{");
514 } else {
515 p.addstr("[");
516 }
517 for i := 0; i < f.Len(); i++ {
518 if i > 0 {
519 if sharp {
520 p.addstr(", ");
521 } else {
522 p.addstr(" ");
523 }
524 }
525 p.printField(f.Elem(i), plus, sharp, depth+1);
526 }
527 if sharp {
528 p.addstr("}");
529 } else {
530 p.addstr("]");
531 }
532 case *reflect.PtrValue:
533 v := f.Get();
534 // pointer to array or slice or struct? ok at top level
535 // but not embedded (avoid loops)
536 if v != 0 && depth == 0 {
537 switch a := f.Elem().(type) {
538 case reflect.ArrayOrSliceValue:
539 p.addstr("&");
540 p.printField(a, plus, sharp, depth+1);
541 break BigSwitch;
542 case *reflect.StructValue:
543 p.addstr("&");
544 p.printField(a, plus, sharp, depth+1);
545 break BigSwitch;
546 }
547 }
548 if sharp {
549 p.addstr("(");
550 p.addstr(field.Type().String());
551 p.addstr(")(");
552 if v == 0 {
553 p.addstr("nil");
554 } else {
555 p.fmt.sharp = true;
556 p.addstr(p.fmt.Fmt_ux64(uint64(v)).Str());
557 }
558 p.addstr(")");
559 break;
560 }
561 if v == 0 {
562 s = "<nil>";
563 break;
564 }
Russ Coxecb863a2009-10-06 15:38:57 -0700565 p.fmt.sharp = true; // turn 0x on
Russ Coxa843b452009-08-31 16:38:30 -0700566 s = p.fmt.Fmt_ux64(uint64(v)).Str();
567 case uintptrGetter:
568 v := f.Get();
569 if sharp {
570 p.addstr("(");
571 p.addstr(field.Type().String());
572 p.addstr(")(");
573 if v == 0 {
574 p.addstr("nil");
575 } else {
576 p.fmt.sharp = true;
577 p.addstr(p.fmt.Fmt_ux64(uint64(v)).Str());
578 }
579 p.addstr(")");
580 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700581 p.fmt.sharp = true; // turn 0x on
Russ Coxa843b452009-08-31 16:38:30 -0700582 p.addstr(p.fmt.Fmt_ux64(uint64(f.Get())).Str());
583 }
Rob Pikee2621b82008-11-13 15:20:52 -0800584 default:
Russ Cox0d400a72009-07-07 11:03:31 -0700585 v, signed, ok := getInt(field);
586 if ok {
587 if signed {
588 s = p.fmt.Fmt_d64(v).Str();
589 } else {
Russ Coxa843b452009-08-31 16:38:30 -0700590 if sharp {
591 p.fmt.sharp = true; // turn on 0x
592 s = p.fmt.Fmt_ux64(uint64(v)).Str();
593 } else {
594 s = p.fmt.Fmt_ud64(uint64(v)).Str();
595 }
Russ Cox0d400a72009-07-07 11:03:31 -0700596 }
597 break;
598 }
Rob Pikee2621b82008-11-13 15:20:52 -0800599 s = "?" + field.Type().String() + "?";
600 }
601 p.addstr(s);
602 return was_string;
603}
604
Russ Cox0d400a72009-07-07 11:03:31 -0700605func (p *pp) doprintf(format string, v *reflect.StructValue) {
Rob Pike418b97c2008-10-24 16:33:29 -0700606 p.ensure(len(format)); // a good starting size
Russ Coxecb863a2009-10-06 15:38:57 -0700607 end := len(format)-1;
Rob Pike418b97c2008-10-24 16:33:29 -0700608 fieldnum := 0; // we process one field per non-trivial format
Russ Coxecb863a2009-10-06 15:38:57 -0700609 for i := 0; i <= end; {
Russ Cox3619f1e2009-05-11 14:10:34 -0700610 c, w := utf8.DecodeRuneInString(format[i:len(format)]);
Rob Pike418b97c2008-10-24 16:33:29 -0700611 if c != '%' || i == end {
612 p.add(c);
613 i += w;
614 continue;
615 }
Russ Cox387df5e2008-11-24 14:51:33 -0800616 i++;
Russ Coxbf67afc2008-12-11 16:53:33 -0800617 // flags and widths
618 p.fmt.clearflags();
Russ Coxecb863a2009-10-06 15:38:57 -0700619 F: for ; i < end; i++ {
Russ Cox387df5e2008-11-24 14:51:33 -0800620 switch format[i] {
621 case '#':
622 p.fmt.sharp = true;
623 case '0':
624 p.fmt.zero = true;
625 case '+':
626 p.fmt.plus = true;
627 case '-':
628 p.fmt.minus = true;
629 case ' ':
630 p.fmt.space = true;
631 default:
632 break F;
633 }
634 }
635 // do we have 20 (width)?
Russ Coxbf67afc2008-12-11 16:53:33 -0800636 p.fmt.wid, p.fmt.wid_present, i = parsenum(format, i, end);
Russ Cox387df5e2008-11-24 14:51:33 -0800637 // do we have .20 (precision)?
Rob Pike418b97c2008-10-24 16:33:29 -0700638 if i < end && format[i] == '.' {
Russ Coxbf67afc2008-12-11 16:53:33 -0800639 p.fmt.prec, p.fmt.prec_present, i = parsenum(format, i+1, end);
Rob Pike418b97c2008-10-24 16:33:29 -0700640 }
Russ Cox3619f1e2009-05-11 14:10:34 -0700641 c, w = utf8.DecodeRuneInString(format[i:len(format)]);
Rob Pike418b97c2008-10-24 16:33:29 -0700642 i += w;
643 // percent is special - absorbs no operand
644 if c == '%' {
645 p.add('%'); // TODO: should we bother with width & prec?
646 continue;
647 }
Russ Cox0d400a72009-07-07 11:03:31 -0700648 if fieldnum >= v.NumField() { // out of operands
Rob Pikef15dfa72008-11-06 10:40:57 -0800649 p.add('%');
650 p.add(c);
651 p.addstr("(missing)");
Rob Pike418b97c2008-10-24 16:33:29 -0700652 continue;
653 }
Russ Cox387df5e2008-11-24 14:51:33 -0800654 field := getField(v, fieldnum);
Rob Pike418b97c2008-10-24 16:33:29 -0700655 fieldnum++;
Russ Coxa843b452009-08-31 16:38:30 -0700656
657 // Try formatter except for %T,
658 // which is special and handled internally.
Rob Pikeac09eb42008-12-11 12:59:49 -0800659 inter := field.Interface();
Russ Coxa843b452009-08-31 16:38:30 -0700660 if inter != nil && c != 'T' {
Rob Pike28ba9772009-06-23 15:20:30 -0700661 if formatter, ok := inter.(Formatter); ok {
Rob Pikee2621b82008-11-13 15:20:52 -0800662 formatter.Format(p, c);
663 continue;
664 }
Rob Pikef15dfa72008-11-06 10:40:57 -0800665 }
Russ Coxa843b452009-08-31 16:38:30 -0700666
Rob Pike418b97c2008-10-24 16:33:29 -0700667 s := "";
668 switch c {
Russ Cox8f2bf202009-09-10 14:18:53 -0700669 // bool
670 case 't':
671 if v, ok := getBool(field); ok {
672 if v {
673 s = "true";
Rob Pike59f029c2008-11-01 16:37:53 -0700674 } else {
Russ Cox8f2bf202009-09-10 14:18:53 -0700675 s = "false";
Rob Pike59f029c2008-11-01 16:37:53 -0700676 }
Russ Cox8f2bf202009-09-10 14:18:53 -0700677 } else {
678 goto badtype;
679 }
Rob Pike59f029c2008-11-01 16:37:53 -0700680
Russ Cox8f2bf202009-09-10 14:18:53 -0700681 // int
682 case 'b':
Russ Coxca6a0fe2009-09-15 09:41:59 -0700683 if v, _, ok := getInt(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700684 s = p.fmt.Fmt_b64(uint64(v)).Str(); // always unsigned
Russ Cox8f2bf202009-09-10 14:18:53 -0700685 } else if v, ok := getFloat32(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700686 s = p.fmt.Fmt_fb32(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700687 } else if v, ok := getFloat64(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700688 s = p.fmt.Fmt_fb64(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700689 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700690 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700691 }
692 case 'c':
Russ Coxca6a0fe2009-09-15 09:41:59 -0700693 if v, _, ok := getInt(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700694 s = p.fmt.Fmt_c(int(v)).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700695 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700696 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700697 }
698 case 'd':
699 if v, signed, ok := getInt(field); ok {
700 if signed {
Russ Coxecb863a2009-10-06 15:38:57 -0700701 s = p.fmt.Fmt_d64(v).Str();
Rob Pike418b97c2008-10-24 16:33:29 -0700702 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700703 s = p.fmt.Fmt_ud64(uint64(v)).Str();
Rob Pikef15dfa72008-11-06 10:40:57 -0800704 }
Russ Cox8f2bf202009-09-10 14:18:53 -0700705 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700706 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700707 }
708 case 'o':
709 if v, signed, ok := getInt(field); ok {
710 if signed {
Russ Coxecb863a2009-10-06 15:38:57 -0700711 s = p.fmt.Fmt_o64(v).Str();
Rob Pikef15dfa72008-11-06 10:40:57 -0800712 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700713 s = p.fmt.Fmt_uo64(uint64(v)).Str();
Rob Pike418b97c2008-10-24 16:33:29 -0700714 }
Russ Cox8f2bf202009-09-10 14:18:53 -0700715 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700716 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700717 }
718 case 'x':
719 if v, signed, ok := getInt(field); ok {
720 if signed {
Russ Coxecb863a2009-10-06 15:38:57 -0700721 s = p.fmt.Fmt_x64(v).Str();
Rob Pike418b97c2008-10-24 16:33:29 -0700722 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700723 s = p.fmt.Fmt_ux64(uint64(v)).Str();
Rob Pike418b97c2008-10-24 16:33:29 -0700724 }
Russ Cox8f2bf202009-09-10 14:18:53 -0700725 } else if v, ok := getString(field); ok {
726 s = p.fmt.Fmt_sx(v).Str();
727 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700728 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700729 }
730 case 'X':
731 if v, signed, ok := getInt(field); ok {
732 if signed {
Russ Coxecb863a2009-10-06 15:38:57 -0700733 s = p.fmt.Fmt_X64(v).Str();
Rob Pike418b97c2008-10-24 16:33:29 -0700734 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700735 s = p.fmt.Fmt_uX64(uint64(v)).Str();
Rob Pike418b97c2008-10-24 16:33:29 -0700736 }
Russ Cox8f2bf202009-09-10 14:18:53 -0700737 } else if v, ok := getString(field); ok {
738 s = p.fmt.Fmt_sX(v).Str();
739 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700740 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700741 }
Rob Pike418b97c2008-10-24 16:33:29 -0700742
Russ Cox8f2bf202009-09-10 14:18:53 -0700743 // float
744 case 'e':
745 if v, ok := getFloat32(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700746 s = p.fmt.Fmt_e32(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700747 } else if v, ok := getFloat64(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700748 s = p.fmt.Fmt_e64(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700749 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700750 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700751 }
752 case 'E':
753 if v, ok := getFloat32(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700754 s = p.fmt.Fmt_E32(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700755 } else if v, ok := getFloat64(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700756 s = p.fmt.Fmt_E64(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700757 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700758 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700759 }
760 case 'f':
761 if v, ok := getFloat32(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700762 s = p.fmt.Fmt_f32(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700763 } else if v, ok := getFloat64(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700764 s = p.fmt.Fmt_f64(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700765 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700766 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700767 }
768 case 'g':
769 if v, ok := getFloat32(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700770 s = p.fmt.Fmt_g32(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700771 } else if v, ok := getFloat64(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700772 s = p.fmt.Fmt_g64(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700773 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700774 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700775 }
776 case 'G':
777 if v, ok := getFloat32(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700778 s = p.fmt.Fmt_G32(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700779 } else if v, ok := getFloat64(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700780 s = p.fmt.Fmt_G64(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700781 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700782 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700783 }
Rob Pike418b97c2008-10-24 16:33:29 -0700784
Russ Cox8f2bf202009-09-10 14:18:53 -0700785 // string
786 case 's':
787 if inter != nil {
788 // if object implements String, use the result.
789 if stringer, ok := inter.(Stringer); ok {
790 s = p.fmt.Fmt_s(stringer.String()).Str();
791 break;
Rob Pike85647c92009-03-06 03:35:38 -0800792 }
Russ Cox8f2bf202009-09-10 14:18:53 -0700793 }
794 if v, ok := getString(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700795 s = p.fmt.Fmt_s(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700796 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700797 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700798 }
799 case 'q':
800 if v, ok := getString(field); ok {
Russ Coxecb863a2009-10-06 15:38:57 -0700801 s = p.fmt.Fmt_q(v).Str();
Russ Cox8f2bf202009-09-10 14:18:53 -0700802 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700803 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700804 }
805
806 // pointer
807 case 'p':
808 if v, ok := getPtr(field); ok {
809 if v == 0 {
Russ Coxecb863a2009-10-06 15:38:57 -0700810 s = "<nil>";
Rob Pike418b97c2008-10-24 16:33:29 -0700811 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700812 s = "0x" + p.fmt.Fmt_uX64(uint64(v)).Str();
Rob Pike418b97c2008-10-24 16:33:29 -0700813 }
Russ Cox8f2bf202009-09-10 14:18:53 -0700814 } else {
Russ Coxecb863a2009-10-06 15:38:57 -0700815 goto badtype;
Russ Cox8f2bf202009-09-10 14:18:53 -0700816 }
Rob Pike418b97c2008-10-24 16:33:29 -0700817
Russ Cox8f2bf202009-09-10 14:18:53 -0700818 // arbitrary value; do your best
819 case 'v':
820 plus, sharp := p.fmt.plus, p.fmt.sharp;
821 p.fmt.plus = false;
822 p.fmt.sharp = false;
823 p.printField(field, plus, sharp, 0);
Rob Pike418b97c2008-10-24 16:33:29 -0700824
Russ Cox8f2bf202009-09-10 14:18:53 -0700825 // the value's type
826 case 'T':
827 s = field.Type().String();
Rob Pikee2621b82008-11-13 15:20:52 -0800828
Russ Cox8f2bf202009-09-10 14:18:53 -0700829 default:
830 badtype:
831 s = "%" + string(c) + "(" + field.Type().String() + "=";
832 p.addstr(s);
833 p.printField(field, false, false, 0);
834 s = ")";
Rob Pike418b97c2008-10-24 16:33:29 -0700835 }
836 p.addstr(s);
837 }
Russ Cox0d400a72009-07-07 11:03:31 -0700838 if fieldnum < v.NumField() {
Rob Pikef15dfa72008-11-06 10:40:57 -0800839 p.addstr("?(extra ");
Russ Cox0d400a72009-07-07 11:03:31 -0700840 for ; fieldnum < v.NumField(); fieldnum++ {
Rob Pikec6540d32009-08-28 13:02:34 -0700841 field := getField(v, fieldnum);
842 p.addstr(field.Type().String());
843 p.addstr("=");
Russ Coxa843b452009-08-31 16:38:30 -0700844 p.printField(field, false, false, 0);
Russ Coxecb863a2009-10-06 15:38:57 -0700845 if fieldnum+1 < v.NumField() {
Rob Pikef15dfa72008-11-06 10:40:57 -0800846 p.addstr(", ");
847 }
848 }
849 p.addstr(")");
850 }
Rob Pike418b97c2008-10-24 16:33:29 -0700851}
852
Russ Cox0d400a72009-07-07 11:03:31 -0700853func (p *pp) doprint(v *reflect.StructValue, addspace, addnewline bool) {
Rob Pike418b97c2008-10-24 16:33:29 -0700854 prev_string := false;
Russ Coxecb863a2009-10-06 15:38:57 -0700855 for fieldnum := 0; fieldnum < v.NumField(); fieldnum++ {
Rob Pike418b97c2008-10-24 16:33:29 -0700856 // always add spaces if we're doing println
Russ Cox387df5e2008-11-24 14:51:33 -0800857 field := getField(v, fieldnum);
Rob Pike2d4f7ba2008-11-02 12:33:02 -0800858 if fieldnum > 0 {
Russ Cox0d400a72009-07-07 11:03:31 -0700859 _, is_string := field.(*reflect.StringValue);
860 if addspace || !is_string && !prev_string {
861 p.add(' ');
Rob Pike418b97c2008-10-24 16:33:29 -0700862 }
Rob Pike418b97c2008-10-24 16:33:29 -0700863 }
Russ Coxa843b452009-08-31 16:38:30 -0700864 prev_string = p.printField(field, false, false, 0);
Rob Pike418b97c2008-10-24 16:33:29 -0700865 }
Rob Pike2d4f7ba2008-11-02 12:33:02 -0800866 if addnewline {
Russ Coxecb863a2009-10-06 15:38:57 -0700867 p.add('\n');
Rob Pike418b97c2008-10-24 16:33:29 -0700868 }
869}