347 lines
		
	
	
	
		
			8.5 KiB
		
	
	
	
		
			Modula-2
		
	
	
	
	
	
			
		
		
	
	
			347 lines
		
	
	
	
		
			8.5 KiB
		
	
	
	
		
			Modula-2
		
	
	
	
	
	
This file is getopts.def, from which is created getopts.c.
 | 
						|
It implements the builtin "getopts" in Bash.
 | 
						|
 | 
						|
Copyright (C) 1987-2015 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 Software Foundation, either version 3 of the License, or
 | 
						|
(at your option) any later 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.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
$PRODUCES getopts.c
 | 
						|
 | 
						|
$BUILTIN getopts
 | 
						|
$FUNCTION getopts_builtin
 | 
						|
$SHORT_DOC getopts optstring name [arg]
 | 
						|
Parse option arguments.
 | 
						|
 | 
						|
Getopts is used by shell procedures to parse positional parameters
 | 
						|
as options.
 | 
						|
 | 
						|
OPTSTRING contains the option letters to be recognized; if a letter
 | 
						|
is followed by a colon, the option is expected to have an argument,
 | 
						|
which should be separated from it by white space.
 | 
						|
 | 
						|
Each time it is invoked, getopts will place the next option in the
 | 
						|
shell variable $name, initializing name if it does not exist, and
 | 
						|
the index of the next argument to be processed into the shell
 | 
						|
variable OPTIND.  OPTIND is initialized to 1 each time the shell or
 | 
						|
a shell script is invoked.  When an option requires an argument,
 | 
						|
getopts places that argument into the shell variable OPTARG.
 | 
						|
 | 
						|
getopts reports errors in one of two ways.  If the first character
 | 
						|
of OPTSTRING is a colon, getopts uses silent error reporting.  In
 | 
						|
this mode, no error messages are printed.  If an invalid option is
 | 
						|
seen, getopts places the option character found into OPTARG.  If a
 | 
						|
required argument is not found, getopts places a ':' into NAME and
 | 
						|
sets OPTARG to the option character found.  If getopts is not in
 | 
						|
silent mode, and an invalid option is seen, getopts places '?' into
 | 
						|
NAME and unsets OPTARG.  If a required argument is not found, a '?'
 | 
						|
is placed in NAME, OPTARG is unset, and a diagnostic message is
 | 
						|
printed.
 | 
						|
 | 
						|
If the shell variable OPTERR has the value 0, getopts disables the
 | 
						|
printing of error messages, even if the first character of
 | 
						|
OPTSTRING is not a colon.  OPTERR has the value 1 by default.
 | 
						|
 | 
						|
Getopts normally parses the positional parameters ($0 - $9), but if
 | 
						|
more arguments are given, they are parsed instead.
 | 
						|
 | 
						|
Exit Status:
 | 
						|
Returns success if an option is found; fails if the end of options is
 | 
						|
encountered or an error occurs.
 | 
						|
$END
 | 
						|
 | 
						|
#include <config.h>
 | 
						|
 | 
						|
#include <stdio.h>
 | 
						|
 | 
						|
#if defined (HAVE_UNISTD_H)
 | 
						|
#  ifdef _MINIX
 | 
						|
#    include <sys/types.h>
 | 
						|
#  endif
 | 
						|
#  include <unistd.h>
 | 
						|
#endif
 | 
						|
 | 
						|
#include "../bashansi.h"
 | 
						|
#include "../bashintl.h"
 | 
						|
 | 
						|
#include "../shell.h"
 | 
						|
#include "common.h"
 | 
						|
#include "bashgetopt.h"
 | 
						|
#include "getopt.h"
 | 
						|
 | 
						|
#define G_EOF		-1
 | 
						|
#define G_INVALID_OPT	-2
 | 
						|
#define G_ARG_MISSING	-3
 | 
						|
 | 
						|
extern char *this_command_name;
 | 
						|
 | 
						|
static int getopts_unbind_variable __P((char *));
 | 
						|
static int getopts_bind_variable __P((char *, char *));
 | 
						|
static int dogetopts __P((int, char **));
 | 
						|
 | 
						|
/* getopts_reset is magic code for when OPTIND is reset.  N is the
 | 
						|
   value that has just been assigned to OPTIND. */
 | 
						|
void
 | 
						|
