vfprintf.c

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

C
1,279
字号
/*- * Copyright (c) 1990, 1993 *	The Regents of the University of California.  All rights reserved. * * This code is derived from software contributed to Berkeley by * Chris Torek. * * 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. */#if defined(LIBC_SCCS) && !defined(lint)#if 0static char sccsid[] = "@(#)vfprintf.c	8.1 (Berkeley) 6/4/93";#endifstatic const char rcsid[] =		"$Id: vfprintf.c,v 1.20.2.1 1999/05/04 13:06:29 dt Exp $";#endif /* LIBC_SCCS and not lint *//* * Actual printf innards. * * This code is large and complicated... */#include <sys/types.h>#include <limits.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#if __STDC__#include <stdarg.h>#else#include <varargs.h>#endif#include "local.h"#include "fvwrite.h"#include "libc_private.h"/* Define FLOATING_POINT to get floating point. */#define	FLOATING_POINTstatic int	__sprint __P((FILE *, struct __suio *));static int	__sbprintf __P((FILE *, const char *, va_list));static char *	__ultoa __P((u_long, char *, int, int, char *));static char *	__uqtoa __P((u_quad_t, char *, int, int, char *));static void	__find_arguments __P((const char *, va_list, void ***));static void	__grow_type_table __P((int, unsigned char **, int *));/* * Flush out all the vectors defined by the given uio, * then reset it so that it can be reused. */static int__sprint(fp, uio)	FILE *fp;	register struct __suio *uio;{	register int err;	if (uio->uio_resid == 0) {		uio->uio_iovcnt = 0;		return (0);	}	err = __sfvwrite(fp, uio);	uio->uio_resid = 0;	uio->uio_iovcnt = 0;	return (err);}/* * Helper function for `fprintf to unbuffered unix file': creates a * temporary buffer.  We only work on write-only files; this avoids * worries about ungetc buffers and so forth. */static int__sbprintf(fp, fmt, ap)	register FILE *fp;	const char *fmt;	va_list ap;{	int ret;	FILE fake;	unsigned char buf[BUFSIZ];	/* copy the important variables */	fake._flags = fp->_flags & ~__SNBF;	fake._file = fp->_file;	fake._cookie = fp->_cookie;	fake._write = fp->_write;	/* set up the buffer */	fake._bf._base = fake._p = buf;	fake._bf._size = fake._w = sizeof(buf);	fake._lbfsize = 0;	/* not actually used, but Just In Case */	/* do the work, then copy any error status */	ret = vfprintf(&fake, fmt, ap);	if (ret >= 0 && fflush(&fake))		ret = EOF;	if (fake._flags & __SERR)		fp->_flags |= __SERR;	return (ret);}/* * Macros for converting digits to letters and vice versa */#define	to_digit(c)	((c) - '0')#define is_digit(c)	((unsigned)to_digit(c) <= 9)#define	to_char(n)	((n) + '0')/* * Convert an unsigned long to ASCII for printf purposes, returning * a pointer to the first character of the string representation. * Octal numbers can be forced to have a leading zero; hex numbers * use the given digits. */static char *__ultoa(val, endp, base, octzero, xdigs)	register u_long val;	char *endp;	int base, octzero;	char *xdigs;{	register char *cp = endp;	register long sval;	/*	 * Handle the three cases separately, in the hope of getting	 * better/faster code.	 */	switch (base) {	case 10:		if (val < 10) {	/* many numbers are 1 digit */			*--cp = to_char(val);			return (cp);		}		/*		 * On many machines, unsigned arithmetic is harder than		 * signed arithmetic, so we do at most one unsigned mod and		 * divide; this is sufficient to reduce the range of		 * the incoming value to where signed arithmetic works.		 */		if (val > LONG_MAX) {			*--cp = to_char(val % 10);			sval = val / 10;		} else			sval = val;		do {			*--cp = to_char(sval % 10);			sval /= 10;		} while (sval != 0);		break;	case 8:		do {			*--cp = to_char(val & 7);			val >>= 3;		} while (val);		if (octzero && *cp != '0')			*--cp = '0';		break;	case 16:		do {			*--cp = xdigs[val & 15];			val >>= 4;		} while (val);		break;	default:			/* oops */		abort();	}	return (cp);}/* Identical to __ultoa, but for quads. */static char *__uqtoa(val, endp, base, octzero, xdigs)	register u_quad_t val;	char *endp;	int base, octzero;	char *xdigs;{	register char *cp = endp;	register quad_t sval;	/* quick test for small values; __ultoa is typically much faster */	/* (perhaps instead we should run until small, then call __ultoa?) */	if (val <= ULONG_MAX)		return (__ultoa((u_long)val, endp, base, octzero, xdigs));	switch (base) {	case 10:		if (val < 10) {			*--cp = to_char(val % 10);			return (cp);		}		if (val > QUAD_MAX) {			*--cp = to_char(val % 10);			sval = val / 10;		} else			sval = val;		do {			*--cp = to_char(sval % 10);			sval /= 10;		} while (sval != 0);		break;	case 8:		do {			*--cp = to_char(val & 7);			val >>= 3;		} while (val);		if (octzero && *cp != '0')			*--cp = '0';		break;	case 16:		do {			*--cp = xdigs[val & 15];			val >>= 4;		} while (val);		break;	default:		abort();	}	return (cp);}#ifdef FLOATING_POINT#include <math.h>#include "floatio.h"#define	BUF		(MAXEXP+MAXFRACT+1)	/* + decimal point */#define	DEFPREC		6static char *cvt __P((double, int, int, char *, int *, int, int *));static int exponent __P((char *, int, int));#else /* no FLOATING_POINT */#define	BUF		68#endif /* FLOATING_POINT */#define STATIC_ARG_TBL_SIZE 8           /* Size of static argument table. *//* * Flags used during conversion. */#define	ALT		0x001		/* alternate form */#define	HEXPREFIX	0x002		/* add 0x or 0X prefix */#define	LADJUST		0x004		/* left adjustment */#define	LONGDBL		0x008		/* long double */#define	LONGINT		0x010		/* long integer */#define	QUADINT		0x020		/* quad integer */#define	SHORTINT	0x040		/* short integer */#define	ZEROPAD		0x080		/* zero (as opposed to blank) pad */#define FPT		0x100		/* Floating point number */intvfprintf(fp, fmt0, ap)	FILE *fp;	const char *fmt0;	va_list ap;{	register char *fmt;	/* format string */	register int ch;	/* character from fmt */	register int n, n2;	/* handy integer (short term usage) */	register char *cp;	/* handy char pointer (short term usage) */	register struct __siov *iovp;/* for PRINT macro */	register int flags;	/* flags as above */	int ret;		/* return value accumulator */	int width;		/* width from format (%8d), or 0 */	int prec;		/* precision from format (%.3d), or -1 */	char sign;		/* sign prefix (' ', '+', '-', or \0) */#ifdef FLOATING_POINT	char softsign;		/* temporary negative sign for floats */	double _double;		/* double precision arguments %[eEfgG] */	int expt;		/* integer value of exponent */	int expsize;		/* character count for expstr */	int ndig;		/* actual number of digits returned by cvt */	char expstr[7];		/* buffer for exponent string */#endif	u_long	ulval;		/* integer arguments %[diouxX] */	u_quad_t uqval;		/* %q integers */	int base;		/* base for [diouxX] conversion */	int dprec;		/* a copy of prec if [diouxX], 0 otherwise */	int realsz;		/* field size expanded by dprec, sign, etc */	int size;		/* size of converted field or string */	int prsize;             /* max size of printed field */	char *xdigs;		/* digits for [xX] conversion */#define NIOV 8	struct __suio uio;	/* output information: summary */	struct __siov iov[NIOV];/* ... and individual io vectors */	char buf[BUF];		/* space for %c, %[diouxX], %[eEfgG] */	char ox[2];		/* space for 0x hex-prefix */        void **argtable;        /* args, built due to positional arg */        void *statargtable [STATIC_ARG_TBL_SIZE];        int nextarg;            /* 1-based argument index */        va_list orgap;          /* original argument pointer */	/*	 * Choose PADSIZE to trade efficiency vs. size.  If larger printf	 * fields occur frequently, increase PADSIZE and make the initialisers	 * below longer.	 */#define	PADSIZE	16		/* pad chunk size */	static char blanks[PADSIZE] =	 {' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' '};	static char zeroes[PADSIZE] =	 {'0','0','0','0','0','0','0','0','0','0','0','0','0','0','0','0'};	/*	 * BEWARE, these `goto error' on error, and PAD uses `n'.	 */#define	PRINT(ptr, len) { \	iovp->iov_base = (ptr); \	iovp->iov_len = (len); \	uio.uio_resid += (len); \	iovp++; \	if (++uio.uio_iovcnt >= NIOV) { \		if (__sprint(fp, &uio)) \			goto error; \		iovp = iov; \	} \}#define	PAD(howmany, with) { \	if ((n = (howmany)) > 0) { \		while (n > PADSIZE) { \			PRINT(with, PADSIZE); \			n -= PADSIZE; \		} \		PRINT(with, n); \	} \}#define	FLUSH() { \	if (uio.uio_resid && __sprint(fp, &uio)) \		goto error; \	uio.uio_iovcnt = 0; \	iovp = iov; \}        /*         * Get the argument indexed by nextarg.   If the argument table is         * built, use it to get the argument.  If its not, get the next         * argument (and arguments must be gotten sequentially).         */#define GETARG(type) \        ((argtable != NULL) ? *((type*)(argtable[nextarg++])) : \            (nextarg++, va_arg(ap, type)))	/*	 * To extend shorts properly, we need both signed and unsigned	 * argument extraction methods.	 */#define	SARG() \	(flags&LONGINT ? GETARG(long) : \	    flags&SHORTINT ? (long)(short)GETARG(int) : \	    (long)GETARG(int))#define	UARG() \	(flags&LONGINT ? GETARG(u_long) : \	    flags&SHORTINT ? (u_long)(u_short)GETARG(int) : \	    (u_long)GETARG(u_int))        /*         * Get * arguments, including the form *nn$.  Preserve the nextarg         * that the argument can be gotten once the type is determined.         */#define GETASTER(val) \        n2 = 0; \        cp = fmt; \        while (is_digit(*cp)) { \                n2 = 10 * n2 + to_digit(*cp); \                cp++; \        } \        if (*cp == '$') { \            	int hold = nextarg; \                if (argtable == NULL) { \                        argtable = statargtable; \                        __find_arguments (fmt0, orgap, &argtable); \                } \                nextarg = n2; \                val = GETARG (int); \                nextarg = hold; \                fmt = ++cp; \        } else { \		val = GETARG (int); \        }        	FLOCKFILE(fp);	/* sorry, fprintf(read_only_file, "") returns EOF, not 0 */	if (cantwrite(fp)) {		FUNLOCKFILE(fp);		return (EOF);	}	/* optimise fprintf(stderr) (and other unbuffered Unix files) */	if ((fp->_flags & (__SNBF|__SWR|__SRW)) == (__SNBF|__SWR) &&	    fp->_file >= 0) {		FUNLOCKFILE(fp);		return (__sbprintf(fp, fmt0, ap));	}	fmt = (char *)fmt0;        argtable = NULL;        nextarg = 1;        orgap = ap;	uio.uio_iov = iovp = iov;	uio.uio_resid = 0;	uio.uio_iovcnt = 0;	ret = 0;	/*	 * Scan the format for conversions (`%' character).	 */	for (;;) {		for (cp = fmt; (ch = *fmt) != '\0' && ch != '%'; fmt++)			/* void */;		if ((n = fmt - cp) != 0) {			if ((unsigned)ret + n > INT_MAX) {				ret = EOF;				goto error;			}			PRINT(cp, n);			ret += n;		}		if (ch == '\0')			goto done;		fmt++;		/* skip over '%' */		flags = 0;		dprec = 0;		width = 0;		prec = -1;		sign = '\0';rflag:		ch = *fmt++;reswitch:	switch (ch) {		case ' ':			/*			 * ``If the space and + flags both appear, the space			 * flag will be ignored.''			 *	-- ANSI X3J11			 */			if (!sign)				sign = ' ';			goto rflag;		case '#':			flags |= ALT;			goto rflag;		case '*':			/*			 * ``A negative field width argument is taken as a			 * - flag followed by a positive field width.''			 *	-- ANSI X3J11			 * They don't exclude field widths read from args.			 */			GETASTER (width);			if (width >= 0)				goto rflag;			width = -width;			/* FALLTHROUGH */		case '-':			flags |= LADJUST;			goto rflag;		case '+':			sign = '+';			goto rflag;		case '.':			if ((ch = *fmt++) == '*') {				GETASTER (n);				prec = n < 0 ? -1 : n;				goto rflag;			}			n = 0;			while (is_digit(ch)) {				n = 10 * n + to_digit(ch);				ch = *fmt++;			}			prec = n < 0 ? -1 : n;			goto reswitch;		case '0':			/*			 * ``Note that 0 is taken as a flag, not as the			 * beginning of a field width.''			 *	-- ANSI X3J11			 */			flags |= ZEROPAD;			goto rflag;		case '1': case '2': case '3': case '4':		case '5': case '6': case '7': case '8': case '9':			n = 0;			do {				n = 10 * n + to_digit(ch);				ch = *fmt++;			} while (is_digit(ch));			if (ch == '$') {				nextarg = n;                        	if (argtable == NULL) {                                	argtable = statargtable;                                	__find_arguments (fmt0, orgap,						&argtable);				}				goto rflag;                        }			width = n;			goto reswitch;#ifdef FLOATING_POINT		case 'L':			flags |= LONGDBL;			goto rflag;#endif		case 'h':			flags |= SHORTINT;			goto rflag;		case 'l':			if (flags & LONGINT)				flags |= QUADINT;			else				flags |= LONGINT;			goto rflag;		case 'q':			flags |= QUADINT;			goto rflag;		case 'c':			*(cp = buf) = GETARG(int);			size = 1;			sign = '\0';			break;		case 'D':			flags |= LONGINT;			/*FALLTHROUGH*/		case 'd':		case 'i':			if (flags & QUADINT) {				uqval = GETARG(quad_t);				if ((quad_t)uqval < 0) {					uqval = -uqval;					sign = '-';				}			} else {				ulval = SARG();				if ((long)ulval < 0) {					ulval = -ulval;					sign = '-';				}			}			base = 10;			goto number;#ifdef FLOATING_POINT		case 'e':		case 'E':		case 'f':			goto fp_begin;		case 'g':		case 'G':			if (prec == 0)				prec = 1;fp_begin:		if (prec == -1)				prec = DEFPREC;			if (flags & LONGDBL)				/* XXX this loses precision. */				_double = (double)GETARG(long double);			else				_double = GETARG(double);			/* do this before tricky precision changes */			if (isinf(_double)) {				if (_double < 0)					sign = '-';				cp = "Inf";				size = 3;				break;			}			if (isnan(_double)) {				cp = "NaN";				size = 3;				break;			}			flags |= FPT;			cp = cvt(_double, prec, flags, &softsign,				&expt, ch, &ndig);			if (ch == 'g' || ch == 'G') {				if (expt <= -4 || expt > prec)					ch = (ch == 'g') ? 'e' : 'E';				else					ch = 'g';			}			if (ch <= 'e') {	/* 'e' or 'E' fmt */				--expt;				expsize = exponent(expstr, expt, ch);				size = expsize + ndig;				if (ndig > 1 || flags & ALT)					++size;			} else if (ch == 'f') {		/* f fmt */				if (expt > 0) {					size = expt;					if (prec || flags & ALT)						size += prec + 1;				} else	/* "0.X" */					size = prec + 2;			} else if (expt >= ndig) {	/* fixed g fmt */				size = expt;				if (flags & ALT)					++size;			} else				size = ndig + (expt > 0 ?					1 : 2 - expt);			if (softsign)

⌨️ 快捷键说明

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