⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 vfprintf.c

📁 在x86平台上运行不可信任代码的sandbox。
💻 C
📖 第 1 页 / 共 2 页
字号:
/*- * Copyright (c) 1990 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. */#define FLOATING_POINT 1#include <math.h>#if defined(LIBC_SCCS) && !defined(lint)static char *rcsid = "$OpenBSD: vfprintf.c,v 1.8 1998/08/14 21:39:42 deraadt Exp $";#endif /* LIBC_SCCS and not lint *//* * Actual printf innards. * * This code is large and complicated... */#include <sys/types.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include <errno.h>#ifdef __STDC__#include <stdarg.h>#else#include <varargs.h>#endif#include "local.h"  #include "fvwrite.h"  static void __find_arguments __P((const char *fmt0, va_list ap,	va_list **argtable));static int __grow_type_table __P((unsigned char **typetable,	int *tablesize));/* * 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);}#ifdef FLOATING_POINT#include <locale.h>#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		40#endif /* FLOATING_POINT */#define STATIC_ARG_TBL_SIZE 8	/* Size of static argument table. *//* * 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')/* * 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; unimplemented */#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;	_BSD_VA_LIST_ ap;{	register char *fmt;	/* format string */	register int ch;	/* character from fmt */	register int n, m, n2;	/* handy integers (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) */	wchar_t wc;#ifdef FLOATING_POINT	char *decimal_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#ifdef __GNUC__			/* gcc has builtin quad type (long long) SOS */#define	quad_t	  long long#define	u_quad_t  unsigned long long#endif	u_quad_t _uquad;	/* integer arguments %[diouxX] */	enum { OCT, DEC, HEX } base;/* base for [diouxX] conversion */	int dprec;		/* a copy of prec if [diouxX], 0 otherwise */	int realsz;		/* field size expanded by dprec */	int size;		/* size of converted field or string */	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 */	va_list *argtable;        /* args, built due to positional arg */	va_list 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; \}	/*	 * To extend shorts properly, we need both signed and unsigned	 * argument extraction methods.	 */#define	SARG() \	(flags&QUADINT ? va_arg(ap, quad_t) : \	    flags&LONGINT ? GETARG(long) : \	    flags&SHORTINT ? (long)(short)GETARG(int) : \	    (long)GETARG(int))#define	UARG() \	(flags&QUADINT ? va_arg(ap, u_quad_t) : \	    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); \	 }/** 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) ? (void)(ap = argtable[nextarg]) : (void)0), \	 nextarg++, va_arg(ap, type)) 	/* sorry, fprintf(read_only_file, "") returns EOF, not 0 */	if (cantwrite(fp)) {		errno = EBADF;		return (EOF);	}	/* optimise fprintf(stderr) (and other unbuffered Unix files) */	if ((fp->_flags & (__SNBF|__SWR|__SRW)) == (__SNBF|__SWR) &&	    fp->_file >= 0)		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 (;;) {		cp = fmt;		while ((wc = *fmt) != 0 && wc != '%')			fmt++;		if ((m = fmt - cp) != 0) {			PRINT(cp, m);			ret += m;		}		if (wc == 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++;			}			if (ch == '$') {				nextarg = n;				if (argtable == NULL) {					argtable = statargtable;					__find_arguments(fmt0, orgap,					    &argtable);				}				goto rflag;			}			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 (*fmt == 'l') {				fmt++;				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':			_uquad = SARG();			if ((quad_t)_uquad < 0) {				_uquad = -_uquad;				sign = '-';			}			base = DEC;			goto number;#ifdef FLOATING_POINT		case 'e':		case 'E':		case 'f':		case 'g':		case 'G':			if (prec == -1) {				prec = DEFPREC;			} else if ((ch == 'g' || ch == 'G') && prec == 0) {				prec = 1;			}			if (flags & LONGDBL) {				_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)				sign = '-';			break;#endif /* FLOATING_POINT */		case 'n':			if (flags & QUADINT)				*GETARG(quad_t *) = ret;			else if (flags & LONGINT)				*GETARG(long *) = ret;			else if (flags & SHORTINT)				*GETARG(short *) = ret;			else				*GETARG(int *) = ret;			continue;	/* no output */		case 'O':			flags |= LONGINT;			/*FALLTHROUGH*/		case 'o':			_uquad = UARG();			base = OCT;			goto nosign;		case 'p':			/*			 * ``The argument shall be a pointer to void.  The			 * value of the pointer is converted to a sequence			 * of printable characters, in an implementation-			 * defined manner.''			 *	-- ANSI X3J11			 */			/* NOSTRICT */			_uquad = (u_long)GETARG(void *);			base = HEX;			xdigs = "0123456789abcdef";			flags |= HEXPREFIX;			ch = 'x';			goto nosign;		case 's':			if ((cp = GETARG(char *)) == NULL)				cp = "(null)";			if (prec >= 0) {				/*				 * can't use strlen; can only look for the				 * NUL in the first `prec' characters, and				 * strlen() will go further.				 */				char *p = memchr(cp, 0, prec);				if (p != NULL) {					size = p - cp;					if (size > prec)						size = prec;				} else					size = prec;			} else				size = strlen(cp);			sign = '\0';			break;		case 'U':			flags |= LONGINT;			/*FALLTHROUGH*/		case 'u':			_uquad = UARG();			base = DEC;			goto nosign;

⌨️ 快捷键说明

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