| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | This file is printf.def, from which is created printf.c. | 
					
						
							|  |  |  | It implements the builtin "printf" in Bash. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Copyright (C) 1997 Free Software Foundation, Inc. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This file is part of GNU Bash, the Bourne Again SHell. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Bash is free software; you can redistribute it and/or modify it under | 
					
						
							|  |  |  | the terms of the GNU General Public License as published by the Free | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | Software Foundation; either version 2, or (at your option) any later | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | version. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Bash is distributed in the hope that it will be useful, but WITHOUT ANY | 
					
						
							|  |  |  | WARRANTY; without even the implied warranty of MERCHANTABILITY or | 
					
						
							|  |  |  | FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License | 
					
						
							|  |  |  | for more details. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | You should have received a copy of the GNU General Public License along | 
					
						
							|  |  |  | with Bash; see the file COPYING.  If not, write to the Free Software | 
					
						
							|  |  |  | Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $PRODUCES printf.c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $BUILTIN printf | 
					
						
							|  |  |  | $FUNCTION printf_builtin | 
					
						
							|  |  |  | $SHORT_DOC printf format [arguments] | 
					
						
							|  |  |  | printf formats and prints ARGUMENTS under control of the FORMAT. FORMAT | 
					
						
							|  |  |  | is a character string which contains three types of objects: plain | 
					
						
							|  |  |  | characters, which are simply copied to standard output, character escape | 
					
						
							|  |  |  | sequences which are converted and copied to the standard output, and | 
					
						
							|  |  |  | format specifications, each of which causes printing of the next successive | 
					
						
							|  |  |  | argument.  In addition to the standard printf(1) formats, %b means to | 
					
						
							|  |  |  | expand backslash escape sequences in the corresponding argument, and %q | 
					
						
							|  |  |  | means to quote the argument in a way that can be reused as shell input. | 
					
						
							|  |  |  | $END | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <config.h> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "../bashtypes.h" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <errno.h> | 
					
						
							|  |  |  | #if defined (HAVE_LIMITS_H) | 
					
						
							|  |  |  | #  include <limits.h> | 
					
						
							|  |  |  | #else | 
					
						
							|  |  |  |    /* Assume 32-bit ints and longs. */ | 
					
						
							|  |  |  | #  define LONG_MAX		2147483647L | 
					
						
							|  |  |  | #  define LONG_MIN		(-2147483647L-1) | 
					
						
							|  |  |  | #  define INT_MAX		2147483647 | 
					
						
							|  |  |  | #  define INT_MIN		(-2147483647-1) | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdio.h> | 
					
						
							|  |  |  | #include <ctype.h> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "../bashansi.h" | 
					
						
							|  |  |  | #include "../shell.h" | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | #include "stdc.h" | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #include "bashgetopt.h" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined (errno) | 
					
						
							|  |  |  | extern int errno; | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define PF(f, func) \ | 
					
						
							|  |  |  |   do { \ | 
					
						
							|  |  |  |     if (fieldwidth && precision) \ | 
					
						
							|  |  |  |       (void)printf(f, fieldwidth, precision, func); \ | 
					
						
							|  |  |  |     else if (fieldwidth && precision == 0) \ | 
					
						
							|  |  |  |       (void)printf(f, fieldwidth, func); \ | 
					
						
							|  |  |  |     else if (precision) \ | 
					
						
							|  |  |  |       (void)printf(f, precision, func); \ | 
					
						
							|  |  |  |     else \ | 
					
						
							|  |  |  |       (void)printf(f, func); \ | 
					
						
							|  |  |  |   } while (0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define PRETURN(value) \ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   do { /* free (format); */ fflush (stdout); return (value); } while (0) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define  SKIP1 "#-+ 0" | 
					
						
							|  |  |  | #define  SKIP2 "*0123456789" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void printstr __P((char *, char *, int, int, int)); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | static int tescape __P((char *, int, char *, int *)); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static char *bexpand __P((char *, int, int *, int *)); | 
					
						
							|  |  |  | static char *mklong __P((char *, int)); | 
					
						
							|  |  |  | static int getchr __P((void)); | 
					
						
							|  |  |  | static char *getstr __P((void)); | 
					
						
							|  |  |  | static int  getint __P((void)); | 
					
						
							|  |  |  | static int getlong __P((long *)); | 
					
						
							|  |  |  | static int getulong __P((unsigned long *)); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | static int getdouble __P((double *)); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static int asciicode __P((void)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static WORD_LIST *garglist; | 
					
						
							|  |  |  | static int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | extern char *sh_backslash_quote (); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | printf_builtin (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   int ch, end, fieldwidth, precision, foundmod, fmtlen; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   char convch, nextch, *format, *fmt, *start; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   retval = EXECUTION_SUCCESS; | 
					
						
							|  |  |  |   reset_internal_getopt (); | 
					
						
							|  |  |  |   while ((ch = internal_getopt (list, "")) != -1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       switch (ch) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	case '?': | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	  builtin_usage(); | 
					
						
							|  |  |  | 	  return (EX_USAGE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   list = loptend; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (list == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       builtin_usage (); | 
					
						
							|  |  |  |       return (EX_USAGE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (list->word->word == 0 || list->word->word[0] == '\0') | 
					
						
							|  |  |  |     return (EXECUTION_SUCCESS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   format = list->word->word; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   garglist = list->next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-07-23 14:37:54 +00:00
										 |  |  |   /* If the format string is empty after preprocessing, return immediately. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (format == 0 || *format == 0) | 
					
						
							| 
									
										
										
										
											1998-07-23 14:37:54 +00:00
										 |  |  |     return (EXECUTION_SUCCESS); | 
					
						
							|  |  |  | 	   | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* Basic algorithm is to scan the format string for conversion | 
					
						
							|  |  |  |      specifications -- once one is found, find out if the field | 
					
						
							|  |  |  |      width or precision is a '*'; if it is, gather up value.  Note, | 
					
						
							|  |  |  |      format strings are reused as necessary to use up the provided | 
					
						
							|  |  |  |      arguments, arguments of zero/null string are provided to use | 
					
						
							|  |  |  |      up the format string. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   do | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* find next format specification */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       for (fmt = format; *fmt; fmt++) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  precision = fieldwidth = foundmod = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	  if (*fmt == '\\') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      fmt++; | 
					
						
							|  |  |  | 	      /* A NULL third argument to tescape means to not do special | 
					
						
							|  |  |  | 		 processing for \c. */ | 
					
						
							|  |  |  | 	      fmt += tescape (fmt, 1, &nextch, (int *)NULL); | 
					
						
							|  |  |  | 	      putchar (nextch); | 
					
						
							|  |  |  | 	      fmt--;	/* for loop will increment it for us again */ | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  if (*fmt != '%') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      putchar (*fmt); | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* ASSERT(*fmt == '%') */ | 
					
						
							|  |  |  | 	  start = fmt++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (*fmt == '%')		/* %% prints a % */ | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      putchar ('%'); | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* found format specification, skip to field width */ | 
					
						
							|  |  |  | 	  for (; *fmt && strchr(SKIP1, *fmt); ++fmt) | 
					
						
							|  |  |  | 	    ; | 
					
						
							|  |  |  | 	  fieldwidth = (*fmt == '*') ? getint () : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* skip to possible '.', get following precision */ | 
					
						
							|  |  |  | 	  for (; *fmt && strchr(SKIP2, *fmt); ++fmt) | 
					
						
							|  |  |  | 	    ; | 
					
						
							|  |  |  | 	  if (*fmt == '.') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      ++fmt; | 
					
						
							|  |  |  | 	      precision = (*fmt == '*') ? getint () : 0; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* skip to conversion char */ | 
					
						
							|  |  |  | 	  for (; *fmt && strchr(SKIP2, *fmt); ++fmt) | 
					
						
							|  |  |  | 	    ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* skip possible format modifiers */ | 
					
						
							|  |  |  | 	  if (*fmt == 'l' || *fmt == 'L' || *fmt == 'h') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      fmt++; | 
					
						
							|  |  |  | 	      foundmod = 1; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	     | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	  if (*fmt == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      builtin_error ("`%s': missing format character", start); | 
					
						
							|  |  |  | 	      PRETURN (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  convch = *fmt; | 
					
						
							|  |  |  | 	  nextch = fmt[1]; | 
					
						
							|  |  |  | 	  fmt[1] = '\0'; | 
					
						
							|  |  |  | 	  switch(convch) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	    case 'c': | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		char p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p = getchr (); | 
					
						
							|  |  |  | 		PF(start, p); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    case 's': | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		char *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p = getstr (); | 
					
						
							|  |  |  | 		PF(start, p); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    case 'b':		/* expand escapes in argument */ | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		char *p, *xp; | 
					
						
							|  |  |  | 		int rlen; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p = getstr (); | 
					
						
							|  |  |  | 		ch = rlen = 0; | 
					
						
							|  |  |  | 		xp = bexpand (p, strlen (p), &ch, &rlen); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (xp) | 
					
						
							|  |  |  | 		  { | 
					
						
							|  |  |  | 		    /* Have to use printstr because of possible NUL bytes | 
					
						
							|  |  |  | 		       in XP -- printf does not handle that well. */ | 
					
						
							|  |  |  | 		    printstr (start, xp, rlen, fieldwidth, precision); | 
					
						
							|  |  |  | 		    free (xp); | 
					
						
							|  |  |  | 		  } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (ch) | 
					
						
							|  |  |  | 		  PRETURN (retval); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    case 'q':		/* print with shell quoting */ | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		char *p, *xp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p = getstr (); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 		xp = sh_backslash_quote (p); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		if (xp) | 
					
						
							|  |  |  | 		  { | 
					
						
							|  |  |  | 		    /* Use printstr to get fieldwidth and precision right. */ | 
					
						
							|  |  |  | 		    printstr (start, xp, strlen (xp), fieldwidth, precision); | 
					
						
							|  |  |  | 		    free (xp); | 
					
						
							|  |  |  | 		  } | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    case 'd': | 
					
						
							|  |  |  | 	    case 'i': | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		long p; | 
					
						
							|  |  |  | 		char *f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (foundmod == 0 && ((f = mklong (start, convch)) == NULL)) | 
					
						
							|  |  |  | 		  PRETURN (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		  f = start; | 
					
						
							|  |  |  | 		if (getlong (&p)) | 
					
						
							|  |  |  | 		  PRETURN (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 		PF(f, p); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    case 'o': | 
					
						
							|  |  |  | 	    case 'u': | 
					
						
							|  |  |  | 	    case 'x': | 
					
						
							|  |  |  | 	    case 'X': | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		unsigned long p; | 
					
						
							|  |  |  | 		char *f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (foundmod == 0 && ((f = mklong (start, convch)) == NULL)) | 
					
						
							|  |  |  | 		  PRETURN (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		  f = start; | 
					
						
							|  |  |  | 		if (getulong (&p)) | 
					
						
							|  |  |  | 		  PRETURN (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 		PF (f, p); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    case 'e': | 
					
						
							|  |  |  | 	    case 'E': | 
					
						
							|  |  |  | 	    case 'f': | 
					
						
							|  |  |  | 	    case 'g': | 
					
						
							|  |  |  | 	    case 'G': | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		double p; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 		if (getdouble (&p)) | 
					
						
							|  |  |  | 		  PRETURN (EXECUTION_FAILURE); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		PF(start, p); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	    /* We don't output unrecognized format characters; we print an | 
					
						
							|  |  |  | 	       error message and return a failure exit status. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    default: | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	      builtin_error ("`%c': invalid format character", convch); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      PRETURN (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  fmt[1] = nextch; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-07-23 14:37:54 +00:00
										 |  |  |   while (garglist && garglist != list->next); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   PRETURN (retval); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* We duplicate a lot of what printf(3) does here. */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | printstr (fmt, string, len, fieldwidth, precision) | 
					
						
							|  |  |  |      char *fmt;			/* format */ | 
					
						
							|  |  |  |      char *string;		/* expanded string argument */ | 
					
						
							|  |  |  |      int len;			/* length of expanded string */ | 
					
						
							|  |  |  |      int fieldwidth;		/* argument for width of `*' */ | 
					
						
							|  |  |  |      int precision;		/* argument for precision of `*' */ | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if 0 | 
					
						
							|  |  |  |   char *s; | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  |   int padlen, nc, ljust, i; | 
					
						
							|  |  |  |   int fw, pr;			/* fieldwidth and precision */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string == 0 || *string == '\0') | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0 | 
					
						
							|  |  |  |   s = fmt; | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  |   if (*fmt == '%') | 
					
						
							|  |  |  |     fmt++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ljust = fw = pr = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* skip flags */ | 
					
						
							|  |  |  |   while (*fmt == '#' || *fmt == '-' || *fmt == '+' || *fmt == ' ' || *fmt == '0') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*fmt == '-') | 
					
						
							|  |  |  | 	ljust = 1; | 
					
						
							|  |  |  |       fmt++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* get fieldwidth, if present */ | 
					
						
							|  |  |  |   if (*fmt == '*') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       fmt++; | 
					
						
							|  |  |  |       fw = fieldwidth; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (isdigit (*fmt)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       fw = *fmt++ - '0'; | 
					
						
							|  |  |  |       while (isdigit (*fmt)) | 
					
						
							|  |  |  | 	fw = (fw * 10) + (*fmt++ - '0'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* get precision, if present */ | 
					
						
							|  |  |  |   if (*fmt == '.') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       fmt++; | 
					
						
							|  |  |  |       if (*fmt == '*') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  fmt++; | 
					
						
							|  |  |  | 	  pr = precision; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (isdigit (*fmt)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pr = *fmt++ - '0'; | 
					
						
							|  |  |  | 	  while (isdigit (*fmt)) | 
					
						
							|  |  |  | 	    pr = (pr * 10) + (*fmt++ - '0'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0 | 
					
						
							|  |  |  |   /* If we remove this, get rid of `s'. */ | 
					
						
							|  |  |  |   if (*fmt != 'b' && *fmt != 'q') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       internal_error ("format parsing problem: %s", s); | 
					
						
							|  |  |  |       fw = pr = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* chars from string to print */ | 
					
						
							|  |  |  |   nc = (pr > 0 && pr <= len) ? pr : len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   padlen = fw - nc; | 
					
						
							|  |  |  |   if (padlen < 0) | 
					
						
							|  |  |  |     padlen = 0; | 
					
						
							|  |  |  |   if (ljust) | 
					
						
							|  |  |  |     padlen = -padlen; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* leading pad characters */ | 
					
						
							|  |  |  |   for (; padlen > 0; padlen--) | 
					
						
							|  |  |  |     putchar (' '); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* output NC characters from STRING */ | 
					
						
							|  |  |  |   for (i = 0; i < nc; i++) | 
					
						
							|  |  |  |     putchar (string[i]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* output any necessary trailing padding */ | 
					
						
							|  |  |  |   for (; padlen < 0; padlen++) | 
					
						
							|  |  |  |     putchar (' '); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  | /* Convert STRING by expanding the escape sequences specified by the | 
					
						
							|  |  |  |    POSIX standard for printf's `%b' format string.  If SAWC is non-null, | 
					
						
							|  |  |  |    recognize `\c' and use that as a string terminator.  If we see \c, set | 
					
						
							|  |  |  |    *SAWC to 1 before returning.  LEN is the length of STRING. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef isoctal | 
					
						
							|  |  |  | #undef isoctal | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define isoctal(c)	((c) >= '0' && (c) <= '7') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define OCTVALUE(c)	((c) - '0') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef isxdigit | 
					
						
							|  |  |  | #  define isxdigit(c)	(isdigit((c)) || ((c) >= 'a' && (c) <= 'f') || ((c) >= 'A' && (c) <= 'F')) | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define HEXVALUE(c) \ | 
					
						
							|  |  |  |   ((c) >= 'a' && (c) <= 'f' ? (c)-'a'+10 : (c) >= 'A' && (c) <= 'F' ? (c)-'A'+10 : (c)-'0') | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Translate a single backslash-escape sequence starting at ESTART (the | 
					
						
							|  |  |  |    character after the backslash) and return the number of characters | 
					
						
							|  |  |  |    consumed by the sequence.  CP is the place to return the translated | 
					
						
							|  |  |  |    value.  *SAWC is set to 1 if the escape sequence was \c, since that means | 
					
						
							|  |  |  |    to short-circuit the rest of the processing.  If SAWC is null, we don't | 
					
						
							|  |  |  |    do the \c short-circuiting, and \c is treated as an unrecognized escape | 
					
						
							|  |  |  |    sequence.  */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | tescape (estart, trans_squote, cp, sawc) | 
					
						
							|  |  |  |      char *estart; | 
					
						
							|  |  |  |      int trans_squote; | 
					
						
							|  |  |  |      char *cp; | 
					
						
							|  |  |  |      int *sawc; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char *p; | 
					
						
							|  |  |  |   int temp, c, evalue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p = estart; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (c = *p++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  | #if defined (__STDC__) | 
					
						
							|  |  |  |       case 'a': *cp = '\a'; break; | 
					
						
							|  |  |  | #else | 
					
						
							|  |  |  |       case 'a': *cp = '\007'; break; | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case 'b': *cp = '\b'; break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       case 'e': | 
					
						
							|  |  |  |       case 'E': *cp = '\033'; break;	/* ESC -- non-ANSI */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       case 'f': *cp = '\f'; break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case 'n': *cp = '\n'; break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case 'r': *cp = '\r'; break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case 't': *cp = '\t'; break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case 'v': *cp = '\v'; break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* %b octal constants are `\0' followed by one, two, or three | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 octal digits... */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       case '0': | 
					
						
							|  |  |  | 	for (temp = 3, evalue = 0; isoctal (*p) && temp--; p++) | 
					
						
							|  |  |  | 	  evalue = (evalue * 8) + OCTVALUE (*p); | 
					
						
							|  |  |  | 	*cp = evalue; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* but, as an extension, the other echo-like octal escape | 
					
						
							|  |  |  | 	 sequences are supported as well. */ | 
					
						
							|  |  |  |       case '1': case '2': case '3': case '4': | 
					
						
							|  |  |  |       case '5': case '6': case '7': | 
					
						
							|  |  |  | 	for (temp = 2, evalue = c - '0'; isoctal (*p) && temp--; p++) | 
					
						
							|  |  |  | 	  evalue = (evalue * 8) + OCTVALUE (*p); | 
					
						
							|  |  |  | 	*cp = evalue; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* And, as another extension, we allow \xNNN, where each N is a | 
					
						
							|  |  |  | 	 hex digit. */ | 
					
						
							|  |  |  |       case 'x': | 
					
						
							|  |  |  | 	for (temp = 3, evalue = 0; isxdigit (*p) && temp--; p++) | 
					
						
							|  |  |  | 	  evalue = (evalue * 16) + HEXVALUE (*p); | 
					
						
							|  |  |  | 	if (temp == 3) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    builtin_error ("missing hex digit for \\x"); | 
					
						
							|  |  |  | 	    *cp = '\\'; | 
					
						
							|  |  |  | 	    return 0; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	*cp = evalue; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case '\\':	/* \\ -> \ */ | 
					
						
							|  |  |  | 	*cp = c; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case '\'':	/* TRANS_SQUOTE != 0 means \' -> ' */ | 
					
						
							|  |  |  | 	if (trans_squote) | 
					
						
							|  |  |  | 	  *cp = c; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    *cp = '\\'; | 
					
						
							|  |  |  | 	    return 0; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case 'c': | 
					
						
							|  |  |  | 	if (sawc) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    *sawc = 1; | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  |       /* other backslash escapes are passed through unaltered */ | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  | 	*cp = '\\'; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |   return (p - estart); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static char * | 
					
						
							|  |  |  | bexpand (string, len, sawc, lenp) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int len, *sawc, *lenp; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   int temp; | 
					
						
							|  |  |  |   char *ret, *r, *s, c; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (string == 0 || *string == '\0') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (sawc) | 
					
						
							|  |  |  | 	*sawc = 0; | 
					
						
							|  |  |  |       if (lenp) | 
					
						
							|  |  |  | 	*lenp = 0; | 
					
						
							|  |  |  |       return ((char *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ret = xmalloc (len + 1); | 
					
						
							|  |  |  |   for (r = ret, s = string; s && *s; ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       c = *s++; | 
					
						
							|  |  |  |       if (c != '\\' || *s == '\0') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  *r++ = c; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       temp = 0; | 
					
						
							|  |  |  |       s += tescape (s, 0, &c, &temp); | 
					
						
							|  |  |  |       if (temp) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  if (sawc) | 
					
						
							|  |  |  | 	    *sawc = 1; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       *r++ = c; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   *r = '\0'; | 
					
						
							|  |  |  |   if (lenp) | 
					
						
							|  |  |  |     *lenp = r - ret; | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | mklong (str, ch) | 
					
						
							|  |  |  |      char *str; | 
					
						
							|  |  |  |      int ch; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   static char copy[64]; | 
					
						
							|  |  |  |   int len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   len = strlen (str) + 2; | 
					
						
							|  |  |  |   FASTCOPY (str, copy, len - 3); | 
					
						
							|  |  |  |   copy[len - 3] = 'l'; | 
					
						
							|  |  |  |   copy[len - 2] = ch; | 
					
						
							|  |  |  |   copy[len - 1] = '\0'; | 
					
						
							|  |  |  |   return (copy); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | getchr () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (garglist == 0) | 
					
						
							|  |  |  |     return ('\0'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ret = (int)garglist->word->word[0]; | 
					
						
							|  |  |  |   garglist = garglist->next; | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | getstr () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (garglist == 0) | 
					
						
							|  |  |  |     return (""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ret = garglist->word->word; | 
					
						
							|  |  |  |   garglist = garglist->next; | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | getint () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   long ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (getlong (&ret)) | 
					
						
							|  |  |  |     return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ret > INT_MAX) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       builtin_error ("warning: %s: %s", garglist->word->word, strerror(ERANGE)); | 
					
						
							|  |  |  |       ret = INT_MAX; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (ret < INT_MIN) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       builtin_error ("warning: %s: %s", garglist->word->word, strerror(ERANGE)); | 
					
						
							|  |  |  |       ret = INT_MIN; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ((int)ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | getlong (lp) | 
					
						
							|  |  |  |      long *lp; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   long ret; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   char *ep; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (garglist == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       *lp = 0L; | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (garglist->word->word[0] == '\'' || garglist->word->word[0] == '"') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       *lp =  (long)asciicode (); | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   errno = 0; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   /* If we use 0 as the third argument, we can handle octal and hex, which | 
					
						
							|  |  |  |      legal_number does not.  (This was | 
					
						
							|  |  |  | 	  if (legal_number (garglist->word->word, &ret) == 0) | 
					
						
							|  |  |  |      ) */ | 
					
						
							|  |  |  |   ret = strtol (garglist->word->word, &ep, 0); | 
					
						
							|  |  |  |   if (*ep != '\0') | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       builtin_error ("%s: invalid number", garglist->word->word); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       return (1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (errno == ERANGE) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |     builtin_error ("warning: %s: %s", garglist->word->word, strerror(ERANGE)); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   *lp = ret; | 
					
						
							|  |  |  |   garglist = garglist->next; | 
					
						
							|  |  |  |   return (0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | getulong (ulp) | 
					
						
							|  |  |  |      unsigned long *ulp; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   unsigned long ret; | 
					
						
							|  |  |  |   char *ep; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (garglist == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       *ulp = (unsigned long)0; | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (garglist->word->word[0] == '\'' || garglist->word->word[0] == '"') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       *ulp =  (unsigned long)asciicode (); | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   errno = 0; | 
					
						
							|  |  |  |   ret = strtoul (garglist->word->word, &ep, 0); | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   if (*ep) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       builtin_error ("%s: invalid number", garglist->word->word); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       return (1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (errno == ERANGE) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |     builtin_error ("warning: %s: %s", garglist->word->word, strerror(ERANGE)); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   *ulp = ret; | 
					
						
							|  |  |  |   garglist = garglist->next; | 
					
						
							|  |  |  |   return (0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | getdouble (dp) | 
					
						
							|  |  |  |      double *dp; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   double ret; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   char *ep; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (garglist == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       *dp = (double)0; | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (garglist->word->word[0] == '\'' || garglist->word->word[0] == '"') | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       *dp = (double)asciicode (); | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   errno = 0; | 
					
						
							|  |  |  |   ret = strtod (garglist->word->word, &ep); | 
					
						
							|  |  |  |   if (*ep) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       builtin_error ("%s: invalid number", garglist->word->word); | 
					
						
							|  |  |  |       return (1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (errno == ERANGE) | 
					
						
							|  |  |  |     builtin_error ("warning: %s: %s", garglist->word->word, strerror(ERANGE)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   *dp = ret; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   garglist = garglist->next; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   return (0); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* NO check is needed for garglist here. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | asciicode () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int ch; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ch = garglist->word->word[1]; | 
					
						
							|  |  |  |   garglist = garglist->next; | 
					
						
							|  |  |  |   return (ch); | 
					
						
							|  |  |  | } |