diff options
Diffstat (limited to '')
| -rw-r--r-- | src/argparser.c | 681 | 
1 files changed, 677 insertions, 4 deletions
| diff --git a/src/argparser.c b/src/argparser.c index a3020d6..b72639f 100644 --- a/src/argparser.c +++ b/src/argparser.c @@ -20,19 +20,692 @@  #include <stdio.h> +#define true  1 +#define false 0 +#define null  0 + +#define ARGUMENTLESS 0 +#define ARGUMENTED   1 +#define VARIADIC     2 + + +static void sort(char** list, long count); +static long cmp(char* a, char* b); + + +/** + * Whether the Linux VT is being used + */  long args_linuxvt; + +/** + * The name of the executed command + */  char* args_program; + +/** + * Whether to free the member of `args_program` + */ +long args_program_dispose; + +/** + * Short, single-line, description of the program + */  char* args_dscription; + +/** + * Formated, multi-line, usage text, `null` if none + */  char* args_usage; + +/** + * Long, multi-line, description of the program, `null` if none + */  char* args_longdscription; +/** + * The error output stream + */ +FILE* args_out; + +/** + * The passed arguments + */ +char** args_arguments; -extern void args_init(char* dscription, char* usage, char* longdscription, char* program, long usestderr) +/** + * The number of passed arguments + */ +long args_arguments_count; + +/** + * The number of unrecognised arguments + */ +long args_unrecognised_count; + +/** + * The concatination of `files` with blankspaces as delimiters, `null` if no files + */ +char* args_message; + +/** + * The arguments passed that is not tied to an option + */ +char** args_files; + +/** + * The number of elements in `args_files` + */ +long args_files_count; + +// Options, in order +// ArrayList<Option> options = new ArrayList<Option>(); + +// Option map +// HashMap<String, Option> optmap = new HashMap<String, Option>(); + +// Parsed arguments, a map from option to arguments, null` if not used, add one `null` element per argumentless use. +// HashMap<String, String[]> opts = new HashMap<String, String[]>(); + + + +/** + * Constructor. + * The short description is printed on same line as the program name + *  + * @param  description      Short, single-line, description of the program + * @param  usage            Formated, multi-line, usage text, may be `null` + * @param  longdescription  Long, multi-line, description of the program, may be `null` + * @param  program          The name of the program, `null` for automatic + * @param  usestderr        Whether to use stderr instead of stdout + */ +extern void args_init(char* description, char* usage, char* longdscription, char* program, long usestderr)  {    char* term = getenv("TERM"); +  args_linuxvt = 0; +  if (term == null) +    if (*(term + 0) == 'l') +      if (*(term + 1) == 'i') +	if (*(term + 2) == 'n') +	  if (*(term + 3) == 'u') +	    if (*(term + 4) == 'x') +	      if (*(term + 5) == 0) +		args_linuxvt = 1; +  args_program_dispose = program == null; +  args_program = program == null ? args_parent_name(0) : program; +  if (args_program == null) +    { +      args_program = "?"; +      args_program_dispose = false; +    } +  args_dscription = description; +  args_usage = usage; +  args_longdscription = longdescription; +  args_out = usestderr ? stderr : stdout; +  args_arguments_count = args_unrecognised_count = args_files_count = 0; +  args_files = args_arguments = null; +  args_message = null; +} + -args_dscription; -args_usage; -args_longdscription; +/** + * Disposes of all resources, run this when you are done + */ +extern void args_dispose() +{ +  if (args_files != null) +    free(args_files); +  if (args_message != null) +    free(args_message); +  if (args_program_dispose) +    free(args_program); +} + + +/** + * Gets the name of the parent process + *  + * @param   levels  The number of parents to walk, 0 for self, and 1 for direct parent + * @return          The name of the parent process, `null` if not found + */ +extern char* args_parent_name(long levels) +{ +  char pid[22]; /* 6 should be enough, but we want to be future proof */ +  ssize_t pid_n = readlink("/proc/self", pid, 21); +  long lvl = levels, i, j, cmdsize; +  size_t n; +  FILE* is; +  char* buf[35]; +  char* cmd; +  char* data; +  if (pid_n <= 0) +    return null; +  pid[pid_n] = 0; +  data = (char*)malloc(2048 * sizeof(char)); +  while (lvl > 0) +    { +      i = 0; +      for (j = 0; *("/proc/" + j);  j++)  *(buf + i++) = *("/proc/" + j); +      for (j = 0; *(pid + j);       j++)  *(buf + i++) = *(pid + j); +      for (j = 0; *("/status" + j); j++)  *(buf + i++) = *("/status" + j); +      *(buf + i++) = 0; +      if ((is = fopen(buf, "r")) == null) +	{ +	  free(data); +	  return null; +	} +      n = fread(data, 1, 2048, is); +      j = 0; +      for (i = 0; i < n; i++) +	{ +	  char c = *(data + i); +	  if (c == '\n') +	    { +	      if (j > 5) +		if (*(buf + 0) == 'P') +		  if (*(buf + 1) == 'P') +		    if (*(buf + 2) == 'i') +		      if (*(buf + 3) == 'd') +			if (*(buf + 4) == ':') +			  { +			    i = 5; +			    while ((*(buf + i) == '\t') || (*(buf + i) == ' ')) +			      i++; +			    j -= n = i; +			    buf += n; +			    for (i = 0; i < j; i++) +			      *(pid + i) = *(buf + i); +			    *(pid + j) = 0; +			    buf -= n; +			    lvl--; +			    break; +			  } +	      j = 0; +	    } +	  if (j < 35) +	    *(buf + j) = c; +	} +      free(data); +      return null; +    } +  free(data); +  i = 0; +  for (j = 0; *("/proc/" + j);   j++)  *(buf + i++) = *("/proc/" + j); +  for (j = 0; *(pid + j);        j++)  *(buf + i++) = *(pid + j); +  for (j = 0; *("/cmdline" + j); j++)  *(buf + i++) = *("/cmdline" + j); +  *(buf + i++) = 0; +  if ((is = fopen(buf, "r")) == null) +    return null; +  i = 0; +  n = 0; +  cmd = (char*)malloc(cmdsize = 128); +  for (;;) +    { +      n += fread(cmd, 1, 128, is); +      for (; i < n; i++) +	if (*(cmd + i) == 0) +	  break; +      if (i == n) +	{ +	  char* tmp = (char*)malloc(cmdsize + 128); +	  for (i = 0; i < n; i++) +	    *(tmp + i) = *(cmd + i); +	  cmdsize += 128; +	  free(cmd); +	  cmd = tmp; +	} +      else +	break; +    } +  if (*cmd == 0) +    { +      free(cmd); +      cmd = 0; +    } +  return cmd; +} + + +/** + * Checks the correctness of the number of used non-option arguments + *  + * @param   min  The minimum number of files + * @return       Whether the usage was correct + */ +extern long args_test_files_min(long min) +{ +  return min <= args_files_count; +} + + +/** + * Checks the correctness of the number of used non-option arguments + *  + * @param   max  The maximum number of files + * @return       Whether the usage was correct + */ +extern long args_test_files_max(long max) +{ +  return args_files_count <= max; +} + + +/** + * Checks the correctness of the number of used non-option arguments + *  + * @param   min  The minimum number of files + * @param   max  The maximum number of files + * @return       Whether the usage was correct + */ +extern long args_test_files(long min, long max) +{ +  return (min <= args_files_count) && (args_files_count <= max); +} + + +/** + * Checks for out of context option usage + *  + * @param   allowed        Allowed options, will be sorted + * @param   allowed_count  The number of elements in `allowed` + * @return                 Whether only allowed options was used + */ +extern long args_test_allowed(char** allowed, long allowed_count) +{ +  char** opts; +  char** a; +  char** o; +  long rc = true, _a, _o; +   +  sort(allowed, _a = allowed_count); +  opts = args_get_opts(); /* TODO */ +  sort(opts, _o = args_get_opts_count()); /* TODO */ +   +  a = allowed + _a; +  o = opts + _o; +   +  while (opts != o) +    { +      if ((allowed == a) || (cmp(*opts, *allowed) < 0)) +	if (args_opts_used(*opt)) /* TODO */ +	  { +	    fprintf(args_out, "%s: option used out of context: %s", args_program, *opts); +	    char* std = args_optmap_get_std(*opt); /* TODO */ +	    if (cmp(std, *opt) != 0) +	      fprintf(args_out, "(%s)", std); +	    fprintf(args_out, "\n"); +	    rc = false; +	  } +      while ((allowed != a) && (cmp(*opts, *allowed) > 0)) +	allowed++; +      opts++; +    } +   +  return rc; +} + + +/** + * Checks for option conflicts + *  + * @param   exclusives        Exclusive options, will be sorted + * @param   exclusives_count  The number of elements in `exclusives` + * @return                    Whether at most one exclusive option was used + */ +extern long args_test_exclusiveness(char** exclusives, long exclusives_count) +{ +  long used_ptr = 0, i = 0; +  char** used = (char**)malloc(args_get_opts_count() * sizeof(char*)); +  char** e; +  char** o; +  char* std; +  long _e, _o; +   +  sort(exclusives, _e = exclusives_count); +  opts = args_get_opts(); +  sort(opts, _o = args_get_opts_count()); +   +  e = exclusives + _e; +  o = opts + _o; +   +  while ((opts != o) && (exclusives != e)) +    { +      while ((opts != o) && (cmp(*opts, *allowed) > 0)) +	opt++; +      while ((allowed != a) && (cmp(*opts, *allowed) > 0)) +	allowed++; +      if ((cmp(*opts, *allowed) == 0) && (args_opts_used(*opt))) +	*(used + used_ptr++) = opt; +      opts++; +    } +   +  if (used_ptr >= 1) +    { +      fprintf(args_out, "%s: conflicting options:", args_program); +      for (; i < used_ptr; i++) +	{ +	  std = args_optmap_get_std(*(used + i)); +	  if (cmp(*(used + i), std) == 0) +	    fprintf(args_out, " %s", *(used + i)); +	  else +	    fprintf(args_out, " %s(%s)", *(used + i), std); +	} +      fprintf(args_out, "\n"); +      free(used); +      return false; +    } +   +  free(used); +  return true; +} + + +/** + * Maps up options that are alternatives to the first alternative for each option + */ +extern void args_support_alternatives() +{ +  char** opts = args_get_optmap(); /* TODO */ +  long n = args_get_optmap_count(); /* TODO */ +  long i; +   +  for (i = 0; i < n; i++) +    args_opts_put(*(opts + 1), args_opts_get(args_optmap_get_std(*opt))); +    /* TODO */                 /* TODO */ +} + + +/** + * Prints a colourful help message + */ +extern void args_help() +{ +  long maxfirstlen = 0, count = 0, copts = args_get_options_count(); /* TODO */ +  char* dash = args_linuxvt ? "-" : "—"; +  char* empty; +  char** lines; +  long* lens; +   +  fprintf(args_out, "\033[01m%s\033[21m %s %s\n", args_program, dash, args_description); +  if (args_longdescription != null) +    fprintf(args_out, "%s\n", args_longdescription); +  fprintf(args_out, "\n"); +   +  if (args_usage != null) +    { +      long n = 0, lines = 0, i = 0; +      char* buf; +      fprintf(args_out, "\033[01mUSAGE:\033[21m\n"); +      while (*(args_usage + n)) +	if (*(args_usage + n++) == '\n') +	  lines++; +      buf = (char*)malloc((n + 2 + lines * 7) * sizeof(char)); +      *buf++ = '\t'; +      while (i < n) +	{ +	  *buf++ = *(args_usage + i); +	  if (*(args_usage + i++) == '\n') +	    { +	      *buf++ = ' '; +	      *buf++ = ' '; +	      *buf++ = ' '; +	      *buf++ = ' '; +	      *buf++ = 'o'; +	      *buf++ = 'r'; +	      *buf++ = '\t'; +	    } +	} +      *buf++ = 0; +      buf -= n + 2 + lines * 7; +      fprintf(args_out, "%s\n\n", buf); +      free(buf); +    } +   +  { +    long i = 0; +    for (i = 0; i < copts; i++) +      { +	if (args_options_get_help(i) == null) /* TODO */ +	  continue; +	if (args_options_get_alternatives_count(i) > 1) /* TODO */ +	  { +	    long n = 0; +	    char* first = *(args_options_get_alternatives(i)); /* TODO */ +	    while (*(first + n)) +	      n++; +	    if (maxfirstlen < n) +	      maxfirstlen = n; +	  } +      } +  } +   +  empty = (char*)malloc((maxfirstlen + 1) * sizeof(char)); +  { +    long i; +    for (i = 0; i < maxfirstlen; i++) +      *(empty + i++) = ' '; +    *(empty + maxfirstlen) = 0; +  } +   +  fprintf(args_out, "\033[01mSYNOPSIS:\033[21m\n"); +  lines = (char**)malloc(copts * sizeof(char*)); +  lens = (long*)malloc(copts * sizeof(long)); +  { +    char* first_extra = 0; +    long i = 0, n, l, j, type; +    for (i = 0; i < copts; i++) +      { +	char* first; +	char* last; +	char* line; +	char* arg; +	if (args_options_get_help(i) == null) +	  continue; +	l = 0; +	arg = *(args_options_get_argument(i)); /* TODO */ +	first = *(args_options_get_alternatives(i)); /* TODO */ +	last = *(args_options_get_alternatives(i) + args_options_get_alternatives_count(i) - 1); /* TODO */ +	type = *(args_options_get_type(i)); /* TODO */ +	if (first == last) +	  first = empty; +	else +	  { +	    n = 0; +	    while (*(first + n)) +	      n++; +	    first_extra = empty + n; +	  } +	n = 0; +	while (*(last + n)) +	  n++; +	if (arg != null) +	  while (*(arg + n)) +	    n++; +	l += maxfirstlen + 6 + n; +	*(lines + count) = line = (char*)malloc((1 + 17 + 16 + maxfirstlen + n) * sizeof(char)); +	for (j = 0; *("    \033[02m" + j); j++) +	  *line++ = *("    \033[02m" + j); +	for (j = 0; *(first + j); j++) +	  *line++ = *(first + j); +	if (first_extra != null) +	  for (j = 0; *(first_extra + j); j++) +	    *line++ = *(first_extra + j); +	for (j = 0; *("\033[22m  \0" + j); j++) +	  *line++ = *("\033[22m  \0" + j); +	for (j = 0; *(last + j); j++) +	  *line++ = *(last + j); +	if (type == VARIADIC) +	  { +	    for (j = 0; *(" [\033[04m" + j); j++) +	      *line++ = *(" [\033[04m" + j); +	    for (j = 0; *(arg + j); j++) +	      *line++ = *(arg + j); +	    for (j = 0; *("\033[24m...]" + j); j++) +	      *line++ = *("\033[24m...]" + j); +	    l += 6; +	  } +	else if (type == ARGUMENTED) +	  { +	    for (j = 0; *(" \033[04m" + j); j++) +	      *line++ = *(" \033[04m" + j); +	    for (j = 0; *(arg + j); j++) +	      *line++ = *(arg + j); +	    for (j = 0; *("\033[24m" + j); j++) +	      *line++ = *("\033[24m" + j); +	    l += 1; +	  } +	*line = 0; +	*(lens + count++) = l; +      } +  } +   +  free(empty); +   +  { +    long col = 0, i = 0, index = 0 +    for (; i < count; i++) +      if (col < *(lens + i)) +	col = *(lens + i); +    col += 8 - ((col - 4) & 7); +     +    empty = (char*)malloc((col + 1) * sizeof(char)); +    for (i = 0; i < col; i++) +      *(empty + i++) = ' '; +    *(empty + col) = 0; +    i = 0; +     +    for (i = 0; i < copts; i++) +      { +	long first = true, j = 0, jptr = 0; +	char* colour = (index & 1) == 0 ? "36" : "34"; +	char* help = args_options_get_help(i); +	char* line; +	char* buf; +	char** jumps; +	char c; +	if (help == null) +	  continue; +	fprintf(args_out, "%s\033[%s;01m", line = *(lines + index), colour); +	while (*line++) +	  ; +	fprintf(args_out, "%s%s", line, empty + *(lens + index)); +	free(*(lines + index++)); +	while ((c = *(help + j++))) +	  if (c == '\n') +	    jptr++; +	jumps = (char**)malloc(jptr * sizeof(char*)); +	*jumps = buf = (char*)malloc(j * sizeof(char)); +	j = 0; +	jptr = 1; +	while ((c = *(help + j))) +	  if (c == '\n') +	    { +	      *(buf + j++) = 0; +	      *(jumps + jptr++) = buf + j; +	    } +	  else +	    *(buf + j++) = c; +	for (j = 0; j < jptr; j++) +	  if (first) +	    { +	      first = false; +	      fprintf(args_out, "%s\033[00m\n", *(jump + j)); +	    } +	  else +	    fprintf(args_out, "%s\033[%sm%s\033[00m\n", empty, colour, *(jump + j)); +	free(buf); +	free(jumps); +      } +  } +   +  free(empty); +  free(lines); +  free(lens); +  fprintf(args_out, "\n"); +} + + +/** + * Parse arguments + *  + * @param   argc  The number of elements in `argv` + * @param   argv  The command line arguments, it should include the execute file at index 0 + * @return        Whether no unrecognised option is used + */ +extern long args_parse(int argc, char** argv) +{ +  /* TODO */ +} + + +/** + * Compare two strings + *  + * @param  a  -1 if returned if this sting is the alphabetically lesser one + * @param  b   1 if returned if this sting is the alphabetically lesser one + * @return    0 is returned if the two string are identical, other -1 or 1 is returned + */ +static long cmp(char* a, char* b) +{ +  char c; +  while (*a && *b) +    { +      if ((c = (*a < *b ? -1 : (*a > *b ? 1 : 0)))) +	return c; +      a++; +      b++; +    } +  return *a < *b ? -1 : (*a > *b ? 1 : 0); +} + + +/** + * Naïve merge sort is best merge sort in C + *  + * @param  list   The list to sort from the point that needs sorting + * @param  count  The number of elements to sort + * @param  temp   Auxiliary memory + */ +static void _sort(char** list, long count, char** temp) +{ +  if (count > 1) +    { +      long i = 0, a = count >> 1; +      long j = a, b = count - a; +      sort(list + 0, a, temp + 0); +      sort(list + a, b, temp + a); +      b += a; +      while ((i < a) && (j < b)) +	{ +          char c = cmp(*(temp + i), *(temp + j)); +          if (c <= 0) +            *list++ = *(temp + i++); +          else +            *list++ = *(temp + j++); +	} +      while (i < a) +	*list++ = *(temp + i++); +      while (j < b) +	*list++ = *(temp + j++); +      list -= count; +      for (i = 0; i < count; i++) +	*(temp + i) = *(list + i); +    } +  else if (count == 1) +    *temp = *list; +} + + +/** + * Naïve merge sort is best merge sort in C + *  + * @param  list   The list to sort + * @param  count  The number of elements to sort + */ +static void sort(char** list, long count) +{ +  char** temp = (char**)malloc(count * sizeof(char*)); +  _sort(list, count, temp); +  free(temp);  } | 
