Source file
src/runtime/tracestack.go
1
2
3
4
5
6
7 package runtime
8
9 import (
10 "internal/abi"
11 "internal/goarch"
12 "internal/trace/tracev2"
13 "unsafe"
14 )
15
16 const (
17
18
19
20
21 logicalStackSentinel = ^uintptr(0)
22 )
23
24
25
26
27
28
29
30
31
32
33
34 func traceStack(skip int, gp *g, gen uintptr) uint64 {
35 var pcBuf [tracev2.MaxFramesPerStack]uintptr
36
37
38 var mp *m
39 if gp == nil {
40 mp = getg().m
41 gp = mp.curg
42 }
43
44
45 if debug.traceCheckStackOwnership != 0 && gp != nil {
46 status := readgstatus(gp)
47
48 if status&_Gscan == 0 {
49
50
51
52
53 switch goStatusToTraceGoStatus(status, gp.waitreason) {
54 case tracev2.GoRunning, tracev2.GoSyscall:
55 if getg() == gp || mp.curg == gp {
56 break
57 }
58 fallthrough
59 default:
60 print("runtime: gp=", unsafe.Pointer(gp), " gp.goid=", gp.goid, " status=", gStatusStrings[status], "\n")
61 throw("attempted to trace stack of a goroutine this thread does not own")
62 }
63 }
64 }
65
66 if gp != nil && mp == nil {
67
68
69 mp = gp.lockedm.ptr()
70 }
71 nstk := 1
72 if tracefpunwindoff() || (mp != nil && mp.hasCgoOnStack()) {
73
74
75
76
77
78
79 pcBuf[0] = logicalStackSentinel
80 if getg() == gp {
81 nstk += callers(skip+1, pcBuf[1:])
82 } else if gp != nil {
83 nstk += gcallers(gp, skip, pcBuf[1:])
84 }
85 } else {
86
87 pcBuf[0] = uintptr(skip)
88 if getg() == gp {
89 nstk += fpTracebackPCs(unsafe.Pointer(getfp()), pcBuf[1:])
90 } else if gp != nil {
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107 if gp.syscallsp != 0 {
108 pcBuf[1] = gp.syscallpc
109 nstk += 1 + fpTracebackPCs(unsafe.Pointer(gp.syscallbp), pcBuf[2:])
110 } else {
111 pcBuf[1] = gp.sched.pc
112 nstk += 1 + fpTracebackPCs(unsafe.Pointer(gp.sched.bp), pcBuf[2:])
113 }
114 }
115 }
116 if nstk > 0 {
117 nstk--
118 }
119 if nstk > 0 && gp.goid == 1 {
120 nstk--
121 }
122 id := trace.stackTab[gen%2].put(pcBuf[:nstk])
123 return id
124 }
125
126
127
128 type traceStackTable struct {
129 tab traceMap
130 }
131
132
133
134 func (t *traceStackTable) put(pcs []uintptr) uint64 {
135 if len(pcs) == 0 {
136 return 0
137 }
138 id, _ := t.tab.put(noescape(unsafe.Pointer(&pcs[0])), uintptr(len(pcs))*unsafe.Sizeof(uintptr(0)))
139 return id
140 }
141
142
143
144
145 func (t *traceStackTable) dump(gen uintptr) {
146 stackBuf := make([]uintptr, tracev2.MaxFramesPerStack)
147 w := unsafeTraceWriter(gen, nil)
148 if root := (*traceMapNode)(t.tab.root.Load()); root != nil {
149 w = dumpStacksRec(root, w, stackBuf)
150 }
151 w.flush().end()
152 t.tab.reset()
153 }
154
155 func dumpStacksRec(node *traceMapNode, w traceWriter, stackBuf []uintptr) traceWriter {
156 stack := unsafe.Slice((*uintptr)(unsafe.Pointer(&node.data[0])), uintptr(len(node.data))/unsafe.Sizeof(uintptr(0)))
157
158
159
160 n := fpunwindExpand(stackBuf, stack)
161 frames := makeTraceFrames(w.gen, stackBuf[:n])
162
163
164
165 maxBytes := 1 + (2+4*len(frames))*traceBytesPerNumber
166
167
168
169
170
171
172 var flushed bool
173 w, flushed = w.ensure(1 + maxBytes)
174 if flushed {
175 w.byte(byte(tracev2.EvStacks))
176 }
177
178
179 w.byte(byte(tracev2.EvStack))
180 w.varint(uint64(node.id))
181 w.varint(uint64(len(frames)))
182 for _, frame := range frames {
183 w.varint(uint64(frame.PC))
184 w.varint(frame.funcID)
185 w.varint(frame.fileID)
186 w.varint(frame.line)
187 }
188
189
190 for i := range node.children {
191 child := node.children[i].Load()
192 if child == nil {
193 continue
194 }
195 w = dumpStacksRec((*traceMapNode)(child), w, stackBuf)
196 }
197 return w
198 }
199
200
201
202 func makeTraceFrames(gen uintptr, pcs []uintptr) []traceFrame {
203 frames := make([]traceFrame, 0, len(pcs))
204 ci := CallersFrames(pcs)
205 for {
206 f, more := ci.Next()
207 frames = append(frames, makeTraceFrame(gen, f))
208 if !more {
209 return frames
210 }
211 }
212 }
213
214 type traceFrame struct {
215 PC uintptr
216 funcID uint64
217 fileID uint64
218 line uint64
219 }
220
221
222 func makeTraceFrame(gen uintptr, f Frame) traceFrame {
223 var frame traceFrame
224 frame.PC = f.PC
225
226 fn := f.Function
227 const maxLen = 1 << 10
228 if len(fn) > maxLen {
229 fn = fn[len(fn)-maxLen:]
230 }
231 frame.funcID = trace.stringTab[gen%2].put(gen, fn)
232 frame.line = uint64(f.Line)
233 file := f.File
234 if len(file) > maxLen {
235 file = file[len(file)-maxLen:]
236 }
237 frame.fileID = trace.stringTab[gen%2].put(gen, file)
238 return frame
239 }
240
241
242
243 func tracefpunwindoff() bool {
244 return debug.tracefpunwindoff != 0 || (goarch.ArchFamily != goarch.AMD64 && goarch.ArchFamily != goarch.ARM64)
245 }
246
247
248
249
250
251 func fpTracebackPCs(fp unsafe.Pointer, pcBuf []uintptr) (i int) {
252 for i = 0; i < len(pcBuf) && fp != nil; i++ {
253
254 pcBuf[i] = *(*uintptr)(unsafe.Pointer(uintptr(fp) + goarch.PtrSize))
255
256 fp = unsafe.Pointer(*(*uintptr)(fp))
257 }
258 return i
259 }
260
261
262 func pprof_fpunwindExpand(dst, src []uintptr) int {
263 return fpunwindExpand(dst, src)
264 }
265
266
267
268
269
270
271
272
273
274
275 func fpunwindExpand(dst, pcBuf []uintptr) int {
276 if len(pcBuf) == 0 {
277 return 0
278 } else if len(pcBuf) > 0 && pcBuf[0] == logicalStackSentinel {
279
280
281 return copy(dst, pcBuf[1:])
282 }
283
284 var (
285 n int
286 lastFuncID = abi.FuncIDNormal
287 skip = pcBuf[0]
288
289
290 skipOrAdd = func(retPC uintptr) bool {
291 if skip > 0 {
292 skip--
293 } else if n < len(dst) {
294 dst[n] = retPC
295 n++
296 }
297 return n < len(dst)
298 }
299 )
300
301 outer:
302 for _, retPC := range pcBuf[1:] {
303 callPC := retPC - 1
304 fi := findfunc(callPC)
305 if !fi.valid() {
306
307
308 if more := skipOrAdd(retPC); !more {
309 break outer
310 }
311 continue
312 }
313
314 u, uf := newInlineUnwinder(fi, callPC)
315 for ; uf.valid(); uf = u.next(uf) {
316 sf := u.srcFunc(uf)
317 if sf.funcID == abi.FuncIDWrapper && elideWrapperCalling(lastFuncID) {
318
319 } else if more := skipOrAdd(uf.pc + 1); !more {
320 break outer
321 }
322 lastFuncID = sf.funcID
323 }
324 }
325 return n
326 }
327
328
329
330
331 func startPCForTrace(pc uintptr) uintptr {
332 f := findfunc(pc)
333 if !f.valid() {
334 return pc
335 }
336 w := funcdata(f, abi.FUNCDATA_WrapInfo)
337 if w == nil {
338 return pc
339 }
340 return f.datap.textAddr(*(*uint32)(w))
341 }
342
View as plain text