Annotation of 43BSDReno/sys/kern/kern_ktrace.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1989 The Regents of the University of California.
        !             3:  * All rights reserved.
        !             4:  *
        !             5:  * Redistribution is only permitted until one year after the first shipment
        !             6:  * of 4.4BSD by the Regents.  Otherwise, redistribution and use in source and
        !             7:  * binary forms are permitted provided that: (1) source distributions retain
        !             8:  * this entire copyright notice and comment, and (2) distributions including
        !             9:  * binaries display the following acknowledgement:  This product includes
        !            10:  * software developed by the University of California, Berkeley and its
        !            11:  * contributors'' in the documentation or other materials provided with the
        !            12:  * distribution and in all advertising materials mentioning features or use
        !            13:  * of this software.  Neither the name of the University nor the names of
        !            14:  * its contributors may be used to endorse or promote products derived from
        !            15:  * this software without specific prior written permission.
        !            16:  * THIS SOFTWARE IS PROVIDED AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
        !            17:  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
        !            18:  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
        !            19:  *
        !            20:  *     @(#)kern_ktrace.c       7.9 (Berkeley) 6/28/90
        !            21:  */
        !            22: 
        !            23: #ifdef KTRACE
        !            24: 
        !            25: #include "param.h"
        !            26: #include "user.h"
        !            27: #include "proc.h"
        !            28: #include "file.h"
        !            29: #include "vnode.h"
        !            30: #include "ktrace.h"
        !            31: #include "malloc.h"
        !            32: 
        !            33: #include "syscalls.c"
        !            34: 
        !            35: extern int nsysent;
        !            36: extern char *syscallnames[];
        !            37: 
        !            38: int ktrace_nocheck = 0;        /* set to 1 when security checks in place */
        !            39: 
        !            40: struct ktr_header *
        !            41: ktrgetheader(type)
        !            42: {
        !            43:        register struct ktr_header *kth;
        !            44: 
        !            45:        MALLOC(kth, struct ktr_header *, sizeof (struct ktr_header), 
        !            46:                M_TEMP, M_WAITOK);
        !            47:        kth->ktr_type = type;
        !            48:        microtime(&kth->ktr_time);
        !            49:        kth->ktr_pid = u.u_procp->p_pid;
        !            50:        bcopy(u.u_procp->p_comm, kth->ktr_comm, MAXCOMLEN);
        !            51:        return (kth);
        !            52: }
        !            53: 
        !            54: ktrsyscall(vp, code, narg, args)
        !            55:        struct vnode *vp;
        !            56:        int code, narg, args[];
        !            57: {
        !            58:        struct  ktr_header *kth = ktrgetheader(KTR_SYSCALL);
        !            59:        struct  ktr_syscall *ktp;
        !            60:        register len = sizeof(struct ktr_syscall) + (narg * sizeof(int));
        !            61:        int     *argp, i;
        !            62: 
        !            63:        if (kth == NULL)
        !            64:                return;
        !            65:        MALLOC(ktp, struct ktr_syscall *, len, M_TEMP, M_WAITOK);
        !            66:        ktp->ktr_code = code;
        !            67:        ktp->ktr_narg = narg;
        !            68:        argp = (int *)((char *)ktp + sizeof(struct ktr_syscall));
        !            69:        for (i = 0; i < narg; i++)
        !            70:                *argp++ = args[i];
        !            71:        kth->ktr_buf = (caddr_t)ktp;
        !            72:        kth->ktr_len = len;
        !            73:        ktrwrite(vp, kth);
        !            74:        FREE(ktp, M_TEMP);
        !            75:        FREE(kth, M_TEMP);
        !            76: }
        !            77: 
        !            78: ktrsysret(vp, code, error, retval)
        !            79:        struct vnode *vp;
        !            80:        int code, error, retval;
        !            81: {
        !            82:        struct ktr_header *kth = ktrgetheader(KTR_SYSRET);
        !            83:        struct ktr_sysret ktp;
        !            84: 
        !            85:        if (kth == NULL)
        !            86:                return;
        !            87:        ktp.ktr_code = code;
        !            88:        ktp.ktr_error = error;
        !            89:        ktp.ktr_retval = retval;                /* what about val2 ? */
        !            90: 
        !            91:        kth->ktr_buf = (caddr_t)&ktp;
        !            92:        kth->ktr_len = sizeof(struct ktr_sysret);
        !            93: 
        !            94:        ktrwrite(vp, kth);
        !            95:        FREE(kth, M_TEMP);
        !            96: }
        !            97: 
        !            98: ktrnamei(vp, path)
        !            99:        struct vnode *vp;
        !           100:        char *path;
        !           101: {
        !           102:        struct ktr_header *kth = ktrgetheader(KTR_NAMEI);
        !           103: 
        !           104:        if (kth == NULL)
        !           105:                return;
        !           106:        kth->ktr_len = strlen(path);
        !           107:        kth->ktr_buf = path;
        !           108: 
        !           109:        ktrwrite(vp, kth);
        !           110:        FREE(kth, M_TEMP);
        !           111: }
        !           112: 
        !           113: ktrgenio(vp, fd, rw, iov, len, error)
        !           114:        struct vnode *vp;
        !           115:        int fd;
        !           116:        enum uio_rw rw;
        !           117:        register struct iovec *iov;
        !           118: {
        !           119:        struct ktr_header *kth = ktrgetheader(KTR_GENIO);
        !           120:        register struct ktr_genio *ktp;
        !           121:        register caddr_t cp;
        !           122:        register int resid = len, cnt;
        !           123:        
        !           124:        if (kth == NULL || error)
        !           125:                return;
        !           126:        MALLOC(ktp, struct ktr_genio *, sizeof(struct ktr_genio) + len,
        !           127:                M_TEMP, M_WAITOK);
        !           128:        ktp->ktr_fd = fd;
        !           129:        ktp->ktr_rw = rw;
        !           130:        cp = (caddr_t)((char *)ktp + sizeof (struct ktr_genio));
        !           131:        while (resid > 0) {
        !           132:                if ((cnt = iov->iov_len) > resid)
        !           133:                        cnt = resid;
        !           134:                if (copyin(iov->iov_base, cp, (unsigned)cnt))
        !           135:                        goto done;
        !           136:                cp += cnt;
        !           137:                resid -= cnt;
        !           138:                iov++;
        !           139:        }
        !           140:        kth->ktr_buf = (caddr_t)ktp;
        !           141:        kth->ktr_len = sizeof (struct ktr_genio) + len;
        !           142: 
        !           143:        ktrwrite(vp, kth);
        !           144: done:
        !           145:        FREE(kth, M_TEMP);
        !           146:        FREE(ktp, M_TEMP);
        !           147: }
        !           148: 
        !           149: ktrpsig(vp, sig, action, mask, code)
        !           150:        struct  vnode *vp;
        !           151:        sig_t   action;
        !           152: {
        !           153:        struct ktr_header *kth = ktrgetheader(KTR_PSIG);
        !           154:        struct ktr_psig kp;
        !           155: 
        !           156:        if (kth == NULL)
        !           157:                return;
        !           158:        kp.signo = (char)sig;
        !           159:        kp.action = action;
        !           160:        kp.mask = mask;
        !           161:        kp.code = code;
        !           162:        kth->ktr_buf = (caddr_t)&kp;
        !           163:        kth->ktr_len = sizeof (struct ktr_psig);
        !           164: 
        !           165:        ktrwrite(vp, kth);
        !           166:        FREE(kth, M_TEMP);
        !           167: }
        !           168: 
        !           169: /* Interface and common routines */
        !           170: 
        !           171: /*
        !           172:  * ktrace system call
        !           173:  */
        !           174: /* ARGSUSED */
        !           175: ktrace(curp, uap, retval)
        !           176:        struct proc *curp;
        !           177:        register struct args {
        !           178:                char    *fname;
        !           179:                int     ops;
        !           180:                int     facs;
        !           181:                int     pid;
        !           182:        } *uap;
        !           183:        int *retval;
        !           184: {
        !           185:        register struct vnode *vp = NULL;
        !           186:        register struct nameidata *ndp = &u.u_nd;
        !           187:        register struct proc *p;
        !           188:        register ops = KTROP(uap->ops);
        !           189:        struct pgrp *pg;
        !           190:        register int facs = uap->facs;
        !           191:        register int ret = 0;
        !           192:        int error = 0;
        !           193: 
        !           194:        /*
        !           195:         * Until security implications are thought through,
        !           196:         * limit tracing to root (unless ktrace_nocheck is set).
        !           197:         */
        !           198:        if (!ktrace_nocheck && (error = suser(u.u_cred, &u.u_acflag)))
        !           199:                return (error);
        !           200:        if (ops != KTROP_CLEAR) {
        !           201:                /*
        !           202:                 * an operation which requires a file argument.
        !           203:                 */
        !           204:                ndp->ni_segflg = UIO_USERSPACE;
        !           205:                ndp->ni_dirp = uap->fname;
        !           206:                if (error = vn_open(ndp, FREAD|FWRITE, 0))
        !           207:                        return (error);
        !           208:                vp = ndp->ni_vp;
        !           209:                if (vp->v_type != VREG) {
        !           210:                        vrele(vp);
        !           211:                        return (EACCES);
        !           212:                }
        !           213:        }
        !           214:        /*
        !           215:         * Clear all uses of the tracefile
        !           216:         */
        !           217:        if (ops == KTROP_CLEARFILE) {
        !           218:                for (p = allproc; p != NULL; p = p->p_nxt) {
        !           219:                        if (p->p_tracep == vp) {
        !           220:                                p->p_tracep = NULL;
        !           221:                                p->p_traceflag = 0;
        !           222:                                vrele(vp);
        !           223:                        }
        !           224:                }
        !           225:                goto done;
        !           226:        }
        !           227:        /*
        !           228:         * need something to (un)trace
        !           229:         */
        !           230:        if (!facs) {
        !           231:                error = EINVAL;
        !           232:                goto done;
        !           233:        }
        !           234:        /* 
        !           235:         * doit
        !           236:         */
        !           237:        if (uap->pid < 0) {
        !           238:                pg = pgfind(-uap->pid);
        !           239:                if (pg == NULL) {
        !           240:                        error = ESRCH;
        !           241:                        goto done;
        !           242:                }
        !           243:                for (p = pg->pg_mem; p != NULL; p = p->p_pgrpnxt)
        !           244:                        if (uap->ops&KTRFLAG_DESCEND)
        !           245:                                ret |= ktrsetchildren(p, ops, facs, vp);
        !           246:                        else 
        !           247:                                ret |= ktrops(p, ops, facs, vp);
        !           248:                                        
        !           249:        } else {
        !           250:                p = pfind(uap->pid);
        !           251:                if (p == NULL) {
        !           252:                        error = ESRCH;
        !           253:                        goto done;
        !           254:                }
        !           255:                if (ops&KTRFLAG_DESCEND)
        !           256:                        ret |= ktrsetchildren(p, ops, facs, vp);
        !           257:                else
        !           258:                        ret |= ktrops(p, ops, facs, vp);
        !           259:        }
        !           260:        if (!ret)
        !           261:                error = EPERM;
        !           262: done:
        !           263:        if (vp != NULL)
        !           264:                vrele(vp);
        !           265:        return (error);
        !           266: }
        !           267: 
        !           268: ktrops(p, ops, facs, vp)
        !           269:        struct proc *p;
        !           270:        struct vnode *vp;
        !           271: {
        !           272: 
        !           273:        if (u.u_uid && u.u_uid != p->p_uid)
        !           274:                return (0);
        !           275:        if (ops == KTROP_SET) {
        !           276:                if (p->p_tracep != vp) { 
        !           277:                        /*
        !           278:                         * if trace file already in use, relinquish
        !           279:                         */
        !           280:                        if (p->p_tracep != NULL)
        !           281:                                vrele(p->p_tracep);
        !           282:                        VREF(vp);
        !           283:                        p->p_tracep = vp;
        !           284:                }
        !           285:                p->p_traceflag |= facs;
        !           286:        } else {        
        !           287:                /* KTROP_CLEAR */
        !           288:                if (((p->p_traceflag &= ~facs) & ~KTRFAC_INHERIT) == 0) {
        !           289:                        /* no more tracing */
        !           290:                        p->p_traceflag = 0;
        !           291:                        if (p->p_tracep != NULL) {
        !           292:                                vrele(p->p_tracep);
        !           293:                                p->p_tracep = NULL;
        !           294:                        }
        !           295:                }
        !           296:        }
        !           297: 
        !           298:        return 1;
        !           299: }
        !           300: 
        !           301: ktrsetchildren(top, ops, facs, vp)
        !           302:        struct proc *top;
        !           303:        struct vnode *vp;
        !           304: {
        !           305:        register struct proc *p;
        !           306:        register int ret = 0;
        !           307: 
        !           308:        p = top;
        !           309:        for (;;) {
        !           310:                ret |= ktrops(p, ops, facs, vp);
        !           311:                /*
        !           312:                 * If this process has children, descend to them next,
        !           313:                 * otherwise do any siblings, and if done with this level,
        !           314:                 * follow back up the tree (but not past top).
        !           315:                 */
        !           316:                if (p->p_cptr)
        !           317:                        p = p->p_cptr;
        !           318:                else if (p == top)
        !           319:                        return ret;
        !           320:                else if (p->p_osptr)
        !           321:                        p = p->p_osptr;
        !           322:                else for (;;) {
        !           323:                        p = p->p_pptr;
        !           324:                        if (p == top)
        !           325:                                return ret;
        !           326:                        if (p->p_osptr) {
        !           327:                                p = p->p_osptr;
        !           328:                                break;
        !           329:                        }
        !           330:                }
        !           331:        }
        !           332:        /*NOTREACHED*/
        !           333: }
        !           334: 
        !           335: ktrwrite(vp, kth)
        !           336:        struct vnode *vp;
        !           337:        register struct ktr_header *kth;
        !           338: {
        !           339:        struct uio auio;
        !           340:        struct iovec aiov[2];
        !           341:        struct proc *p;
        !           342:        int error;
        !           343: 
        !           344:        if (vp == NULL)
        !           345:                return;
        !           346:        auio.uio_iov = &aiov[0];
        !           347:        auio.uio_offset = 0;
        !           348:        auio.uio_segflg = UIO_SYSSPACE;
        !           349:        auio.uio_rw = UIO_WRITE;
        !           350:        aiov[0].iov_base = (caddr_t)kth;
        !           351:        aiov[0].iov_len = sizeof(struct ktr_header);
        !           352:        auio.uio_resid = sizeof(struct ktr_header);
        !           353:        auio.uio_iovcnt = 1;
        !           354:        if (kth->ktr_len > 0) {
        !           355:                auio.uio_iovcnt++;
        !           356:                aiov[1].iov_base = kth->ktr_buf;
        !           357:                aiov[1].iov_len = kth->ktr_len;
        !           358:                auio.uio_resid += kth->ktr_len;
        !           359:        }
        !           360:        VOP_LOCK(vp);
        !           361:        error = VOP_WRITE(vp, &auio, IO_UNIT|IO_APPEND, u.u_cred);
        !           362:        VOP_UNLOCK(vp);
        !           363:        if (!error)
        !           364:                return;
        !           365:        /*
        !           366:         * If error encountered, give up tracing on this vnode.
        !           367:         */
        !           368:        uprintf("\ntrace write failed with errno %d, tracing stopped\n", error);
        !           369:        for (p = allproc; p != NULL; p = p->p_nxt) {
        !           370:                if (p->p_tracep == vp) {
        !           371:                        p->p_tracep = NULL;
        !           372:                        p->p_traceflag = 0;
        !           373:                        vrele(vp);
        !           374:                }
        !           375:        }
        !           376: }
        !           377: #endif

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.