kernel: fix sysexec() error handling compiler problem, sysrendez() busyloop
the variables elem and file0 and commited are explicitely set to avoid that they get freed in ther waserror() handlers. but it turns out the compiler optimizes this out as he thinks the variables arent used any further. (the compiler is not aware of the waserror() / longjmp() semantics). rearrange the code to account for this. instead of using a local variable to check for point of no return (commited), we use up->seg[SSEG] to figure it out. for file0 and elem, we just rearrange the code. elem can be checked in the error handler if it was already assigned to up->text, and file0 is just free()'d after the poperror(). remove silly busy loop in sysrendez. it is not needed. dequeueproc() will make sure that the process has come to rest.
This commit is contained in:
parent
257c7e958e
commit
3e567afed5
1 changed files with 17 additions and 22 deletions
|
@ -226,7 +226,7 @@ sysexec(ulong *arg)
|
||||||
char *a, *charp, *args, *file, *file0;
|
char *a, *charp, *args, *file, *file0;
|
||||||
char *progarg[sizeof(Exec)/2+1], *elem, progelem[64];
|
char *progarg[sizeof(Exec)/2+1], *elem, progelem[64];
|
||||||
ulong ssize, tstk, nargs, nbytes, n, bssend;
|
ulong ssize, tstk, nargs, nbytes, n, bssend;
|
||||||
int indir, commit;
|
int indir;
|
||||||
Exec exec;
|
Exec exec;
|
||||||
char line[sizeof(Exec)];
|
char line[sizeof(Exec)];
|
||||||
Fgrp *f;
|
Fgrp *f;
|
||||||
|
@ -234,16 +234,16 @@ sysexec(ulong *arg)
|
||||||
ulong magic, text, entry, data, bss;
|
ulong magic, text, entry, data, bss;
|
||||||
Tos *tos;
|
Tos *tos;
|
||||||
|
|
||||||
commit = 0;
|
|
||||||
indir = 0;
|
indir = 0;
|
||||||
elem = nil;
|
elem = nil;
|
||||||
validaddr(arg[0], 1, 0);
|
validaddr(arg[0], 1, 0);
|
||||||
file0 = validnamedup((char*)arg[0], 1);
|
file0 = validnamedup((char*)arg[0], 1);
|
||||||
if(waserror()){
|
if(waserror()){
|
||||||
free(file0);
|
free(file0);
|
||||||
|
if(elem != up->text)
|
||||||
free(elem);
|
free(elem);
|
||||||
/* Disaster after commit */
|
/* Disaster after commit */
|
||||||
if(commit)
|
if(!up->seg[SSEG])
|
||||||
pexit(up->errstr, 1);
|
pexit(up->errstr, 1);
|
||||||
nexterror();
|
nexterror();
|
||||||
}
|
}
|
||||||
|
@ -386,14 +386,9 @@ sysexec(ulong *arg)
|
||||||
memmove(charp, *argp++, n);
|
memmove(charp, *argp++, n);
|
||||||
charp += n;
|
charp += n;
|
||||||
}
|
}
|
||||||
free(file0);
|
|
||||||
file0 = nil; /* so waserror() won't free file0 */
|
|
||||||
USED(file0);
|
|
||||||
|
|
||||||
free(up->text);
|
free(up->text);
|
||||||
up->text = elem;
|
up->text = elem;
|
||||||
elem = nil; /* so waserror() won't free elem */
|
|
||||||
USED(elem);
|
|
||||||
|
|
||||||
/* copy args; easiest from new process's stack */
|
/* copy args; easiest from new process's stack */
|
||||||
n = charp - args;
|
n = charp - args;
|
||||||
|
@ -415,9 +410,6 @@ sysexec(ulong *arg)
|
||||||
}
|
}
|
||||||
up->nargs = n;
|
up->nargs = n;
|
||||||
|
|
||||||
commit = 1;
|
|
||||||
USED(commit);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Committed.
|
* Committed.
|
||||||
* Free old memory.
|
* Free old memory.
|
||||||
|
@ -428,9 +420,9 @@ sysexec(ulong *arg)
|
||||||
/* prevent a second free if we have an error */
|
/* prevent a second free if we have an error */
|
||||||
up->seg[i] = 0;
|
up->seg[i] = 0;
|
||||||
}
|
}
|
||||||
for(i = BSEG+1; i < NSEG; i++) {
|
for(i = ESEG+1; i < NSEG; i++) {
|
||||||
s = up->seg[i];
|
s = up->seg[i];
|
||||||
if(s != 0 && (s->type&SG_CEXEC)) {
|
if(s != 0 && (s->type&SG_CEXEC) != 0) {
|
||||||
putseg(s);
|
putseg(s);
|
||||||
up->seg[i] = 0;
|
up->seg[i] = 0;
|
||||||
}
|
}
|
||||||
|
@ -439,9 +431,10 @@ sysexec(ulong *arg)
|
||||||
/*
|
/*
|
||||||
* Close on exec
|
* Close on exec
|
||||||
*/
|
*/
|
||||||
f = up->fgrp;
|
if((f = up->fgrp) != nil){
|
||||||
for(i=0; i<=f->maxfd; i++)
|
for(i=0; i<=f->maxfd; i++)
|
||||||
fdclose(i, CCEXEC);
|
fdclose(i, CCEXEC);
|
||||||
|
}
|
||||||
|
|
||||||
/* Text. Shared. Attaches to cache image if possible */
|
/* Text. Shared. Attaches to cache image if possible */
|
||||||
/* attachimage returns a locked cache image */
|
/* attachimage returns a locked cache image */
|
||||||
|
@ -484,9 +477,10 @@ sysexec(ulong *arg)
|
||||||
if(devtab[tc->type]->dc == L'/')
|
if(devtab[tc->type]->dc == L'/')
|
||||||
up->basepri = PriRoot;
|
up->basepri = PriRoot;
|
||||||
up->priority = up->basepri;
|
up->priority = up->basepri;
|
||||||
cclose(tc);
|
|
||||||
poperror(); /* tc */
|
poperror(); /* tc */
|
||||||
poperror(); /* elem */
|
cclose(tc);
|
||||||
|
poperror(); /* file0 */
|
||||||
|
free(file0);
|
||||||
|
|
||||||
qlock(&up->debug);
|
qlock(&up->debug);
|
||||||
up->nnote = 0;
|
up->nnote = 0;
|
||||||
|
@ -830,9 +824,9 @@ sysrendezvous(ulong *arg)
|
||||||
val = p->rendval;
|
val = p->rendval;
|
||||||
p->rendval = arg[1];
|
p->rendval = arg[1];
|
||||||
unlock(up->rgrp);
|
unlock(up->rgrp);
|
||||||
while(p->mach != 0)
|
|
||||||
;
|
|
||||||
ready(p);
|
ready(p);
|
||||||
|
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
l = &p->rendhash;
|
l = &p->rendhash;
|
||||||
|
@ -1140,7 +1134,8 @@ syssemrelease(ulong *arg)
|
||||||
|
|
||||||
if((s = seg(up, (ulong)addr, 0)) == nil)
|
if((s = seg(up, (ulong)addr, 0)) == nil)
|
||||||
error(Ebadarg);
|
error(Ebadarg);
|
||||||
|
/* delta == 0 is a no-op, not a release */
|
||||||
if(delta < 0 || *addr < 0)
|
if(delta < 0 || *addr < 0)
|
||||||
error(Ebadarg);
|
error(Ebadarg);
|
||||||
return semrelease(s, addr, arg[1]);
|
return semrelease(s, addr, delta);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue