summaryrefslogtreecommitdiff
path: root/src/pkg/runtime/linux/arm/signal.c
blob: d1d8bc08c7b817e0d573721cae4d31f33c20e4e1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
// Copyright 2009 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

#include "runtime.h"
#include "defs.h"
#include "signals.h"
#include "os.h"

void
dumpregs(Sigcontext *r)
{
	printf("trap    %x\n", r->trap_no);
	printf("error   %x\n", r->error_code);
	printf("oldmask %x\n", r->oldmask);
	printf("r0      %x\n", r->arm_r0);
	printf("r1      %x\n", r->arm_r1);
	printf("r2      %x\n", r->arm_r2);
	printf("r3      %x\n", r->arm_r3);
	printf("r4      %x\n", r->arm_r4);
	printf("r5      %x\n", r->arm_r5);
	printf("r6      %x\n", r->arm_r6);
	printf("r7      %x\n", r->arm_r7);
	printf("r8      %x\n", r->arm_r8);
	printf("r9      %x\n", r->arm_r9);
	printf("r10     %x\n", r->arm_r10);
	printf("fp      %x\n", r->arm_fp);
	printf("ip      %x\n", r->arm_ip);
	printf("sp      %x\n", r->arm_sp);
	printf("lr      %x\n", r->arm_lr);
	printf("pc      %x\n", r->arm_pc);
	printf("cpsr    %x\n", r->arm_cpsr);
	printf("fault   %x\n", r->fault_address);
}

/*
 * This assembler routine takes the args from registers, puts them on the stack,
 * and calls sighandler().
 */
extern void sigtramp(void);
extern void sigignore(void);	// just returns
extern void sigreturn(void);	// calls sigreturn

String
signame(int32 sig)
{
	if(sig < 0 || sig >= NSIG)
		return emptystring;
	return gostring((byte*)sigtab[sig].name);
}

void
sighandler(int32 sig, Siginfo *info, void *context)
{
	Ucontext *uc;
	Sigcontext *sc;

	if(sigtab[sig].flags & SigQueue) {
		if(sigsend(sig) || (sigtab[sig].flags & SigIgnore))
			return;
		exit(2);	// SIGINT, SIGTERM, etc
	}

	if(panicking)	// traceback already printed
		exit(2);
	panicking = 1;

	if(sig < 0 || sig >= NSIG)
		printf("Signal %d\n", sig);
	else
		printf("%s\n", sigtab[sig].name);

	uc = context;
	sc = &uc->uc_mcontext;

	printf("Faulting address: %p\n", sc->fault_address);
	printf("PC=%x\n", sc->arm_pc);
	printf("\n");

	if(gotraceback()){
		traceback((void*)sc->arm_pc, (void*)sc->arm_sp, (void*)sc->arm_lr, m->curg);
		tracebackothers(m->curg);
		printf("\n");
		dumpregs(sc);
	}

//	breakpoint();
	exit(2);
}

void
signalstack(byte *p, int32 n)
{
	Sigaltstack st;

	st.ss_sp = p;
	st.ss_size = n;
	st.ss_flags = 0;
	sigaltstack(&st, nil);
}

void
initsig(void)
{
	static Sigaction sa;

	siginit();

	int32 i;
	sa.sa_flags = SA_ONSTACK | SA_SIGINFO | SA_RESTORER;
	sa.sa_mask.sig[0] = 0xFFFFFFFF;
	sa.sa_mask.sig[1] = 0xFFFFFFFF;
	sa.sa_restorer = (void*)sigreturn;
	for(i = 0; i<NSIG; i++) {
		if(sigtab[i].flags) {
			if(sigtab[i].flags & (SigCatch | SigQueue))
				sa.sa_handler = (void*)sigtramp;
			else
				sa.sa_handler = (void*)sigignore;
			if(sigtab[i].flags & SigRestart)
				sa.sa_flags |= SA_RESTART;
			else
				sa.sa_flags &= ~SA_RESTART;
			rt_sigaction(i, &sa, nil, 8);
		}
	}
}