251 lines
		
	
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			251 lines
		
	
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* $Header$ */
 | |
| #include <stdio.h>
 | |
| #include <varargs.h>
 | |
| 
 | |
| #ifndef NOFLOAT
 | |
| extern char     *_pfloat();
 | |
| extern char     *_pscien();
 | |
| extern char     *gcvt();
 | |
| #endif
 | |
| 
 | |
| # define wsize(par) ( (sizeof par) / sizeof (int) )
 | |
| 
 | |
| 
 | |
| static char *gnum(f,ip,app) register char *f; int *ip; va_list *app; {
 | |
| 	register int    i,c;
 | |
| 
 | |
| 	if (*f == '*') {
 | |
| 		*ip = va_arg((*app), int);
 | |
| 		f++;
 | |
| 	} else {
 | |
| 		i = 0;
 | |
| 		while ((c = *f - '0') >= 0 && c <= 9) {
 | |
| 			i = i*10 + c;
 | |
| 			f++;
 | |
| 		}
 | |
| 		*ip = i;
 | |
| 	}
 | |
| 	return(f);
 | |
| }
 | |
| 
 | |
| #define signbit(par) (1L<<(sizeof par*8 -1))
 | |
| 
 | |
| static char *i_compute(val,base,s) unsigned val; char *s; {
 | |
| 	int c;
 | |
| 
 | |
| 	c= val % base ;
 | |
| 	val/= base ;
 | |
| 	if (val)
 | |
| 		s = i_compute(val,base,s);
 | |
| 	*s++ = (c>9 ? c-10+'a' : c+'0');
 | |
| 	return(s);
 | |
| }
 | |
| 
 | |
| #ifndef NOLONG
 | |
