URL
https://opencores.org/ocsvn/xenie/xenie/trunk
Subversion Repositories xenie
[/] [xenie/] [trunk/] [examples/] [Eth_example/] [mb_fw/] [xenie_eth_test_womtd/] [src/] [uprintf.c] - Rev 4
Compare with Previous | Blame | View Log
/* * Sprintf is taken from dip122 project. * I just adopted it that msp430-gcc can handle it * The code is about 1500 bytes (depends on -O options) * * (c) msp gcc team http://mspgcc.sourceforge.net */ /****************************************************************************** This file is a patched version of printf called _printf_P It is made to work with avr-gcc for Atmel AVR MCUs. There are some differences from standard printf: 1. There is no floating point support (with fp the code is about 8K!) 2. Return type is void 3. Format string must be in program memory (by using macro printf this is done automaticaly) 4. %n is not implemented (just remove the comment around it if you need it) 5. If LIGHTPRINTF is defined, the code is about 550 bytes smaller and the folowing specifiers are disabled : space # * . - + p s o O 6. A function void uart_sendchar(char c) is used for output. The UART must be initialized before using printf. Alexander Popov sasho@vip.orbitel.bg small modifications to make it work with liblcd_dip122.a replaced uart_sendchar with (*write_char)(char) 2001 Michael Schaenzler *******************************************************************************/ /* * Actual printf innards. * * This code is large and complicated... */ #include <string.h> #ifdef __STDC__ #include <stdarg.h> #else #include <varargs.h> #endif static char *Bufs; static void (*uprintf_print_fn)(void *inst, const char *buf, int len); static void *uprintf_print_fn_inst; void uprintf_init(void (*print)(void *inst, const char *buf, int len), void *inst) { uprintf_print_fn = print; uprintf_print_fn_inst = inst; } #define PRINTP PRINT static void PRINT_f(const char * ptr, unsigned int len) { if (uprintf_print_fn) uprintf_print_fn(uprintf_print_fn_inst, ptr, len); else while(1); /* Uprintf was not initialized */ } static void PRINTS_f(const char * ptr, unsigned int len) { for(;len;len--) *Bufs++ = *ptr++; } #define PAD_SP(x) __write_pad(PRINT, ' ',x); #define PAD_0(x) __write_pad(PRINT,'0',x); static void __write_pad(void PRINT(const char * ptr, unsigned int len), char c, signed char howmany) { for(;howmany>0;howmany--) PRINT(&c, 1); } #define BUF 20 #define PRG_RDB(x) *x /* * Macros for converting digits to letters and vice versa */ #define to_digit(c) ((c) - '0') #define is_digit(c) ((c)<='9' && (c)>='0') #define to_char(n) ((n) + '0') /* * Flags used during conversion. */ #define LONGINT 0x01 /* long integer */ #define LONGDBL 0x02 /* long double; unimplemented */ #define SHORTINT 0x04 /* short integer */ #define ALT 0x08 /* alternate form */ #define LADJUST 0x10 /* left adjustment */ #define ZEROPAD 0x20 /* zero (as opposed to blank) pad */ #define HEXPREFIX 0x40 /* add 0x or 0X prefix */ int vuprintf (void PRINT(const char * ptr, unsigned int len), char const *fmt0, va_list ap); void uprintf (char const *fmt0, ...) { va_list ap; va_start(ap, fmt0); vuprintf(PRINT_f, fmt0, ap); va_end(ap); } int suprintf (char *buf, char const *fmt0, ...) { int ret; va_list ap; va_start(ap, fmt0); ret = vuprintf(PRINTS_f, fmt0, ap); buf[ret] = '\0'; va_end(ap); return ret; } int vuprintf (void PRINT(const char * ptr, unsigned int len), char const *fmt0, va_list ap) { register const char *fmt; /* format string */ register char ch; /* character from fmt */ register int n; /* handy integer (short term usage) */ register char *cp; /* handy char pointer (short term usage) */ const char *fmark; /* for remembering a place in fmt */ register unsigned char flags; /* flags as above */ signed char width; /* width from format (%8d), or 0 */ signed char prec; /* precision from format (%.3d), or -1 */ char sign; /* sign prefix (' ', '+', '-', or \0) */ unsigned long _ulong=0; /* integer arguments %[diouxX] */ #define OCT 8 #define DEC 10 #define HEX 16 unsigned char base; /* base for [diouxX] conversion */ signed char dprec; /* a copy of prec if [diouxX], 0 otherwise */ signed char dpad; /* extra 0 padding needed for integers */ signed char fieldsz; /* field size expanded by sign, dpad etc */ /* The initialization of 'size' is to suppress a warning that 'size' might be used unitialized. It seems gcc can't quite grok this spaghetti code ... */ signed char size = 0; /* size of converted field or string */ char buf[BUF]; /* space for %c, %[diouxX], %[eEfgG] */ char ox[2]; /* space for 0x hex-prefix */ int ret = 0; /* Number of bytes printed */ fmt = fmt0; /* * Scan the format for conversions (`%' character). */ for (;;) { for (fmark = fmt; (ch = PRG_RDB(fmt)) != '\0' && ch != '%'; fmt++) /* void */; if ((n = fmt - fmark) != 0) { PRINTP(fmark, n); ret += n; } if (ch == '\0') goto done; fmt++; /* skip over '%' */ flags = 0; dprec = 0; width = 0; prec = -1; sign = '\0'; rflag: ch = PRG_RDB(fmt++); reswitch: if (ch=='u' || (ch|0x20)=='x') { if (flags&LONGINT) { _ulong=va_arg(ap, unsigned long); } else { register unsigned int _d; _d=va_arg(ap, unsigned int); _ulong = flags&SHORTINT ? (unsigned long)(unsigned short)_d : (unsigned long)_d; } } if(ch==' ') { /* * ``If the space and + flags both appear, the space * flag will be ignored.'' * -- ANSI X3J11 */ if (!sign) sign = ' '; goto rflag; } else if (ch=='#') { flags |= ALT; goto rflag; } else if (ch=='*'||ch=='-') { if (ch=='*') { /* * ``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; } flags |= LADJUST; flags &= ~ZEROPAD; /* '-' disables '0' */ goto rflag; } else if (ch=='+') { sign = '+'; goto rflag; } else if (ch=='.') { if ((ch = PRG_RDB(fmt++)) == '*') { n = va_arg(ap, int); prec = n < 0 ? -1 : n; goto rflag; } n = 0; while (is_digit(ch)) { n = n*10 + to_digit(ch); ch = PRG_RDB(fmt++); } prec = n < 0 ? -1 : n; goto reswitch; } else if (ch=='0') { /* * ``Note that 0 is taken as a flag, not as the * beginning of a field width.'' * -- ANSI X3J11 */ if (!(flags & LADJUST)) flags |= ZEROPAD; /* '-' disables '0' */ goto rflag; } else if (ch>='1' && ch<='9') { n = 0; do { n = 10 * n + to_digit(ch); ch = PRG_RDB(fmt++); } while (is_digit(ch)); width = n; goto reswitch; } else if (ch=='h') { flags |= SHORTINT; goto rflag; } else if (ch=='l') { flags |= LONGINT; goto rflag; } else if (ch=='c') { *(cp = buf) = va_arg(ap, int); size = 1; sign = '\0'; } else if (ch=='D'||ch=='d'||ch=='i') { if(ch=='D') flags |= LONGINT; if (flags&LONGINT) { _ulong=va_arg(ap, long); } else { register int _d; _d=va_arg(ap, int); _ulong = flags&SHORTINT ? (long)(short)_d : (long)_d; } if ((long)_ulong < 0) { _ulong = -_ulong; sign = '-'; } base = DEC; goto number; } else if (ch=='O'||ch=='o') { if (ch=='O') flags |= LONGINT; base = OCT; goto nosign; } else if (ch=='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 */ /* NOSTRICT */ _ulong = (unsigned int)va_arg(ap, void *); base = HEX; flags |= HEXPREFIX; ch = 'x'; goto nosign; } else if (ch=='s') { /* print a string from RAM */ if ((cp = va_arg(ap, char *)) == NULL) { cp=buf; cp[0] = '('; cp[1] = 'n'; cp[2] = 'u'; cp[4] = cp[3] = 'l'; cp[5] = ')'; cp[6] = '\0'; } 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 = (char*)memchr(cp, 0, prec); if (p != NULL) { size = p - cp; if (size > prec) size = prec; } else size = prec; } else size = strlen(cp); sign = '\0'; } else if(ch=='U'||ch=='u') { if (ch=='U') flags |= LONGINT; base = DEC; goto nosign; } else if (ch=='X'||ch=='x') { base = HEX; /* leading 0x/X only if non-zero */ if ((flags & ALT) && _ulong != 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 (_ulong != 0 || prec != 0) { register unsigned char _d,notlastdigit; do { notlastdigit=(_ulong>=base); _d = _ulong % base; if (_d<10) { _d+='0'; } else { _d+='a'-10; if (ch=='X') _d&=~0x20; } *--cp=_d; _ulong /= base; } while (notlastdigit); /* handle octal leading 0 */ if (base==OCT && (flags & ALT) && *cp != '0') *--cp = '0'; } size = buf + BUF - cp; } else { /* 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'; } /* * 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 = size; dpad = dprec - size; if (dpad < 0) dpad = 0; if (sign) fieldsz++; else if (flags & HEXPREFIX) fieldsz += 2; fieldsz += dpad; /* right-adjusting blank padding */ if ((flags & (LADJUST|ZEROPAD)) == 0) { PAD_SP(width - fieldsz); ret += (width - fieldsz); } /* prefix */ if (sign) { PRINT(&sign, 1); ret += 1; } else if (flags & HEXPREFIX) { ox[0] = '0'; ox[1] = ch; PRINT(ox, 2); ret += 2; } /* right-adjusting zero padding */ if ((flags & (LADJUST|ZEROPAD)) == ZEROPAD) { PAD_0(width - fieldsz); ret += (width - fieldsz); } /* leading zeroes from decimal precision */ PAD_0(dpad); ret += dpad; /* the string or number proper */ PRINT(cp, size); ret += size; /* left-adjusting padding (always blank) */ if (flags & LADJUST) { PAD_SP(width - fieldsz); ret += (width - fieldsz); } } done: ; return ret; }