Blame lib/isc/print.c

Packit Service ae04f2
/*
Packit Service ae04f2
 * Copyright (C) Internet Systems Consortium, Inc. ("ISC")
Packit Service ae04f2
 *
Packit Service ae04f2
 * This Source Code Form is subject to the terms of the Mozilla Public
Packit Service ae04f2
 * License, v. 2.0. If a copy of the MPL was not distributed with this
Packit Service ae04f2
 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
Packit Service ae04f2
 *
Packit Service ae04f2
 * See the COPYRIGHT file distributed with this work for additional
Packit Service ae04f2
 * information regarding copyright ownership.
Packit Service ae04f2
 */
Packit Service ae04f2
Packit Service ae04f2
/*! \file */
Packit Service ae04f2
Packit Service ae04f2
#include <config.h>
Packit Service ae04f2
Packit Service ae04f2
#include <ctype.h>
Packit Service ae04f2
#include <inttypes.h>
Packit Service ae04f2
#include <stdbool.h>
Packit Service ae04f2
#include <stdio.h>		/* for sprintf() */
Packit Service ae04f2
#include <string.h>		/* for strlen() */
Packit Service ae04f2
#include <assert.h>		/* for assert() */
Packit Service ae04f2
Packit Service ae04f2
#define	ISC__PRINT_SOURCE	/* Used to get the isc_print_* prototypes. */
Packit Service ae04f2
Packit Service ae04f2
#include <isc/assertions.h>
Packit Service ae04f2
#include <isc/msgs.h>
Packit Service ae04f2
#include <isc/print.h>
Packit Service ae04f2
#include <isc/stdlib.h>
Packit Service ae04f2
#include <isc/util.h>
Packit Service ae04f2
Packit Service ae04f2
/*
Packit Service ae04f2
 * We use the system's sprintf so we undef it here.
Packit Service ae04f2
 */
Packit Service ae04f2
#undef sprintf
Packit Service ae04f2
Packit Service ae04f2
static int
Packit Service ae04f2
isc__print_printf(void (*emit)(char, void *), void *arg,
Packit Service ae04f2
		  const char *format, va_list ap);
