Source file
src/runtime/signal_windows.go
1
2
3
4
5 package runtime
6
7 import (
8 "internal/abi"
9 "unsafe"
10 )
11
12 const (
13 _SEM_FAILCRITICALERRORS = 0x0001
14 _SEM_NOGPFAULTERRORBOX = 0x0002
15 _SEM_NOOPENFILEERRORBOX = 0x8000
16
17 _WER_FAULT_REPORTING_NO_UI = 0x0020
18 )
19
20 func preventErrorDialogs() {
21 errormode := stdcall0(_GetErrorMode)
22 stdcall1(_SetErrorMode, errormode|_SEM_FAILCRITICALERRORS|_SEM_NOGPFAULTERRORBOX|_SEM_NOOPENFILEERRORBOX)
23
24
25
26
27
28 var werflags uintptr
29 stdcall2(_WerGetFlags, currentProcess, uintptr(unsafe.Pointer(&werflags)))
30 stdcall1(_WerSetFlags, werflags|_WER_FAULT_REPORTING_NO_UI)
31 }
32
33
34 func enableWER() {
35
36 errormode := stdcall0(_GetErrorMode)
37 if errormode&_SEM_NOGPFAULTERRORBOX != 0 {
38 stdcall1(_SetErrorMode, errormode^_SEM_NOGPFAULTERRORBOX)
39 }
40 }
41
42
43 func exceptiontramp()
44 func firstcontinuetramp()
45 func lastcontinuetramp()
46 func sehtramp()
47 func sigresume()
48
49 func initExceptionHandler() {
50 stdcall2(_AddVectoredExceptionHandler, 1, abi.FuncPCABI0(exceptiontramp))
51 if GOARCH == "386" {
52
53
54 stdcall1(_SetUnhandledExceptionFilter, abi.FuncPCABI0(lastcontinuetramp))
55 } else {
56 stdcall2(_AddVectoredContinueHandler, 1, abi.FuncPCABI0(firstcontinuetramp))
57 stdcall2(_AddVectoredContinueHandler, 0, abi.FuncPCABI0(lastcontinuetramp))
58 }
59 }
60
61
62
63
64
65 func isAbort(r *context) bool {
66 pc := r.ip()
67 if GOARCH == "386" || GOARCH == "amd64" || GOARCH == "arm" {
68
69
70
71 pc--
72 }
73 return isAbortPC(pc)
74 }
75
76
77
78
79
80
81
82
83 func isgoexception(info *exceptionrecord, r *context) bool {
84
85
86
87 if r.ip() < firstmoduledata.text || firstmoduledata.etext < r.ip() {
88 return false
89 }
90
91
92 switch info.exceptioncode {
93 default:
94 return false
95 case _EXCEPTION_ACCESS_VIOLATION:
96 case _EXCEPTION_IN_PAGE_ERROR:
97 case _EXCEPTION_INT_DIVIDE_BY_ZERO:
98 case _EXCEPTION_INT_OVERFLOW:
99 case _EXCEPTION_FLT_DENORMAL_OPERAND:
100 case _EXCEPTION_FLT_DIVIDE_BY_ZERO:
101 case _EXCEPTION_FLT_INEXACT_RESULT:
102 case _EXCEPTION_FLT_OVERFLOW:
103 case _EXCEPTION_FLT_UNDERFLOW:
104 case _EXCEPTION_BREAKPOINT:
105 case _EXCEPTION_ILLEGAL_INSTRUCTION:
106 }
107 return true
108 }
109
110 const (
111 callbackVEH = iota
112 callbackFirstVCH
113 callbackLastVCH
114 )
115
116
117
118
119
120
121 func sigFetchGSafe() *g
122
123 func sigFetchG() *g {
124 if GOARCH == "386" {
125 return sigFetchGSafe()
126 }
127 return getg()
128 }
129
130
131
132
133
134
135
136
137
138 func sigtrampgo(ep *exceptionpointers, kind int) int32 {
139 gp := sigFetchG()
140 if gp == nil {
141 return _EXCEPTION_CONTINUE_SEARCH
142 }
143
144 var fn func(info *exceptionrecord, r *context, gp *g) int32
145 switch kind {
146 case callbackVEH:
147 fn = exceptionhandler
148 case callbackFirstVCH:
149 fn = firstcontinuehandler
150 case callbackLastVCH:
151 fn = lastcontinuehandler
152 default:
153 throw("unknown sigtramp callback")
154 }
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170 var ret int32
171 if gp != gp.m.g0 {
172 systemstack(func() {
173 ret = fn(ep.record, ep.context, gp)
174 })
175 } else {
176 ret = fn(ep.record, ep.context, gp)
177 }
178 if ret == _EXCEPTION_CONTINUE_SEARCH {
179 return ret
180 }
181
182
183
184
185
186
187
188
189
190
191
192
193 if ep.context.ip() == abi.FuncPCABI0(sigresume) {
194
195 return ret
196 }
197 prepareContextForSigResume(ep.context)
198 ep.context.set_sp(gp.m.g0.sched.sp)
199 ep.context.set_ip(abi.FuncPCABI0(sigresume))
200 return ret
201 }
202
203
204
205
206
207
208
209
210
211 func exceptionhandler(info *exceptionrecord, r *context, gp *g) int32 {
212 if !isgoexception(info, r) {
213 return _EXCEPTION_CONTINUE_SEARCH
214 }
215
216 if gp.throwsplit || isAbort(r) {
217
218
219
220
221 winthrow(info, r, gp)
222 }
223
224
225
226
227
228
229
230 gp.sig = info.exceptioncode
231 gp.sigcode0 = info.exceptioninformation[0]
232 gp.sigcode1 = info.exceptioninformation[1]
233 gp.sigpc = r.ip()
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248 if r.ip() != 0 && r.ip() != abi.FuncPCABI0(asyncPreempt) {
249 r.pushCall(abi.FuncPCABI0(sigpanic0), r.ip())
250 } else {
251
252 r.set_ip(abi.FuncPCABI0(sigpanic0))
253 }
254 return _EXCEPTION_CONTINUE_EXECUTION
255 }
256
257
258
259
260
261
262 func sehhandler(_ *exceptionrecord, _ uint64, _ *context, dctxt *_DISPATCHER_CONTEXT) int32 {
263 g0 := getg()
264 if g0 == nil || g0.m.curg == nil {
265
266 return _EXCEPTION_CONTINUE_SEARCH_SEH
267 }
268
269
270
271
272
273
274
275
276
277
278 gp := g0.m.curg
279 ctxt := dctxt.ctx()
280 var base, sp uintptr
281 for {
282 entry := stdcall3(_RtlLookupFunctionEntry, ctxt.ip(), uintptr(unsafe.Pointer(&base)), 0)
283 if entry == 0 {
284 break
285 }
286 stdcall8(_RtlVirtualUnwind, 0, base, ctxt.ip(), entry, uintptr(unsafe.Pointer(ctxt)), 0, uintptr(unsafe.Pointer(&sp)), 0)
287 if sp < gp.stack.lo || gp.stack.hi <= sp {
288 break
289 }
290 }
291 return _EXCEPTION_CONTINUE_SEARCH_SEH
292 }
293
294
295
296
297
298
299
300
301
302 func firstcontinuehandler(info *exceptionrecord, r *context, gp *g) int32 {
303 if !isgoexception(info, r) {
304 return _EXCEPTION_CONTINUE_SEARCH
305 }
306 return _EXCEPTION_CONTINUE_EXECUTION
307 }
308
309
310
311
312
313
314
315 func lastcontinuehandler(info *exceptionrecord, r *context, gp *g) int32 {
316 if islibrary || isarchive {
317
318
319
320 return _EXCEPTION_CONTINUE_SEARCH
321 }
322
323
324
325
326
327
328
329 if GOARCH == "arm64" && info.exceptioncode == _EXCEPTION_ILLEGAL_INSTRUCTION &&
330 (r.ip() < firstmoduledata.text || firstmoduledata.etext < r.ip()) {
331 return _EXCEPTION_CONTINUE_SEARCH
332 }
333
334 winthrow(info, r, gp)
335 return 0
336 }
337
338
339
340
341 func winthrow(info *exceptionrecord, r *context, gp *g) {
342 g0 := getg()
343
344 if panicking.Load() != 0 {
345 exit(2)
346 }
347 panicking.Store(1)
348
349
350
351
352 g0.stack.lo = 0
353 g0.stackguard0 = g0.stack.lo + stackGuard
354 g0.stackguard1 = g0.stackguard0
355
356 print("Exception ", hex(info.exceptioncode), " ", hex(info.exceptioninformation[0]), " ", hex(info.exceptioninformation[1]), " ", hex(r.ip()), "\n")
357
358 print("PC=", hex(r.ip()), "\n")
359 if g0.m.incgo && gp == g0.m.g0 && g0.m.curg != nil {
360 if iscgo {
361 print("signal arrived during external code execution\n")
362 }
363 gp = g0.m.curg
364 }
365 print("\n")
366
367 g0.m.throwing = throwTypeRuntime
368 g0.m.caughtsig.set(gp)
369
370 level, _, docrash := gotraceback()
371 if level > 0 {
372 tracebacktrap(r.ip(), r.sp(), r.lr(), gp)
373 tracebackothers(gp)
374 dumpregs(r)
375 }
376
377 if docrash {
378 dieFromException(info, r)
379 }
380
381 exit(2)
382 }
383
384 func sigpanic() {
385 gp := getg()
386 if !canpanic() {
387 throw("unexpected signal during runtime execution")
388 }
389
390 switch gp.sig {
391 case _EXCEPTION_ACCESS_VIOLATION, _EXCEPTION_IN_PAGE_ERROR:
392 if gp.sigcode1 < 0x1000 {
393 panicmem()
394 }
395 if gp.paniconfault {
396 panicmemAddr(gp.sigcode1)
397 }
398 if inUserArenaChunk(gp.sigcode1) {
399
400
401
402 print("accessed data from freed user arena ", hex(gp.sigcode1), "\n")
403 } else {
404 print("unexpected fault address ", hex(gp.sigcode1), "\n")
405 }
406 throw("fault")
407 case _EXCEPTION_INT_DIVIDE_BY_ZERO:
408 panicdivide()
409 case _EXCEPTION_INT_OVERFLOW:
410 panicoverflow()
411 case _EXCEPTION_FLT_DENORMAL_OPERAND,
412 _EXCEPTION_FLT_DIVIDE_BY_ZERO,
413 _EXCEPTION_FLT_INEXACT_RESULT,
414 _EXCEPTION_FLT_OVERFLOW,
415 _EXCEPTION_FLT_UNDERFLOW:
416 panicfloat()
417 }
418 throw("fault")
419 }
420
421
422
423 func initsig(preinit bool) {
424 }
425
426 func sigenable(sig uint32) {
427 }
428
429 func sigdisable(sig uint32) {
430 }
431
432 func sigignore(sig uint32) {
433 }
434
435 func signame(sig uint32) string {
436 return ""
437 }
438
439
440 func crash() {
441 dieFromException(nil, nil)
442 }
443
444
445
446
447
448 func dieFromException(info *exceptionrecord, r *context) {
449 if info == nil {
450 gp := getg()
451 if gp.sig != 0 {
452
453
454 info = &exceptionrecord{
455 exceptionaddress: gp.sigpc,
456 exceptioncode: gp.sig,
457 numberparameters: 2,
458 }
459 info.exceptioninformation[0] = gp.sigcode0
460 info.exceptioninformation[1] = gp.sigcode1
461 } else {
462
463
464 info = &exceptionrecord{
465 exceptioncode: 2,
466 }
467 }
468 }
469 const FAIL_FAST_GENERATE_EXCEPTION_ADDRESS = 0x1
470 stdcall3(_RaiseFailFastException, uintptr(unsafe.Pointer(info)), uintptr(unsafe.Pointer(r)), FAIL_FAST_GENERATE_EXCEPTION_ADDRESS)
471 }
472
473
474 type gsignalStack struct{}
475
View as plain text