getopts_reset (newind)
 | 
						|
     int newind;
 | 
						|
{
 | 
						|
  sh_optind = newind;
 | 
						|
  sh_badopt = 0;
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
getopts_unbind_variable (name)
 | 
						|
     char *name;
 | 
						|
{
 | 
						|
#if 0
 | 
						|
  return (unbind_variable (name));
 | 
						|
#else
 | 
						|
  return (unbind_variable_noref (name));
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
getopts_bind_variable (name, value)
 | 
						|
     char *name, *value;
 | 
						|
{
 | 
						|
  SHELL_VAR *v;
 | 
						|
 | 
						|
  if (legal_identifier (name))
 | 
						|
    {
 | 
						|
      v = bind_variable (name, value, 0);
 | 
						|
      if (v && (readonly_p (v) || noassign_p (v)))
 | 
						|
	return (EX_MISCERROR);
 | 
						|
      return (v ? EXECUTION_SUCCESS : EXECUTION_FAILURE);
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      sh_invalidid (name);
 | 
						|
      return (EXECUTION_FAILURE);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/* Error handling is now performed as specified by Posix.2, draft 11
 | 
						|
   (identical to that of ksh-88).  The special handling is enabled if
 | 
						|
   the first character of the option string is a colon; this handling
 | 
						|
   disables diagnostic messages concerning missing option arguments
 | 
						|
   and invalid option characters.  The handling is as follows.
 | 
						|
 | 
						|
   INVALID OPTIONS:
 | 
						|
        name -> "?"
 | 
						|
        if (special_error) then
 | 
						|
                OPTARG = option character found
 | 
						|
                no error output
 | 
						|
        else
 | 
						|
                OPTARG unset
 | 
						|
                diagnostic message
 | 
						|
        fi
 | 
						|
 
 | 
						|
  MISSING OPTION ARGUMENT;
 | 
						|
        if (special_error) then
 | 
						|
                name -> ":"
 | 
						|
                OPTARG = option character found
 | 
						|
        else
 | 
						|
                name -> "?"
 | 
						|
                OPTARG unset
 | 
						|
                diagnostic message
 | 
						|
        fi
 | 
						|
 */
 | 
						|
 | 
						|
static int
 | 
						|
dogetopts (argc, argv)
 | 
						|
     int argc;
 | 
						|
     char **argv;
 | 
						|
{
 | 
						|
  int ret, special_error, old_opterr, i, n;
 | 
						|
  char strval[2], numval[16];
 | 
						|
  char *optstr;			/* list of options */
 | 
						|
  char *name;			/* variable to get flag val */
 | 
						|
  char *t;
 | 
						|
 | 
						|
  if (argc < 3)
 | 
						|
    {
 | 
						|
      builtin_usage ();
 | 
						|
      return (EX_USAGE);
 | 
						|
    }
 | 
						|
 | 
						|
  /* argv[0] is "getopts". */
 | 
						|
 | 
						|
  optstr = argv[1];
 | 
						|
  name = argv[2];
 | 
						|
  argc -= 2;
 | 
						|
  argv += 2;
 | 
						|
 | 
						|
  special_error = optstr[0] == ':';
 | 
						|
 | 
						|
  if (special_error)
 | 
						|
    {
 | 
						|
      old_opterr = sh_opterr;
 | 
						|
      optstr++;
 | 
						|
      sh_opterr = 0;		/* suppress diagnostic messages */
 | 
						|
    }
 | 
						|
 | 
						|
  if (argc > 1)
 | 
						|
    {
 | 
						|
      sh_getopt_restore_state (argv);
 | 
						|
      t = argv[0];
 | 
						|
      argv[0] = dollar_vars[0];
 | 
						|
      ret = sh_getopt (argc, argv, optstr);
 | 
						|
      argv[0] = t;
 | 
						|
    }
 | 
						|
  else if (rest_of_args == (WORD_LIST *)NULL)
 | 
						|
    {
 | 
						|
      for (i = 0; i < 10 && dollar_vars[i]; i++)
 | 
						|
	;
 | 
						|
 | 
						|
      sh_getopt_restore_state (dollar_vars);
 | 
						|
      ret = sh_getopt (i, dollar_vars, optstr);
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      register WORD_LIST *words;
 | 
						|
      char **v;
 | 
						|
 | 
						|
      for (i = 0; i < 10 && dollar_vars[i]; i++)
 | 
						|
	;
 | 
						|
      for (words = rest_of_args; words; words = words->next, i++)
 | 
						|
	;
 | 
						|
      v = strvec_create (i + 1);
 | 
						|
      for (i = 0; i < 10 && dollar_vars[i]; i++)
 | 
						|
	v[i] = dollar_vars[i];
 | 
						|
      for (words = rest_of_args; words; words = words->next, i++)
 | 
						|
	v[i] = words->word->word;
 | 
						|
      v[i] = (char *)NULL;
 | 
						|
      sh_getopt_restore_state (v);
 | 
						|
      ret = sh_getopt (i, v, optstr);
 | 
						|
      free (v);
 | 
						|
    }
 | 
						|
 | 
						|
  if (special_error)
 | 
						|
    sh_opterr = old_opterr;
 | 
						|
 | 
						|
  /* Set the OPTIND variable in any case, to handle "--" skipping.  It's
 | 
						|
     highly unlikely that 14 digits will be too few. */
 | 
						|
  if (sh_optind < 10)
 | 
						|
    {
 | 
						|
      numval[14] = sh_optind + '0';
 | 
						|
      numval[15] = '\0';
 | 
						|
      i = 14;
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      numval[i = 15] = '\0';
 | 
						|
      n = sh_optind;
 | 
						|
      do
 | 
						|
	{
 | 
						|
	  numval[--i] = (n % 10) + '0';
 | 
						|
	}
 | 
						|
      while (n /= 10);
 | 
						|
    }
 | 
						|
  bind_variable ("OPTIND", numval + i, 0);
 | 
						|
 | 
						|
  /* If an error occurred, decide which one it is and set the return
 | 
						|
     code appropriately.  In all cases, the option character in error
 | 
						|
     is in OPTOPT.  If an invalid option was encountered, OPTARG is
 | 
						|
     NULL.  If a required option argument was missing, OPTARG points
 | 
						|
     to a NULL string (that is, sh_optarg[0] == 0). */
 | 
						|
  if (ret == '?')
 | 
						|
    {
 | 
						|
      if (sh_optarg == NULL)
 | 
						|
	ret = G_INVALID_OPT;
 | 
						|
      else if (sh_optarg[0] == '\0')
 | 
						|
	ret = G_ARG_MISSING;
 | 
						|
    }
 | 
						|
	    
 | 
						|
  if (ret == G_EOF)
 | 
						|
    {
 | 
						|
      getopts_unbind_variable ("OPTARG");
 | 
						|
      getopts_bind_variable (name, "?");
 | 
						|
      return (EXECUTION_FAILURE);
 | 
						|
    }
 | 
						|
 | 
						|
  if (ret == G_INVALID_OPT)
 | 
						|
    {
 | 
						|
      /* Invalid option encountered. */
 | 
						|
      ret = getopts_bind_variable (name, "?");
 | 
						|
 | 
						|
      if (special_error)
 | 
						|
	{
 | 
						|
	  strval[0] = (char)sh_optopt;
 | 
						|
	  strval[1] = '\0';
 | 
						|
	  bind_variable ("OPTARG", strval, 0);
 | 
						|
	}
 | 
						|
      else
 | 
						|
	getopts_unbind_variable ("OPTARG");
 | 
						|
 | 
						|
      return (ret);
 | 
						|
    }
 | 
						|
 | 
						|
  if (ret == G_ARG_MISSING)
 | 
						|
    {
 | 
						|
      /* Required argument missing. */
 | 
						|
      if (special_error)
 | 
						|
	{
 | 
						|
	  ret = getopts_bind_variable (name, ":");
 | 
						|
 | 
						|
	  strval[0] = (char)sh_optopt;
 | 
						|
	  strval[1] = '\0';
 | 
						|
	  bind_variable ("OPTARG", strval, 0);
 | 
						|
	}
 | 
						|
      else
 | 
						|
	{
 | 
						|
	  ret = getopts_bind_variable (name, "?");
 | 
						|
	  getopts_unbind_variable ("OPTARG");
 | 
						|
	}
 | 
						|
      return (ret);
 | 
						|
    }			
 | 
						|
 | 
						|
  bind_variable ("OPTARG", sh_optarg, 0);
 | 
						|
 | 
						|
  strval[0] = (char) ret;
 | 
						|
  strval[1] = '\0';
 | 
						|
  return (getopts_bind_variable (name, strval));
 | 
						|
}
 | 
						|
 | 
						|
/* The getopts builtin.  Build an argv, and call dogetopts with it. */
 | 
						|
int
 | 
						|
getopts_builtin (list)
 | 
						|
     WORD_LIST *list;
 | 
						|
{
 | 
						|
  char **av;
 | 
						|
  int ac, ret;
 | 
						|
 | 
						|
  if (list == 0)
 | 
						|
    {
 | 
						|
      builtin_usage ();
 | 
						|
      return EX_USAGE;
 | 
						|
    }
 | 
						|
 | 
						|
  reset_internal_getopt ();
 | 
						|
  if ((ret = internal_getopt (list, "")) != -1)
 | 
						|
    {
 | 
						|
      if (ret == GETOPT_HELP)
 | 
						|
	builtin_help ();
 | 
						|
      else
 | 
						|
	builtin_usage ();
 | 
						|
      return (EX_USAGE);
 | 
						|
    }
 | 
						|
  list = loptend;
 | 
						|
 | 
						|
  av = make_builtin_argv (list, &ac);
 | 
						|
  ret = dogetopts (ac, av);
 | 
						|
  free ((char *)av);
 | 
						|
 | 
						|
  return (ret);
 | 
						|
}
 |