| | |
| | |
| | |
| |
|
| | package main |
| |
|
| | import ( |
| | "internal/trace" |
| | ) |
| |
|
| | var _ generator = &goroutineGenerator{} |
| |
|
| | type goroutineGenerator struct { |
| | globalRangeGenerator |
| | globalMetricGenerator |
| | stackSampleGenerator[trace.GoID] |
| | logEventGenerator[trace.GoID] |
| |
|
| | gStates map[trace.GoID]*gState[trace.GoID] |
| | focus trace.GoID |
| | filter map[trace.GoID]struct{} |
| | } |
| |
|
| | func newGoroutineGenerator(ctx *traceContext, focus trace.GoID, filter map[trace.GoID]struct{}) *goroutineGenerator { |
| | gg := new(goroutineGenerator) |
| | rg := func(ev *trace.Event) trace.GoID { |
| | return ev.Goroutine() |
| | } |
| | gg.stackSampleGenerator.getResource = rg |
| | gg.logEventGenerator.getResource = rg |
| | gg.gStates = make(map[trace.GoID]*gState[trace.GoID]) |
| | gg.focus = focus |
| | gg.filter = filter |
| |
|
| | |
| | if filter != nil { |
| | ctx.SetResourceFilter(func(resource uint64) bool { |
| | _, ok := filter[trace.GoID(resource)] |
| | return ok |
| | }) |
| | } |
| | return gg |
| | } |
| |
|
| | func (g *goroutineGenerator) Sync() { |
| | g.globalRangeGenerator.Sync() |
| | } |
| |
|
| | func (g *goroutineGenerator) GoroutineLabel(ctx *traceContext, ev *trace.Event) { |
| | l := ev.Label() |
| | g.gStates[l.Resource.Goroutine()].setLabel(l.Label) |
| | } |
| |
|
| | func (g *goroutineGenerator) GoroutineRange(ctx *traceContext, ev *trace.Event) { |
| | r := ev.Range() |
| | switch ev.Kind() { |
| | case trace.EventRangeBegin: |
| | g.gStates[r.Scope.Goroutine()].rangeBegin(ev.Time(), r.Name, ev.Stack()) |
| | case trace.EventRangeActive: |
| | g.gStates[r.Scope.Goroutine()].rangeActive(r.Name) |
| | case trace.EventRangeEnd: |
| | gs := g.gStates[r.Scope.Goroutine()] |
| | gs.rangeEnd(ev.Time(), r.Name, ev.Stack(), ctx) |
| | } |
| | } |
| |
|
| | func (g *goroutineGenerator) GoroutineTransition(ctx *traceContext, ev *trace.Event) { |
| | st := ev.StateTransition() |
| | goID := st.Resource.Goroutine() |
| |
|
| | |
| | |
| | gs, ok := g.gStates[goID] |
| | if !ok { |
| | gs = newGState[trace.GoID](goID) |
| | g.gStates[goID] = gs |
| | } |
| |
|
| | |
| | gs.augmentName(st.Stack) |
| |
|
| | |
| | from, to := st.Goroutine() |
| | if from == to { |
| | |
| | return |
| | } |
| | if from.Executing() && !to.Executing() { |
| | if to == trace.GoWaiting { |
| | |
| | gs.block(ev.Time(), ev.Stack(), st.Reason, ctx) |
| | } else { |
| | gs.stop(ev.Time(), ev.Stack(), ctx) |
| | } |
| | } |
| | if !from.Executing() && to.Executing() { |
| | start := ev.Time() |
| | if from == trace.GoUndetermined { |
| | |
| | start = ctx.startTime |
| | } |
| | gs.start(start, goID, ctx) |
| | } |
| |
|
| | if from == trace.GoWaiting { |
| | |
| | gs.unblock(ev.Time(), ev.Stack(), ev.Goroutine(), ctx) |
| | } |
| | if from == trace.GoNotExist && to == trace.GoRunnable { |
| | |
| | gs.created(ev.Time(), ev.Goroutine(), ev.Stack()) |
| | } |
| | if from == trace.GoSyscall && to != trace.GoRunning { |
| | |
| | gs.syscallEnd(ev.Time(), true, ctx) |
| | gs.blockedSyscallEnd(ev.Time(), ev.Stack(), ctx) |
| | } else if from == trace.GoSyscall { |
| | |
| | gs.syscallEnd(ev.Time(), false, ctx) |
| | } |
| |
|
| | |
| | if to == trace.GoSyscall { |
| | start := ev.Time() |
| | if from == trace.GoUndetermined { |
| | |
| | start = ctx.startTime |
| | } |
| | |
| | |
| | |
| | gs.syscallBegin(start, goID, ev.Stack()) |
| | } |
| |
|
| | |
| | _, inMarkAssist := gs.activeRanges["GC mark assist"] |
| | ctx.GoroutineTransition(ctx.elapsed(ev.Time()), viewerGState(from, inMarkAssist), viewerGState(to, inMarkAssist)) |
| | } |
| |
|
| | func (g *goroutineGenerator) ProcRange(ctx *traceContext, ev *trace.Event) { |
| | |
| | |
| | } |
| |
|
| | func (g *goroutineGenerator) ProcTransition(ctx *traceContext, ev *trace.Event) { |
| | |
| | } |
| |
|
| | func (g *goroutineGenerator) Finish(ctx *traceContext) { |
| | ctx.SetResourceType("G") |
| |
|
| | |
| | g.globalRangeGenerator.Finish(ctx) |
| |
|
| | |
| | for id, gs := range g.gStates { |
| | gs.finish(ctx) |
| |
|
| | |
| | ctx.Resource(uint64(id), gs.name()) |
| | } |
| |
|
| | |
| | if g.focus != trace.NoGoroutine { |
| | ctx.Focus(uint64(g.focus)) |
| | } |
| | } |
| |
|