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