aboutsummaryrefslogtreecommitdiff
path: root/src/cmd
diff options
context:
space:
mode:
Diffstat (limited to 'src/cmd')
-rw-r--r--src/cmd/getflags/funcgetflags.c267
1 files changed, 267 insertions, 0 deletions
diff --git a/src/cmd/getflags/funcgetflags.c b/src/cmd/getflags/funcgetflags.c
new file mode 100644
index 00000000..a88e99f2
--- /dev/null
+++ b/src/cmd/getflags/funcgetflags.c
@@ -0,0 +1,267 @@
+#include <u.h>
+#include <libc.h>
+#include <ctype.h>
+#include "getflags.h"
+
+char **flag[NFLAG];
+char cmdline[NCMDLINE+1];
+char *cmdname;
+char *flagset[];
+char *flagset[]={"<flag>"};
+static char *flagarg="";
+static void reverse(char **, char **);
+static int scanflag(int, char *);
+static int reason;
+#define RESET 1
+#define ARGCCOUNT 2
+#define FLAGSYN 3
+#define BADFLAG 4
+static int badflag;
+char *getflagsargv[NGETFLAGSARGV+2]; /* original argv stored here for people who need it */
+
+int
+getflags(int argc, char *argv[], char *flags)
+{
+ char *s, *t;
+ int i, j, c, count;
+ flagarg=flags;
+ if(cmdname==0){
+ cmdname=argv[0];
+ for(i=0;i!=argc && i!=NGETFLAGSARGV;i++) getflagsargv[i]=argv[i];
+ if(argc>NGETFLAGSARGV) getflagsargv[i++]="...";
+ getflagsargv[i]=0;
+ }
+ s=cmdline;
+ for(i=0;i!=argc;i++){
+ for(t=argv[i];*t;)
+ if(s!=&cmdline[NCMDLINE])
+ *s++=*t++;
+ else
+ break;
+ if(i!=argc-1 && s!=&cmdline[NCMDLINE])
+ *s++=' ';
+ }
+ *s='\0';
+ i=1;
+ while(i!=argc && argv[i][0]=='-'){
+ s=argv[i]+1;
+ if(*s=='\0'){ /* if argument is "-", stop scanning and delete it */
+ for(j=i+1;j<=argc;j++)
+ argv[j-1]=argv[j];
+ return argc-1;
+ }
+ while(*s){
+ c=*s++;
+ count=scanflag(c, flags);
+ if(count==-1) return -1;
+ if(flag[c]){ reason=RESET; badflag=c; return -1; }
+ if(count==0){
+ flag[c]=flagset;
+ if(*s=='\0'){
+ for(j=i+1;j<=argc;j++)
+ argv[j-1]=argv[j];
+ --argc;
+ }
+ }
+ else{
+ if(*s=='\0'){
+ for(j=i+1;j<=argc;j++)
+ argv[j-1]=argv[j];
+ --argc;
+ s=argv[i];
+ }
+ if(argc-i<count){
+ reason=ARGCCOUNT;
+ badflag=c;
+ return -1;
+ }
+ reverse(argv+i, argv+argc);
+ reverse(argv+i, argv+argc-count);
+ reverse(argv+argc-count+1, argv+argc);
+ argc-=count;
+ flag[c]=argv+argc+1;
+ flag[c][0]=s;
+ s="";
+ }
+ }
+ }
+ return argc;
+}
+
+void
+static reverse(char **p, char **q)
+{
+ register char *t;
+ for(;p<q;p++,--q){ t=*p; *p=*q; *q=t; }
+}
+
+static int
+scanflag(int c, char *f)
+{
+ int fc, count;
+ if(0<=c && c<NFLAG) while(*f){
+ if(*f==' '){
+ f++;
+ continue;
+ }
+ fc=*f++;
+ if(*f==':'){
+ f++;
+ if(!isdigit(*f)){ reason=FLAGSYN; return -1; }
+ count=strtol(f, &f, 10);
+ }
+ else
+ count=0;
+ if(*f=='['){
+ int depth=1;
+ do{
+ f++;
+ if(*f=='\0'){ reason=FLAGSYN; return -1; }
+ if(*f=='[') depth++;
+ if(*f==']') depth--;
+ }while(depth>0);
+ f++;
+ }
+ if(c==fc) return count;
+ }
+ reason=BADFLAG;
+ badflag=c;
+ return -1;
+}
+
+static void errn(char *, int), errs(char *), errc(int);
+
+void
+usage(char *tail)
+{
+ char *s, *t, c;
+ int count, nflag=0;
+ switch(reason){
+ case RESET:
+ errs("Flag -");
+ errc(badflag);
+ errs(": set twice\n");
+ break;
+ case ARGCCOUNT:
+ errs("Flag -");
+ errc(badflag);
+ errs(": too few arguments\n");
+ break;
+ case FLAGSYN:
+ errs("Bad argument to getflags!\n");
+ break;
+ case BADFLAG:
+ errs("Illegal flag -");
+ errc(badflag);
+ errc('\n');
+ break;
+ }
+ errs("Usage: ");
+ errs(cmdname);
+ for(s=flagarg;*s;){
+ c=*s;
+ if(*s++==' ') continue;
+ if(*s==':'){
+ s++;
+ count=strtol(s, &s, 10);
+ }
+ else count=0;
+ if(count==0){
+ if(nflag==0) errs(" [-");
+ nflag++;
+ errc(c);
+ }
+ if(*s=='['){
+ int depth=1;
+ s++;
+ for(;*s!='\0' && depth>0; s++)
+ if (*s==']') depth--;
+ else if (*s=='[') depth++;
+ }
+ }
+ if(nflag) errs("]");
+ for(s=flagarg;*s;){
+ c=*s;
+ if(*s++==' ') continue;
+ if(*s==':'){
+ s++;
+ count=strtol(s, &s, 10);
+ }
+ else count=0;
+ if(count!=0){
+ errs(" [-");
+ errc(c);
+ if(*s=='['){
+ int depth=1;
+ s++;
+ t=s;
+ for(;*s!='\0' && depth>0; s++)
+ if (*s==']') depth--;
+ else if (*s=='[') depth++;
+ errs(" ");
+ errn(t, s-t);
+ }
+ else
+ while(count--) errs(" arg");
+ errs("]");
+ }
+ else if(*s=='['){
+ int depth=1;
+ s++;
+ for(;*s!='\0' && depth>0; s++)
+ if (*s==']') depth--;
+ else if (*s=='[') depth++;
+ }
+ }
+ if(tail){
+ errs(" ");
+ errs(tail);
+ }
+ errs("\n");
+ exits("usage");
+}
+
+static void
+errn(char *s, int count)
+{
+ while(count){ errc(*s++); --count; }
+}
+
+static void
+errs(char *s)
+{
+ while(*s) errc(*s++);
+}
+
+#define NBUF 80
+static char buf[NBUF], *bufp=buf;
+
+static void
+errc(int c){
+ *bufp++=c;
+ if(bufp==&buf[NBUF] || c=='\n'){
+ write(2, buf, bufp-buf);
+ bufp=buf;
+ }
+}
+
+#ifdef TEST
+#include <stdio.h>
+main(int argc, char *argv[])
+{
+ int c, i, n;
+ if(argc<3){
+ fprint(2, "Usage: %s flags cmd ...\n", argv[0]);
+ exits("usage");
+ }
+ n=getflags(argc-2, argv+2, argv[1]);
+ if(n<0) usage("...");
+ putchar('\n');
+ for(c=0;c!=128;c++) if(flag[c]){
+ print("\t-.%c. ", c);
+ n=scanflag(c, argv[1]);
+ for(i=0;i!=n;i++) print(" <%s>", flag[c][i]);
+ putchar('\n');
+ }
+}
+#endif