subr_prf.c

来自「基于组件方式开发操作系统的OSKIT源代码」· C语言 代码 · 共 1,296 行 · 第 1/2 页

C
1,296
字号
/*	$NetBSD: subr_prf.c,v 1.76 2000/08/09 10:22:31 tv Exp $	*//*- * Copyright (c) 1986, 1988, 1991, 1993 *	The Regents of the University of California.  All rights reserved. * (c) UNIX System Laboratories, Inc. * All or some portions of this file are derived from material licensed * to the University of California by American Telephone and Telegraph * Co. or Unix System Laboratories, Inc. and are reproduced herein with * the permission of UNIX System Laboratories, Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright *    notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright *    notice, this list of conditions and the following disclaimer in the *    documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software *    must display the following acknowledgement: *	This product includes software developed by the University of *	California, Berkeley and its contributors. * 4. Neither the name of the University nor the names of its contributors *    may be used to endorse or promote products derived from this software *    without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * *	@(#)subr_prf.c	8.4 (Berkeley) 5/4/95 */#include "opt_ddb.h"#include "opt_ipkdb.h"#include "opt_multiprocessor.h"#include <sys/param.h>#include <sys/systm.h>#include <sys/buf.h>#include <sys/reboot.h>#include <sys/msgbuf.h>#include <sys/proc.h>#include <sys/ioctl.h>#include <sys/vnode.h>#include <sys/file.h>#include <sys/tty.h>#include <sys/tprintf.h>#include <sys/syslog.h>#include <sys/malloc.h>#include <sys/lock.h>#ifndef OSKIT#include <dev/cons.h>#endif#ifdef DDB#include <ddb/ddbvar.h>#include <machine/db_machdep.h>#include <ddb/db_command.h>#include <ddb/db_interface.h>#endif#ifdef IPKDB#include <ipkdb/ipkdb.h>#endif#if defined(MULTIPROCESSOR)struct simplelock kprintf_slock = SIMPLELOCK_INITIALIZER;/* * Use cpu_simple_lock() and cpu_simple_unlock().  These are the actual * atomic locking operations, and never attempt to print debugging * information. */#define	KPRINTF_MUTEX_ENTER(s)						\do {									\	(s) = splhigh();						\	__cpu_simple_lock(&kprintf_slock.lock_data);			\} while (0)#define	KPRINTF_MUTEX_EXIT(s)						\do {									\	__cpu_simple_unlock(&kprintf_slock.lock_data);			\	splx((s));							\} while (0)#else /* ! MULTIPROCESSOR */#define	KPRINTF_MUTEX_ENTER(s)	(s) = splhigh()#define	KPRINTF_MUTEX_EXIT(s)	splx((s))#endif /* MULTIPROCESSOR *//* * note that stdarg.h and the ansi style va_start macro is used for both * ansi and traditional c complers. * XXX: this requires that stdarg.h define: va_alist and va_dcl */#include <machine/stdarg.h>#ifdef KGDB#include <sys/kgdb.h>#include <machine/cpu.h>#endif#ifdef DDB#include <ddb/db_output.h>	/* db_printf, db_putchar prototypes */#endif/* * defines *//* flags for kprintf */#define TOCONS		0x01	/* to the console */#define TOTTY		0x02	/* to the process' tty */#define TOLOG		0x04	/* to the kernel message buffer */#define TOBUFONLY	0x08	/* to the buffer (only) [for snprintf] */#define TODDB		0x10	/* to ddb console *//* max size buffer kprintf needs to print quad_t [size in base 8 + \0] */#define KPRINTF_BUFSIZE		(sizeof(quad_t) * NBBY / 3 + 2)#ifndef OSKIT/* * local prototypes */static int	 kprintf __P((const char *, int, void *, 				char *, va_list));static void	 putchar __P((int, int, struct tty *));static void	 klogpri __P((int));/* * globals */struct	tty *constty;	/* pointer to console "window" tty */extern	int log_open;	/* subr_log: is /dev/klog open? */#endif /*OSKIT*/const	char *panicstr; /* arg to first call to panic (used as a flag			   to indicate that panic has already been called). */int	doing_shutdown;	/* set to indicate shutdown in progress */#ifndef OSKIT/* * v_putc: routine to putc on virtual console * * the v_putc pointer can be used to redirect the console cnputc elsewhere * [e.g. to a "virtual console"]. */void (*v_putc) __P((int)) = cnputc;	/* start with cnputc (normal cons) *//* * functions *//* * tablefull: warn that a system table is full */voidtablefull(tab, hint)	const char *tab, *hint;{	if (hint)		log(LOG_ERR, "%s: table is full - %s\n", tab, hint);	else		log(LOG_ERR, "%s: table is full\n", tab);}/* * panic: handle an unresolvable fatal error * * prints "panic: <message>" and reboots.   if called twice (i.e. recursive * call) we avoid trying to sync the disk and just reboot (to avoid  * recursive panics). */void#ifdef __STDC__panic(const char *fmt, ...)#elsepanic(fmt, va_alist)	char *fmt;	va_dcl#endif{	int bootopt;	va_list ap;	bootopt = RB_AUTOBOOT | RB_DUMP;	if (doing_shutdown)		bootopt |= RB_NOSYNC;	if (!panicstr)		panicstr = fmt;	doing_shutdown = 1;		va_start(ap, fmt);	printf("panic: ");	vprintf(fmt, ap);	printf("\n");	va_end(ap);#ifdef IPKDB	ipkdb_panic();#endif#ifdef KGDB	kgdb_panic();#endif#ifdef KADB	if (boothowto & RB_KDB)		kdbpanic();#endif#ifdef DDB	if (db_onpanic)		Debugger();	else {		static int intrace = 0;		if (intrace==0) {			intrace=1;			printf("Begin traceback...\n");			db_stack_trace_print(			    (db_expr_t)__builtin_frame_address(0),			    TRUE, 65535, "", printf);			printf("End traceback...\n");			intrace=0;		} else			printf("Faulted in mid-traceback; aborting...");	}#endif	cpu_reboot(bootopt, NULL);}/* * kernel logging functions: log, logpri, addlog *//* * log: write to the log buffer * * => will not sleep [so safe to call from interrupt] * => will log to console if /dev/klog isn't open */void#ifdef __STDC__log(int level, const char *fmt, ...)#elselog(level, fmt, va_alist)	int level;	char *fmt;	va_dcl#endif{	int s;	va_list ap;	KPRINTF_MUTEX_ENTER(s);	klogpri(level);		/* log the level first */	va_start(ap, fmt);	kprintf(fmt, TOLOG, NULL, NULL, ap);	va_end(ap);	if (!log_open) {		va_start(ap, fmt);		kprintf(fmt, TOCONS, NULL, NULL, ap);		va_end(ap);	}	KPRINTF_MUTEX_EXIT(s);	logwakeup();		/* wake up anyone waiting for log msgs */}/* * vlog: write to the log buffer [already have va_alist] */voidvlog(level, fmt, ap)	int level;	const char *fmt;	va_list ap;{	int s;	KPRINTF_MUTEX_ENTER(s);	klogpri(level);		/* log the level first */	kprintf(fmt, TOLOG, NULL, NULL, ap);	if (!log_open)		kprintf(fmt, TOCONS, NULL, NULL, ap);	KPRINTF_MUTEX_EXIT(s);	logwakeup();		/* wake up anyone waiting for log msgs */}/* * logpri: log the priority level to the klog */voidlogpri(level)	int level;{	int s;	KPRINTF_MUTEX_ENTER(s);	klogpri(level);	KPRINTF_MUTEX_EXIT(s);}/* * Note: we must be in the mutex here! */static voidklogpri(level)	int level;{	char *p;	char snbuf[KPRINTF_BUFSIZE];	putchar('<', TOLOG, NULL);	snprintf(snbuf, sizeof(snbuf), "%d", level);	for (p = snbuf ; *p ; p++)		putchar(*p, TOLOG, NULL);	putchar('>', TOLOG, NULL);}/* * addlog: add info to previous log message */void#ifdef __STDC__addlog(const char *fmt, ...)#elseaddlog(fmt, va_alist)	char *fmt;	va_dcl#endif{	int s;	va_list ap;	KPRINTF_MUTEX_ENTER(s);	va_start(ap, fmt);	kprintf(fmt, TOLOG, NULL, NULL, ap);	va_end(ap);	if (!log_open) {		va_start(ap, fmt);		kprintf(fmt, TOCONS, NULL, NULL, ap);		va_end(ap);	}	KPRINTF_MUTEX_EXIT(s);	logwakeup();}/* * putchar: print a single character on console or user terminal. * * => if console, then the last MSGBUFS chars are saved in msgbuf *	for inspection later (e.g. dmesg/syslog) * => we must already be in the mutex! */static voidputchar(c, flags, tp)	int c;	int flags;	struct tty *tp;{	struct kern_msgbuf *mbp;	if (panicstr)		constty = NULL;	if ((flags & TOCONS) && tp == NULL && constty) {		tp = constty;		flags |= TOTTY;	}	if ((flags & TOTTY) && tp && tputchar(c, tp) < 0 &&	    (flags & TOCONS) && tp == constty)		constty = NULL;	if ((flags & TOLOG) &&	    c != '\0' && c != '\r' && c != 0177 && msgbufenabled) {		mbp = msgbufp;		if (mbp->msg_magic != MSG_MAGIC) {			/*			 * Arguably should panic or somehow notify the			 * user...  but how?  Panic may be too drastic,			 * and would obliterate the message being kicked			 * out (maybe a panic itself), and printf			 * would invoke us recursively.  Silently punt			 * for now.  If syslog is running, it should			 * notice.			 */			msgbufenabled = 0;		} else {			mbp->msg_bufc[mbp->msg_bufx++] = c;			if (mbp->msg_bufx < 0 || mbp->msg_bufx >= mbp->msg_bufs)				mbp->msg_bufx = 0;			/* If the buffer is full, keep the most recent data. */			if (mbp->msg_bufr == mbp->msg_bufx) {				 if (++mbp->msg_bufr >= mbp->msg_bufs)					mbp->msg_bufr = 0;			}		}	}	if ((flags & TOCONS) && constty == NULL && c != '\0')		(*v_putc)(c);#ifdef DDB	if (flags & TODDB)		db_putchar(c);#endif}/* * uprintf: print to the controlling tty of the current process * * => we may block if the tty queue is full * => no message is printed if the queue doesn't clear in a reasonable *	time */void#ifdef __STDC__uprintf(const char *fmt, ...)#elseuprintf(fmt, va_alist)	char *fmt;	va_dcl#endif{	struct proc *p = curproc;	va_list ap;	if (p->p_flag & P_CONTROLT && p->p_session->s_ttyvp) {		/* No mutex needed; going to process TTY. */		va_start(ap, fmt);		kprintf(fmt, TOTTY, p->p_session->s_ttyp, NULL, ap);		va_end(ap);	}}/* * tprintf functions: used to send messages to a specific process * * usage: *   get a tpr_t handle on a process "p" by using "tprintf_open(p)" *   use the handle when calling "tprintf" *   when done, do a "tprintf_close" to drop the handle *//* * tprintf_open: get a tprintf handle on a process "p" * * => returns NULL if process can't be printed to */tpr_ttprintf_open(p)	struct proc *p;{	if (p->p_flag & P_CONTROLT && p->p_session->s_ttyvp) {		SESSHOLD(p->p_session);		return ((tpr_t) p->p_session);	}	return ((tpr_t) NULL);}/* * tprintf_close: dispose of a tprintf handle obtained with tprintf_open */voidtprintf_close(sess)	tpr_t sess;{	if (sess)		SESSRELE((struct session *) sess);}/* * tprintf: given tprintf handle to a process [obtained with tprintf_open],  * send a message to the controlling tty for that process. * * => also sends message to /dev/klog */void#ifdef __STDC__tprintf(tpr_t tpr, const char *fmt, ...)#elsetprintf(tpr, fmt, va_alist)	tpr_t tpr;	char *fmt;	va_dcl#endif{	struct session *sess = (struct session *)tpr;	struct tty *tp = NULL;	int s, flags = TOLOG;	va_list ap;	if (sess && sess->s_ttyvp && ttycheckoutq(sess->s_ttyp, 0)) {		flags |= TOTTY;		tp = sess->s_ttyp;	}	KPRINTF_MUTEX_ENTER(s);	klogpri(LOG_INFO);	va_start(ap, fmt);	kprintf(fmt, flags, tp, NULL, ap);	va_end(ap);	KPRINTF_MUTEX_EXIT(s);	logwakeup();}/* * ttyprintf: send a message to a specific tty * * => should be used only by tty driver or anything that knows the *    underlying tty will not be revoked(2)'d away.  [otherwise, *    use tprintf] */void#ifdef __STDC__ttyprintf(struct tty *tp, const char *fmt, ...)#elsettyprintf(tp, fmt, va_alist)	struct tty *tp;	char *fmt;	va_dcl#endif{	va_list ap;	/* No mutex needed; going to process TTY. */	va_start(ap, fmt);	kprintf(fmt, TOTTY, tp, NULL, ap);	va_end(ap);}#ifdef DDB/* * db_printf: printf for DDB (via db_putchar) */void#ifdef __STDC__db_printf(const char *fmt, ...)#elsedb_printf(fmt, va_alist)	char *fmt;	va_dcl#endif{	va_list ap;	/* No mutex needed; DDB pauses all processors. */	va_start(ap, fmt);	kprintf(fmt, TODDB, NULL, NULL, ap);	va_end(ap);}#endif /* DDB *//* * normal kernel printf functions: printf, vprintf, snprintf, vsnprintf *//* * printf: print a message to the console and the log */void#ifdef __STDC__printf(const char *fmt, ...)#elseprintf(fmt, va_alist)	char *fmt;	va_dcl#endif{	va_list ap;	int s;	KPRINTF_MUTEX_ENTER(s);	va_start(ap, fmt);	kprintf(fmt, TOCONS | TOLOG, NULL, NULL, ap);	va_end(ap);	KPRINTF_MUTEX_EXIT(s);	if (!panicstr)		logwakeup();}/* * vprintf: print a message to the console and the log [already have *	va_alist] */voidvprintf(fmt, ap)	const char *fmt;	va_list ap;{	int s;	KPRINTF_MUTEX_ENTER(s);	kprintf(fmt, TOCONS | TOLOG, NULL, NULL, ap);	KPRINTF_MUTEX_EXIT(s);	if (!panicstr)		logwakeup();}/* * sprintf: print a message to a buffer

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?