| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by Michael Van Canneyt,    member of the Free Pascal development team.    Getopt implementation for Free Pascal, modeled after GNU getopt    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program 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. **********************************************************************}{$IFNDEF FPC_DOTTEDUNITS}unit getopts;{$ENDIF FPC_DOTTEDUNITS}{$modeswitch advancedrecords}{$modeswitch defaultparameters}{$h+}InterfaceConst  No_Argument       = 0;  Required_Argument = 1;  Optional_Argument = 2;  EndOfOptions      = #255;Type  POption  = ^TOption;  TOption = Record    Name    : String;    Has_arg : Integer;    Flag    : PAnsiChar;    Value   : AnsiChar;    Procedure SetOption(const aName:String;AHas_Arg:integer=0;AFlag:PAnsiChar=nil;AValue:AnsiChar=#0);  end;  Orderings = (require_order,permute,return_in_order);Const  OptSpecifier : set of AnsiChar=['-'];Var  OptArg : String;  OptInd : Longint;  OptErr : Boolean;  OptOpt : AnsiChar;Function GetOpt (ShortOpts : String) : AnsiChar;Function GetLongOpts (ShortOpts : String;LongOpts : POption;var Longind : Longint) : AnsiChar;Implementation{$IFNDEF FPC}{***************************************************************************                               Create an ArgV***************************************************************************}uses SysUtils;    type PtrInt = Integer;type  PPAnsiChar = ^pansichar;  apchar = array[0..127] of pansichar;var  argc  : longint;  argv  : apchar;const  CHAR_SIZE = SizeOf(AnsiChar);procedure setup_arguments;var  arglen,  count   : longint;  argstart,  cmdline : pansichar;  quote   : set of ansichar;  argsbuf : array[0..127] of ansipchar;  s       : string;  i       : integer;begin{ create argv[0] which is the started filename }  s:=paramstr(0);  arglen:=length(s);  getmem(argsbuf[0], ( ( arglen + 1 ) * CHAR_SIZE ) );  strpcopy(argsbuf[0],s);{ create commandline }  s:='';  for i:=1 to paramcount do    begin    if Pos(' ', paramstr(i)) > 0 then       s := s + '"' + paramstr(i) + '" '    else       s:=s+paramstr(i)+' ';    end;  s:=s+#0;  cmdline:=@s[1];  count:=1;  repeat  { skip leading spaces }    while cmdline^ in [' ',#9,#13] do     inc(PtrInt(cmdline),CHAR_SIZE);    case cmdline^ of      #0 : break;     '"' : begin             quote:=['"'];             inc(PtrInt(cmdline),CHAR_SIZE);           end;    '''' : begin             quote:=[''''];             inc(PtrInt(cmdline),CHAR_SIZE);           end;    else     quote:=[' ',#9,#13];    end;  { scan until the end of the argument }    argstart:=cmdline;    while (cmdline^<>#0) and not(cmdline^ in quote) do     inc(PtrInt(cmdline),CHAR_SIZE);  { reserve some memory }    arglen:=cmdline-argstart;    getmem(argsbuf[count],(arglen+1) * CHAR_SIZE);    move(argstart^,argsbuf[count]^,arglen * CHAR_SIZE);    argsbuf[count][arglen]:=#0;  { skip quote }    if cmdline^ in quote then     inc(PtrInt(cmdline),CHAR_SIZE);    inc(count);  until false;{ create argc }  argc:=count;{ create an nil entry }  argsbuf[count]:=nil;  inc(count);{ create the argv }  move(argsbuf,argv,count shl 2);end;{$ENDIF}function strpas(p : pansichar) : ansistring;begin  if p=nil then     strpas:=''  else    strpas:=p;end;Procedure TOption.SetOption(const aName:String;AHas_Arg:integer=0;AFlag:PAnsiChar=nil;AValue:AnsiChar=#0);begin  Name:=aName; Has_Arg:=AHas_Arg; Flag:=AFlag; Value:=Avalue;end;{***************************************************************************                               Real Getopts***************************************************************************}Var  NextChar,  Nrargs,  first_nonopt,  last_nonopt   : Longint;  Ordering      : Orderings;Procedure Exchange;var  bottom,  middle,  top,i,len : longint;  temp      : pansichar;begin  bottom:=first_nonopt;  middle:=last_nonopt;  top:=optind;  while (top>middle) and (middle>bottom) do    begin    if (top-middle>middle-bottom) then      begin      len:=middle-bottom;      for i:=0 to len-1 do        begin        temp:=argv[bottom+i];        argv[bottom+i]:=argv[top-(middle-bottom)+i];        argv[top-(middle-bottom)+i]:=temp;        end;      top:=top-len;      end    else      begin      len:=top-middle;      for i:=0 to len-1 do        begin        temp:=argv[bottom+i];        argv[bottom+i]:=argv[middle+i];        argv[middle+i]:=temp;        end;      bottom:=bottom+len;      end;    end;  first_nonopt:=first_nonopt + optind-last_nonopt;  last_nonopt:=optind;end; { exchange }procedure getopt_init (var opts : string);begin{ Initialize some defaults. }  Optarg:='';  Optind:=1;  First_nonopt:=1;  Last_nonopt:=1;  OptOpt:='?';  Nextchar:=0;  ordering:=permute;  if length(opts)>0 then   case opts[1] of   '-' : begin           ordering:=return_in_order;           delete(opts,1,1);         end;   '+' : begin           ordering:=require_order;           delete(opts,1,1);         end;  else    ordering:=permute;   end;end;Function Internal_getopt (Var Optstring : string;LongOpts : POption;                          LongInd : pointer;Long_only : boolean ) : AnsiChar;var  temp,endopt,  option_index : byte;  indfound     : integer;  currentarg,  optname      : string;  p,pfound     : POption;  exact,ambig  : boolean;  c            : AnsiChar;begin  optarg:='';  if optind=0 then   getopt_init(optstring);{ Check if We need the next argument. }  if (optind<nrargs) then   currentarg:=strpas(argv[optind])  else   currentarg:='';  if (nextchar=0) then   begin     if ordering=permute then      begin      { If we processed options following non-options : exchange }        if (first_nonopt<>last_nonopt) and (last_nonopt<>optind) then         exchange        else         if last_nonopt<>optind then          first_nonopt:=optind;        while (optind<nrargs) and (not(argv[optind][0] in OptSpecifier) or              (length(strpas(argv[optind]))=1)) do         inc(optind);        last_nonopt:=optind;      end;   { Check for '--' argument }     if optind<nrargs then      currentarg:=strpas(argv[optind])     else      currentarg:='';     if (optind<>nrargs) and (currentarg='--') then      begin        inc(optind);        if (first_nonopt<>last_nonopt) and (last_nonopt<>optind) then         exchange        else         if first_nonopt=last_nonopt then          first_nonopt:=optind;        last_nonopt:=nrargs;        optind:=nrargs;      end;   { Are we at the end of all arguments ? }     if optind>=nrargs then      begin        if first_nonopt<>last_nonopt then         optind:=first_nonopt;        Internal_getopt:=EndOfOptions;        exit;      end;     if optind<nrargs then      currentarg:=strpas(argv[optind])     else      currentarg:='';   { Are we at a non-option ? }     if not(currentarg[1] in OptSpecifier) or (length(currentarg)=1) then      begin        if ordering=require_order then         begin           Internal_getopt:=EndOfOptions;           exit;         end        else         begin           optarg:=strpas(argv[optind]);           inc(optind);           Internal_getopt:=#0;           exit;         end;      end;   { At this point we're at an option ...}     nextchar:=2;     if (longopts<>nil) and ((currentarg[2]='-') and                             (currentArg[1]='-')) then      inc(nextchar);   { So, now nextchar points at the first character of an option }   end;{ Check if we have a long option }  if longopts<>nil then   if length(currentarg)>1 then    if ((currentarg[2]='-') and (currentArg[1]='-'))       or       ((not long_only) and (pos(currentarg[2],optstring)<>0)) then     begin     { Get option name }       endopt:=pos('=',currentarg);       if endopt=0 then        endopt:=length(currentarg)+1;       optname:=copy(currentarg,nextchar,endopt-nextchar);     { Match partial or full }       p:=longopts;       pfound:=nil;       exact:=false;       ambig:=false;       option_index:=0;       indfound:=0;       while (p^.name<>'') and (not exact) do        begin          if pos(optname,p^.name)<>0 then           begin             if length(optname)=length(p^.name) then              begin                exact:=true;                pfound:=p;                indfound:=option_index;              end             else              if pfound=nil then               begin                 indfound:=option_index;                 pfound:=p               end              else               ambig:=true;           end;          inc(PByte(p),sizeof(toption)); //inc(pointer(p),sizeof(toption)); // for Delphi compatibility          inc(option_index);        end;       if ambig and not exact then        begin          if opterr then           writeln(argv[0],': option "',optname,'" is ambiguous');          nextchar:=0;          inc(optind);          Internal_getopt:='?';          exit;        end;       if pfound<>nil then        begin          inc(optind);          if endopt<=length(currentarg) then           begin             if pfound^.has_arg>0 then              optarg:=copy(currentarg,endopt+1,length(currentarg)-endopt)             else              begin                if opterr then                 if currentarg[2]='-' then                  writeln(argv[0],': option "--',pfound^.name,'" doesn''t allow an argument')                 else                  writeln(argv[0],': option "',currentarg[1],pfound^.name,'" doesn''t allow an argument');                nextchar:=0;                internal_getopt:='?';                exit;              end;           end          else { argument in next paramstr...  }           begin             if pfound^.has_arg=1 then              begin                if optind<nrargs then                 begin                   optarg:=strpas(argv[optind]);                   inc(optind);                 end { required argument }                else                 begin { no req argument}                   if opterr then                    writeln(argv[0],': option ',pfound^.name,' requires an argument');                   nextchar:=0;                   if optstring[1]=':' then                    Internal_getopt:=':'                   else                    Internal_getopt:='?';                   exit;                 end;              end;           end; { argument in next parameter end;}          nextchar:=0;          if longind<>nil then           plongint(longind)^:=indfound+1;          if pfound^.flag<>nil then           begin             pfound^.flag^:=pfound^.value;             internal_getopt:=#0;             exit;           end;          internal_getopt:=pfound^.value;          exit;        end; { pfound<>nil }      { We didn't find it as an option }        if (not long_only) or           ((currentarg[2]='-') or (pos(CurrentArg[nextchar],optstring)=0)) then         begin           if opterr then            if currentarg[2]='-' then             writeln(argv[0],' unrecognized option "--',optname,'"')            else             writeln(argv[0],' unrecognized option "',currentarg[1],optname,'"');           nextchar:=0;           inc(optind);           Internal_getopt:='?';           exit;        end;     end; { Of long options.}{ We check for a short option. }  temp:=pos(currentarg[nextchar],optstring);  c:=currentarg[nextchar];  inc(nextchar);  if nextchar>length(currentarg) then   begin     inc(optind);     nextchar:=0;   end;  if (temp=0) or (c=':') then   begin     if opterr then      writeln(argv[0],': illegal option -- ',c);     optopt:=c;     internal_getopt:='?';     exit;   end;  Internal_getopt:=optstring[temp];  if (length(optstring)>temp) and (optstring[temp+1]=':') then   if (length(optstring)>temp+1) and (optstring[temp+2]=':') then    begin { optional argument }      if nextchar>0 then       begin        optarg:=copy (currentarg,nextchar,length(currentarg)-nextchar+1);        inc(optind);        nextchar:=0;       end else if (optind<>nrargs) then       begin        optarg:=strpas(argv[optind]);        if optarg[1]='-' then          optarg:=''         else          inc(optind);        nextchar:=0;       end;    end   else    begin { required argument }      if nextchar>0 then       begin         optarg:=copy (currentarg,nextchar,length(currentarg)-nextchar+1);         inc(optind);       end      else       if (optind=nrargs) then        begin          if opterr then           writeln (argv[0],': option requires an argument -- ',optstring[temp]);          optopt:=optstring[temp];          if optstring[1]=':' then           Internal_getopt:=':'          else           Internal_Getopt:='?';        end       else        begin          optarg:=strpas(argv[optind]);          inc(optind)        end;       nextchar:=0;    end; { End of required argument}end; { End of internal getopt...}Function GetOpt(ShortOpts : String) : AnsiChar;begin  getopt:=internal_getopt(shortopts,nil,nil,false);end;Function GetLongOpts(ShortOpts : String;LongOpts : POption;var Longind : Longint) : AnsiChar;begin  getlongopts:=internal_getopt(shortopts,longopts,@longind,true);end;initialization{ create argv if not running under FPC }{$ifndef FPC}  setup_arguments;{$endif}{ Needed to detect startup }  Opterr:=true;  Optind:=0;  nrargs:=argc;end.
 |