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

📄 vfprintf.c

📁 早期freebsd实现
💻 C
📖 第 1 页 / 共 2 页
字号:
/*- * 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)static char sccsid[] = "@(#)vfprintf.c	8.1 (Berkeley) 6/4/93";#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"/* Define FLOATING_POINT to get floating point. */#define	FLOATING_POINT/* * 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 *//* * 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;	va_list ap;{	register char *fmt;	/* format string */	register int ch;	/* character from fmt */	register int n;		/* 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 fieldsz;		/* field size expanded by sign, etc */	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 */	/*	 * 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&LONGINT ? va_arg(ap, long) : \	    flags&SHORTINT ? (long)(short)va_arg(ap, int) : \	    (long)va_arg(ap, int))#define	UARG() \	(flags&LONGINT ? va_arg(ap, u_long) : \	    flags&SHORTINT ? (u_long)(u_short)va_arg(ap, int) : \	    (u_long)va_arg(ap, u_int))	/* sorry, fprintf(read_only_file, "") returns EOF, not 0 */	if (cantwrite(fp))		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;	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) {			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.			 */			if ((width = va_arg(ap, int)) >= 0)				goto rflag;			width = -width;			/* FALLTHROUGH */		case '-':			flags |= LADJUST;

⌨️ 快捷键说明

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