Skip to content
Snippets Groups Projects
vsprintf.c 47.2 KiB
Newer Older
  • Learn to ignore specific revisions
  • Linus Torvalds's avatar
    Linus Torvalds committed
    /*
     *  linux/lib/vsprintf.c
     *
     *  Copyright (C) 1991, 1992  Linus Torvalds
     */
    
    /* vsprintf.c -- Lars Wirzenius & Linus Torvalds. */
    /*
     * Wirzenius wrote this portably, Torvalds fucked it up :-)
     */
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
     * Fri Jul 13 2001 Crutcher Dunnavant <crutcher+kernel@datastacks.com>
     * - changed to provide snprintf and vsnprintf functions
     * So Feb  1 16:51:32 CET 2004 Juergen Quade <quade@hsnr.de>
     * - scnprintf and vscnprintf
     */
    
    #include <stdarg.h>
    #include <linux/module.h>
    #include <linux/types.h>
    #include <linux/string.h>
    #include <linux/ctype.h>
    #include <linux/kernel.h>
    
    #include <linux/kallsyms.h>
    #include <linux/uaccess.h>
    
    #include <linux/ioport.h>
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    #include <asm/page.h>		/* for PAGE_SIZE */
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #include <asm/div64.h>
    
    #include <asm/sections.h>	/* for dereference_function_descriptor() */
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    /* Works only for digits and letters, but small and fast */
    #define TOLOWER(x) ((x) | 0x20)
    
    
    static unsigned int simple_guess_base(const char *cp)
    {
    	if (cp[0] == '0') {
    		if (TOLOWER(cp[1]) == 'x' && isxdigit(cp[2]))
    			return 16;
    		else
    			return 8;
    	} else {
    		return 10;
    	}
    }
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    /**
    
     * simple_strtoull - convert a string to an unsigned long long
    
    Linus Torvalds's avatar
    Linus Torvalds committed
     * @cp: The start of the string
     * @endp: A pointer to the end of the parsed string will be placed here
     * @base: The number base to use
     */
    
    unsigned long long simple_strtoull(const char *cp, char **endp, unsigned int base)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    
    	unsigned long long result = 0;
    
    
    	if (!base)
    		base = simple_guess_base(cp);
    
    	if (base == 16 && cp[0] == '0' && TOLOWER(cp[1]) == 'x')
    		cp += 2;
    
    	while (isxdigit(*cp)) {
    		unsigned int value;
    
    		value = isdigit(*cp) ? *cp - '0' : TOLOWER(*cp) - 'a' + 10;
    		if (value >= base)
    			break;
    		result = result * base + value;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		cp++;
    	}
    	if (endp)
    		*endp = (char *)cp;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	return result;
    }
    
    EXPORT_SYMBOL(simple_strtoull);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    /**
    
     * simple_strtoul - convert a string to an unsigned long
    
    Linus Torvalds's avatar
    Linus Torvalds committed
     * @cp: The start of the string
     * @endp: A pointer to the end of the parsed string will be placed here
     * @base: The number base to use
     */
    
    unsigned long simple_strtoul(const char *cp, char **endp, unsigned int base)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    
    	return simple_strtoull(cp, endp, base);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    EXPORT_SYMBOL(simple_strtoul);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    /**
    
     * simple_strtol - convert a string to a signed long
    
    Linus Torvalds's avatar
    Linus Torvalds committed
     * @cp: The start of the string
     * @endp: A pointer to the end of the parsed string will be placed here
     * @base: The number base to use
     */
    
    long simple_strtol(const char *cp, char **endp, unsigned int base)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    
    	if (*cp == '-')
    		return -simple_strtoul(cp + 1, endp, base);
    
    	return simple_strtoul(cp, endp, base);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    /**
     * simple_strtoll - convert a string to a signed long long
     * @cp: The start of the string
     * @endp: A pointer to the end of the parsed string will be placed here
     * @base: The number base to use
     */
    
    long long simple_strtoll(const char *cp, char **endp, unsigned int base)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    
    		return -simple_strtoull(cp + 1, endp, base);
    
    	return simple_strtoull(cp, endp, base);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    }
    
    EXPORT_SYMBOL(simple_strtoll);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    static noinline_for_stack
    int skip_atoi(const char **s)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    	while (isdigit(**s))
    		i = i*10 + *((*s)++) - '0';
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	return i;
    }
    
    
    /* Decimal conversion is by far the most typical, and is used
     * for /proc and /sys data. This directly impacts e.g. top performance
     * with many processes running. We optimize it for speed
     * using code from
     * http://www.cs.uiowa.edu/~jones/bcd/decimal.html
     * (with permission from the author, Douglas W. Jones). */
    
    /* Formats correctly any integer in [0,99999].
     * Outputs from one to five digits depending on input.
     * On i386 gcc 4.1.2 -O2: ~250 bytes of code. */
    
    static noinline_for_stack
    char *put_dec_trunc(char *buf, unsigned q)
    
    {
    	unsigned d3, d2, d1, d0;
    	d1 = (q>>4) & 0xf;
    	d2 = (q>>8) & 0xf;
    	d3 = (q>>12);
    
    	d0 = 6*(d3 + d2 + d1) + (q & 0xf);
    	q = (d0 * 0xcd) >> 11;
    	d0 = d0 - 10*q;
    	*buf++ = d0 + '0'; /* least significant digit */
    	d1 = q + 9*d3 + 5*d2 + d1;
    	if (d1 != 0) {
    		q = (d1 * 0xcd) >> 11;
    		d1 = d1 - 10*q;
    		*buf++ = d1 + '0'; /* next digit */
    
    		d2 = q + 2*d2;
    		if ((d2 != 0) || (d3 != 0)) {
    			q = (d2 * 0xd) >> 7;
    			d2 = d2 - 10*q;
    			*buf++ = d2 + '0'; /* next digit */
    
    			d3 = q + 4*d3;
    			if (d3 != 0) {
    				q = (d3 * 0xcd) >> 11;
    				d3 = d3 - 10*q;
    				*buf++ = d3 + '0';  /* next digit */
    				if (q != 0)
    
    					*buf++ = q + '0'; /* most sign. digit */
    
    	return buf;
    }
    /* Same with if's removed. Always emits five digits */
    
    static noinline_for_stack
    char *put_dec_full(char *buf, unsigned q)
    
    {
    	/* BTW, if q is in [0,9999], 8-bit ints will be enough, */
    	/* but anyway, gcc produces better code with full-sized ints */
    	unsigned d3, d2, d1, d0;
    	d1 = (q>>4) & 0xf;
    	d2 = (q>>8) & 0xf;
    	d3 = (q>>12);
    
    
    	/*
    	 * Possible ways to approx. divide by 10
    	 * gcc -O2 replaces multiply with shifts and adds
    	 * (x * 0xcd) >> 11: 11001101 - shorter code than * 0x67 (on i386)
    	 * (x * 0x67) >> 10:  1100111
    	 * (x * 0x34) >> 9:    110100 - same
    	 * (x * 0x1a) >> 8:     11010 - same
    	 * (x * 0x0d) >> 7:      1101 - same, shortest code (on i386)
    	 */
    
    	d0 = 6*(d3 + d2 + d1) + (q & 0xf);
    	q = (d0 * 0xcd) >> 11;
    	d0 = d0 - 10*q;
    	*buf++ = d0 + '0';
    	d1 = q + 9*d3 + 5*d2 + d1;
    		q = (d1 * 0xcd) >> 11;
    		d1 = d1 - 10*q;
    		*buf++ = d1 + '0';
    
    		d2 = q + 2*d2;
    			q = (d2 * 0xd) >> 7;
    			d2 = d2 - 10*q;
    			*buf++ = d2 + '0';
    
    			d3 = q + 4*d3;
    				q = (d3 * 0xcd) >> 11; /* - shorter code */
    				/* q = (d3 * 0x67) >> 10; - would also work */
    				d3 = d3 - 10*q;
    				*buf++ = d3 + '0';
    					*buf++ = q + '0';
    
    	return buf;
    }
    /* No inlining helps gcc to use registers better */
    
    static noinline_for_stack
    char *put_dec(char *buf, unsigned long long num)
    
    {
    	while (1) {
    		unsigned rem;
    		if (num < 100000)
    			return put_dec_trunc(buf, num);
    		rem = do_div(num, 100000);
    		buf = put_dec_full(buf, rem);
    	}
    }
    
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    #define ZEROPAD	1		/* pad with zero */
    #define SIGN	2		/* unsigned/signed long */
    #define PLUS	4		/* show plus */
    #define SPACE	8		/* space if plus */
    #define LEFT	16		/* left justified */
    
    #define SMALL	32		/* use lowercase in hex (must be 32 == 0x20) */
    #define SPECIAL	64		/* prefix hex with "0x", octal with "0" */
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    
    
    enum format_type {
    	FORMAT_TYPE_NONE, /* Just a string part */
    
    	FORMAT_TYPE_PRECISION,
    	FORMAT_TYPE_CHAR,
    	FORMAT_TYPE_STR,
    	FORMAT_TYPE_PTR,
    	FORMAT_TYPE_PERCENT_CHAR,
    	FORMAT_TYPE_INVALID,
    	FORMAT_TYPE_LONG_LONG,
    	FORMAT_TYPE_ULONG,
    	FORMAT_TYPE_LONG,
    
    	FORMAT_TYPE_UBYTE,
    	FORMAT_TYPE_BYTE,
    
    	FORMAT_TYPE_USHORT,
    	FORMAT_TYPE_SHORT,
    	FORMAT_TYPE_UINT,
    	FORMAT_TYPE_INT,
    	FORMAT_TYPE_NRCHARS,
    	FORMAT_TYPE_SIZE_T,
    	FORMAT_TYPE_PTRDIFF
    };
    
    struct printf_spec {
    
    	u8	type;		/* format_type enum */
    
    	u8	flags;		/* flags to number() */
    
    	u8	base;		/* number base, 8, 10 or 16 only */
    	u8	qualifier;	/* number qualifier, one of 'hHlLtzZ' */
    	s16	field_width;	/* width of output field */
    	s16	precision;	/* # of digits/chars */
    
    static noinline_for_stack
    char *number(char *buf, char *end, unsigned long long num,
    	     struct printf_spec spec)
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    {
    
    	/* we are called with base 8, 10 or 16, only, thus don't need "G..."  */
    	static const char digits[16] = "0123456789ABCDEF"; /* "GHIJKLMNOPQRSTUVWXYZ"; */
    
    	char tmp[66];
    	char sign;
    	char locase;
    
    	int need_pfx = ((spec.flags & SPECIAL) && spec.base != 10);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	int i;
    
    
    	/* locase = 0 or 0x20. ORing digits or letters with 'locase'
    	 * produces same digits or (maybe lowercased) letters */
    
    	locase = (spec.flags & SMALL);
    	if (spec.flags & LEFT)
    		spec.flags &= ~ZEROPAD;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	sign = 0;
    
    		if ((signed long long)num < 0) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			sign = '-';
    
    			num = -(signed long long)num;
    
    			spec.field_width--;
    		} else if (spec.flags & PLUS) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			sign = '+';
    
    			spec.field_width--;
    		} else if (spec.flags & SPACE) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			sign = ' ';
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    		}
    	}
    
    		spec.field_width--;
    		if (spec.base == 16)
    			spec.field_width--;
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	}
    
    
    	/* generate full string in tmp[], in reverse order */
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	i = 0;
    	if (num == 0)
    
    	/* Generic code, for any base:
    	else do {
    
    		tmp[i++] = (digits[do_div(num,base)] | locase);
    
    	else if (spec.base != 10) { /* 8 or 16 */
    		int mask = spec.base - 1;
    
    			tmp[i++] = (digits[((unsigned char)num) & mask] | locase);
    
    			num >>= shift;
    		} while (num);
    
    	} else { /* base 10 */
    		i = put_dec(tmp, num) - tmp;
    	}
    
    
    	/* printing 100 using %2d gives "100", not "00" */
    
    	if (i > spec.precision)
    		spec.precision = i;
    
    	/* leading space padding */
    
    	spec.field_width -= spec.precision;
    	if (!(spec.flags & (ZEROPAD+LEFT))) {
    
    		while (--spec.field_width >= 0) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    				*buf = ' ';
    			++buf;
    		}
    	}
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	if (sign) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			*buf = sign;
    		++buf;
    	}
    
    	/* "0x" / "0" prefix */
    	if (need_pfx) {
    		if (buf < end)
    			*buf = '0';
    		++buf;
    
    				*buf = ('X' | locase);
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			++buf;
    		}
    	}
    
    	/* zero or space padding */
    
    	if (!(spec.flags & LEFT)) {
    		char c = (spec.flags & ZEROPAD) ? '0' : ' ';
    		while (--spec.field_width >= 0) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    				*buf = c;
    			++buf;
    		}
    	}
    
    	/* hmm even more zero padding? */
    
    	while (i <= --spec.precision) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			*buf = '0';
    		++buf;
    	}
    
    	/* actual digits of result */
    	while (--i >= 0) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			*buf = tmp[i];
    		++buf;
    	}
    
    	/* trailing space padding */
    
    	while (--spec.field_width >= 0) {
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    			*buf = ' ';
    		++buf;
    	}
    
    Linus Torvalds's avatar
    Linus Torvalds committed
    	return buf;
    }
    
    
    static noinline_for_stack
    char *string(char *buf, char *end, const char *s, struct printf_spec spec)
    
    {
    	int len, i;
    
    	if ((unsigned long)s < PAGE_SIZE)
    
    	len = strnlen(s, spec.precision);
    
    	if (!(spec.flags & LEFT)) {
    		while (len < spec.field_width--) {
    
    			if (buf < end)
    				*buf = ' ';
    			++buf;
    		}
    	}
    	for (i = 0; i < len; ++i) {
    		if (buf < end)
    			*buf = *s;
    		++buf; ++s;
    	}
    
    	while (len < spec.field_width--) {
    
    		if (buf < end)
    			*buf = ' ';
    		++buf;
    	}
    
    static noinline_for_stack
    char *symbol_string(char *buf, char *end, void *ptr,
    		    struct printf_spec spec, char ext)
    
    {
    	unsigned long value = (unsigned long) ptr;
    #ifdef CONFIG_KALLSYMS
    	char sym[KSYM_SYMBOL_LEN];
    
    	if (ext == 'B')
    		sprint_backtrace(sym, value);
    	else if (ext != 'f' && ext != 's')
    
    		sprint_symbol(sym, value);
    	else
    		kallsyms_lookup(value, NULL, NULL, NULL, sym);
    
    	return string(buf, end, sym, spec);
    
    	spec.field_width = 2 * sizeof(void *);
    
    	spec.flags |= SPECIAL | SMALL | ZEROPAD;
    	spec.base = 16;
    
    	return number(buf, end, value, spec);
    
    static noinline_for_stack
    char *resource_string(char *buf, char *end, struct resource *res,
    		      struct printf_spec spec, const char *fmt)
    
    {
    #ifndef IO_RSRC_PRINTK_SIZE
    
    #define IO_RSRC_PRINTK_SIZE	6
    
    #endif
    
    #ifndef MEM_RSRC_PRINTK_SIZE
    
    #define MEM_RSRC_PRINTK_SIZE	10
    
    	static const struct printf_spec io_spec = {
    
    		.field_width = IO_RSRC_PRINTK_SIZE,
    
    		.precision = -1,
    		.flags = SPECIAL | SMALL | ZEROPAD,
    	};
    
    	static const struct printf_spec mem_spec = {
    		.base = 16,
    		.field_width = MEM_RSRC_PRINTK_SIZE,
    		.precision = -1,
    		.flags = SPECIAL | SMALL | ZEROPAD,
    	};
    
    	static const struct printf_spec bus_spec = {
    		.base = 16,
    		.field_width = 2,
    		.precision = -1,
    		.flags = SMALL | ZEROPAD,
    	};
    
    	static const struct printf_spec dec_spec = {
    
    		.base = 10,
    		.precision = -1,
    		.flags = 0,
    	};
    
    	static const struct printf_spec str_spec = {
    
    		.field_width = -1,
    		.precision = 10,
    		.flags = LEFT,
    	};
    
    	static const struct printf_spec flag_spec = {
    
    		.base = 16,
    		.precision = -1,
    		.flags = SPECIAL | SMALL,
    	};
    
    
    	/* 32-bit res (sizeof==4): 10 chars in dec, 10 in hex ("0x" + 8)
    	 * 64-bit res (sizeof==8): 20 chars in dec, 18 in hex ("0x" + 16) */
    #define RSRC_BUF_SIZE		((2 * sizeof(resource_size_t)) + 4)
    #define FLAG_BUF_SIZE		(2 * sizeof(res->flags))
    
    #define DECODED_BUF_SIZE	sizeof("[mem - 64bit pref window disabled]")
    
    #define RAW_BUF_SIZE		sizeof("[mem - flags 0x]")
    	char sym[max(2*RSRC_BUF_SIZE + DECODED_BUF_SIZE,
    		     2*RSRC_BUF_SIZE + FLAG_BUF_SIZE + RAW_BUF_SIZE)];
    
    
    	char *p = sym, *pend = sym + sizeof(sym);
    
    	int decode = (fmt[0] == 'R') ? 1 : 0;
    
    	const struct printf_spec *specp;
    
    	if (res->flags & IORESOURCE_IO) {
    
    		p = string(p, pend, "io  ", str_spec);
    
    		specp = &io_spec;
    	} else if (res->flags & IORESOURCE_MEM) {
    
    		p = string(p, pend, "mem ", str_spec);
    
    		specp = &mem_spec;
    	} else if (res->flags & IORESOURCE_IRQ) {
    
    		p = string(p, pend, "irq ", str_spec);
    
    		specp = &dec_spec;
    	} else if (res->flags & IORESOURCE_DMA) {
    
    		p = string(p, pend, "dma ", str_spec);
    
    	} else if (res->flags & IORESOURCE_BUS) {
    		p = string(p, pend, "bus ", str_spec);
    		specp = &bus_spec;
    
    		p = string(p, pend, "??? ", str_spec);
    
    	p = number(p, pend, res->start, *specp);
    
    	if (res->start != res->end) {
    		*p++ = '-';
    
    		p = number(p, pend, res->end, *specp);
    
    		if (res->flags & IORESOURCE_MEM_64)
    			p = string(p, pend, " 64bit", str_spec);
    		if (res->flags & IORESOURCE_PREFETCH)
    			p = string(p, pend, " pref", str_spec);
    
    		if (res->flags & IORESOURCE_WINDOW)
    			p = string(p, pend, " window", str_spec);
    
    		if (res->flags & IORESOURCE_DISABLED)
    			p = string(p, pend, " disabled", str_spec);
    
    	} else {
    		p = string(p, pend, " flags ", str_spec);
    		p = number(p, pend, res->flags, flag_spec);
    
    	return string(buf, end, sym, spec);
    
    static noinline_for_stack
    char *mac_address_string(char *buf, char *end, u8 *addr,
    			 struct printf_spec spec, const char *fmt)
    
    	char mac_addr[sizeof("xx:xx:xx:xx:xx:xx")];
    
    	char *p = mac_addr;
    	int i;
    
    	char separator;
    
    	if (fmt[1] == 'F') {		/* FDDI canonical format */
    		separator = '-';
    	} else {
    		separator = ':';
    	}
    
    		p = pack_hex_byte(p, addr[i]);
    
    	return string(buf, end, mac_addr, spec);
    
    static noinline_for_stack
    char *ip4_string(char *p, const u8 *addr, const char *fmt)
    
    	bool leading_zeros = (fmt[0] == 'i');
    	int index;
    	int step;
    
    	switch (fmt[2]) {
    	case 'h':
    #ifdef __BIG_ENDIAN
    		index = 0;
    		step = 1;
    #else
    		index = 3;
    		step = -1;
    #endif
    		break;
    	case 'l':
    		index = 3;
    		step = -1;
    		break;
    	case 'n':
    	case 'b':
    	default:
    		index = 0;
    		step = 1;
    		break;
    	}
    
    	for (i = 0; i < 4; i++) {
    		char temp[3];	/* hold each IP quad in reverse order */
    
    		int digits = put_dec_trunc(temp, addr[index]) - temp;
    
    		if (leading_zeros) {
    			if (digits < 3)
    				*p++ = '0';
    			if (digits < 2)
    				*p++ = '0';
    		}
    		/* reverse the digits in the quad */
    		while (digits--)
    			*p++ = temp[digits];
    		if (i < 3)
    			*p++ = '.';
    
    static noinline_for_stack
    char *ip6_compressed_string(char *p, const char *addr)
    
    	unsigned char zerolength[8];
    	int longest = 1;
    	int colonpos = -1;
    	u16 word;
    
    	bool useIPv4;
    	struct in6_addr in6;
    
    	memcpy(&in6, addr, sizeof(struct in6_addr));
    
    	useIPv4 = ipv6_addr_v4mapped(&in6) || ipv6_addr_is_isatap(&in6);
    
    
    	memset(zerolength, 0, sizeof(zerolength));
    
    	if (useIPv4)
    		range = 6;
    	else
    		range = 8;
    
    	/* find position of longest 0 run */
    	for (i = 0; i < range; i++) {
    		for (j = i; j < range; j++) {
    
    			if (in6.s6_addr16[j] != 0)
    
    				break;
    			zerolength[i]++;
    		}
    	}
    	for (i = 0; i < range; i++) {
    		if (zerolength[i] > longest) {
    			longest = zerolength[i];
    			colonpos = i;
    		}
    	}
    
    	/* emit address */
    	for (i = 0; i < range; i++) {
    		if (i == colonpos) {
    			if (needcolon || i == 0)
    				*p++ = ':';
    			*p++ = ':';
    			needcolon = false;
    			i += longest - 1;
    			continue;
    		}
    		if (needcolon) {
    			*p++ = ':';
    			needcolon = false;
    		}
    		/* hex u16 without leading 0s */
    
    		word = ntohs(in6.s6_addr16[i]);
    
    		hi = word >> 8;
    		lo = word & 0xff;
    		if (hi) {
    			if (hi > 0x0f)
    				p = pack_hex_byte(p, hi);
    			else
    				*p++ = hex_asc_lo(hi);
    
    			p = pack_hex_byte(p, lo);
    
    			p = pack_hex_byte(p, lo);
    		else
    			*p++ = hex_asc_lo(lo);
    		needcolon = true;
    	}
    
    	if (useIPv4) {
    		if (needcolon)
    			*p++ = ':';
    
    		p = ip4_string(p, &in6.s6_addr[12], "I4");
    
    static noinline_for_stack
    char *ip6_string(char *p, const char *addr, const char *fmt)
    
    	for (i = 0; i < 8; i++) {
    
    		p = pack_hex_byte(p, *addr++);
    		p = pack_hex_byte(p, *addr++);
    
    static noinline_for_stack
    char *ip6_addr_string(char *buf, char *end, const u8 *addr,
    		      struct printf_spec spec, const char *fmt)
    
    {
    	char ip6_addr[sizeof("xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:255.255.255.255")];
    
    	if (fmt[0] == 'I' && fmt[2] == 'c')
    
    		ip6_compressed_string(ip6_addr, addr);
    
    		ip6_string(ip6_addr, addr, fmt);
    
    	return string(buf, end, ip6_addr, spec);
    
    static noinline_for_stack
    char *ip4_addr_string(char *buf, char *end, const u8 *addr,
    		      struct printf_spec spec, const char *fmt)
    
    	char ip4_addr[sizeof("255.255.255.255")];
    
    	return string(buf, end, ip4_addr, spec);
    
    static noinline_for_stack
    char *uuid_string(char *buf, char *end, const u8 *addr,
    		  struct printf_spec spec, const char *fmt)
    
    {
    	char uuid[sizeof("xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx")];
    	char *p = uuid;
    	int i;
    	static const u8 be[16] = {0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15};
    	static const u8 le[16] = {3,2,1,0,5,4,7,6,8,9,10,11,12,13,14,15};
    	const u8 *index = be;
    	bool uc = false;
    
    	switch (*(++fmt)) {
    	case 'L':
    		uc = true;		/* fall-through */
    	case 'l':
    		index = le;
    		break;
    	case 'B':
    		uc = true;
    		break;
    	}
    
    	for (i = 0; i < 16; i++) {
    		p = pack_hex_byte(p, addr[index[i]]);
    		switch (i) {
    		case 3:
    		case 5:
    		case 7:
    		case 9:
    			*p++ = '-';
    			break;
    		}
    	}
    
    	*p = 0;
    
    	if (uc) {
    		p = uuid;
    		do {
    			*p = toupper(*p);
    		} while (*(++p));
    	}
    
    	return string(buf, end, uuid, spec);
    }
    
    
    /*
     * Show a '%p' thing.  A kernel extension is that the '%p' is followed
     * by an extra set of alphanumeric characters that are extended format
     * specifiers.
     *
    
     * Right now we handle:
     *
    
     * - 'F' For symbolic function descriptor pointers with offset
     * - 'f' For simple symbolic function names without offset
    
     * - 'S' For symbolic direct pointers with offset
     * - 's' For symbolic direct pointers without offset
    
     * - 'B' For backtraced symbolic direct pointers with offset
    
     * - 'R' For decoded struct resource, e.g., [mem 0x0-0x1f 64bit pref]
     * - 'r' For raw struct resource, e.g., [mem 0x0-0x1f flags 0x201]
    
     * - 'M' For a 6-byte MAC address, it prints the address in the
     *       usual colon-separated hex notation
    
     * - 'm' For a 6-byte MAC address, it prints the hex address without colons
    
     * - 'MF' For a 6-byte MAC FDDI address, it prints the address
    
     *       with a dash-separated hex notation
    
     * - 'I' [46] for IPv4/IPv6 addresses printed in the usual way
     *       IPv4 uses dot-separated decimal without leading 0's (1.2.3.4)
     *       IPv6 uses colon separated network-order 16 bit hex with leading 0's
     * - 'i' [46] for 'raw' IPv4/IPv6 addresses
     *       IPv6 omits the colons (01020304...0f)
     *       IPv4 uses dot-separated decimal with leading 0's (010.123.045.006)
    
     * - '[Ii]4[hnbl]' IPv4 addresses in host, network, big or little endian order
    
     * - 'I6c' for IPv6 addresses printed as specified by
    
     *       http://tools.ietf.org/html/draft-ietf-6man-text-addr-representation-00
    
     * - 'U' For a 16 byte UUID/GUID, it prints the UUID/GUID in the form
     *       "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"
     *       Options for %pU are:
     *         b big endian lower case hex (default)
     *         B big endian UPPER case hex
     *         l little endian lower case hex
     *         L little endian UPPER case hex
     *           big endian output byte order is:
     *             [0][1][2][3]-[4][5]-[6][7]-[8][9]-[10][11][12][13][14][15]
     *           little endian output byte order is:
     *             [3][2][1][0]-[5][4]-[7][6]-[8][9]-[10][11][12][13][14][15]
    
     * - 'V' For a struct va_format which contains a format string * and va_list *,
     *       call vsnprintf(->format, *->va_list).
     *       Implements a "recursive vsnprintf".
     *       Do not use this feature without some mechanism to verify the
     *       correctness of the format string and va_list arguments.
    
     * - 'K' For a kernel pointer that should be hidden from unprivileged users
    
     * Note: The difference between 'S' and 'F' is that on ia64 and ppc64
     * function pointers are really function descriptors, which contain a
     * pointer to the real address.
    
    static noinline_for_stack
    char *pointer(const char *fmt, char *buf, char *end, void *ptr,
    	      struct printf_spec spec)
    
    	if (!ptr && *fmt != 'K') {
    
    		/*
    		 * Print (null) with the same width as a pointer so it makes
    		 * tabular output look nice.
    		 */
    		if (spec.field_width == -1)
    			spec.field_width = 2 * sizeof(void *);
    
    		return string(buf, end, "(null)", spec);
    
    		ptr = dereference_function_descriptor(ptr);
    		/* Fallthrough */
    	case 'S':
    
    		return symbol_string(buf, end, ptr, spec, *fmt);
    
    		return resource_string(buf, end, ptr, spec, fmt);
    
    	case 'M':			/* Colon separated: 00:01:02:03:04:05 */
    	case 'm':			/* Contiguous: 000102030405 */
    
    					/* [mM]F (FDDI, bit reversed) */
    
    		return mac_address_string(buf, end, ptr, spec, fmt);
    	case 'I':			/* Formatted IP supported
    					 * 4:	1.2.3.4
    					 * 6:	0001:0203:...:0708
    					 * 6c:	1::708 or 1::1.2.3.4
    					 */
    	case 'i':			/* Contiguous:
    					 * 4:	001.002.003.004
    					 * 6:   000102...0f
    					 */
    		switch (fmt[1]) {
    		case '6':
    			return ip6_addr_string(buf, end, ptr, spec, fmt);
    		case '4':
    			return ip4_addr_string(buf, end, ptr, spec, fmt);
    		}
    
    	case 'U':
    		return uuid_string(buf, end, ptr, spec, fmt);
    
    	case 'V':
    		return buf + vsnprintf(buf, end - buf,
    				       ((struct va_format *)ptr)->fmt,
    				       *(((struct va_format *)ptr)->va));
    
    	case 'K':
    		/*
    		 * %pK cannot be used in IRQ context because its test
    		 * for CAP_SYSLOG would be meaningless.
    		 */
    		if (in_irq() || in_serving_softirq() || in_nmi()) {
    			if (spec.field_width == -1)
    				spec.field_width = 2 * sizeof(void *);
    			return string(buf, end, "pK-error", spec);
    		}
    
    		if (!((kptr_restrict == 0) ||
    		      (kptr_restrict == 1 &&
    		       has_capability_noaudit(current, CAP_SYSLOG))))
    			ptr = NULL;
    		break;
    
    	}
    	spec.flags |= SMALL;
    	if (spec.field_width == -1) {
    
    		spec.field_width = 2 * sizeof(void *);
    
    		spec.flags |= ZEROPAD;
    	}
    	spec.base = 16;
    
    	return number(buf, end, (unsigned long) ptr, spec);
    }
    
    /*
     * Helper function to decode printf style format.
     * Each call decode a token from the format and return the
     * number of characters read (or likely the delta where it wants
     * to go on the next call).
     * The decoded token is returned through the parameters
     *
     * 'h', 'l', or 'L' for integer fields
     * 'z' support added 23/7/1999 S.H.
     * 'z' changed to 'Z' --davidm 1/25/99
     * 't' added for ptrdiff_t
     *
     * @fmt: the format string
     * @type of the token returned
     * @flags: various flags such as +, -, # tokens..
     * @field_width: overwritten width
     * @base: base of the number (octal, hex, ...)
     * @precision: precision of a number
     * @qualifier: qualifier of a number (long, size_t, ...)
     */
    
    static noinline_for_stack
    int format_decode(const char *fmt, struct printf_spec *spec)
    
    {
    	const char *start = fmt;
    
    	/* we finished early by reading the field width */
    
    	if (spec->type == FORMAT_TYPE_WIDTH) {
    
    		if (spec->field_width < 0) {
    			spec->field_width = -spec->field_width;
    			spec->flags |= LEFT;
    		}
    		spec->type = FORMAT_TYPE_NONE;
    		goto precision;
    	}
    
    	/* we finished early by reading the precision */
    	if (spec->type == FORMAT_TYPE_PRECISION) {
    		if (spec->precision < 0)
    			spec->precision = 0;
    
    		spec->type = FORMAT_TYPE_NONE;
    		goto qualifier;
    	}
    
    	/* By default */
    	spec->type = FORMAT_TYPE_NONE;
    
    	for (; *fmt ; ++fmt) {
    		if (*fmt == '%')
    			break;
    	}
    
    	/* Return the current non-format string */
    	if (fmt != start || !*fmt)
    		return fmt - start;
    
    	/* Process flags */
    	spec->flags = 0;
    
    	while (1) { /* this also skips first '%' */
    		bool found = true;
    
    		++fmt;
    
    		switch (*fmt) {
    		case '-': spec->flags |= LEFT;    break;
    		case '+': spec->flags |= PLUS;    break;
    		case ' ': spec->flags |= SPACE;   break;
    		case '#': spec->flags |= SPECIAL; break;
    		case '0': spec->flags |= ZEROPAD; break;
    		default:  found = false;