| static char *l_compute(l1,d,s) long l1; char *s; {
 | |
| 	int c;
 | |
| 	long l2;
 | |
| 
 | |
| 	if ( l1<0 ) {
 | |
| 		/* assumption: d is a multiple of 2 */
 | |
| 		c= l1&1 ;
 | |
| 		l2= ( (l1>>1) & ~signbit(l1) );
 | |
| 		l1= l2/(d>>1) ;
 | |
| 		c += (l2%(d>>1))<<1 ;
 | |
| 	} else {
 | |
| 		c= l1 % d ;
 | |
| 		l1= l1 / d ;
 | |
| 	}
 | |
| 	if (l1)
 | |
| 		s = l_compute(l1,d,s);
 | |
| 	*s++ = (c>9 ? c-10+'a' : c+'0');
 | |
| 	return(s);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| _doprnt(fmt,ap,stream)
 | |
| 	register char *fmt; va_list ap ; FILE *stream;
 | |
| {
 | |
| 	register char   *s;
 | |
| #ifndef NOLONG
 | |
| 	long            l;
 | |
| 	int             lflag ;
 | |
| #else
 | |
| #define lflag 0
 | |
| #endif
 | |
| #ifndef NOFLOAT
 | |
| 	double          dbl ;
 | |
| 	int		capitalE = 0;
 | |
| #endif
 | |
| 	int             inte ;
 | |
| 	unsigned int    uint ;
 | |
| 	register int    j ;
 | |
| 	int             i,c,rjust,width,ndigit,ndfnd,zfill;
 | |
| 	char            *oldfmt,*s1,buf[1025];
 | |
| 
 | |
| 	while (c = *fmt++) {
 | |
| 		if (c != '%') {
 | |
| #ifdef  CPM
 | |
| 			if (c == '\n') putc('\r',stream);
 | |
| #endif
 | |
| 			putc(c,stream);
 | |
| 			continue;
 | |
| 		}
 | |
| #ifndef NOLONG
 | |
| 		lflag = 0 ;
 | |
| #endif
 | |
| 		j = 10 ;
 | |
| 		rjust = 0;
 | |
| 		if (*fmt == '-') {
 | |
| 			fmt++;
 | |
| 			rjust++;
 | |
| 		}
 | |
| 		zfill = ' ';
 | |
| 		if (*fmt == '0') {
 | |
| 			fmt++;
 | |
| 			zfill = '0';
 | |
| 		}
 | |
| 		fmt = gnum(fmt,&width,&ap);
 | |
| 		ndigit = 0; ndfnd = 0;
 | |
| 		if (*fmt == '.') {
 | |
| 			fmt++; oldfmt = fmt;
 | |
| 			fmt = gnum(fmt,&ndigit,&ap);
 | |
| 			ndfnd = (fmt != oldfmt);
 | |
| 		}
 | |
| 		s = s1 = buf;
 | |
| #ifndef NOLONG
 | |
| 		if ( *fmt == 'l' || *fmt == 'L' ) {
 | |
| 			fmt++ ; lflag++ ;
 | |
| 		}
 | |
| #endif
 | |
| 		switch (c = *fmt++) {
 | |
| 		default:
 | |
| #ifdef  CPM
 | |
| 			if (c == '\n') putc('r',stream);
 | |
| #endif
 | |
| 			putc(c,stream);
 | |
| 			continue;
 | |
| 		case 's':
 | |
| 			s1 = va_arg(ap, char *);
 | |
| 			if (s1 == 0)
 | |
| 				s1 = "(null)";
 | |
| 			s = s1;
 | |
| 			do {
 | |
| 				if (*s == 0)
 | |
| 					break;
 | |
| 				s++;
 | |
| 			} while (--ndigit);
 | |
| 			break;
 | |
| 		case 'b':
 | |
| 			j = 2;
 | |
| 		case 'u':
 | |
| 		getu:
 | |
| 			if ( !lflag ) {
 | |
| 				inte = va_arg(ap, int);
 | |
| 				goto i_unsignd ;
 | |
| 			}
 | |
| #ifndef NOLONG
 | |
| 		case 'U':
 | |
| 		getlu:
 | |
| 			l = va_arg(ap, long);
 | |
| 			goto l_unsignd ;
 | |
| 		case 'B':
 | |
| 			j = 2 ;
 | |
| 			goto getlu ;
 | |
| 		case 'X':
 | |
| 			j = 16;
 | |
| 			goto getlu ;
 | |
| 		case 'O':
 | |
| 			j = 8;
 | |
| 			goto getlu ;
 | |
| 		case 'D':
 | |
| 		    l_signed:
 | |
| 			l = va_arg(ap, long);
 | |
| 			if (l < 0) {
 | |
| 				*s++ = '-';
 | |
| 				l = -l;
 | |
| 			}
 | |
| 			goto do_l;
 | |
| 		    l_unsignd:
 | |
| 			if (l && ndigit)
 | |
| 				*s++ = '0';
 | |
| 		    do_l:
 | |
| 			s = l_compute(l,j,s);
 | |
| 			break;
 | |
| #endif
 | |
| 
 | |
| 		case 'x':
 | |
| 			j = 16;
 | |
| 			goto getu ;
 | |
| 		case 'o':
 | |
| 			j = 8;
 | |
| 			goto getu ;
 | |
| 		case 'd':
 | |
| 			if ( lflag ) goto l_signed; ;
 | |
| 			inte = va_arg(ap, int);
 | |
| 			if ( inte<0 ) {
 | |
| 				*s++ = '-';
 | |
| 				inte= -inte ;
 | |
| 			}
 | |
| 			goto do_i ;
 | |
| 		    i_unsignd:
 | |
| 			if (inte && ndigit)
 | |
| 				*s++ = '0';
 | |
| 		    do_i:
 | |
| 			s = i_compute(inte,j,s);
 | |
| 			break;
 | |
| 		case 'c':
 | |
| 			uint = va_arg(ap, unsigned int);
 | |
| 			for ( i= sizeof uint -1  ; i>=0 ; i-- ) {
 | |
| 				if ( *s = uint%256 ) s++;
 | |
| 				uint/= 256 ;
 | |
| 			}
 | |
| 			break;
 | |
| #ifndef NOFLOAT
 | |
| 		case 'E':
 | |
| 			capitalE = 1;
 | |
| 			/* fall through */
 | |
| 		case 'e':
 | |
| 			if (ndigit >= sizeof(buf)) ndigit = sizeof(buf) - 1;
 | |
| 			dbl = va_arg(ap, double);
 | |
| 			s = _pscien(dbl,s,ndigit,ndfnd);
 | |
| 			break;
 | |
| 		case 'f':
 | |
| 			if (ndigit >= sizeof(buf)) ndigit = sizeof(buf) - 1;
 | |
| 			dbl = va_arg(ap, double);
 | |
| 			s = _pfloat(dbl,s,ndigit,ndfnd);
 | |
| 			break;
 | |
| 		case 'G':
 | |
| 			capitalE = 1;
 | |
| 			/* fall through */
 | |
| 		case 'g':
 | |
| 			if (ndigit >= sizeof(buf)) ndigit = sizeof(buf) - 1;
 | |
| 			dbl = va_arg(ap, double);
 | |
| 			s = gcvt(dbl, ndigit ? ndigit : 6, s) + strlen(s);
 | |
| 			break;
 | |
| #endif
 | |
| 		case 'r':
 | |
| 			ap = va_arg(ap, char *);
 | |
| 			fmt = va_arg(ap, char *);
 | |
| 			fmt=oldfmt;
 | |
| 			continue;
 | |
| 		}
 | |
| #ifndef NOFLOAT
 | |
| 		if (capitalE) {
 | |
| 			register char *p = buf;
 | |
| 			capitalE=0;
 | |
| 			while (*p && *p != 'e') p++;
 | |
| 			if (*p == 'e') *p = 'E';
 | |
| 		}
 | |
| #endif
 | |
| 		j = s - s1;
 | |
| 		if ((c = width - j) > 0)
 | |
| 			if (rjust == 0)
 | |
| 				do putc(zfill,stream);
 | |
| 				while (--c);
 | |
| 		while (--j >= 0)
 | |
| 			putc(*s1++,stream);
 | |
| 		while (--c >= 0)
 | |
| 			putc(zfill,stream);
 | |
| 	}
 | |
| }
 |