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