github.com/euank/go@v0.0.0-20160829210321-495514729181/src/internal/trace/parser.go (about) 1 // Copyright 2014 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 5 package trace 6 7 import ( 8 "bufio" 9 "bytes" 10 "fmt" 11 "io" 12 "math/rand" 13 "os" 14 "os/exec" 15 "strconv" 16 "strings" 17 _ "unsafe" 18 ) 19 20 // Event describes one event in the trace. 21 type Event struct { 22 Off int // offset in input file (for debugging and error reporting) 23 Type byte // one of Ev* 24 seq int64 // sequence number 25 Ts int64 // timestamp in nanoseconds 26 P int // P on which the event happened (can be one of TimerP, NetpollP, SyscallP) 27 G uint64 // G on which the event happened 28 StkID uint64 // unique stack ID 29 Stk []*Frame // stack trace (can be empty) 30 Args [3]uint64 // event-type-specific arguments 31 // linked event (can be nil), depends on event type: 32 // for GCStart: the GCStop 33 // for GCScanStart: the GCScanDone 34 // for GCSweepStart: the GCSweepDone 35 // for GoCreate: first GoStart of the created goroutine 36 // for GoStart: the associated GoEnd, GoBlock or other blocking event 37 // for GoSched/GoPreempt: the next GoStart 38 // for GoBlock and other blocking events: the unblock event 39 // for GoUnblock: the associated GoStart 40 // for blocking GoSysCall: the associated GoSysExit 41 // for GoSysExit: the next GoStart 42 Link *Event 43 } 44 45 // Frame is a frame in stack traces. 46 type Frame struct { 47 PC uint64 48 Fn string 49 File string 50 Line int 51 } 52 53 const ( 54 // Special P identifiers: 55 FakeP = 1000000 + iota 56 TimerP // depicts timer unblocks 57 NetpollP // depicts network unblocks 58 SyscallP // depicts returns from syscalls 59 ) 60 61 // Parse parses, post-processes and verifies the trace. 62 func Parse(r io.Reader, bin string) ([]*Event, error) { 63 ver, events, err := parse(r, bin) 64 if err != nil { 65 return nil, err 66 } 67 if ver < 1007 && bin == "" { 68 return nil, fmt.Errorf("for traces produced by go 1.6 or below, the binary argument must be provided") 69 } 70 return events, nil 71 } 72 73 // parse parses, post-processes and verifies the trace. It returns the 74 // trace version and the list of events. 75 func parse(r io.Reader, bin string) (int, []*Event, error) { 76 ver, rawEvents, strings, err := readTrace(r) 77 if err != nil { 78 return 0, nil, err 79 } 80 events, stacks, err := parseEvents(ver, rawEvents, strings) 81 if err != nil { 82 return 0, nil, err 83 } 84 events, err = removeFutile(events) 85 if err != nil { 86 return 0, nil, err 87 } 88 err = postProcessTrace(ver, events) 89 if err != nil { 90 return 0, nil, err 91 } 92 // Attach stack traces. 93 for _, ev := range events { 94 if ev.StkID != 0 { 95 ev.Stk = stacks[ev.StkID] 96 } 97 } 98 if ver < 1007 && bin != "" { 99 if err := symbolize(events, bin); err != nil { 100 return 0, nil, err 101 } 102 } 103 return ver, events, nil 104 } 105 106 // rawEvent is a helper type used during parsing. 107 type rawEvent struct { 108 off int 109 typ byte 110 args []uint64 111 } 112 113 // readTrace does wire-format parsing and verification. 114 // It does not care about specific event types and argument meaning. 115 func readTrace(r io.Reader) (ver int, events []rawEvent, strings map[uint64]string, err error) { 116 // Read and validate trace header. 117 var buf [16]byte 118 off, err := io.ReadFull(r, buf[:]) 119 if err != nil { 120 err = fmt.Errorf("failed to read header: read %v, err %v", off, err) 121 return 122 } 123 ver, err = parseHeader(buf[:]) 124 if err != nil { 125 return 126 } 127 switch ver { 128 case 1005, 1007: 129 break 130 default: 131 err = fmt.Errorf("unsupported trace file version %v.%v (update Go toolchain) %v", ver/1000, ver%1000, ver) 132 return 133 } 134 135 // Read events. 136 strings = make(map[uint64]string) 137 for { 138 // Read event type and number of arguments (1 byte). 139 off0 := off 140 var n int 141 n, err = r.Read(buf[:1]) 142 if err == io.EOF { 143 err = nil 144 break 145 } 146 if err != nil || n != 1 { 147 err = fmt.Errorf("failed to read trace at offset 0x%x: n=%v err=%v", off0, n, err) 148 return 149 } 150 off += n 151 typ := buf[0] << 2 >> 2 152 narg := buf[0]>>6 + 1 153 inlineArgs := byte(4) 154 if ver < 1007 { 155 narg++ 156 inlineArgs++ 157 } 158 if typ == EvNone || typ >= EvCount || EventDescriptions[typ].minVersion > ver { 159 err = fmt.Errorf("unknown event type %v at offset 0x%x", typ, off0) 160 return 161 } 162 if typ == EvString { 163 // String dictionary entry [ID, length, string]. 164 var id uint64 165 id, off, err = readVal(r, off) 166 if err != nil { 167 return 168 } 169 if id == 0 { 170 err = fmt.Errorf("string at offset %d has invalid id 0", off) 171 return 172 } 173 if strings[id] != "" { 174 err = fmt.Errorf("string at offset %d has duplicate id %v", off, id) 175 return 176 } 177 var ln uint64 178 ln, off, err = readVal(r, off) 179 if err != nil { 180 return 181 } 182 if ln == 0 { 183 err = fmt.Errorf("string at offset %d has invalid length 0", off) 184 return 185 } 186 if ln > 1e6 { 187 err = fmt.Errorf("string at offset %d has too large length %v", off, ln) 188 return 189 } 190 buf := make([]byte, ln) 191 var n int 192 n, err = io.ReadFull(r, buf) 193 if err != nil { 194 err = fmt.Errorf("failed to read trace at offset %d: read %v, want %v, error %v", off, n, ln, err) 195 return 196 } 197 off += n 198 strings[id] = string(buf) 199 continue 200 } 201 ev := rawEvent{typ: typ, off: off0} 202 if narg < inlineArgs { 203 for i := 0; i < int(narg); i++ { 204 var v uint64 205 v, off, err = readVal(r, off) 206 if err != nil { 207 err = fmt.Errorf("failed to read event %v argument at offset %v (%v)", typ, off, err) 208 return 209 } 210 ev.args = append(ev.args, v) 211 } 212 } else { 213 // More than inlineArgs args, the first value is length of the event in bytes. 214 var v uint64 215 v, off, err = readVal(r, off) 216 if err != nil { 217 err = fmt.Errorf("failed to read event %v argument at offset %v (%v)", typ, off, err) 218 return 219 } 220 evLen := v 221 off1 := off 222 for evLen > uint64(off-off1) { 223 v, off, err = readVal(r, off) 224 if err != nil { 225 err = fmt.Errorf("failed to read event %v argument at offset %v (%v)", typ, off, err) 226 return 227 } 228 ev.args = append(ev.args, v) 229 } 230 if evLen != uint64(off-off1) { 231 err = fmt.Errorf("event has wrong length at offset 0x%x: want %v, got %v", off0, evLen, off-off1) 232 return 233 } 234 } 235 events = append(events, ev) 236 } 237 return 238 } 239 240 // parseHeader parses trace header of the form "go 1.7 trace\x00\x00\x00\x00" 241 // and returns parsed version as 1007. 242 func parseHeader(buf []byte) (int, error) { 243 if len(buf) != 16 { 244 return 0, fmt.Errorf("bad header length") 245 } 246 if buf[0] != 'g' || buf[1] != 'o' || buf[2] != ' ' || 247 buf[3] < '1' || buf[3] > '9' || 248 buf[4] != '.' || 249 buf[5] < '1' || buf[5] > '9' { 250 return 0, fmt.Errorf("not a trace file") 251 } 252 ver := int(buf[5] - '0') 253 i := 0 254 for ; buf[6+i] >= '0' && buf[6+i] <= '9' && i < 2; i++ { 255 ver = ver*10 + int(buf[6+i]-'0') 256 } 257 ver += int(buf[3]-'0') * 1000 258 if !bytes.Equal(buf[6+i:], []byte(" trace\x00\x00\x00\x00")[:10-i]) { 259 return 0, fmt.Errorf("not a trace file") 260 } 261 return ver, nil 262 } 263 264 // Parse events transforms raw events into events. 265 // It does analyze and verify per-event-type arguments. 266 func parseEvents(ver int, rawEvents []rawEvent, strings map[uint64]string) (events []*Event, stacks map[uint64][]*Frame, err error) { 267 var ticksPerSec, lastSeq, lastTs int64 268 var lastG, timerGoid uint64 269 var lastP int 270 lastGs := make(map[int]uint64) // last goroutine running on P 271 stacks = make(map[uint64][]*Frame) 272 batches := make(map[int][]*Event) // events by P 273 for _, raw := range rawEvents { 274 desc := EventDescriptions[raw.typ] 275 if desc.Name == "" { 276 err = fmt.Errorf("missing description for event type %v", raw.typ) 277 return 278 } 279 narg := argNum(raw, ver) 280 if len(raw.args) != narg { 281 err = fmt.Errorf("%v has wrong number of arguments at offset 0x%x: want %v, got %v", 282 desc.Name, raw.off, narg, len(raw.args)) 283 return 284 } 285 switch raw.typ { 286 case EvBatch: 287 lastGs[lastP] = lastG 288 lastP = int(raw.args[0]) 289 lastG = lastGs[lastP] 290 if ver < 1007 { 291 lastSeq = int64(raw.args[1]) 292 lastTs = int64(raw.args[2]) 293 } else { 294 lastTs = int64(raw.args[1]) 295 } 296 case EvFrequency: 297 ticksPerSec = int64(raw.args[0]) 298 if ticksPerSec <= 0 { 299 // The most likely cause for this is tick skew on different CPUs. 300 // For example, solaris/amd64 seems to have wildly different 301 // ticks on different CPUs. 302 err = ErrTimeOrder 303 return 304 } 305 case EvTimerGoroutine: 306 timerGoid = raw.args[0] 307 case EvStack: 308 if len(raw.args) < 2 { 309 err = fmt.Errorf("EvStack has wrong number of arguments at offset 0x%x: want at least 2, got %v", 310 raw.off, len(raw.args)) 311 return 312 } 313 size := raw.args[1] 314 if size > 1000 { 315 err = fmt.Errorf("EvStack has bad number of frames at offset 0x%x: %v", 316 raw.off, size) 317 return 318 } 319 want := 2 + 4*size 320 if ver < 1007 { 321 want = 2 + size 322 } 323 if uint64(len(raw.args)) != want { 324 err = fmt.Errorf("EvStack has wrong number of arguments at offset 0x%x: want %v, got %v", 325 raw.off, want, len(raw.args)) 326 return 327 } 328 id := raw.args[0] 329 if id != 0 && size > 0 { 330 stk := make([]*Frame, size) 331 for i := 0; i < int(size); i++ { 332 if ver < 1007 { 333 stk[i] = &Frame{PC: raw.args[2+i]} 334 } else { 335 pc := raw.args[2+i*4+0] 336 fn := raw.args[2+i*4+1] 337 file := raw.args[2+i*4+2] 338 line := raw.args[2+i*4+3] 339 stk[i] = &Frame{PC: pc, Fn: strings[fn], File: strings[file], Line: int(line)} 340 } 341 } 342 stacks[id] = stk 343 } 344 default: 345 e := &Event{Off: raw.off, Type: raw.typ, P: lastP, G: lastG} 346 var argOffset int 347 if ver < 1007 { 348 e.seq = lastSeq + int64(raw.args[0]) 349 e.Ts = lastTs + int64(raw.args[1]) 350 lastSeq = e.seq 351 argOffset = 2 352 } else { 353 e.Ts = lastTs + int64(raw.args[0]) 354 argOffset = 1 355 } 356 lastTs = e.Ts 357 for i := argOffset; i < narg; i++ { 358 if i == narg-1 && desc.Stack { 359 e.StkID = raw.args[i] 360 } else { 361 e.Args[i-argOffset] = raw.args[i] 362 } 363 } 364 switch raw.typ { 365 case EvGoStart, EvGoStartLocal: 366 lastG = e.Args[0] 367 e.G = lastG 368 case EvGCStart, EvGCDone, EvGCScanStart, EvGCScanDone: 369 e.G = 0 370 case EvGoEnd, EvGoStop, EvGoSched, EvGoPreempt, 371 EvGoSleep, EvGoBlock, EvGoBlockSend, EvGoBlockRecv, 372 EvGoBlockSelect, EvGoBlockSync, EvGoBlockCond, EvGoBlockNet, 373 EvGoSysBlock: 374 lastG = 0 375 case EvGoSysExit, EvGoWaiting, EvGoInSyscall: 376 e.G = e.Args[0] 377 } 378 batches[lastP] = append(batches[lastP], e) 379 } 380 } 381 if len(batches) == 0 { 382 err = fmt.Errorf("trace is empty") 383 return 384 } 385 if ticksPerSec == 0 { 386 err = fmt.Errorf("no EvFrequency event") 387 return 388 } 389 if BreakTimestampsForTesting { 390 var batchArr [][]*Event 391 for _, batch := range batches { 392 batchArr = append(batchArr, batch) 393 } 394 for i := 0; i < 5; i++ { 395 batch := batchArr[rand.Intn(len(batchArr))] 396 batch[rand.Intn(len(batch))].Ts += int64(rand.Intn(2000) - 1000) 397 } 398 } 399 if ver < 1007 { 400 events, err = order1005(batches) 401 } else { 402 events, err = order1007(batches) 403 } 404 if err != nil { 405 return 406 } 407 408 // Translate cpu ticks to real time. 409 minTs := events[0].Ts 410 // Use floating point to avoid integer overflows. 411 freq := 1e9 / float64(ticksPerSec) 412 for _, ev := range events { 413 ev.Ts = int64(float64(ev.Ts-minTs) * freq) 414 // Move timers and syscalls to separate fake Ps. 415 if timerGoid != 0 && ev.G == timerGoid && ev.Type == EvGoUnblock { 416 ev.P = TimerP 417 } 418 if ev.Type == EvGoSysExit { 419 ev.P = SyscallP 420 } 421 } 422 423 return 424 } 425 426 // removeFutile removes all constituents of futile wakeups (block, unblock, start). 427 // For example, a goroutine was unblocked on a mutex, but another goroutine got 428 // ahead and acquired the mutex before the first goroutine is scheduled, 429 // so the first goroutine has to block again. Such wakeups happen on buffered 430 // channels and sync.Mutex, but are generally not interesting for end user. 431 func removeFutile(events []*Event) ([]*Event, error) { 432 // Two non-trivial aspects: 433 // 1. A goroutine can be preempted during a futile wakeup and migrate to another P. 434 // We want to remove all of that. 435 // 2. Tracing can start in the middle of a futile wakeup. 436 // That is, we can see a futile wakeup event w/o the actual wakeup before it. 437 // postProcessTrace runs after us and ensures that we leave the trace in a consistent state. 438 439 // Phase 1: determine futile wakeup sequences. 440 type G struct { 441 futile bool 442 wakeup []*Event // wakeup sequence (subject for removal) 443 } 444 gs := make(map[uint64]G) 445 futile := make(map[*Event]bool) 446 for _, ev := range events { 447 switch ev.Type { 448 case EvGoUnblock: 449 g := gs[ev.Args[0]] 450 g.wakeup = []*Event{ev} 451 gs[ev.Args[0]] = g 452 case EvGoStart, EvGoPreempt, EvFutileWakeup: 453 g := gs[ev.G] 454 g.wakeup = append(g.wakeup, ev) 455 if ev.Type == EvFutileWakeup { 456 g.futile = true 457 } 458 gs[ev.G] = g 459 case EvGoBlock, EvGoBlockSend, EvGoBlockRecv, EvGoBlockSelect, EvGoBlockSync, EvGoBlockCond: 460 g := gs[ev.G] 461 if g.futile { 462 futile[ev] = true 463 for _, ev1 := range g.wakeup { 464 futile[ev1] = true 465 } 466 } 467 delete(gs, ev.G) 468 } 469 } 470 471 // Phase 2: remove futile wakeup sequences. 472 newEvents := events[:0] // overwrite the original slice 473 for _, ev := range events { 474 if !futile[ev] { 475 newEvents = append(newEvents, ev) 476 } 477 } 478 return newEvents, nil 479 } 480 481 // ErrTimeOrder is returned by Parse when the trace contains 482 // time stamps that do not respect actual event ordering. 483 var ErrTimeOrder = fmt.Errorf("time stamps out of order") 484 485 // postProcessTrace does inter-event verification and information restoration. 486 // The resulting trace is guaranteed to be consistent 487 // (for example, a P does not run two Gs at the same time, or a G is indeed 488 // blocked before an unblock event). 489 func postProcessTrace(ver int, events []*Event) error { 490 const ( 491 gDead = iota 492 gRunnable 493 gRunning 494 gWaiting 495 ) 496 type gdesc struct { 497 state int 498 ev *Event 499 evStart *Event 500 evCreate *Event 501 } 502 type pdesc struct { 503 running bool 504 g uint64 505 evScan *Event 506 evSweep *Event 507 } 508 509 gs := make(map[uint64]gdesc) 510 ps := make(map[int]pdesc) 511 gs[0] = gdesc{state: gRunning} 512 var evGC *Event 513 514 checkRunning := func(p pdesc, g gdesc, ev *Event, allowG0 bool) error { 515 name := EventDescriptions[ev.Type].Name 516 if g.state != gRunning { 517 return fmt.Errorf("g %v is not running while %v (offset %v, time %v)", ev.G, name, ev.Off, ev.Ts) 518 } 519 if p.g != ev.G { 520 return fmt.Errorf("p %v is not running g %v while %v (offset %v, time %v)", ev.P, ev.G, name, ev.Off, ev.Ts) 521 } 522 if !allowG0 && ev.G == 0 { 523 return fmt.Errorf("g 0 did %v (offset %v, time %v)", EventDescriptions[ev.Type].Name, ev.Off, ev.Ts) 524 } 525 return nil 526 } 527 528 for _, ev := range events { 529 g := gs[ev.G] 530 p := ps[ev.P] 531 532 switch ev.Type { 533 case EvProcStart: 534 if p.running { 535 return fmt.Errorf("p %v is running before start (offset %v, time %v)", ev.P, ev.Off, ev.Ts) 536 } 537 p.running = true 538 case EvProcStop: 539 if !p.running { 540 return fmt.Errorf("p %v is not running before stop (offset %v, time %v)", ev.P, ev.Off, ev.Ts) 541 } 542 if p.g != 0 { 543 return fmt.Errorf("p %v is running a goroutine %v during stop (offset %v, time %v)", ev.P, p.g, ev.Off, ev.Ts) 544 } 545 p.running = false 546 case EvGCStart: 547 if evGC != nil { 548 return fmt.Errorf("previous GC is not ended before a new one (offset %v, time %v)", ev.Off, ev.Ts) 549 } 550 evGC = ev 551 case EvGCDone: 552 if evGC == nil { 553 return fmt.Errorf("bogus GC end (offset %v, time %v)", ev.Off, ev.Ts) 554 } 555 evGC.Link = ev 556 evGC = nil 557 case EvGCScanStart: 558 if p.evScan != nil { 559 return fmt.Errorf("previous scanning is not ended before a new one (offset %v, time %v)", ev.Off, ev.Ts) 560 } 561 p.evScan = ev 562 case EvGCScanDone: 563 if p.evScan == nil { 564 return fmt.Errorf("bogus scanning end (offset %v, time %v)", ev.Off, ev.Ts) 565 } 566 p.evScan.Link = ev 567 p.evScan = nil 568 case EvGCSweepStart: 569 if p.evSweep != nil { 570 return fmt.Errorf("previous sweeping is not ended before a new one (offset %v, time %v)", ev.Off, ev.Ts) 571 } 572 p.evSweep = ev 573 case EvGCSweepDone: 574 if p.evSweep == nil { 575 return fmt.Errorf("bogus sweeping end (offset %v, time %v)", ev.Off, ev.Ts) 576 } 577 p.evSweep.Link = ev 578 p.evSweep = nil 579 case EvGoWaiting: 580 if g.state != gRunnable { 581 return fmt.Errorf("g %v is not runnable before EvGoWaiting (offset %v, time %v)", ev.G, ev.Off, ev.Ts) 582 } 583 g.state = gWaiting 584 g.ev = ev 585 case EvGoInSyscall: 586 if g.state != gRunnable { 587 return fmt.Errorf("g %v is not runnable before EvGoInSyscall (offset %v, time %v)", ev.G, ev.Off, ev.Ts) 588 } 589 g.state = gWaiting 590 g.ev = ev 591 case EvGoCreate: 592 if err := checkRunning(p, g, ev, true); err != nil { 593 return err 594 } 595 if _, ok := gs[ev.Args[0]]; ok { 596 return fmt.Errorf("g %v already exists (offset %v, time %v)", ev.Args[0], ev.Off, ev.Ts) 597 } 598 gs[ev.Args[0]] = gdesc{state: gRunnable, ev: ev, evCreate: ev} 599 case EvGoStart: 600 if g.state != gRunnable { 601 return fmt.Errorf("g %v is not runnable before start (offset %v, time %v)", ev.G, ev.Off, ev.Ts) 602 } 603 if p.g != 0 { 604 return fmt.Errorf("p %v is already running g %v while start g %v (offset %v, time %v)", ev.P, p.g, ev.G, ev.Off, ev.Ts) 605 } 606 g.state = gRunning 607 g.evStart = ev 608 p.g = ev.G 609 if g.evCreate != nil { 610 if ver < 1007 { 611 // +1 because symbolizer expects return pc. 612 ev.Stk = []*Frame{{PC: g.evCreate.Args[1] + 1}} 613 } else { 614 ev.StkID = g.evCreate.Args[1] 615 } 616 g.evCreate = nil 617 } 618 619 if g.ev != nil { 620 g.ev.Link = ev 621 g.ev = nil 622 } 623 case EvGoEnd, EvGoStop: 624 if err := checkRunning(p, g, ev, false); err != nil { 625 return err 626 } 627 g.evStart.Link = ev 628 g.evStart = nil 629 g.state = gDead 630 p.g = 0 631 case EvGoSched, EvGoPreempt: 632 if err := checkRunning(p, g, ev, false); err != nil { 633 return err 634 } 635 g.state = gRunnable 636 g.evStart.Link = ev 637 g.evStart = nil 638 p.g = 0 639 g.ev = ev 640 case EvGoUnblock: 641 if g.state != gRunning { 642 return fmt.Errorf("g %v is not running while unpark (offset %v, time %v)", ev.G, ev.Off, ev.Ts) 643 } 644 if ev.P != TimerP && p.g != ev.G { 645 return fmt.Errorf("p %v is not running g %v while unpark (offset %v, time %v)", ev.P, ev.G, ev.Off, ev.Ts) 646 } 647 g1 := gs[ev.Args[0]] 648 if g1.state != gWaiting { 649 return fmt.Errorf("g %v is not waiting before unpark (offset %v, time %v)", ev.Args[0], ev.Off, ev.Ts) 650 } 651 if g1.ev != nil && g1.ev.Type == EvGoBlockNet && ev.P != TimerP { 652 ev.P = NetpollP 653 } 654 if g1.ev != nil { 655 g1.ev.Link = ev 656 } 657 g1.state = gRunnable 658 g1.ev = ev 659 gs[ev.Args[0]] = g1 660 case EvGoSysCall: 661 if err := checkRunning(p, g, ev, false); err != nil { 662 return err 663 } 664 g.ev = ev 665 case EvGoSysBlock: 666 if err := checkRunning(p, g, ev, false); err != nil { 667 return err 668 } 669 g.state = gWaiting 670 g.evStart.Link = ev 671 g.evStart = nil 672 p.g = 0 673 case EvGoSysExit: 674 if g.state != gWaiting { 675 return fmt.Errorf("g %v is not waiting during syscall exit (offset %v, time %v)", ev.G, ev.Off, ev.Ts) 676 } 677 if g.ev != nil && g.ev.Type == EvGoSysCall { 678 g.ev.Link = ev 679 } 680 g.state = gRunnable 681 g.ev = ev 682 case EvGoSleep, EvGoBlock, EvGoBlockSend, EvGoBlockRecv, 683 EvGoBlockSelect, EvGoBlockSync, EvGoBlockCond, EvGoBlockNet: 684 if err := checkRunning(p, g, ev, false); err != nil { 685 return err 686 } 687 g.state = gWaiting 688 g.ev = ev 689 g.evStart.Link = ev 690 g.evStart = nil 691 p.g = 0 692 } 693 694 gs[ev.G] = g 695 ps[ev.P] = p 696 } 697 698 // TODO(dvyukov): restore stacks for EvGoStart events. 699 // TODO(dvyukov): test that all EvGoStart events has non-nil Link. 700 701 return nil 702 } 703 704 // symbolize attaches func/file/line info to stack traces. 705 func symbolize(events []*Event, bin string) error { 706 // First, collect and dedup all pcs. 707 pcs := make(map[uint64]*Frame) 708 for _, ev := range events { 709 for _, f := range ev.Stk { 710 pcs[f.PC] = nil 711 } 712 } 713 714 // Start addr2line. 715 cmd := exec.Command("go", "tool", "addr2line", bin) 716 in, err := cmd.StdinPipe() 717 if err != nil { 718 return fmt.Errorf("failed to pipe addr2line stdin: %v", err) 719 } 720 cmd.Stderr = os.Stderr 721 out, err := cmd.StdoutPipe() 722 if err != nil { 723 return fmt.Errorf("failed to pipe addr2line stdout: %v", err) 724 } 725 err = cmd.Start() 726 if err != nil { 727 return fmt.Errorf("failed to start addr2line: %v", err) 728 } 729 outb := bufio.NewReader(out) 730 731 // Write all pcs to addr2line. 732 // Need to copy pcs to an array, because map iteration order is non-deterministic. 733 var pcArray []uint64 734 for pc := range pcs { 735 pcArray = append(pcArray, pc) 736 _, err := fmt.Fprintf(in, "0x%x\n", pc-1) 737 if err != nil { 738 return fmt.Errorf("failed to write to addr2line: %v", err) 739 } 740 } 741 in.Close() 742 743 // Read in answers. 744 for _, pc := range pcArray { 745 fn, err := outb.ReadString('\n') 746 if err != nil { 747 return fmt.Errorf("failed to read from addr2line: %v", err) 748 } 749 file, err := outb.ReadString('\n') 750 if err != nil { 751 return fmt.Errorf("failed to read from addr2line: %v", err) 752 } 753 f := &Frame{PC: pc} 754 f.Fn = fn[:len(fn)-1] 755 f.File = file[:len(file)-1] 756 if colon := strings.LastIndex(f.File, ":"); colon != -1 { 757 ln, err := strconv.Atoi(f.File[colon+1:]) 758 if err == nil { 759 f.File = f.File[:colon] 760 f.Line = ln 761 } 762 } 763 pcs[pc] = f 764 } 765 cmd.Wait() 766 767 // Replace frames in events array. 768 for _, ev := range events { 769 for i, f := range ev.Stk { 770 ev.Stk[i] = pcs[f.PC] 771 } 772 } 773 774 return nil 775 } 776 777 // readVal reads unsigned base-128 value from r. 778 func readVal(r io.Reader, off0 int) (v uint64, off int, err error) { 779 off = off0 780 for i := 0; i < 10; i++ { 781 var buf [1]byte 782 var n int 783 n, err = r.Read(buf[:]) 784 if err != nil || n != 1 { 785 return 0, 0, fmt.Errorf("failed to read trace at offset %d: read %v, error %v", off0, n, err) 786 } 787 off++ 788 v |= uint64(buf[0]&0x7f) << (uint(i) * 7) 789 if buf[0]&0x80 == 0 { 790 return 791 } 792 } 793 return 0, 0, fmt.Errorf("bad value at offset 0x%x", off0) 794 } 795 796 // Print dumps events to stdout. For debugging. 797 func Print(events []*Event) { 798 for _, ev := range events { 799 PrintEvent(ev) 800 } 801 } 802 803 // PrintEvent dumps the event to stdout. For debugging. 804 func PrintEvent(ev *Event) { 805 desc := EventDescriptions[ev.Type] 806 fmt.Printf("%v %v p=%v g=%v off=%v", ev.Ts, desc.Name, ev.P, ev.G, ev.Off) 807 for i, a := range desc.Args { 808 fmt.Printf(" %v=%v", a, ev.Args[i]) 809 } 810 fmt.Printf("\n") 811 } 812 813 // argNum returns total number of args for the event accounting for timestamps, 814 // sequence numbers and differences between trace format versions. 815 func argNum(raw rawEvent, ver int) int { 816 desc := EventDescriptions[raw.typ] 817 if raw.typ == EvStack { 818 return len(raw.args) 819 } 820 narg := len(desc.Args) 821 if desc.Stack { 822 narg++ 823 } 824 switch raw.typ { 825 case EvBatch, EvFrequency, EvTimerGoroutine: 826 if ver < 1007 { 827 narg++ // there was an unused arg before 1.7 828 } 829 case EvGCStart, EvGoStart, EvGoUnblock: 830 if ver < 1007 { 831 narg-- // 1.7 added an additional seq arg 832 } 833 fallthrough 834 default: 835 narg++ // timestamp 836 if ver < 1007 { 837 narg++ // sequence 838 } 839 } 840 return narg 841 } 842 843 // BreakTimestampsForTesting causes the parser to randomly alter timestamps (for testing of broken cputicks). 844 var BreakTimestampsForTesting bool 845 846 // Event types in the trace. 847 // Verbatim copy from src/runtime/trace.go. 848 const ( 849 EvNone = 0 // unused 850 EvBatch = 1 // start of per-P batch of events [pid, timestamp] 851 EvFrequency = 2 // contains tracer timer frequency [frequency (ticks per second)] 852 EvStack = 3 // stack [stack id, number of PCs, array of {PC, func string ID, file string ID, line}] 853 EvGomaxprocs = 4 // current value of GOMAXPROCS [timestamp, GOMAXPROCS, stack id] 854 EvProcStart = 5 // start of P [timestamp, thread id] 855 EvProcStop = 6 // stop of P [timestamp] 856 EvGCStart = 7 // GC start [timestamp, seq, stack id] 857 EvGCDone = 8 // GC done [timestamp] 858 EvGCScanStart = 9 // GC scan start [timestamp] 859 EvGCScanDone = 10 // GC scan done [timestamp] 860 EvGCSweepStart = 11 // GC sweep start [timestamp, stack id] 861 EvGCSweepDone = 12 // GC sweep done [timestamp] 862 EvGoCreate = 13 // goroutine creation [timestamp, new goroutine id, new stack id, stack id] 863 EvGoStart = 14 // goroutine starts running [timestamp, goroutine id, seq] 864 EvGoEnd = 15 // goroutine ends [timestamp] 865 EvGoStop = 16 // goroutine stops (like in select{}) [timestamp, stack] 866 EvGoSched = 17 // goroutine calls Gosched [timestamp, stack] 867 EvGoPreempt = 18 // goroutine is preempted [timestamp, stack] 868 EvGoSleep = 19 // goroutine calls Sleep [timestamp, stack] 869 EvGoBlock = 20 // goroutine blocks [timestamp, stack] 870 EvGoUnblock = 21 // goroutine is unblocked [timestamp, goroutine id, seq, stack] 871 EvGoBlockSend = 22 // goroutine blocks on chan send [timestamp, stack] 872 EvGoBlockRecv = 23 // goroutine blocks on chan recv [timestamp, stack] 873 EvGoBlockSelect = 24 // goroutine blocks on select [timestamp, stack] 874 EvGoBlockSync = 25 // goroutine blocks on Mutex/RWMutex [timestamp, stack] 875 EvGoBlockCond = 26 // goroutine blocks on Cond [timestamp, stack] 876 EvGoBlockNet = 27 // goroutine blocks on network [timestamp, stack] 877 EvGoSysCall = 28 // syscall enter [timestamp, stack] 878 EvGoSysExit = 29 // syscall exit [timestamp, goroutine id, seq, real timestamp] 879 EvGoSysBlock = 30 // syscall blocks [timestamp] 880 EvGoWaiting = 31 // denotes that goroutine is blocked when tracing starts [timestamp, goroutine id] 881 EvGoInSyscall = 32 // denotes that goroutine is in syscall when tracing starts [timestamp, goroutine id] 882 EvHeapAlloc = 33 // memstats.heap_live change [timestamp, heap_alloc] 883 EvNextGC = 34 // memstats.next_gc change [timestamp, next_gc] 884 EvTimerGoroutine = 35 // denotes timer goroutine [timer goroutine id] 885 EvFutileWakeup = 36 // denotes that the previous wakeup of this goroutine was futile [timestamp] 886 EvString = 37 // string dictionary entry [ID, length, string] 887 EvGoStartLocal = 38 // goroutine starts running on the same P as the last event [timestamp, goroutine id] 888 EvGoUnblockLocal = 39 // goroutine is unblocked on the same P as the last event [timestamp, goroutine id, stack] 889 EvGoSysExitLocal = 40 // syscall exit on the same P as the last event [timestamp, goroutine id, real timestamp] 890 EvCount = 41 891 ) 892 893 var EventDescriptions = [EvCount]struct { 894 Name string 895 minVersion int 896 Stack bool 897 Args []string 898 }{ 899 EvNone: {"None", 1005, false, []string{}}, 900 EvBatch: {"Batch", 1005, false, []string{"p", "ticks"}}, // in 1.5 format it was {"p", "seq", "ticks"} 901 EvFrequency: {"Frequency", 1005, false, []string{"freq"}}, // in 1.5 format it was {"freq", "unused"} 902 EvStack: {"Stack", 1005, false, []string{"id", "siz"}}, 903 EvGomaxprocs: {"Gomaxprocs", 1005, true, []string{"procs"}}, 904 EvProcStart: {"ProcStart", 1005, false, []string{"thread"}}, 905 EvProcStop: {"ProcStop", 1005, false, []string{}}, 906 EvGCStart: {"GCStart", 1005, true, []string{"seq"}}, // in 1.5 format it was {} 907 EvGCDone: {"GCDone", 1005, false, []string{}}, 908 EvGCScanStart: {"GCScanStart", 1005, false, []string{}}, 909 EvGCScanDone: {"GCScanDone", 1005, false, []string{}}, 910 EvGCSweepStart: {"GCSweepStart", 1005, true, []string{}}, 911 EvGCSweepDone: {"GCSweepDone", 1005, false, []string{}}, 912 EvGoCreate: {"GoCreate", 1005, true, []string{"g", "stack"}}, 913 EvGoStart: {"GoStart", 1005, false, []string{"g", "seq"}}, // in 1.5 format it was {"g"} 914 EvGoEnd: {"GoEnd", 1005, false, []string{}}, 915 EvGoStop: {"GoStop", 1005, true, []string{}}, 916 EvGoSched: {"GoSched", 1005, true, []string{}}, 917 EvGoPreempt: {"GoPreempt", 1005, true, []string{}}, 918 EvGoSleep: {"GoSleep", 1005, true, []string{}}, 919 EvGoBlock: {"GoBlock", 1005, true, []string{}}, 920 EvGoUnblock: {"GoUnblock", 1005, true, []string{"g", "seq"}}, // in 1.5 format it was {"g"} 921 EvGoBlockSend: {"GoBlockSend", 1005, true, []string{}}, 922 EvGoBlockRecv: {"GoBlockRecv", 1005, true, []string{}}, 923 EvGoBlockSelect: {"GoBlockSelect", 1005, true, []string{}}, 924 EvGoBlockSync: {"GoBlockSync", 1005, true, []string{}}, 925 EvGoBlockCond: {"GoBlockCond", 1005, true, []string{}}, 926 EvGoBlockNet: {"GoBlockNet", 1005, true, []string{}}, 927 EvGoSysCall: {"GoSysCall", 1005, true, []string{}}, 928 EvGoSysExit: {"GoSysExit", 1005, false, []string{"g", "seq", "ts"}}, 929 EvGoSysBlock: {"GoSysBlock", 1005, false, []string{}}, 930 EvGoWaiting: {"GoWaiting", 1005, false, []string{"g"}}, 931 EvGoInSyscall: {"GoInSyscall", 1005, false, []string{"g"}}, 932 EvHeapAlloc: {"HeapAlloc", 1005, false, []string{"mem"}}, 933 EvNextGC: {"NextGC", 1005, false, []string{"mem"}}, 934 EvTimerGoroutine: {"TimerGoroutine", 1005, false, []string{"g"}}, // in 1.5 format it was {"g", "unused"} 935 EvFutileWakeup: {"FutileWakeup", 1005, false, []string{}}, 936 EvString: {"String", 1007, false, []string{}}, 937 EvGoStartLocal: {"GoStartLocal", 1007, false, []string{"g"}}, 938 EvGoUnblockLocal: {"GoUnblockLocal", 1007, true, []string{"g"}}, 939 EvGoSysExitLocal: {"GoSysExitLocal", 1007, false, []string{"g", "ts"}}, 940 }