xref: /netbsd-src/external/mpl/dhcp/dist/omapip/iscprint.c (revision ce893c7eea48d62b6f0d8ef9958789ad27d03f6f)
1*ce893c7eSchristos /*	$NetBSD: iscprint.c,v 1.2 2018/04/07 22:37:30 christos Exp $	*/
26fb29d29Schristos 
36fb29d29Schristos /*
46fb29d29Schristos  * Copyright (C) 2004-2017  Internet Systems Consortium, Inc. ("ISC")
56fb29d29Schristos  * Copyright (C) 1999-2003  Internet Software Consortium.
66fb29d29Schristos  *
76fb29d29Schristos  * This Source Code Form is subject to the terms of the Mozilla Public
86fb29d29Schristos  * License, v. 2.0. If a copy of the MPL was not distributed with this
96fb29d29Schristos  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
106fb29d29Schristos  *
116fb29d29Schristos  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
126fb29d29Schristos  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
136fb29d29Schristos  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
146fb29d29Schristos  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
156fb29d29Schristos  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
166fb29d29Schristos  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
176fb29d29Schristos  * PERFORMANCE OF THIS SOFTWARE.
186fb29d29Schristos  */
196fb29d29Schristos 
206fb29d29Schristos #include <sys/cdefs.h>
21*ce893c7eSchristos __RCSID("$NetBSD: iscprint.c,v 1.2 2018/04/07 22:37:30 christos Exp $");
226fb29d29Schristos 
236fb29d29Schristos 
246fb29d29Schristos /* Id: iscprint.c,v 1.2 2005/03/17 20:30:41 dhankins Exp  */
256fb29d29Schristos 
266fb29d29Schristos #include "dhcpd.h"
276fb29d29Schristos 
286fb29d29Schristos #ifdef NO_SNPRINTF
296fb29d29Schristos 
306fb29d29Schristos #ifndef LINT
316fb29d29Schristos static char copyright[] =
326fb29d29Schristos "Id: iscprint.c,v 1.2 2005/03/17 20:30:41 dhankins Exp  Copyright (c) 2004 Internet Systems Consortium, Inc.  All rights reserved.";
336fb29d29Schristos #endif
346fb29d29Schristos 
356fb29d29Schristos #define INSIST(cond)	REQUIRE(cond)
366fb29d29Schristos #define REQUIRE(cond)	if (!(cond)) { return 0; }
376fb29d29Schristos 
386fb29d29Schristos /*
396fb29d29Schristos  * Return length of string that would have been written if not truncated.
406fb29d29Schristos  */
416fb29d29Schristos 
426fb29d29Schristos int
isc_print_snprintf(char * str,size_t size,const char * format,...)436fb29d29Schristos isc_print_snprintf(char *str, size_t size, const char *format, ...) {
446fb29d29Schristos 	va_list ap;
456fb29d29Schristos 	int ret;
466fb29d29Schristos 
476fb29d29Schristos 	va_start(ap, format);
486fb29d29Schristos 	ret = vsnprintf(str, size, format, ap);
496fb29d29Schristos 	va_end(ap);
506fb29d29Schristos 	return (ret);
516fb29d29Schristos }
526fb29d29Schristos 
536fb29d29Schristos /*
546fb29d29Schristos  * Return length of string that would have been written if not truncated.
556fb29d29Schristos  */
566fb29d29Schristos 
576fb29d29Schristos int
isc_print_vsnprintf(char * str,size_t size,const char * format,va_list ap)586fb29d29Schristos isc_print_vsnprintf(char *str, size_t size, const char *format, va_list ap) {
596fb29d29Schristos 	int h;
606fb29d29Schristos 	int l;
616fb29d29Schristos 	int q;
626fb29d29Schristos 	int alt;
636fb29d29Schristos 	int zero;
646fb29d29Schristos 	int left;
656fb29d29Schristos 	int plus;
666fb29d29Schristos 	int space;
676fb29d29Schristos 	int neg;
686fb29d29Schristos 	isc_int64_t tmpi;
696fb29d29Schristos 	isc_uint64_t tmpui;
706fb29d29Schristos 	unsigned long width;
716fb29d29Schristos 	unsigned long precision;
726fb29d29Schristos 	unsigned int length;
736fb29d29Schristos 	char buf[1024];
746fb29d29Schristos 	char c;
756fb29d29Schristos 	void *v;
766fb29d29Schristos 	char *save = str;
776fb29d29Schristos 	const char *cp;
786fb29d29Schristos 	const char *head;
796fb29d29Schristos 	int count = 0;
806fb29d29Schristos 	int pad;
816fb29d29Schristos 	int zeropad;
826fb29d29Schristos 	int dot;
836fb29d29Schristos 	double dbl;
846fb29d29Schristos #ifdef HAVE_LONG_DOUBLE
856fb29d29Schristos 	long double ldbl;
866fb29d29Schristos #endif
876fb29d29Schristos 	char fmt[32];
886fb29d29Schristos 
896fb29d29Schristos 	INSIST(str != NULL);
906fb29d29Schristos 	INSIST(format != NULL);
916fb29d29Schristos 
926fb29d29Schristos 	while (*format != '\0') {
936fb29d29Schristos 		if (*format != '%') {
946fb29d29Schristos 			if (size > 1) {
956fb29d29Schristos 				*str++ = *format;
966fb29d29Schristos 				size--;
976fb29d29Schristos 			}
986fb29d29Schristos 			count++;
996fb29d29Schristos 			format++;
1006fb29d29Schristos 			continue;
1016fb29d29Schristos 		}
1026fb29d29Schristos 		format++;
1036fb29d29Schristos 
1046fb29d29Schristos 		/*
1056fb29d29Schristos 		 * Reset flags.
1066fb29d29Schristos 		 */
1076fb29d29Schristos 		dot = neg = space = plus = left = zero = alt = h = l = q = 0;
1086fb29d29Schristos 		width = precision = 0;
1096fb29d29Schristos 		head = "";
1106fb29d29Schristos 		length = pad = zeropad = 0;
1116fb29d29Schristos 
1126fb29d29Schristos 		do {
1136fb29d29Schristos 			if (*format == '#') {
1146fb29d29Schristos 				alt = 1;
1156fb29d29Schristos 				format++;
1166fb29d29Schristos 			} else if (*format == '-') {
1176fb29d29Schristos 				left = 1;
1186fb29d29Schristos 				zero = 0;
1196fb29d29Schristos 				format++;
1206fb29d29Schristos 			} else if (*format == ' ') {
1216fb29d29Schristos 				if (!plus)
1226fb29d29Schristos 					space = 1;
1236fb29d29Schristos 				format++;
1246fb29d29Schristos 			} else if (*format == '+') {
1256fb29d29Schristos 				plus = 1;
1266fb29d29Schristos 				space = 0;
1276fb29d29Schristos 				format++;
1286fb29d29Schristos 			} else if (*format == '0') {
1296fb29d29Schristos 				if (!left)
1306fb29d29Schristos 					zero = 1;
1316fb29d29Schristos 				format++;
1326fb29d29Schristos 			} else
1336fb29d29Schristos 				break;
1346fb29d29Schristos 		} while (1);
1356fb29d29Schristos 
1366fb29d29Schristos 		/*
1376fb29d29Schristos 		 * Width.
1386fb29d29Schristos 		 */
1396fb29d29Schristos 		if (*format == '*') {
1406fb29d29Schristos 			width = va_arg(ap, int);
1416fb29d29Schristos 			format++;
1426fb29d29Schristos 		} else if (isdigit((unsigned char)*format)) {
1436fb29d29Schristos 			char *e;
1446fb29d29Schristos 			width = strtoul(format, &e, 10);
1456fb29d29Schristos 			format = e;
1466fb29d29Schristos 		}
1476fb29d29Schristos 
1486fb29d29Schristos 		/*
1496fb29d29Schristos 		 * Precision.
1506fb29d29Schristos 		 */
1516fb29d29Schristos 		if (*format == '.') {
1526fb29d29Schristos 			format++;
1536fb29d29Schristos 			dot = 1;
1546fb29d29Schristos 			if (*format == '*') {
1556fb29d29Schristos 				precision = va_arg(ap, int);
1566fb29d29Schristos 				format++;
1576fb29d29Schristos 			} else if (isdigit((unsigned char)*format)) {
1586fb29d29Schristos 				char *e;
1596fb29d29Schristos 				precision = strtoul(format, &e, 10);
1606fb29d29Schristos 				format = e;
1616fb29d29Schristos 			}
1626fb29d29Schristos 		}
1636fb29d29Schristos 
1646fb29d29Schristos 		switch (*format) {
1656fb29d29Schristos 		case '\0':
1666fb29d29Schristos 			continue;
1676fb29d29Schristos 		case '%':
1686fb29d29Schristos 			if (size > 1) {
1696fb29d29Schristos 				*str++ = *format;
1706fb29d29Schristos 				size--;
1716fb29d29Schristos 			}
1726fb29d29Schristos 			count++;
1736fb29d29Schristos 			break;
1746fb29d29Schristos 		case 'q':
1756fb29d29Schristos 			q = 1;
1766fb29d29Schristos 			format++;
1776fb29d29Schristos 			goto doint;
1786fb29d29Schristos 		case 'h':
1796fb29d29Schristos 			h = 1;
1806fb29d29Schristos 			format++;
1816fb29d29Schristos 			goto doint;
1826fb29d29Schristos 		case 'l':
1836fb29d29Schristos 			l = 1;
1846fb29d29Schristos 			format++;
1856fb29d29Schristos 			if (*format == 'l') {
1866fb29d29Schristos 				q = 1;
1876fb29d29Schristos 				format++;
1886fb29d29Schristos 			}
1896fb29d29Schristos 			goto doint;
1906fb29d29Schristos 		case 'n':
1916fb29d29Schristos 		case 'i':
1926fb29d29Schristos 		case 'd':
1936fb29d29Schristos 		case 'o':
1946fb29d29Schristos 		case 'u':
1956fb29d29Schristos 		case 'x':
1966fb29d29Schristos 		case 'X':
1976fb29d29Schristos 		doint:
1986fb29d29Schristos 			if (precision != 0)
1996fb29d29Schristos 				zero = 0;
2006fb29d29Schristos 			switch (*format) {
2016fb29d29Schristos 			case 'n':
2026fb29d29Schristos 				if (h) {
2036fb29d29Schristos 					short int *p;
2046fb29d29Schristos 					p = va_arg(ap, short *);
2056fb29d29Schristos 					REQUIRE(p != NULL);
2066fb29d29Schristos 					*p = str - save;
2076fb29d29Schristos 				} else if (l) {
2086fb29d29Schristos 					long int *p;
2096fb29d29Schristos 					p = va_arg(ap, long *);
2106fb29d29Schristos 					REQUIRE(p != NULL);
2116fb29d29Schristos 					*p = str - save;
2126fb29d29Schristos 				} else {
2136fb29d29Schristos 					int *p;
2146fb29d29Schristos 					p = va_arg(ap, int *);
2156fb29d29Schristos 					REQUIRE(p != NULL);
2166fb29d29Schristos 					*p = str - save;
2176fb29d29Schristos 				}
2186fb29d29Schristos 				break;
2196fb29d29Schristos 			case 'i':
2206fb29d29Schristos 			case 'd':
2216fb29d29Schristos 				if (q)
2226fb29d29Schristos 					tmpi = va_arg(ap, isc_int64_t);
2236fb29d29Schristos 				else if (l)
2246fb29d29Schristos 					tmpi = va_arg(ap, long int);
2256fb29d29Schristos 				else
2266fb29d29Schristos 					tmpi = va_arg(ap, int);
2276fb29d29Schristos 				if (tmpi < 0) {
2286fb29d29Schristos 					head = "-";
2296fb29d29Schristos 					tmpui = -tmpi;
2306fb29d29Schristos 				} else {
2316fb29d29Schristos 					if (plus)
2326fb29d29Schristos 						head = "+";
2336fb29d29Schristos 					else if (space)
2346fb29d29Schristos 						head = " ";
2356fb29d29Schristos 					else
2366fb29d29Schristos 						head = "";
2376fb29d29Schristos 					tmpui = tmpi;
2386fb29d29Schristos 				}
2396fb29d29Schristos 				sprintf(buf, "%u", tmpui);
2406fb29d29Schristos 				goto printint;
2416fb29d29Schristos 			case 'o':
2426fb29d29Schristos 				if (q)
2436fb29d29Schristos 					tmpui = va_arg(ap, isc_uint64_t);
2446fb29d29Schristos 				else if (l)
2456fb29d29Schristos 					tmpui = va_arg(ap, long int);
2466fb29d29Schristos 				else
2476fb29d29Schristos 					tmpui = va_arg(ap, int);
2486fb29d29Schristos 				sprintf(buf, alt ? "%#o"
2496fb29d29Schristos 						 : "%o", tmpui);
2506fb29d29Schristos 				goto printint;
2516fb29d29Schristos 			case 'u':
2526fb29d29Schristos 				if (q)
2536fb29d29Schristos 					tmpui = va_arg(ap, isc_uint64_t);
2546fb29d29Schristos 				else if (l)
2556fb29d29Schristos 					tmpui = va_arg(ap, unsigned long int);
2566fb29d29Schristos 				else
2576fb29d29Schristos 					tmpui = va_arg(ap, unsigned int);
2586fb29d29Schristos 				sprintf(buf, "%u", tmpui);
2596fb29d29Schristos 				goto printint;
2606fb29d29Schristos 			case 'x':
2616fb29d29Schristos 				if (q)
2626fb29d29Schristos 					tmpui = va_arg(ap, isc_uint64_t);
2636fb29d29Schristos 				else if (l)
2646fb29d29Schristos 					tmpui = va_arg(ap, unsigned long int);
2656fb29d29Schristos 				else
2666fb29d29Schristos 					tmpui = va_arg(ap, unsigned int);
2676fb29d29Schristos 				if (alt) {
2686fb29d29Schristos 					head = "0x";
2696fb29d29Schristos 					if (precision > 2)
2706fb29d29Schristos 						precision -= 2;
2716fb29d29Schristos 				}
2726fb29d29Schristos 				sprintf(buf, "%x", tmpui);
2736fb29d29Schristos 				goto printint;
2746fb29d29Schristos 			case 'X':
2756fb29d29Schristos 				if (q)
2766fb29d29Schristos 					tmpui = va_arg(ap, isc_uint64_t);
2776fb29d29Schristos 				else if (l)
2786fb29d29Schristos 					tmpui = va_arg(ap, unsigned long int);
2796fb29d29Schristos 				else
2806fb29d29Schristos 					tmpui = va_arg(ap, unsigned int);
2816fb29d29Schristos 				if (alt) {
2826fb29d29Schristos 					head = "0X";
2836fb29d29Schristos 					if (precision > 2)
2846fb29d29Schristos 						precision -= 2;
2856fb29d29Schristos 				}
2866fb29d29Schristos 				sprintf(buf, "%X", tmpui);
2876fb29d29Schristos 				goto printint;
2886fb29d29Schristos 			printint:
2896fb29d29Schristos 				if (precision != 0 || width != 0) {
2906fb29d29Schristos 					length = strlen(buf);
2916fb29d29Schristos 					if (length < precision)
2926fb29d29Schristos 						zeropad = precision - length;
2936fb29d29Schristos 					else if (length < width && zero)
2946fb29d29Schristos 						zeropad = width - length;
2956fb29d29Schristos 					if (width != 0) {
2966fb29d29Schristos 						pad = width - length -
2976fb29d29Schristos 						      zeropad - strlen(head);
2986fb29d29Schristos 						if (pad < 0)
2996fb29d29Schristos 							pad = 0;
3006fb29d29Schristos 					}
3016fb29d29Schristos 				}
3026fb29d29Schristos 				count += strlen(head) + strlen(buf) + pad +
3036fb29d29Schristos 					 zeropad;
3046fb29d29Schristos 				if (!left) {
3056fb29d29Schristos 					while (pad > 0 && size > 1) {
3066fb29d29Schristos 						*str++ = ' ';
3076fb29d29Schristos 						size--;
3086fb29d29Schristos 						pad--;
3096fb29d29Schristos 					}
3106fb29d29Schristos 				}
3116fb29d29Schristos 				cp = head;
3126fb29d29Schristos 				while (*cp != '\0' && size > 1) {
3136fb29d29Schristos 					*str++ = *cp++;
3146fb29d29Schristos 					size--;
3156fb29d29Schristos 				}
3166fb29d29Schristos 				while (zeropad > 0 && size > 1) {
3176fb29d29Schristos 					*str++ = '0';
3186fb29d29Schristos 					size--;
3196fb29d29Schristos 					zeropad--;
3206fb29d29Schristos 				}
3216fb29d29Schristos 				cp = buf;
3226fb29d29Schristos 				while (*cp != '\0' && size > 1) {
3236fb29d29Schristos 					*str++ = *cp++;
3246fb29d29Schristos 					size--;
3256fb29d29Schristos 				}
3266fb29d29Schristos 				while (pad > 0 && size > 1) {
3276fb29d29Schristos 					*str++ = ' ';
3286fb29d29Schristos 					size--;
3296fb29d29Schristos 					pad--;
3306fb29d29Schristos 				}
3316fb29d29Schristos 				break;
3326fb29d29Schristos 			default:
3336fb29d29Schristos 				break;
3346fb29d29Schristos 			}
3356fb29d29Schristos 			break;
3366fb29d29Schristos 		case 's':
3376fb29d29Schristos 			cp = va_arg(ap, char *);
3386fb29d29Schristos 			REQUIRE(cp != NULL);
3396fb29d29Schristos 
3406fb29d29Schristos 			if (precision != 0) {
3416fb29d29Schristos 				/*
3426fb29d29Schristos 				 * cp need not be NULL terminated.
3436fb29d29Schristos 				 */
3446fb29d29Schristos 				const char *tp;
3456fb29d29Schristos 				unsigned long n;
3466fb29d29Schristos 
3476fb29d29Schristos 				n = precision;
3486fb29d29Schristos 				tp = cp;
3496fb29d29Schristos 				while (n != 0 && *tp != '\0')
3506fb29d29Schristos 					n--, tp++;
3516fb29d29Schristos 				length = precision - n;
3526fb29d29Schristos 			} else {
3536fb29d29Schristos 				length = strlen(cp);
3546fb29d29Schristos 			}
3556fb29d29Schristos 			if (width != 0) {
3566fb29d29Schristos 				pad = width - length;
3576fb29d29Schristos 				if (pad < 0)
3586fb29d29Schristos 					pad = 0;
3596fb29d29Schristos 			}
3606fb29d29Schristos 			count += pad + length;
3616fb29d29Schristos 			if (!left)
3626fb29d29Schristos 				while (pad > 0 && size > 1) {
3636fb29d29Schristos 					*str++ = ' ';
3646fb29d29Schristos 					size--;
3656fb29d29Schristos 					pad--;
3666fb29d29Schristos 				}
3676fb29d29Schristos 			if (precision != 0)
3686fb29d29Schristos 				while (precision > 0 && *cp != '\0' &&
3696fb29d29Schristos 				       size > 1) {
3706fb29d29Schristos 					*str++ = *cp++;
3716fb29d29Schristos 					size--;
3726fb29d29Schristos 					precision--;
3736fb29d29Schristos 				}
3746fb29d29Schristos 			else
3756fb29d29Schristos 				while (*cp != '\0' && size > 1) {
3766fb29d29Schristos 					*str++ = *cp++;
3776fb29d29Schristos 					size--;
3786fb29d29Schristos 				}
3796fb29d29Schristos 			while (pad > 0 && size > 1) {
3806fb29d29Schristos 				*str++ = ' ';
3816fb29d29Schristos 				size--;
3826fb29d29Schristos 				pad--;
3836fb29d29Schristos 			}
3846fb29d29Schristos 			break;
3856fb29d29Schristos 		case 'c':
3866fb29d29Schristos 			c = va_arg(ap, int);
3876fb29d29Schristos 			if (width > 0) {
3886fb29d29Schristos 				count += width;
3896fb29d29Schristos 				width--;
3906fb29d29Schristos 				if (left) {
3916fb29d29Schristos 					*str++ = c;
3926fb29d29Schristos 					size--;
3936fb29d29Schristos 				}
3946fb29d29Schristos 				while (width-- > 0 && size > 1) {
3956fb29d29Schristos 					*str++ = ' ';
3966fb29d29Schristos 					size--;
3976fb29d29Schristos 				}
3986fb29d29Schristos 				if (!left && size > 1) {
3996fb29d29Schristos 					*str++ = c;
4006fb29d29Schristos 					size--;
4016fb29d29Schristos 				}
4026fb29d29Schristos 			} else {
4036fb29d29Schristos 				count++;
4046fb29d29Schristos 				if (size > 1) {
4056fb29d29Schristos 					*str++ = c;
4066fb29d29Schristos 					size--;
4076fb29d29Schristos 				}
4086fb29d29Schristos 			}
4096fb29d29Schristos 			break;
4106fb29d29Schristos 		case 'p':
4116fb29d29Schristos 			v = va_arg(ap, void *);
4126fb29d29Schristos 			sprintf(buf, "%p", v);
4136fb29d29Schristos 			length = strlen(buf);
4146fb29d29Schristos 			if (precision > length)
4156fb29d29Schristos 				zeropad = precision - length;
4166fb29d29Schristos 			if (width > 0) {
4176fb29d29Schristos 				pad = width - length - zeropad;
4186fb29d29Schristos 				if (pad < 0)
4196fb29d29Schristos 					pad = 0;
4206fb29d29Schristos 			}
4216fb29d29Schristos 			count += length + pad + zeropad;
4226fb29d29Schristos 			if (!left)
4236fb29d29Schristos 				while (pad > 0 && size > 1) {
4246fb29d29Schristos 					*str++ = ' ';
4256fb29d29Schristos 					size--;
4266fb29d29Schristos 					pad--;
4276fb29d29Schristos 				}
4286fb29d29Schristos 			cp = buf;
4296fb29d29Schristos 			if (zeropad > 0 && buf[0] == '0' &&
4306fb29d29Schristos 			    (buf[1] == 'x' || buf[1] == 'X')) {
4316fb29d29Schristos 				if (size > 1) {
4326fb29d29Schristos 					*str++ = *cp++;
4336fb29d29Schristos 					size--;
4346fb29d29Schristos 				}
4356fb29d29Schristos 				if (size > 1) {
4366fb29d29Schristos 					*str++ = *cp++;
4376fb29d29Schristos 					size--;
4386fb29d29Schristos 				}
4396fb29d29Schristos 				while (zeropad > 0 && size > 1) {
4406fb29d29Schristos 					*str++ = '0';
4416fb29d29Schristos 					size--;
4426fb29d29Schristos 					zeropad--;
4436fb29d29Schristos 				}
4446fb29d29Schristos 			}
4456fb29d29Schristos 			while (*cp != '\0' && size > 1) {
4466fb29d29Schristos 				*str++ = *cp++;
4476fb29d29Schristos 				size--;
4486fb29d29Schristos 			}
4496fb29d29Schristos 			while (pad > 0 && size > 1) {
4506fb29d29Schristos 				*str++ = ' ';
4516fb29d29Schristos 				size--;
4526fb29d29Schristos 				pad--;
4536fb29d29Schristos 			}
4546fb29d29Schristos 			break;
4556fb29d29Schristos 		case 'D':	/*deprecated*/
4566fb29d29Schristos 			INSIST("use %ld instead of %D" == NULL);
4576fb29d29Schristos 		case 'O':	/*deprecated*/
4586fb29d29Schristos 			INSIST("use %lo instead of %O" == NULL);
4596fb29d29Schristos 		case 'U':	/*deprecated*/
4606fb29d29Schristos 			INSIST("use %lu instead of %U" == NULL);
4616fb29d29Schristos 
4626fb29d29Schristos 		case 'L':
4636fb29d29Schristos #ifdef HAVE_LONG_DOUBLE
4646fb29d29Schristos 			l = 1;
4656fb29d29Schristos #else
4666fb29d29Schristos 			INSIST("long doubles are not supported" == NULL);
4676fb29d29Schristos #endif
4686fb29d29Schristos 			/*FALLTHROUGH*/
4696fb29d29Schristos 		case 'e':
4706fb29d29Schristos 		case 'E':
4716fb29d29Schristos 		case 'f':
4726fb29d29Schristos 		case 'g':
4736fb29d29Schristos 		case 'G':
4746fb29d29Schristos 			if (!dot)
4756fb29d29Schristos 				precision = 6;
4766fb29d29Schristos 			/*
4776fb29d29Schristos 			 * IEEE floating point.
4786fb29d29Schristos 			 * MIN 2.2250738585072014E-308
4796fb29d29Schristos 			 * MAX 1.7976931348623157E+308
4806fb29d29Schristos 			 * VAX floating point has a smaller range than IEEE.
4816fb29d29Schristos 			 *
4826fb29d29Schristos 			 * precisions > 324 don't make much sense.
4836fb29d29Schristos 			 * if we cap the precision at 512 we will not
4846fb29d29Schristos 			 * overflow buf.
4856fb29d29Schristos 			 */
4866fb29d29Schristos 			if (precision > 512)
4876fb29d29Schristos 				precision = 512;
4886fb29d29Schristos 			sprintf(fmt, "%%%s%s.%lu%s%c", alt ? "#" : "",
4896fb29d29Schristos 				plus ? "+" : space ? " " : "",
4906fb29d29Schristos 				precision, l ? "L" : "", *format);
4916fb29d29Schristos 			switch (*format) {
4926fb29d29Schristos 			case 'e':
4936fb29d29Schristos 			case 'E':
4946fb29d29Schristos 			case 'f':
4956fb29d29Schristos 			case 'g':
4966fb29d29Schristos 			case 'G':
4976fb29d29Schristos #ifdef HAVE_LONG_DOUBLE
4986fb29d29Schristos 				if (l) {
4996fb29d29Schristos 					ldbl = va_arg(ap, long double);
5006fb29d29Schristos 					sprintf(buf, fmt, ldbl);
5016fb29d29Schristos 				} else
5026fb29d29Schristos #endif
5036fb29d29Schristos 				{
5046fb29d29Schristos 					dbl = va_arg(ap, double);
5056fb29d29Schristos 					sprintf(buf, fmt, dbl);
5066fb29d29Schristos 				}
5076fb29d29Schristos 				length = strlen(buf);
5086fb29d29Schristos 				if (width > 0) {
5096fb29d29Schristos 					pad = width - length;
5106fb29d29Schristos 					if (pad < 0)
5116fb29d29Schristos 						pad = 0;
5126fb29d29Schristos 				}
5136fb29d29Schristos 				count += length + pad;
5146fb29d29Schristos 				if (!left)
5156fb29d29Schristos 					while (pad > 0 && size > 1) {
5166fb29d29Schristos 						*str++ = ' ';
5176fb29d29Schristos 						size--;
5186fb29d29Schristos 						pad--;
5196fb29d29Schristos 					}
5206fb29d29Schristos 				cp = buf;
5216fb29d29Schristos 				while (*cp != ' ' && size > 1) {
5226fb29d29Schristos 					*str++ = *cp++;
5236fb29d29Schristos 					size--;
5246fb29d29Schristos 				}
5256fb29d29Schristos 				while (pad > 0 && size > 1) {
5266fb29d29Schristos 					*str++ = ' ';
5276fb29d29Schristos 					size--;
5286fb29d29Schristos 					pad--;
5296fb29d29Schristos 				}
5306fb29d29Schristos 				break;
5316fb29d29Schristos 			default:
5326fb29d29Schristos 				continue;
5336fb29d29Schristos 			}
5346fb29d29Schristos 			break;
5356fb29d29Schristos 		default:
5366fb29d29Schristos 			continue;
5376fb29d29Schristos 		}
5386fb29d29Schristos 		format++;
5396fb29d29Schristos 	}
5406fb29d29Schristos 	if (size > 0)
5416fb29d29Schristos 		*str = '\0';
5426fb29d29Schristos 	return (count);
5436fb29d29Schristos }
5446fb29d29Schristos 
5456fb29d29Schristos #endif
546