Annotation of XNU/bsd/kern/subr_log.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
                      3:  *
                      4:  * @APPLE_LICENSE_HEADER_START@
                      5:  * 
                      6:  * The contents of this file constitute Original Code as defined in and
                      7:  * are subject to the Apple Public Source License Version 1.1 (the
                      8:  * "License").  You may not use this file except in compliance with the
                      9:  * License.  Please obtain a copy of the License at
                     10:  * http://www.apple.com/publicsource and read it before using this file.
                     11:  * 
                     12:  * This Original Code and all software distributed under the License are
                     13:  * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
                     14:  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
                     15:  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
                     16:  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
                     17:  * License for the specific language governing rights and limitations
                     18:  * under the License.
                     19:  * 
                     20:  * @APPLE_LICENSE_HEADER_END@
                     21:  */
                     22: /* Copyright (c) 1995 NeXT Computer, Inc. All Rights Reserved */
                     23: /*
                     24:  * Copyright (c) 1982, 1986, 1993
                     25:  *     The Regents of the University of California.  All rights reserved.
                     26:  *
                     27:  * Redistribution and use in source and binary forms, with or without
                     28:  * modification, are permitted provided that the following conditions
                     29:  * are met:
                     30:  * 1. Redistributions of source code must retain the above copyright
                     31:  *    notice, this list of conditions and the following disclaimer.
                     32:  * 2. Redistributions in binary form must reproduce the above copyright
                     33:  *    notice, this list of conditions and the following disclaimer in the
                     34:  *    documentation and/or other materials provided with the distribution.
                     35:  * 3. All advertising materials mentioning features or use of this software
                     36:  *    must display the following acknowledgement:
                     37:  *     This product includes software developed by the University of
                     38:  *     California, Berkeley and its contributors.
                     39:  * 4. Neither the name of the University nor the names of its contributors
                     40:  *    may be used to endorse or promote products derived from this software
                     41:  *    without specific prior written permission.
                     42:  *
                     43:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
                     44:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
                     45:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
                     46:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
                     47:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
                     48:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
                     49:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
                     50:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
                     51:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
                     52:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
                     53:  * SUCH DAMAGE.
                     54:  *
                     55:  *     @(#)subr_log.c  8.3 (Berkeley) 2/14/95
                     56:  */
                     57: 
                     58: /*
                     59:  * Error log buffer for kernel printf's.
                     60:  */
                     61: 
                     62: #include <sys/param.h>
                     63: #include <sys/systm.h>
                     64: #include <sys/proc.h>
                     65: #include <sys/vnode.h>
                     66: #include <sys/ioctl.h>
                     67: #include <sys/msgbuf.h>
                     68: #include <sys/file.h>
                     69: #include <sys/errno.h>
                     70: #include <sys/select.h>
                     71: #include <kern/thread.h>
                     72: #if 0 /* [ */
                     73: #include <kern/sched_prim.h>
                     74: #include <kern/parallel.h>
                     75: #endif  /* 0 ] */
                     76: 
                     77: #define LOG_RDPRI      (PZERO + 1)
                     78: 
                     79: #define LOG_NBIO       0x02
                     80: #define LOG_ASYNC      0x04
                     81: #define LOG_RDWAIT     0x08
                     82: 
                     83: struct logsoftc {
                     84:        int     sc_state;               /* see above for possibilities */
                     85:        struct  selinfo sc_selp;        /* thread waiting for select */
                     86:        int     sc_pgid;                /* process/group for async I/O */
                     87: } logsoftc;
                     88: 
                     89: int    log_open;                       /* also used in log() */
                     90: struct msgbuf temp_msgbuf;
                     91: struct msgbuf *msgbufp;
                     92: 
                     93: /*
                     94:  * Serialize log access.  Note that the log can be written at interrupt level,
                     95:  * so any log manipulations that can be done from, or affect, another processor
                     96:  * at interrupt level must be guarded with a spin lock.
                     97:  */
                     98: decl_simple_lock_data(,log_lock);      /* stop races dead in their tracks */
                     99: #define        LOG_LOCK()      simple_lock(&log_lock)
                    100: #define        LOG_UNLOCK()    simple_unlock(&log_lock)
                    101: #define        LOG_LOCK_INIT() simple_lock_init(&log_lock)
                    102: 
                    103: /*ARGSUSED*/
                    104: logopen(dev, flags, mode, p)
                    105:        dev_t dev;
                    106:        int flags, mode;
                    107:        struct proc *p;
                    108: {
                    109:        unix_master();                  /* for pg_id, sigh */
                    110:        LOG_LOCK();
                    111:        if (log_open) {
                    112:                LOG_UNLOCK();
                    113:                unix_release();
                    114:                return (EBUSY);
                    115:        }
                    116:        log_open = 1;
                    117:        logsoftc.sc_pgid = p->p_pid;            /* signal process only */
                    118:        /*
                    119:         * Potential race here with putchar() but since putchar should be
                    120:         * called by autoconf, msg_magic should be initialized by the time
                    121:         * we get here.
                    122:         */
                    123:        if (msgbufp->msg_magic != MSG_MAGIC) {
                    124:                register int i;
                    125: 
                    126:                msgbufp->msg_magic = MSG_MAGIC;
                    127:                msgbufp->msg_bufx = msgbufp->msg_bufr = 0;
                    128:                for (i=0; i < MSG_BSIZE; i++)
                    129:                        msgbufp->msg_bufc[i] = 0;
                    130:        }
                    131:        LOG_UNLOCK();
                    132:        unix_release();
                    133:        return (0);
                    134: }
                    135: 
                    136: /*ARGSUSED*/
                    137: int
                    138: logclose(dev, flag)
                    139:        dev_t dev;
                    140: {
                    141:        int oldpri;
                    142:        LOG_LOCK();
                    143:        log_open = 0;
                    144:        selwakeup(&logsoftc.sc_selp);
                    145:        oldpri = splhigh();
                    146:        selthreadclear(&logsoftc.sc_selp);
                    147:        splx(oldpri);
                    148:        LOG_UNLOCK();
                    149:        return (0);
                    150: }
                    151: 
                    152: /*ARGSUSED*/
                    153: int
                    154: logread(dev, uio, flag)
                    155:        dev_t dev;
                    156:        struct uio *uio;
                    157:        int flag;
                    158: {
                    159:        register long l;
                    160:        register int s;
                    161:        int error = 0;
                    162: 
                    163:        s = splhigh();
                    164:        while (msgbufp->msg_bufr == msgbufp->msg_bufx) {
                    165:                if (flag & IO_NDELAY) {
                    166:                        splx(s);
                    167:                        return (EWOULDBLOCK);
                    168:                }
                    169:                if (logsoftc.sc_state & LOG_NBIO) {
                    170:                        splx(s);
                    171:                        return (EWOULDBLOCK);
                    172:                }
                    173:                logsoftc.sc_state |= LOG_RDWAIT;
                    174:                if (error = tsleep((caddr_t)msgbufp, LOG_RDPRI | PCATCH,
                    175:                                "klog", 0)) {
                    176:                        splx(s);
                    177:                        return (error);
                    178:                }
                    179:        }
                    180:        splx(s);
                    181:        logsoftc.sc_state &= ~LOG_RDWAIT;
                    182: 
                    183:        while (uio->uio_resid > 0) {
                    184:                l = msgbufp->msg_bufx - msgbufp->msg_bufr;
                    185:                if (l < 0)
                    186:                        l = MSG_BSIZE - msgbufp->msg_bufr;
                    187:                l = min(l, uio->uio_resid);
                    188:                if (l == 0)
                    189:                        break;
                    190:                error = uiomove((caddr_t)&msgbufp->msg_bufc[msgbufp->msg_bufr],
                    191:                        (int)l, uio);
                    192:                if (error)
                    193:                        break;
                    194:                msgbufp->msg_bufr += l;
                    195:                if (msgbufp->msg_bufr < 0 || msgbufp->msg_bufr >= MSG_BSIZE)
                    196:                        msgbufp->msg_bufr = 0;
                    197:        }
                    198:        return (error);
                    199: }
                    200: 
                    201: /*ARGSUSED*/
                    202: int
                    203: logselect(dev, rw, p)
                    204:        dev_t dev;
                    205:        int rw;
                    206:        struct proc *p;
                    207: {
                    208:        int s = splhigh();
                    209: 
                    210:        switch (rw) {
                    211: 
                    212:        case FREAD:
                    213:                if (msgbufp->msg_bufr != msgbufp->msg_bufx) {
                    214:                        splx(s);
                    215:                        return (1);
                    216:                }
                    217:                selrecord(p, &logsoftc.sc_selp);
                    218:                break;
                    219:        }
                    220:        splx(s);
                    221:        return (0);
                    222: }
                    223: 
                    224: void
                    225: logwakeup()
                    226: {
                    227:        struct proc *p;
                    228:        int pgid;
                    229:        boolean_t funnel_state;
                    230: 
                    231:        if (!log_open)
                    232:                return;
                    233:        funnel_state = thread_set_funneled(TRUE);
                    234:        selwakeup(&logsoftc.sc_selp);
                    235:        if (logsoftc.sc_state & LOG_ASYNC) {
                    236:                unix_master();
                    237:                LOG_LOCK();
                    238:                pgid = logsoftc.sc_pgid;
                    239:                LOG_UNLOCK();
                    240:                if (pgid < 0)
                    241:                        gsignal(-pgid, SIGIO); 
                    242:                else if (p = pfind(pgid))
                    243:                        psignal(p, SIGIO);
                    244:                unix_release();
                    245:        }
                    246:        if (logsoftc.sc_state & LOG_RDWAIT) {
                    247:                wakeup((caddr_t)msgbufp);
                    248:                logsoftc.sc_state &= ~LOG_RDWAIT;
                    249:        }
                    250:        (void) thread_set_funneled(funnel_state);
                    251: }
                    252: 
                    253: /*ARGSUSED*/
                    254: int
                    255: logioctl(com, data, flag)
                    256:        caddr_t data;
                    257: {
                    258:        long l;
                    259:        int s;
                    260: 
                    261:        switch (com) {
                    262: 
                    263:        /* return number of characters immediately available */
                    264:        case FIONREAD:
                    265:                s = splhigh();
                    266:                l = msgbufp->msg_bufx - msgbufp->msg_bufr;
                    267:                splx(s);
                    268:                if (l < 0)
                    269:                        l += MSG_BSIZE;
                    270:                *(off_t *)data = l;
                    271:                break;
                    272: 
                    273:        case FIONBIO:
                    274:                if (*(int *)data)
                    275:                        logsoftc.sc_state |= LOG_NBIO;
                    276:                else
                    277:                        logsoftc.sc_state &= ~LOG_NBIO;
                    278:                break;
                    279: 
                    280:        case FIOASYNC:
                    281:                if (*(int *)data)
                    282:                        logsoftc.sc_state |= LOG_ASYNC;
                    283:                else
                    284:                        logsoftc.sc_state &= ~LOG_ASYNC;
                    285:                break;
                    286: 
                    287:        case TIOCSPGRP:
                    288:                LOG_LOCK();
                    289:                logsoftc.sc_pgid = *(int *)data;
                    290:                LOG_UNLOCK();
                    291:                break;
                    292: 
                    293:        case TIOCGPGRP:
                    294:                LOG_LOCK();
                    295:                *(int *)data = logsoftc.sc_pgid;
                    296:                LOG_UNLOCK();
                    297:                break;
                    298: 
                    299:        default:
                    300:                return (-1);
                    301:        }
                    302:        return (0);
                    303: }
                    304: 
                    305: void
                    306: log_init()
                    307: {
                    308:        msgbufp = &temp_msgbuf;
                    309:        LOG_LOCK_INIT();
                    310: }
                    311: 
                    312: void
                    313: log_putc(char c)
                    314: {
                    315:        register struct msgbuf *mbp;
                    316: 
                    317:        if (msgbufp == NULL)
                    318:                msgbufp =&temp_msgbuf;
                    319: 
                    320:        mbp = msgbufp; 
                    321:        if (mbp-> msg_magic != MSG_MAGIC) { 
                    322:                register int i;
                    323: 
                    324:                mbp->msg_magic = MSG_MAGIC;
                    325:                mbp->msg_bufx = mbp->msg_bufr = 0;
                    326:                for (i=0; i < MSG_BSIZE; i++)
                    327:                        mbp->msg_bufc[i] = 0;
                    328:        }
                    329:        mbp->msg_bufc[mbp->msg_bufx++] = c;
                    330:        if (mbp->msg_bufx < 0 || mbp->msg_bufx >= MSG_BSIZE)
                    331:                mbp->msg_bufx = 0;
                    332: }

unix.superglobalmegacorp.com

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