Packit Service ae04f2
Packit Service ae04f2
static void
Packit Service ae04f2
file_emit(char c, void *arg) {
Packit Service ae04f2
	FILE *fp = arg;
Packit Service ae04f2
	int i = c & 0xff;
Packit Service ae04f2
Packit Service ae04f2
	putc(i, fp);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
#if 0
Packit Service ae04f2
static int
Packit Service ae04f2
isc_print_vfprintf(FILE *fp, const char *format, va_list ap) {
Packit Service ae04f2
	assert(fp != NULL);
Packit Service ae04f2
	assert(format != NULL);
Packit Service ae04f2
Packit Service ae04f2
	return (isc__print_printf(file_emit, fp, format, ap));
Packit Service ae04f2
}
Packit Service ae04f2
#endif
Packit Service ae04f2
Packit Service ae04f2
int
Packit Service ae04f2
isc_print_printf(const char *format, ...) {
Packit Service ae04f2
	va_list ap;
Packit Service ae04f2
	int n;
Packit Service ae04f2
Packit Service ae04f2
	assert(format != NULL);
Packit Service ae04f2
Packit Service ae04f2
	va_start(ap, format);
Packit Service ae04f2
	n = isc__print_printf(file_emit, stdout, format, ap);
Packit Service ae04f2
	va_end(ap);
Packit Service ae04f2
	return (n);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
int
Packit Service ae04f2
isc_print_fprintf(FILE *fp, const char *format, ...) {
Packit Service ae04f2
	va_list ap;
Packit Service ae04f2
	int n;
Packit Service ae04f2
Packit Service ae04f2
	assert(fp != NULL);
Packit Service ae04f2
	assert(format != NULL);
Packit Service ae04f2
Packit Service ae04f2
	va_start(ap, format);
Packit Service ae04f2
	n = isc__print_printf(file_emit, fp, format, ap);
Packit Service ae04f2
	va_end(ap);
Packit Service ae04f2
	return (n);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
static void
Packit Service ae04f2
nocheck_emit(char c, void *arg) {
Packit Service ae04f2
	struct { char *str; } *a = arg;
Packit Service ae04f2
Packit Service ae04f2
	*(a->str)++ = c;
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
int
Packit Service ae04f2
isc_print_sprintf(char *str, const char *format, ...) {
Packit Service ae04f2
	struct { char *str; } arg;
Packit Service ae04f2
	int n;
Packit Service ae04f2
	va_list ap;
Packit Service ae04f2
Packit Service ae04f2
	arg.str = str;
Packit Service ae04f2
Packit Service ae04f2
	va_start(ap, format);
Packit Service ae04f2
	n = isc__print_printf(nocheck_emit, &arg, format, ap);
Packit Service ae04f2
	va_end(ap);
Packit Service ae04f2
	return (n);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
/*!
Packit Service ae04f2
 * Return length of string that would have been written if not truncated.
Packit Service ae04f2
 */
Packit Service ae04f2
Packit Service ae04f2
int
Packit Service ae04f2
isc_print_snprintf(char *str, size_t size, const char *format, ...) {
Packit Service ae04f2
	va_list ap;
Packit Service ae04f2
	int ret;
Packit Service ae04f2
Packit Service ae04f2
	va_start(ap, format);
Packit Service ae04f2
	ret = isc_print_vsnprintf(str, size, format, ap);
Packit Service ae04f2
	va_end(ap);
Packit Service ae04f2
	return (ret);
Packit Service ae04f2
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
/*!
Packit Service ae04f2
 * Return length of string that would have been written if not truncated.
Packit Service ae04f2
 */
Packit Service ae04f2
Packit Service ae04f2
static void
Packit Service ae04f2
string_emit(char c, void *arg) {
Packit Service ae04f2
	struct { char *str; size_t size; } *p = arg;
Packit Service ae04f2
Packit Service ae04f2
	if (p->size > 0U) {
Packit Service ae04f2
		*(p->str)++ = c;
Packit Service ae04f2
		p->size--;
Packit Service ae04f2
	}
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
int
Packit Service ae04f2
isc_print_vsnprintf(char *str, size_t size, const char *format, va_list ap) {
Packit Service ae04f2
	struct { char *str; size_t size; } arg;
Packit Service ae04f2
	int n;
Packit Service ae04f2
Packit Service ae04f2
	assert(str != NULL);
Packit Service ae04f2
	assert(format != NULL);
Packit Service ae04f2
Packit Service ae04f2
	arg.str = str;
Packit Service ae04f2
	arg.size = size;
Packit Service ae04f2
Packit Service ae04f2
	n = isc__print_printf(string_emit, &arg, format, ap);
Packit Service ae04f2
	if (arg.size > 0U)
Packit Service ae04f2
		*arg.str = '\0';
Packit Service ae04f2
	return (n);
Packit Service ae04f2
}
Packit Service ae04f2
Packit Service ae04f2
static int
Packit Service ae04f2
isc__print_printf(void (*emit)(char, void *), void *arg,
Packit Service ae04f2
		  const char *format, va_list ap)
Packit Service ae04f2
{
Packit Service ae04f2
	int h;
Packit Service ae04f2
	int l;
Packit Service ae04f2
	int z;
Packit Service ae04f2
	int q;
Packit Service ae04f2
	int alt;
Packit Service ae04f2
	int zero;
Packit Service ae04f2
	int left;
Packit Service ae04f2
	int plus;
Packit Service ae04f2
	int space;
Packit Service ae04f2
	int64_t tmpi;
Packit Service ae04f2
	uint64_t tmpui;
Packit Service ae04f2
	unsigned long width;
Packit Service ae04f2
	unsigned long precision;
Packit Service ae04f2
	unsigned int length;
Packit Service ae04f2
	char buf[1024];
Packit Service ae04f2
	char c;
Packit Service ae04f2
	void *v;
Packit Service ae04f2
	const char *cp;
Packit Service ae04f2
	const char *head;
Packit Service ae04f2
	int count = 0;
Packit Service ae04f2
	int pad;
Packit Service ae04f2
	int zeropad;
Packit Service ae04f2
	int dot;
Packit Service ae04f2
	double dbl;
Packit Service ae04f2
	bool precision_set;
Packit Service ae04f2
#ifdef HAVE_LONG_DOUBLE
Packit Service ae04f2
	long double ldbl;
Packit Service ae04f2
#endif
Packit Service ae04f2
	char fmt[32];
Packit Service ae04f2
Packit Service ae04f2
	assert(emit != NULL);
Packit Service ae04f2
	assert(arg != NULL);
Packit Service ae04f2
	assert(format != NULL);
Packit Service ae04f2
Packit Service ae04f2
	while (*format != '\0') {
Packit Service ae04f2
		if (*format != '%') {
Packit Service ae04f2
			emit(*format++, arg);
Packit Service ae04f2
			count++;
Packit Service ae04f2
			continue;
Packit Service ae04f2
		}
Packit Service ae04f2
		format++;
Packit Service ae04f2
Packit Service ae04f2
		/*
Packit Service ae04f2
		 * Reset flags.
Packit Service ae04f2
		 */
Packit Service ae04f2
		dot = space = plus = left = zero = alt = h = l = q = z = 0;
Packit Service ae04f2
		width = precision = 0;
Packit Service ae04f2
		head = "";
Packit Service ae04f2
		pad = zeropad = 0;
Packit Service ae04f2
		precision_set = false;
Packit Service ae04f2
Packit Service ae04f2
		do {
Packit Service ae04f2
			if (*format == '#') {
Packit Service ae04f2
				alt = 1;
Packit Service ae04f2
				format++;
Packit Service ae04f2
			} else if (*format == '-') {
Packit Service ae04f2
				left = 1;
Packit Service ae04f2
				zero = 0;
Packit Service ae04f2
				format++;
Packit Service ae04f2
			} else if (*format == ' ') {
Packit Service ae04f2
				if (!plus)
Packit Service ae04f2
					space = 1;
Packit Service ae04f2
				format++;
Packit Service ae04f2
			} else if (*format == '+') {
Packit Service ae04f2
				plus = 1;
Packit Service ae04f2
				space = 0;
Packit Service ae04f2
				format++;
Packit Service ae04f2
			} else if (*format == '0') {
Packit Service ae04f2
				if (!left)
Packit Service ae04f2
					zero = 1;
Packit Service ae04f2
				format++;
Packit Service ae04f2
			} else
Packit Service ae04f2
				break;
Packit Service ae04f2
		} while (1);
Packit Service ae04f2
Packit Service ae04f2
		/*
Packit Service ae04f2
		 * Width.
Packit Service ae04f2
		 */
Packit Service ae04f2
		if (*format == '*') {
Packit Service ae04f2
			width = va_arg(ap, int);
Packit Service ae04f2
			format++;
Packit Service ae04f2
		} else if (isdigit((unsigned char)*format)) {
Packit Service ae04f2
			char *e;
Packit Service ae04f2
			width = strtoul(format, &e, 10);
Packit Service ae04f2
			format = e;
Packit Service ae04f2
		}
Packit Service ae04f2
Packit Service ae04f2
		/*
Packit Service ae04f2
		 * Precision.
Packit Service ae04f2
		 */
Packit Service ae04f2
		if (*format == '.') {
Packit Service ae04f2
			format++;
Packit Service ae04f2
			dot = 1;
Packit Service ae04f2
			if (*format == '*') {
Packit Service ae04f2
				precision = va_arg(ap, int);
Packit Service ae04f2
				precision_set = true;
Packit Service ae04f2
				format++;
Packit Service ae04f2
			} else if (isdigit((unsigned char)*format)) {
Packit Service ae04f2
				char *e;
Packit Service ae04f2
				precision = strtoul(format, &e, 10);
Packit Service ae04f2
				precision_set = true;
Packit Service ae04f2
				format = e;
Packit Service ae04f2
			}
Packit Service ae04f2
		}
Packit Service ae04f2
Packit Service ae04f2
		switch (*format) {
Packit Service ae04f2
		case '\0':
Packit Service ae04f2
			continue;
Packit Service ae04f2
		case '%':
Packit Service ae04f2
			emit(*format, arg);
Packit Service ae04f2
			count++;
Packit Service ae04f2
			break;
Packit Service ae04f2
		case 'q':
Packit Service ae04f2
			q = 1;
Packit Service ae04f2
			format++;
Packit Service ae04f2
			goto doint;
Packit Service ae04f2
		case 'h':
Packit Service ae04f2
			h = 1;
Packit Service ae04f2
			format++;
Packit Service ae04f2
			goto doint;
Packit Service ae04f2
		case 'l':
Packit Service ae04f2
			l = 1;
Packit Service ae04f2
			format++;
Packit Service ae04f2
			if (*format == 'l') {
Packit Service ae04f2
				q = 1;
Packit Service ae04f2
				format++;
Packit Service ae04f2
			}
Packit Service ae04f2
			goto doint;
Packit Service ae04f2
		case 'z':
Packit Service ae04f2
			z = 1;
Packit Service ae04f2
			format++;
Packit Service ae04f2
			goto doint;
Packit Service ae04f2
#ifdef WIN32
Packit Service ae04f2
		case 'I':
Packit Service ae04f2
			/* Windows has I64 as a modifier for a quad. */
Packit Service ae04f2
			if (format[1] == '6' && format[2] == '4') {
Packit Service ae04f2
				q = 1;
Packit Service ae04f2
				format += 3;
Packit Service ae04f2
				goto doint;
Packit Service ae04f2
			}
Packit Service ae04f2
			continue;
Packit Service ae04f2
#endif
Packit Service ae04f2
		case 'n':
Packit Service ae04f2
		case 'i':
Packit Service ae04f2
		case 'd':
Packit Service ae04f2
		case 'o':
Packit Service ae04f2
		case 'u':
Packit Service ae04f2
		case 'x':
Packit Service ae04f2
		case 'X':
Packit Service ae04f2
		doint:
Packit Service ae04f2
			if (precision != 0U)
Packit Service ae04f2
				zero = 0;
Packit Service ae04f2
			switch (*format) {
Packit Service ae04f2
			case 'n':
Packit Service ae04f2
				if (h) {
Packit Service ae04f2
					short int *p;
Packit Service ae04f2
					p = va_arg(ap, short *);
Packit Service ae04f2
					assert(p != NULL);
Packit Service ae04f2
					*p = count;
Packit Service ae04f2
				} else if (l) {
Packit Service ae04f2
					long int *p;
Packit Service ae04f2
					p = va_arg(ap, long *);
Packit Service ae04f2
					assert(p != NULL);
Packit Service ae04f2
					*p = count;
Packit Service ae04f2
				} else if (z) {
Packit Service ae04f2
					size_t *p;
Packit Service ae04f2
					p = va_arg(ap, size_t *);
Packit Service ae04f2
					assert(p != NULL);
Packit Service ae04f2
					*p = count;
Packit Service ae04f2
				} else {
Packit Service ae04f2
					int *p;
Packit Service ae04f2
					p = va_arg(ap, int *);
Packit Service ae04f2
					assert(p != NULL);
Packit Service ae04f2
					*p = count;
Packit Service ae04f2
				}
Packit Service ae04f2
				break;
Packit Service ae04f2
			case 'i':
Packit Service ae04f2
			case 'd':
Packit Service ae04f2
				if (q)
Packit Service ae04f2
					tmpi = va_arg(ap, int64_t);
Packit Service ae04f2
				else if (l)
Packit Service ae04f2
					tmpi = va_arg(ap, long int);
Packit Service ae04f2
				else if (z)
Packit Service ae04f2
					tmpi = va_arg(ap, ssize_t);
Packit Service ae04f2
				else
Packit Service ae04f2
					tmpi = va_arg(ap, int);
Packit Service ae04f2
				if (tmpi < 0) {
Packit Service ae04f2
					head = "-";
Packit Service ae04f2
					tmpui = -tmpi;
Packit Service ae04f2
				} else {
Packit Service ae04f2
					if (plus)
Packit Service ae04f2
						head = "+";
Packit Service ae04f2
					else if (space)
Packit Service ae04f2
						head = " ";
Packit Service ae04f2
					else
Packit Service ae04f2
						head = "";
Packit Service ae04f2
					tmpui = tmpi;
Packit Service ae04f2
				}
Packit Service ae04f2
				if (tmpui <= 0xffffffffU)
Packit Service ae04f2
					sprintf(buf, "%lu",
Packit Service ae04f2
						(unsigned long)tmpui);
Packit Service ae04f2
				else {
Packit Service ae04f2
					unsigned long mid;
Packit Service ae04f2
					unsigned long lo;
Packit Service ae04f2
					unsigned long hi;
Packit Service ae04f2
					lo = tmpui % 1000000000;
Packit Service ae04f2
					tmpui /= 1000000000;
Packit Service ae04f2
					mid = tmpui % 1000000000;
Packit Service ae04f2
					hi = tmpui / 1000000000;
Packit Service ae04f2
					if (hi != 0U) {
Packit Service ae04f2
						sprintf(buf, "%lu", hi);
Packit Service ae04f2
						sprintf(buf + strlen(buf),
Packit Service ae04f2
							"%09lu", mid);
Packit Service ae04f2
					} else
Packit Service ae04f2
						sprintf(buf, "%lu", mid);
Packit Service ae04f2
					sprintf(buf + strlen(buf), "%09lu",
Packit Service ae04f2
						lo);
Packit Service ae04f2
				}
Packit Service ae04f2
				goto printint;
Packit Service ae04f2
			case 'o':
Packit Service ae04f2
				if (q)
Packit Service ae04f2
					tmpui = va_arg(ap, uint64_t);
Packit Service ae04f2
				else if (l)
Packit Service ae04f2
					tmpui = va_arg(ap, long int);
Packit Service ae04f2
				else if (z)
Packit Service ae04f2
					tmpui = va_arg(ap, size_t);
Packit Service ae04f2
				else
Packit Service ae04f2
					tmpui = va_arg(ap, int);
Packit Service ae04f2
				if (tmpui <= 0xffffffffU)
Packit Service ae04f2
					sprintf(buf, alt ?  "%#lo" : "%lo",
Packit Service ae04f2
						(unsigned long)tmpui);
Packit Service ae04f2
				else {
Packit Service ae04f2
					unsigned long mid;
Packit Service ae04f2
					unsigned long lo;
Packit Service ae04f2
					unsigned long hi;
Packit Service ae04f2
					lo = tmpui % 010000000000;
Packit Service ae04f2
					tmpui /= 010000000000;
Packit Service ae04f2
					mid = tmpui % 010000000000;
Packit Service ae04f2
					hi = tmpui / 010000000000;
Packit Service ae04f2
					if (hi != 0U) {
Packit Service ae04f2
						sprintf(buf,
Packit Service ae04f2
							alt ?  "%#lo" : "%lo",
Packit Service ae04f2
							hi);
Packit Service ae04f2
						sprintf(buf + strlen(buf),
Packit Service ae04f2
							"%09lo", mid);
Packit Service ae04f2
					} else
Packit Service ae04f2
						sprintf(buf,
Packit Service ae04f2
							alt ?  "%#lo" : "%lo",
Packit Service ae04f2
							mid);
Packit Service ae04f2
					sprintf(buf + strlen(buf), "%09lo", lo);
Packit Service ae04f2
				}
Packit Service ae04f2
				goto printint;
Packit Service ae04f2
			case 'u':
Packit Service ae04f2
				if (q)
Packit Service ae04f2
					tmpui = va_arg(ap, uint64_t);
Packit Service ae04f2
				else if (l)
Packit Service ae04f2
					tmpui = va_arg(ap, unsigned long int);
Packit Service ae04f2
				else if (z)
Packit Service ae04f2
					tmpui = va_arg(ap, size_t);
Packit Service ae04f2
				else
Packit Service ae04f2
					tmpui = va_arg(ap, unsigned int);
Packit Service ae04f2
				if (tmpui <= 0xffffffffU)
Packit Service ae04f2
					sprintf(buf, "%lu",
Packit Service ae04f2
						(unsigned long)tmpui);
Packit Service ae04f2
				else {
Packit Service ae04f2
					unsigned long mid;
Packit Service ae04f2
					unsigned long lo;
Packit Service ae04f2
					unsigned long hi;
Packit Service ae04f2
					lo = tmpui % 1000000000;
Packit Service ae04f2
					tmpui /= 1000000000;
Packit Service ae04f2
					mid = tmpui % 1000000000;
Packit Service ae04f2
					hi = tmpui / 1000000000;
Packit Service ae04f2
					if (hi != 0U) {
Packit Service ae04f2
						sprintf(buf, "%lu", hi);
Packit Service ae04f2
						sprintf(buf + strlen(buf),
Packit Service ae04f2
							"%09lu", mid);
Packit Service ae04f2
					 } else
Packit Service ae04f2
						sprintf(buf, "%lu", mid);
Packit Service ae04f2
					sprintf(buf + strlen(buf), "%09lu",
Packit Service ae04f2
						lo);
Packit Service ae04f2
				}
Packit Service ae04f2
				goto printint;
Packit Service ae04f2
			case 'x':
Packit Service ae04f2
				if (q)
Packit Service ae04f2
					tmpui = va_arg(ap, uint64_t);
Packit Service ae04f2
				else if (l)
Packit Service ae04f2
					tmpui = va_arg(ap, unsigned long int);
Packit Service ae04f2
				else if (z)
Packit Service ae04f2
					tmpui = va_arg(ap, size_t);
Packit Service ae04f2
				else
Packit Service ae04f2
					tmpui = va_arg(ap, unsigned int);
Packit Service ae04f2
				if (alt) {
Packit Service ae04f2
					head = "0x";
Packit Service ae04f2
					if (precision > 2U)
Packit Service ae04f2
						precision -= 2;
Packit Service ae04f2
				}
Packit Service ae04f2
				if (tmpui <= 0xffffffffU)
Packit Service ae04f2
					sprintf(buf, "%lx",
Packit Service ae04f2
						(unsigned long)tmpui);
Packit Service ae04f2
				else {
Packit Service ae04f2
					unsigned long hi = tmpui>>32;
Packit Service ae04f2
					unsigned long lo = tmpui & 0xffffffff;
Packit Service ae04f2
					sprintf(buf, "%lx", hi);
Packit Service ae04f2
					sprintf(buf + strlen(buf), "%08lx", lo);
Packit Service ae04f2
				}
Packit Service ae04f2
				goto printint;
Packit Service ae04f2
			case 'X':
Packit Service ae04f2
				if (q)
Packit Service ae04f2
					tmpui = va_arg(ap, uint64_t);
Packit Service ae04f2
				else if (l)
Packit Service ae04f2
					tmpui = va_arg(ap, unsigned long int);
Packit Service ae04f2
				else if (z)
Packit Service ae04f2
					tmpui = va_arg(ap, size_t);
Packit Service ae04f2
				else
Packit Service ae04f2
					tmpui = va_arg(ap, unsigned int);
Packit Service ae04f2
				if (alt) {
Packit Service ae04f2
					head = "0X";
Packit Service ae04f2
					if (precision > 2U)
Packit Service ae04f2
						precision -= 2;
Packit Service ae04f2
				}
Packit Service ae04f2
				if (tmpui <= 0xffffffffU)
Packit Service ae04f2
					sprintf(buf, "%lX",
Packit Service ae04f2
						(unsigned long)tmpui);
Packit Service ae04f2
				else  {
Packit Service ae04f2
					unsigned long hi = tmpui>>32;
Packit Service ae04f2
					unsigned long lo = tmpui & 0xffffffff;
Packit Service ae04f2
					sprintf(buf, "%lX", hi);
Packit Service ae04f2
					sprintf(buf + strlen(buf), "%08lX", lo);
Packit Service ae04f2
				}
Packit Service ae04f2
				goto printint;
Packit Service ae04f2
			printint:
Packit Service ae04f2
				if (precision_set || width != 0U) {
Packit Service ae04f2
					length = strlen(buf);
Packit Service ae04f2
					if (length < precision)
Packit Service ae04f2
						zeropad = precision - length;
Packit Service ae04f2
					else if (length < width && zero)
Packit Service ae04f2
						zeropad = width - length;
Packit Service ae04f2
					if (width != 0U) {
Packit Service ae04f2
						pad = width - length -
Packit Service ae04f2
						      zeropad - strlen(head);
Packit Service ae04f2
						if (pad < 0)
Packit Service ae04f2
							pad = 0;
Packit Service ae04f2
					}
Packit Service ae04f2
				}
Packit Service ae04f2
				count += strlen(head) + strlen(buf) + pad +
Packit Service ae04f2
					 zeropad;
Packit Service ae04f2
				if (!left) {
Packit Service ae04f2
					while (pad > 0) {
Packit Service ae04f2
						emit(' ', arg);
Packit Service ae04f2
						pad--;
Packit Service ae04f2
					}
Packit Service ae04f2
				}
Packit Service ae04f2
				cp = head;
Packit Service ae04f2
				while (*cp != '\0')
Packit Service ae04f2
					emit(*cp++, arg);
Packit Service ae04f2
				while (zeropad > 0) {
Packit Service ae04f2
					emit('0', arg);
Packit Service ae04f2
					zeropad--;
Packit Service ae04f2
				}
Packit Service ae04f2
				cp = buf;
Packit Service ae04f2
				while (*cp != '\0')
Packit Service ae04f2
					emit(*cp++, arg);
Packit Service ae04f2
				while (pad > 0) {
Packit Service ae04f2
					emit(' ', arg);
Packit Service ae04f2
					pad--;
Packit Service ae04f2
				}
Packit Service ae04f2
				break;
Packit Service ae04f2
			default:
Packit Service ae04f2
				break;
Packit Service ae04f2
			}
Packit Service ae04f2
			break;
Packit Service ae04f2
		case 's':
Packit Service ae04f2
			cp = va_arg(ap, char *);
Packit Service ae04f2
Packit Service ae04f2
			if (precision_set) {
Packit Service ae04f2
				/*
Packit Service ae04f2
				 * cp need not be NULL terminated.
Packit Service ae04f2
				 */
Packit Service ae04f2
				const char *tp;
Packit Service ae04f2
				unsigned long n;
Packit Service ae04f2
Packit Service ae04f2
				if (precision != 0U)
Packit Service ae04f2
					assert(cp != NULL);
Packit Service ae04f2
				n = precision;
Packit Service ae04f2
				tp = cp;
Packit Service ae04f2
				while (n != 0U && *tp != '\0')
Packit Service ae04f2
					n--, tp++;
Packit Service ae04f2
				length = precision - n;
Packit Service ae04f2
			} else {
Packit Service ae04f2
				assert(cp != NULL);
Packit Service ae04f2
				length = strlen(cp);
Packit Service ae04f2
			}
Packit Service ae04f2
			if (width != 0U) {
Packit Service ae04f2
				pad = width - length;
Packit Service ae04f2
				if (pad < 0)
Packit Service ae04f2
					pad = 0;
Packit Service ae04f2
			}
Packit Service ae04f2
			count += pad + length;
Packit Service ae04f2
			if (!left)
Packit Service ae04f2
				while (pad > 0) {
Packit Service ae04f2
					emit(' ', arg);
Packit Service ae04f2
					pad--;
Packit Service ae04f2
				}
Packit Service ae04f2
			if (precision_set)
Packit Service ae04f2
				while (precision > 0U && *cp != '\0') {
Packit Service ae04f2
					emit(*cp++, arg);
Packit Service ae04f2
					precision--;
Packit Service ae04f2
				}
Packit Service ae04f2
			else
Packit Service ae04f2
				while (*cp != '\0')
Packit Service ae04f2
					emit(*cp++, arg);
Packit Service ae04f2
			while (pad > 0) {
Packit Service ae04f2
				emit(' ', arg);
Packit Service ae04f2
				pad--;
Packit Service ae04f2
			}
Packit Service ae04f2
			break;
Packit Service ae04f2
		case 'c':
Packit Service ae04f2
			c = va_arg(ap, int);
Packit Service ae04f2
			if (width > 0U) {
Packit Service ae04f2
				count += width;
Packit Service ae04f2
				width--;
Packit Service ae04f2
				if (left)
Packit Service ae04f2
					emit(c, arg);
Packit Service ae04f2
				while (width-- > 0U)
Packit Service ae04f2
					emit(' ', arg);
Packit Service ae04f2
				if (!left)
Packit Service ae04f2
					emit(c, arg);
Packit Service ae04f2
			} else {
Packit Service ae04f2
				count++;
Packit Service ae04f2
				emit(c, arg);
Packit Service ae04f2
			}
Packit Service ae04f2
			break;
Packit Service ae04f2
		case 'p':
Packit Service ae04f2
			v = va_arg(ap, void *);
Packit Service ae04f2
			sprintf(buf, "%p", v);
Packit Service ae04f2
			length = strlen(buf);
Packit Service ae04f2
			if (precision > length)
Packit Service ae04f2
				zeropad = precision - length;
Packit Service ae04f2
			if (width > 0U) {
Packit Service ae04f2
				pad = width - length - zeropad;
Packit Service ae04f2
				if (pad < 0)
Packit Service ae04f2
					pad = 0;
Packit Service ae04f2
			}
Packit Service ae04f2
			count += length + pad + zeropad;
Packit Service ae04f2
			if (!left)
Packit Service ae04f2
				while (pad > 0) {
Packit Service ae04f2
					emit(' ', arg);
Packit Service ae04f2
					pad--;
Packit Service ae04f2
				}
Packit Service ae04f2
			cp = buf;
Packit Service ae04f2
			if (zeropad > 0 && buf[0] == '0' &&
Packit Service ae04f2
			    (buf[1] == 'x' || buf[1] == 'X')) {
Packit Service ae04f2
				emit(*cp++, arg);
Packit Service ae04f2
				emit(*cp++, arg);
Packit Service ae04f2
				while (zeropad > 0) {
Packit Service ae04f2
					emit('0', arg);
Packit Service ae04f2
					zeropad--;
Packit Service ae04f2
				}
Packit Service ae04f2
			}
Packit Service ae04f2
			while (*cp != '\0')
Packit Service ae04f2
				emit(*cp++, arg);
Packit Service ae04f2
			while (pad > 0) {
Packit Service ae04f2
				emit(' ', arg);
Packit Service ae04f2
				pad--;
Packit Service ae04f2
			}
Packit Service ae04f2
			break;
Packit Service ae04f2
		case 'D':	/*deprecated*/
Packit Service ae04f2
			/* cppcheck-suppress literalWithCharPtrCompare */
Packit Service ae04f2
			assert("use %ld instead of %D" == NULL);
Packit Service ae04f2
		case 'O':	/*deprecated*/
Packit Service ae04f2
			/* cppcheck-suppress literalWithCharPtrCompare */
Packit Service ae04f2
			assert("use %lo instead of %O" == NULL);
Packit Service ae04f2
		case 'U':	/*deprecated*/
Packit Service ae04f2
			/* cppcheck-suppress literalWithCharPtrCompare */
Packit Service ae04f2
			assert("use %lu instead of %U" == NULL);
Packit Service ae04f2
Packit Service ae04f2
		case 'L':
Packit Service ae04f2
#ifdef HAVE_LONG_DOUBLE
Packit Service ae04f2
			l = 1;
Packit Service ae04f2
#else
Packit Service ae04f2
			/* cppcheck-suppress literalWithCharPtrCompare */
Packit Service ae04f2
			assert("long doubles are not supported" == NULL);
Packit Service ae04f2
#endif
Packit Service ae04f2
			/* FALLTHROUGH */
Packit Service ae04f2
		case 'e':
Packit Service ae04f2
		case 'E':
Packit Service ae04f2
		case 'f':
Packit Service ae04f2
		case 'g':
Packit Service ae04f2
		case 'G':
Packit Service ae04f2
			if (!dot)
Packit Service ae04f2
				precision = 6;
Packit Service ae04f2
			/*
Packit Service ae04f2
			 * IEEE floating point.
Packit Service ae04f2
			 * MIN 2.2250738585072014E-308
Packit Service ae04f2
			 * MAX 1.7976931348623157E+308
Packit Service ae04f2
			 * VAX floating point has a smaller range than IEEE.
Packit Service ae04f2
			 *
Packit Service ae04f2
			 * precisions > 324 don't make much sense.
Packit Service ae04f2
			 * if we cap the precision at 512 we will not
Packit Service ae04f2
			 * overflow buf.
Packit Service ae04f2
			 */
Packit Service ae04f2
			if (precision > 512U)
Packit Service ae04f2
				precision = 512;
Packit Service ae04f2
			sprintf(fmt, "%%%s%s.%lu%s%c", alt ? "#" : "",
Packit Service ae04f2
				plus ? "+" : space ? " " : "",
Packit Service ae04f2
				precision, l ? "L" : "", *format);
Packit Service ae04f2
			switch (*format) {
Packit Service ae04f2
			case 'e':
Packit Service ae04f2
			case 'E':
Packit Service ae04f2
			case 'f':
Packit Service ae04f2
			case 'g':
Packit Service ae04f2
			case 'G':
Packit Service ae04f2
#ifdef HAVE_LONG_DOUBLE
Packit Service ae04f2
				if (l) {
Packit Service ae04f2
					ldbl = va_arg(ap, long double);
Packit Service ae04f2
					sprintf(buf, fmt, ldbl);
Packit Service ae04f2
				} else
Packit Service ae04f2
#endif
Packit Service ae04f2
				{
Packit Service ae04f2
					dbl = va_arg(ap, double);
Packit Service ae04f2
					sprintf(buf, fmt, dbl);
Packit Service ae04f2
				}
Packit Service ae04f2
				length = strlen(buf);
Packit Service ae04f2
				if (width > 0U) {
Packit Service ae04f2
					pad = width - length;
Packit Service ae04f2
					if (pad < 0)
Packit Service ae04f2
						pad = 0;
Packit Service ae04f2
				}
Packit Service ae04f2
				count += length + pad;
Packit Service ae04f2
				if (!left)
Packit Service ae04f2
					while (pad > 0) {
Packit Service ae04f2
						emit(' ', arg);
Packit Service ae04f2
						pad--;
Packit Service ae04f2
					}
Packit Service ae04f2
				cp = buf;
Packit Service ae04f2
				while (*cp != '\0')
Packit Service ae04f2
					emit(*cp++, arg);
Packit Service ae04f2
				while (pad > 0) {
Packit Service ae04f2
					emit(' ', arg);
Packit Service ae04f2
					pad--;
Packit Service ae04f2
				}
Packit Service ae04f2
				break;
Packit Service ae04f2
			default:
Packit Service ae04f2
				continue;
Packit Service ae04f2
			}
Packit Service ae04f2
			break;
Packit Service ae04f2
		default:
Packit Service ae04f2
			continue;
Packit Service ae04f2
		}
Packit Service ae04f2
		format++;
Packit Service ae04f2
	}
Packit Service ae04f2
	return (count);
Packit Service ae04f2
}