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

📄 vfprintf.c

📁 早期freebsd实现
💻 C
📖 第 1 页 / 共 2 页
字号:
			goto rflag;		case '+':			sign = '+';			goto rflag;		case '.':			if ((ch = *fmt++) == '*') {				n = va_arg(ap, int);				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));			width = n;			goto reswitch;#ifdef FLOATING_POINT		case 'L':			flags |= LONGDBL;			goto rflag;#endif		case 'h':			flags |= SHORTINT;			goto rflag;		case 'l':			flags |= LONGINT;			goto rflag;		case 'q':			flags |= QUADINT;			goto rflag;		case 'c':			*(cp = buf) = va_arg(ap, int);			size = 1;			sign = '\0';			break;		case 'D':			flags |= LONGINT;			/*FALLTHROUGH*/		case 'd':		case 'i':			if (flags & QUADINT) {				uqval = va_arg(ap, 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':		/* anomalous precision */		case 'E':			prec = (prec == -1) ?				DEFPREC + 1 : prec + 1;			/* FALLTHROUGH */			goto fp_begin;		case 'f':		/* always print trailing zeroes */			if (prec != 0)				flags |= ALT;		case 'g':		case 'G':			if (prec == -1)				prec = DEFPREC;fp_begin:		_double = va_arg(ap, 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)				*va_arg(ap, quad_t *) = ret;			else if (flags & LONGINT)				*va_arg(ap, long *) = ret;			else if (flags & SHORTINT)				*va_arg(ap, short *) = ret;			else				*va_arg(ap, int *) = ret;			continue;	/* no output */		case 'O':			flags |= LONGINT;			/*FALLTHROUGH*/		case 'o':			if (flags & QUADINT)				uqval = va_arg(ap, u_quad_t);			else				ulval = UARG();			base = 8;			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			 */			ulval = (u_long)va_arg(ap, void *);			base = 16;			xdigs = "0123456789abcdef";			flags = (flags & ~QUADINT) | HEXPREFIX;			ch = 'x';			goto nosign;		case 's':			if ((cp = va_arg(ap, 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':			if (flags & QUADINT)				uqval = va_arg(ap, u_quad_t);			else				ulval = UARG();			base = 10;			goto nosign;		case 'X':			xdigs = "0123456789ABCDEF";			goto hex;		case 'x':			xdigs = "0123456789abcdef";hex:			if (flags & QUADINT)				uqval = va_arg(ap, u_quad_t);			else				ulval = UARG();			base = 16;			/* leading 0x/X only if non-zero */			if (flags & ALT &&			    (flags & QUADINT ? uqval != 0 : ulval != 0))				flags |= HEXPREFIX;			/* unsigned conversions */nosign:			sign = '\0';			/*			 * ``... diouXx conversions ... if a precision is			 * specified, the 0 flag will be ignored.''			 *	-- ANSI X3J11			 */number:			if ((dprec = prec) >= 0)				flags &= ~ZEROPAD;			/*			 * ``The result of converting a zero value with an			 * explicit precision of zero is no characters.''			 *	-- ANSI X3J11			 */			cp = buf + BUF;			if (flags & QUADINT) {				if (uqval != 0 || prec != 0)					cp = __uqtoa(uqval, cp, base,					    flags & ALT, xdigs);			} else {				if (ulval != 0 || prec != 0)					cp = __ultoa(ulval, cp, base,					    flags & ALT, xdigs);			}			size = buf + BUF - cp;			break;		default:	/* "%?" prints ?, unless ? is NUL */			if (ch == '\0')				goto done;			/* pretend it was %c with argument ch */			cp = buf;			*cp = ch;			size = 1;			sign = '\0';			break;		}		/*		 * All reasonable formats wind up here.  At this point, `cp'		 * points to a string which (if not flags&LADJUST) should be		 * padded out to `width' places.  If flags&ZEROPAD, it should		 * first be prefixed by any sign or other prefix; otherwise,		 * it should be blank padded before the prefix is emitted.		 * After any left-hand padding and prefixing, emit zeroes		 * required by a decimal [diouxX] precision, then print the		 * string proper, then emit zeroes required by any leftover		 * floating precision; finally, if LADJUST, pad with blanks.		 *		 * Compute actual size, so we know how much to pad.		 * fieldsz excludes decimal prec; realsz includes it.		 */		fieldsz = size;		if (sign)			fieldsz++;		else if (flags & HEXPREFIX)			fieldsz += 2;		realsz = dprec > fieldsz ? dprec : fieldsz;		/* right-adjusting blank padding */		if ((flags & (LADJUST|ZEROPAD)) == 0)			PAD(width - realsz, blanks);		/* prefix */		if (sign) {			PRINT(&sign, 1);		} else if (flags & HEXPREFIX) {			ox[0] = '0';			ox[1] = ch;			PRINT(ox, 2);		}		/* right-adjusting zero padding */		if ((flags & (LADJUST|ZEROPAD)) == ZEROPAD)			PAD(width - realsz, zeroes);		/* leading zeroes from decimal precision */		PAD(dprec - fieldsz, zeroes);		/* the string or number proper */#ifdef FLOATING_POINT		if ((flags & FPT) == 0) {			PRINT(cp, size);		} else {	/* glue together f_p fragments */			if (ch >= 'f') {	/* 'f' or 'g' */				if (_double == 0) {				/* kludge for __dtoa irregularity */					if (prec == 0 ||					    (flags & ALT) == 0) {						PRINT("0", 1);					} else {						PRINT("0.", 2);						PAD(ndig - 1, zeroes);					}				} else if (expt <= 0) {					PRINT("0.", 2);					PAD(-expt, zeroes);					PRINT(cp, ndig);				} else if (expt >= ndig) {					PRINT(cp, ndig);					PAD(expt - ndig, zeroes);					if (flags & ALT)						PRINT(".", 1);				} else {					PRINT(cp, expt);					cp += expt;					PRINT(".", 1);					PRINT(cp, ndig-expt);				}			} else {	/* 'e' or 'E' */				if (ndig > 1 || flags & ALT) {					ox[0] = *cp++;					ox[1] = '.';					PRINT(ox, 2);					if (_double || flags & ALT == 0) {						PRINT(cp, ndig-1);					} else	/* 0.[0..] */						/* __dtoa irregularity */						PAD(ndig - 1, zeroes);				} else	/* XeYYY */					PRINT(cp, 1);				PRINT(expstr, expsize);			}		}#else		PRINT(cp, size);#endif		/* left-adjusting padding (always blank) */		if (flags & LADJUST)			PAD(width - realsz, blanks);		/* finally, adjust ret */		ret += width > realsz ? width : realsz;		FLUSH();	/* copy out the I/O vectors */	}done:	FLUSH();error:	return (__sferror(fp) ? EOF : ret);	/* NOTREACHED */}#ifdef FLOATING_POINTextern char *__dtoa __P((double, int, int, int *, int *, char **));static char *cvt(value, ndigits, flags, sign, decpt, ch, length)	double value;	int ndigits, flags, *decpt, ch, *length;	char *sign;{	int mode, dsgn;	char *digits, *bp, *rve;	if (ch == 'f')		mode = 3;	else {		mode = 2;	}	if (value < 0) {		value = -value;		*sign = '-';	} else		*sign = '\000';	digits = __dtoa(value, mode, ndigits, decpt, &dsgn, &rve);	if (flags & ALT) {	/* Print trailing zeros */		bp = digits + ndigits;		if (ch == 'f') {			if (*digits == '0' && value)				*decpt = -ndigits + 1;			bp += *decpt;		}		if (value == 0)	/* kludge for __dtoa irregularity */			rve = bp;		while (rve < bp)			*rve++ = '0';	}	*length = rve - digits;	return (digits);}static intexponent(p0, exp, fmtch)	char *p0;	int exp, fmtch;{	register char *p, *t;	char expbuf[MAXEXP];	p = p0;	*p++ = fmtch;	if (exp < 0) {		exp = -exp;		*p++ = '-';	}	else		*p++ = '+';	t = expbuf + MAXEXP;	if (exp > 9) {		do {			*--t = to_char(exp % 10);		} while ((exp /= 10) > 9);		*--t = to_char(exp);		for (; t < expbuf + MAXEXP; *p++ = *t++);	}	else {		*p++ = '0';		*p++ = to_char(exp);	}	return (p - p0);}#endif /* FLOATING_POINT */

⌨️ 快捷键说明

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