1154abd99SDavid du Colombier #include "u.h"
2154abd99SDavid du Colombier #include "../port/lib.h"
3154abd99SDavid du Colombier #include "mem.h"
4154abd99SDavid du Colombier #include "dat.h"
5154abd99SDavid du Colombier #include "fns.h"
6154abd99SDavid du Colombier #include "../port/error.h"
7154abd99SDavid du Colombier #include "../port/systab.h"
8154abd99SDavid du Colombier
9154abd99SDavid du Colombier #include <tos.h>
10154abd99SDavid du Colombier #include "ureg.h"
11154abd99SDavid du Colombier
12154abd99SDavid du Colombier #include "arm.h"
13154abd99SDavid du Colombier
14154abd99SDavid du Colombier typedef struct {
15154abd99SDavid du Colombier uintptr ip;
16154abd99SDavid du Colombier Ureg* arg0;
17154abd99SDavid du Colombier char* arg1;
18154abd99SDavid du Colombier char msg[ERRMAX];
19154abd99SDavid du Colombier Ureg* old;
20154abd99SDavid du Colombier Ureg ureg;
21154abd99SDavid du Colombier } NFrame;
22154abd99SDavid du Colombier
23154abd99SDavid du Colombier /*
24154abd99SDavid du Colombier * Return user to state before notify()
25154abd99SDavid du Colombier */
26154abd99SDavid du Colombier static void
noted(Ureg * cur,uintptr arg0)27154abd99SDavid du Colombier noted(Ureg* cur, uintptr arg0)
28154abd99SDavid du Colombier {
29154abd99SDavid du Colombier NFrame *nf;
30154abd99SDavid du Colombier Ureg *nur;
31154abd99SDavid du Colombier
32154abd99SDavid du Colombier qlock(&up->debug);
33154abd99SDavid du Colombier if(arg0 != NRSTR && !up->notified){
34154abd99SDavid du Colombier qunlock(&up->debug);
35154abd99SDavid du Colombier pprint("call to noted() when not notified\n");
36154abd99SDavid du Colombier pexit("Suicide", 0);
37154abd99SDavid du Colombier }
38154abd99SDavid du Colombier up->notified = 0;
39154abd99SDavid du Colombier fpunoted();
40154abd99SDavid du Colombier
41154abd99SDavid du Colombier nf = up->ureg;
42154abd99SDavid du Colombier
43154abd99SDavid du Colombier /* sanity clause */
44154abd99SDavid du Colombier if(!okaddr(PTR2UINT(nf), sizeof(NFrame), 0)){
45154abd99SDavid du Colombier qunlock(&up->debug);
462f205b96SDavid du Colombier pprint("bad ureg in noted %#p\n", nf);
47154abd99SDavid du Colombier pexit("Suicide", 0);
48154abd99SDavid du Colombier }
49154abd99SDavid du Colombier
50154abd99SDavid du Colombier /* don't let user change system flags */
51154abd99SDavid du Colombier nur = &nf->ureg;
52154abd99SDavid du Colombier nur->psr &= PsrMask|PsrDfiq|PsrDirq;
53154abd99SDavid du Colombier nur->psr |= (cur->psr & ~(PsrMask|PsrDfiq|PsrDirq));
54154abd99SDavid du Colombier
55154abd99SDavid du Colombier memmove(cur, nur, sizeof(Ureg));
56154abd99SDavid du Colombier
57154abd99SDavid du Colombier switch((int)arg0){
58154abd99SDavid du Colombier case NCONT:
59154abd99SDavid du Colombier case NRSTR:
60472c7937SDavid du Colombier if(!okaddr(nur->pc, BY2WD, 0) || !okaddr(nur->sp, BY2WD, 0)){
61154abd99SDavid du Colombier qunlock(&up->debug);
62154abd99SDavid du Colombier pprint("suicide: trap in noted\n");
63154abd99SDavid du Colombier pexit("Suicide", 0);
64154abd99SDavid du Colombier }
65154abd99SDavid du Colombier up->ureg = nf->old;
66154abd99SDavid du Colombier qunlock(&up->debug);
67154abd99SDavid du Colombier break;
68154abd99SDavid du Colombier case NSAVE:
69472c7937SDavid du Colombier if(!okaddr(nur->pc, BY2WD, 0) || !okaddr(nur->sp, BY2WD, 0)){
70154abd99SDavid du Colombier qunlock(&up->debug);
71154abd99SDavid du Colombier pprint("suicide: trap in noted\n");
72154abd99SDavid du Colombier pexit("Suicide", 0);
73154abd99SDavid du Colombier }
74154abd99SDavid du Colombier qunlock(&up->debug);
75154abd99SDavid du Colombier
76154abd99SDavid du Colombier splhi();
77154abd99SDavid du Colombier nf->arg1 = nf->msg;
78154abd99SDavid du Colombier nf->arg0 = &nf->ureg;
79154abd99SDavid du Colombier nf->ip = 0;
80154abd99SDavid du Colombier cur->sp = PTR2UINT(nf);
81*c02f0a41SDavid du Colombier cur->r0 = PTR2UINT(nf->arg0);
82154abd99SDavid du Colombier break;
83154abd99SDavid du Colombier default:
84154abd99SDavid du Colombier pprint("unknown noted arg %#p\n", arg0);
85154abd99SDavid du Colombier up->lastnote.flag = NDebug;
86154abd99SDavid du Colombier /*FALLTHROUGH*/
87154abd99SDavid du Colombier case NDFLT:
88154abd99SDavid du Colombier if(up->lastnote.flag == NDebug){
89154abd99SDavid du Colombier qunlock(&up->debug);
90154abd99SDavid du Colombier pprint("suicide: %s\n", up->lastnote.msg);
91154abd99SDavid du Colombier }
92154abd99SDavid du Colombier else
93154abd99SDavid du Colombier qunlock(&up->debug);
94154abd99SDavid du Colombier pexit(up->lastnote.msg, up->lastnote.flag != NDebug);
95154abd99SDavid du Colombier }
96154abd99SDavid du Colombier }
97154abd99SDavid du Colombier
98154abd99SDavid du Colombier /*
99154abd99SDavid du Colombier * Call user, if necessary, with note.
100154abd99SDavid du Colombier * Pass user the Ureg struct and the note on his stack.
101154abd99SDavid du Colombier */
102154abd99SDavid du Colombier int
notify(Ureg * ureg)103154abd99SDavid du Colombier notify(Ureg* ureg)
104154abd99SDavid du Colombier {
105154abd99SDavid du Colombier int l;
106154abd99SDavid du Colombier Note *n;
107154abd99SDavid du Colombier u32int s;
108154abd99SDavid du Colombier uintptr sp;
109154abd99SDavid du Colombier NFrame *nf;
110154abd99SDavid du Colombier
111154abd99SDavid du Colombier if(up->procctl)
112154abd99SDavid du Colombier procctl(up);
113154abd99SDavid du Colombier if(up->nnote == 0)
114154abd99SDavid du Colombier return 0;
115154abd99SDavid du Colombier
116154abd99SDavid du Colombier fpunotify(ureg);
117154abd99SDavid du Colombier
118154abd99SDavid du Colombier s = spllo();
119154abd99SDavid du Colombier qlock(&up->debug);
120154abd99SDavid du Colombier
121154abd99SDavid du Colombier up->notepending = 0;
122154abd99SDavid du Colombier n = &up->note[0];
123154abd99SDavid du Colombier if(strncmp(n->msg, "sys:", 4) == 0){
124154abd99SDavid du Colombier l = strlen(n->msg);
125154abd99SDavid du Colombier if(l > ERRMAX-23) /* " pc=0x0123456789abcdef\0" */
126154abd99SDavid du Colombier l = ERRMAX-23;
1277bb09086SDavid du Colombier snprint(n->msg + l, sizeof n->msg - l, " pc=%#lux", ureg->pc);
128154abd99SDavid du Colombier }
129154abd99SDavid du Colombier
130154abd99SDavid du Colombier if(n->flag != NUser && (up->notified || up->notify == 0)){
131154abd99SDavid du Colombier if(n->flag == NDebug)
132154abd99SDavid du Colombier pprint("suicide: %s\n", n->msg);
133154abd99SDavid du Colombier qunlock(&up->debug);
134154abd99SDavid du Colombier pexit(n->msg, n->flag != NDebug);
135154abd99SDavid du Colombier }
136154abd99SDavid du Colombier
137154abd99SDavid du Colombier if(up->notified){
138154abd99SDavid du Colombier qunlock(&up->debug);
139154abd99SDavid du Colombier splhi();
140154abd99SDavid du Colombier return 0;
141154abd99SDavid du Colombier }
142154abd99SDavid du Colombier
143154abd99SDavid du Colombier if(up->notify == nil){
144154abd99SDavid du Colombier qunlock(&up->debug);
145154abd99SDavid du Colombier pexit(n->msg, n->flag != NDebug);
146154abd99SDavid du Colombier }
147154abd99SDavid du Colombier if(!okaddr(PTR2UINT(up->notify), 1, 0)){
148154abd99SDavid du Colombier pprint("suicide: notify function address %#p\n", up->notify);
149154abd99SDavid du Colombier qunlock(&up->debug);
150154abd99SDavid du Colombier pexit("Suicide", 0);
151154abd99SDavid du Colombier }
152154abd99SDavid du Colombier
153154abd99SDavid du Colombier sp = ureg->sp - sizeof(NFrame);
154154abd99SDavid du Colombier if(!okaddr(sp, sizeof(NFrame), 1)){
155154abd99SDavid du Colombier qunlock(&up->debug);
1562f205b96SDavid du Colombier pprint("suicide: notify stack address %#p\n", sp);
157154abd99SDavid du Colombier pexit("Suicide", 0);
158154abd99SDavid du Colombier }
159154abd99SDavid du Colombier
160154abd99SDavid du Colombier nf = UINT2PTR(sp);
161154abd99SDavid du Colombier memmove(&nf->ureg, ureg, sizeof(Ureg));
162154abd99SDavid du Colombier nf->old = up->ureg;
163154abd99SDavid du Colombier up->ureg = nf;
164154abd99SDavid du Colombier memmove(nf->msg, up->note[0].msg, ERRMAX);
165154abd99SDavid du Colombier nf->arg1 = nf->msg;
166154abd99SDavid du Colombier nf->arg0 = &nf->ureg;
167ddc74443SDavid du Colombier ureg->r0 = PTR2UINT(nf->arg0);
168154abd99SDavid du Colombier nf->ip = 0;
169154abd99SDavid du Colombier
170154abd99SDavid du Colombier ureg->sp = sp;
171154abd99SDavid du Colombier ureg->pc = PTR2UINT(up->notify);
172154abd99SDavid du Colombier up->notified = 1;
173154abd99SDavid du Colombier up->nnote--;
174154abd99SDavid du Colombier memmove(&up->lastnote, &up->note[0], sizeof(Note));
175154abd99SDavid du Colombier memmove(&up->note[0], &up->note[1], up->nnote*sizeof(Note));
176154abd99SDavid du Colombier
177154abd99SDavid du Colombier qunlock(&up->debug);
178154abd99SDavid du Colombier splx(s);
179154abd99SDavid du Colombier
180154abd99SDavid du Colombier return 1;
181154abd99SDavid du Colombier }
182154abd99SDavid du Colombier
183154abd99SDavid du Colombier void
syscall(Ureg * ureg)184154abd99SDavid du Colombier syscall(Ureg* ureg)
185154abd99SDavid du Colombier {
186154abd99SDavid du Colombier char *e;
187154abd99SDavid du Colombier u32int s;
188154abd99SDavid du Colombier ulong sp;
189154abd99SDavid du Colombier long ret;
190154abd99SDavid du Colombier int i, scallnr;
191fcbb35d1SDavid du Colombier vlong startns, stopns;
192154abd99SDavid du Colombier
193154abd99SDavid du Colombier if(!userureg(ureg))
1947bb09086SDavid du Colombier panic("syscall: from kernel: pc %#lux r14 %#lux psr %#lux",
195154abd99SDavid du Colombier ureg->pc, ureg->r14, ureg->psr);
196154abd99SDavid du Colombier
197154abd99SDavid du Colombier cycles(&up->kentry);
198154abd99SDavid du Colombier
199154abd99SDavid du Colombier m->syscall++;
200154abd99SDavid du Colombier up->insyscall = 1;
201154abd99SDavid du Colombier up->pc = ureg->pc;
202154abd99SDavid du Colombier up->dbgreg = ureg;
203154abd99SDavid du Colombier
204154abd99SDavid du Colombier scallnr = ureg->r0;
205154abd99SDavid du Colombier up->scallnr = scallnr;
206154abd99SDavid du Colombier if(scallnr == RFORK)
207154abd99SDavid du Colombier fpusysrfork(ureg);
208154abd99SDavid du Colombier spllo();
209154abd99SDavid du Colombier sp = ureg->sp;
210fcbb35d1SDavid du Colombier
211fcbb35d1SDavid du Colombier if(up->procctl == Proc_tracesyscall){
212fcbb35d1SDavid du Colombier /*
213fcbb35d1SDavid du Colombier * Redundant validaddr. Do we care?
214fcbb35d1SDavid du Colombier * Tracing syscalls is not exactly a fast path...
215fcbb35d1SDavid du Colombier * Beware, validaddr currently does a pexit rather
216fcbb35d1SDavid du Colombier * than an error if there's a problem; that might
217fcbb35d1SDavid du Colombier * change in the future.
218fcbb35d1SDavid du Colombier */
219fcbb35d1SDavid du Colombier if(sp < (USTKTOP-BY2PG) || sp > (USTKTOP-sizeof(Sargs)-BY2WD))
220fcbb35d1SDavid du Colombier validaddr(sp, sizeof(Sargs)+BY2WD, 0);
221fcbb35d1SDavid du Colombier
222472c7937SDavid du Colombier syscallfmt(scallnr, ureg->pc, (va_list)(sp+BY2WD));
223fcbb35d1SDavid du Colombier up->procctl = Proc_stopme;
224fcbb35d1SDavid du Colombier procctl(up);
225fcbb35d1SDavid du Colombier if (up->syscalltrace)
226fcbb35d1SDavid du Colombier free(up->syscalltrace);
227fcbb35d1SDavid du Colombier up->syscalltrace = nil;
228fcbb35d1SDavid du Colombier }
229fcbb35d1SDavid du Colombier
230154abd99SDavid du Colombier up->nerrlab = 0;
231154abd99SDavid du Colombier ret = -1;
232fcbb35d1SDavid du Colombier startns = todget(nil);
233154abd99SDavid du Colombier if(!waserror()){
234154abd99SDavid du Colombier if(scallnr >= nsyscall){
2357bb09086SDavid du Colombier pprint("bad sys call number %d pc %#lux\n",
236154abd99SDavid du Colombier scallnr, ureg->pc);
237154abd99SDavid du Colombier postnote(up, 1, "sys: bad sys call", NDebug);
238154abd99SDavid du Colombier error(Ebadarg);
239154abd99SDavid du Colombier }
240154abd99SDavid du Colombier
241154abd99SDavid du Colombier if(sp < (USTKTOP-BY2PG) || sp > (USTKTOP-sizeof(Sargs)-BY2WD))
242154abd99SDavid du Colombier validaddr(sp, sizeof(Sargs)+BY2WD, 0);
243154abd99SDavid du Colombier
244154abd99SDavid du Colombier up->s = *((Sargs*)(sp+BY2WD));
245154abd99SDavid du Colombier up->psstate = sysctab[scallnr];
246154abd99SDavid du Colombier
247154abd99SDavid du Colombier /* iprint("%s: syscall %s\n", up->text, sysctab[scallnr]?sysctab[scallnr]:"huh?"); */
248154abd99SDavid du Colombier
249154abd99SDavid du Colombier ret = systab[scallnr](up->s.args);
250154abd99SDavid du Colombier poperror();
251154abd99SDavid du Colombier }else{
252154abd99SDavid du Colombier /* failure: save the error buffer for errstr */
253154abd99SDavid du Colombier e = up->syserrstr;
254154abd99SDavid du Colombier up->syserrstr = up->errstr;
255154abd99SDavid du Colombier up->errstr = e;
256154abd99SDavid du Colombier }
257154abd99SDavid du Colombier if(up->nerrlab){
258154abd99SDavid du Colombier print("bad errstack [%d]: %d extra\n", scallnr, up->nerrlab);
259154abd99SDavid du Colombier for(i = 0; i < NERR; i++)
260154abd99SDavid du Colombier print("sp=%#p pc=%#p\n",
261154abd99SDavid du Colombier up->errlab[i].sp, up->errlab[i].pc);
262154abd99SDavid du Colombier panic("error stack");
263154abd99SDavid du Colombier }
264154abd99SDavid du Colombier
265154abd99SDavid du Colombier /*
266154abd99SDavid du Colombier * Put return value in frame. On the x86 the syscall is
267154abd99SDavid du Colombier * just another trap and the return value from syscall is
268154abd99SDavid du Colombier * ignored. On other machines the return value is put into
269154abd99SDavid du Colombier * the results register by caller of syscall.
270154abd99SDavid du Colombier */
271154abd99SDavid du Colombier ureg->r0 = ret;
272154abd99SDavid du Colombier
273154abd99SDavid du Colombier if(up->procctl == Proc_tracesyscall){
274fcbb35d1SDavid du Colombier stopns = todget(nil);
275154abd99SDavid du Colombier up->procctl = Proc_stopme;
276472c7937SDavid du Colombier sysretfmt(scallnr, (va_list)(sp+BY2WD), ret, startns, stopns);
277154abd99SDavid du Colombier s = splhi();
278154abd99SDavid du Colombier procctl(up);
279154abd99SDavid du Colombier splx(s);
280fcbb35d1SDavid du Colombier if(up->syscalltrace)
281fcbb35d1SDavid du Colombier free(up->syscalltrace);
282fcbb35d1SDavid du Colombier up->syscalltrace = nil;
283154abd99SDavid du Colombier }
284154abd99SDavid du Colombier
285154abd99SDavid du Colombier up->insyscall = 0;
286154abd99SDavid du Colombier up->psstate = 0;
287154abd99SDavid du Colombier
288154abd99SDavid du Colombier if(scallnr == NOTED)
289154abd99SDavid du Colombier noted(ureg, *(ulong*)(sp+BY2WD));
290154abd99SDavid du Colombier
291154abd99SDavid du Colombier splhi();
292154abd99SDavid du Colombier if(scallnr != RFORK && (up->procctl || up->nnote))
293154abd99SDavid du Colombier notify(ureg);
294154abd99SDavid du Colombier
295154abd99SDavid du Colombier /* if we delayed sched because we held a lock, sched now */
296154abd99SDavid du Colombier if(up->delaysched){
297154abd99SDavid du Colombier sched();
298154abd99SDavid du Colombier splhi();
299154abd99SDavid du Colombier }
300154abd99SDavid du Colombier kexit(ureg);
301154abd99SDavid du Colombier }
302154abd99SDavid du Colombier
303b1c4f505SDavid du Colombier long
execregs(ulong entry,ulong ssize,ulong nargs)304154abd99SDavid du Colombier execregs(ulong entry, ulong ssize, ulong nargs)
305154abd99SDavid du Colombier {
306154abd99SDavid du Colombier ulong *sp;
307154abd99SDavid du Colombier Ureg *ureg;
308154abd99SDavid du Colombier
309154abd99SDavid du Colombier sp = (ulong*)(USTKTOP - ssize);
310154abd99SDavid du Colombier *--sp = nargs;
311154abd99SDavid du Colombier
312154abd99SDavid du Colombier ureg = up->dbgreg;
313154abd99SDavid du Colombier // memset(ureg, 0, 15*sizeof(ulong));
314154abd99SDavid du Colombier ureg->r13 = (ulong)sp;
315154abd99SDavid du Colombier ureg->pc = entry;
316154abd99SDavid du Colombier //print("%lud: EXECREGS pc %#ux sp %#ux nargs %ld\n", up->pid, ureg->pc, ureg->r13, nargs);
317154abd99SDavid du Colombier
318154abd99SDavid du Colombier /*
319154abd99SDavid du Colombier * return the address of kernel/user shared data
320154abd99SDavid du Colombier * (e.g. clock stuff)
321154abd99SDavid du Colombier */
322154abd99SDavid du Colombier return USTKTOP-sizeof(Tos);
323154abd99SDavid du Colombier }
324154abd99SDavid du Colombier
325154abd99SDavid du Colombier void
sysprocsetup(Proc * p)326154abd99SDavid du Colombier sysprocsetup(Proc* p)
327154abd99SDavid du Colombier {
328154abd99SDavid du Colombier fpusysprocsetup(p);
329154abd99SDavid du Colombier }
330154abd99SDavid du Colombier
331154abd99SDavid du Colombier /*
332154abd99SDavid du Colombier * Craft a return frame which will cause the child to pop out of
333154abd99SDavid du Colombier * the scheduler in user mode with the return register zero. Set
334154abd99SDavid du Colombier * pc to point to a l.s return function.
335154abd99SDavid du Colombier */
336154abd99SDavid du Colombier void
forkchild(Proc * p,Ureg * ureg)337154abd99SDavid du Colombier forkchild(Proc *p, Ureg *ureg)
338154abd99SDavid du Colombier {
339154abd99SDavid du Colombier Ureg *cureg;
340154abd99SDavid du Colombier
341154abd99SDavid du Colombier //print("%lud setting up for forking child %lud\n", up->pid, p->pid);
342154abd99SDavid du Colombier p->sched.sp = (ulong)p->kstack+KSTACK-sizeof(Ureg);
343154abd99SDavid du Colombier p->sched.pc = (ulong)forkret;
344154abd99SDavid du Colombier
345154abd99SDavid du Colombier cureg = (Ureg*)(p->sched.sp);
346154abd99SDavid du Colombier memmove(cureg, ureg, sizeof(Ureg));
347154abd99SDavid du Colombier
348154abd99SDavid du Colombier /* syscall returns 0 for child */
349154abd99SDavid du Colombier cureg->r0 = 0;
350154abd99SDavid du Colombier
351154abd99SDavid du Colombier /* Things from bottom of syscall which were never executed */
352154abd99SDavid du Colombier p->psstate = 0;
353154abd99SDavid du Colombier p->insyscall = 0;
354b1c4f505SDavid du Colombier
355b1c4f505SDavid du Colombier fpusysrforkchild(p, cureg, up);
356154abd99SDavid du Colombier }
357