PWB1/sys/sys/os/sys4.c
#
/*
 * Everything in this file is a routine implementing a system call.
 */
#include "../hd/param.h"
#include "../hd/user.h"
#include "../hd/reg.h"
#include "../hd/inode.h"
#include "../hd/systm.h"
#include "../hd/proc.h"
#include "../hd/var.h"
getswit()
{
	u.u_ar0[R0] = SW->integ;
}
gtime()
{
	u.u_ar0[R0] = time.hiword;
	u.u_ar0[R1] = time.loword;
}
stime()
{
	if(suser()) {
		time.hiword = u.u_ar0[R0];
		time.loword = u.u_ar0[R1];
		wakeup(&tout);
	}
}
setuid()
{
	register uid;
	uid = u.u_ar0[R0].lobyte;
	if(u.u_ruid == uid.lobyte || suser()) {
		u.u_uid = uid;
		u.u_procp->p_uid = uid;
		u.u_ruid = uid;
	}
}
getuid()
{
	u.u_ar0[R0].lobyte = u.u_ruid;
	u.u_ar0[R0].hibyte = u.u_uid;
}
setgid()
{
	register gid;
	gid = u.u_ar0[R0].lobyte;
	if(u.u_rgid == gid.lobyte || suser()) {
		u.u_gid = gid;
		u.u_rgid = gid;
	}
}
getgid()
{
	u.u_ar0[R0].lobyte = u.u_rgid;
	u.u_ar0[R0].hibyte = u.u_gid;
}
getpid()
{
	u.u_ar0[R0] = u.u_procp->p_pid;
	u.u_ar0[R1] = u.u_procp->p_pgrp;
}
setpgrp()
{
	u.u_procp->p_pgrp = u.u_procp->p_pid;
}
sync()
{
	update();
}
nice()
{
	register n;
	n = u.u_ar0[R0];
	if(n > 20)
		n = 20;
	if(n < 0 && !suser())
		n = 0;
	u.u_procp->p_nice = n;
}
/*
 * Unlink system call.
 * Hard to avoid races here, especially
 * in unlinking directories.
 */
unlink()
{
	register *ip, *pp;
	extern uchar;
	pp = namei(&uchar, 2);
	if(pp == NULL)
		return;
	/*
	 * Check for unlink(".")
	 * to avoid hanging on the iget
	 */
	if (pp->i_number != u.u_dent.u_ino)
		ip = iget(pp->i_dev, u.u_dent.u_ino);
	else {
		ip = pp;
		ip->i_count++;
	}
	if(ip == NULL)
		goto out1;
	if((ip->i_mode&IFMT)==IFDIR && !suser())
		goto out;
	/*
	 * Don't unlink a mounted file.
	 */
	if (ip->i_dev != pp->i_dev) {
		u.u_error = EBUSY;
		goto out;
	}
	if (ip->i_flag&ITEXT && ip->i_nlink==1) {
		u.u_error = ETXTBSY;
		goto out;
	}
	u.u_offset[1] =- DIRSIZ+2;
	u.u_base = &u.u_dent;
	u.u_count = DIRSIZ+2;
	u.u_dent.u_ino = 0;
	writei(pp);
	ip->i_nlink--;
	ip->i_flag =| IUPD;
out:
	iput(ip);
out1:
	iput(pp);
}
chdir()
{
	register *ip;
	extern uchar;
	ip = namei(&uchar, 0);
	if(ip == NULL)
		return;
	if((ip->i_mode&IFMT) != IFDIR) {
		u.u_error = ENOTDIR;
	bad:
		iput(ip);
		return;
	}
	if(access(ip, IEXEC))
		goto bad;
	prele(ip);
	plock(u.u_cdir);
	iput(u.u_cdir);
	u.u_cdir = ip;
}
chmod()
{
	register *ip, a;
	if ((ip = owner()) == NULL)
		return;
	ip->i_mode =& ~07777;
	a = u.u_arg[1];
	if (u.u_uid) {
		a =& ~ISVTX;
		if (u.u_gid != ip->i_gid)
			a =& ~ISGID;
	}
	ip->i_mode =| a&07777;
	ip->i_flag =| IUPD;
	iput(ip);
}
chown()
{
	register *ip;
	if ((ip = owner()) == NULL)
		return;
	ip->i_uid = u.u_arg[1].lobyte;
	ip->i_gid = u.u_arg[1].hibyte;
	if(u.u_uid != 0)
		ip->i_mode =& ~(ISUID|ISGID);
	ip->i_flag =| IUPD;
	iput(ip);
}
ssig()
{
	register a;
	a = u.u_arg[0];
	if(a<=0 || a>NSIG || a==SIGKIL) {
		u.u_error = EINVAL;
		return;
	}
	u.u_ar0[R0] = u.u_signal[a-1];
	u.u_signal[a-1] = u.u_arg[1];
	u.u_procp->p_sig =& ~(1<<(a-1));
}
kill()
{
	register struct proc *p;
	register a, f;
	f = 0;
	a = u.u_ar0[R0];
	if (a>0)
		p = &proc[0]; else
		p = &proc[2];
	for(; p < v.ve_proc; p++) {
		if(a > 0 && p->p_pid != a)
			continue;
		if(a == 0 && p->p_pgrp != u.u_procp->p_pgrp)
			continue;
		if(u.u_uid != 0 && u.u_uid != p->p_uid)
			if (a>0) {
				u.u_error = EPERM;
				return;
			} else
				continue;
		f++;
		psignal(p, u.u_arg[0]);
	}
	if(f == 0)
		u.u_error = ESRCH;
}
times()
{
	register *p;
	for(p = &u.u_utime; p  < &u.u_utime+4;) {
		suword(u.u_arg[0], *p++);
		u.u_arg[0] =+ 2;
	}
}
profil()
{
	u.u_prof[0] = u.u_arg[0] & ~1;	/* base of sample buf */
	u.u_prof[1] = u.u_arg[1];	/* size of same */
	u.u_prof[2] = u.u_arg[2];	/* pc offset */
	u.u_prof[3] = (u.u_arg[3]>>1) & 077777; /* pc scale */
}
/*
 * alarm clock signal
 */
alarm()
{
	register c, *p;
	p = u.u_procp;
	c = p->p_clktim;
	p->p_clktim = u.u_ar0[R0];
	u.u_ar0[R0] = c;
}
/*
 * indefinite wait.
 * no one should wakeup(&u)
 */
pause()
{
	for(;;)
		sleep(&u, PSLEP);
}