aboutsummaryrefslogtreecommitdiff
path: root/scripts/config2help.c
diff options
context:
space:
mode:
Diffstat (limited to 'scripts/config2help.c')
-rw-r--r--scripts/config2help.c131
1 files changed, 104 insertions, 27 deletions
diff --git a/scripts/config2help.c b/scripts/config2help.c
index cf65641c..30e7bbab 100644
--- a/scripts/config2help.c
+++ b/scripts/config2help.c
@@ -10,29 +10,54 @@ void toy_exec(char *argv[]) {;}
struct symbol {
struct symbol *next;
- int enabled;
+ int enabled, help_indent;
char *name, *depends;
struct double_list *help;
} *sym;
+char *trim(char *s)
+{
+ while (isspace(*s)) s++;
+
+ return s;
+}
+
char *keyword(char *name, char *line)
{
int len = strlen(name);
- while (isspace(*line)) line++;
+ line = trim(line);
if (strncmp(name, line, len)) return 0;
line += len;
if (*line && !isspace(*line)) return 0;
- while (isspace(*line)) line++;
+ line = trim(line);
return line;
}
+char *dlist_zap(struct double_list **help)
+{
+ struct double_list *dd = dlist_pop(help);
+ char *s = dd->data;
+
+ free(dd);
+ return s;
+}
+
+void zap_blank_lines(struct double_list **help)
+{
+ for(;;) {
+ char *s = trim((*help)->data);;
+
+ if (*s) break;
+ free(dlist_zap(help));
+ }
+}
+
void parse(char *filename)
{
FILE *fp = xfopen(filename, "r");
struct symbol *new = 0;
- int help = 0;
for (;;) {
char *s, *line = NULL;
@@ -48,7 +73,6 @@ void parse(char *filename)
// source or config keyword at left edge?
if (*line && !isspace(*line)) {
- help = 0;
if ((s = keyword("config", line))) {
new = xzalloc(sizeof(struct symbol));
new->next = sym;
@@ -60,15 +84,39 @@ void parse(char *filename)
}
if (!new) continue;
- if (help) dlist_add(&(new->help), line);
+ if (sym && sym->help_indent) {
+ dlist_add(&(new->help), line);
+ if (sym->help_indent < 0) {
+ sym->help_indent = 0;
+ while (isspace(line[sym->help_indent])) sym->help_indent++;
+ }
+ }
else if ((s = keyword("depends", line)) && (s = keyword("on", s)))
new->depends = s;
- else if (keyword("help", line)) help++;
+ else if (keyword("help", line)) sym->help_indent = -1;
}
fclose(fp);
}
+int charsort(void *a, void *b)
+{
+ char *aa = a, *bb = b;
+
+ if (*aa < *bb) return -1;
+ if (*aa > *bb) return 1;
+ return 0;
+}
+
+int dashsort(void *a, void *b)
+{
+ char *aa = *(char **)a, *bb = *(char **)b;
+
+ if (aa[1] < bb[1]) return -1;
+ if (aa[1] > bb[1]) return 1;
+ return 0;
+}
+
int main(int argc, char *argv[])
{
FILE *fp;
@@ -109,30 +157,65 @@ int main(int argc, char *argv[])
for (;;) {
struct symbol *throw = 0, *catch;
- char *this, *that, *name;
+ char *this, *that, *cusage, *tusage, *name;
int len;
// find a usage: name and collate all enabled entries with that name
for (catch = sym; catch; catch = catch->next) {
if (catch->enabled != 1) continue;
- if (catch->help && (this = keyword("usage:", catch->help->data))) {
+ if (catch->help && (that = keyword("usage:", catch->help->data))) {
struct double_list *bang;
+ char *try;
+ // Suck help text out of throw into catch, copying from this to that
+
+ if (!throw) name = that;
+ else if (strncmp(name, that, len) || !isspace(that[len])) continue;
+ catch->enabled++;
+ while (!isspace(*that) && *that) that++;
+ if (!throw) len = that-name;
+ that = trim(that);
if (!throw) {
throw = catch;
- catch->enabled++;
- name = this;
- while (!isspace(*this) && *this) this++;
- len = (that = this)-name;
- while (isspace(*that)) that++;
+ this = that;
continue;
}
- if (strncmp(name, this, len) || !isspace(this[len])) continue;
- catch->enabled++;
+ // Grab usage: lines to collate
+ cusage = dlist_zap(&catch->help);
+ zap_blank_lines(&catch->help);
+ tusage = dlist_zap(&throw->help);
+ zap_blank_lines(&throw->help);
+
+ // Collate first [-abc] option block
+
+ try = 0;
+ if (*this == '[' && this[1] == '-' && this[2] != '-' &&
+ *that == '[' && that[1] == '-' && that[2] != '-')
+ {
+ char *from = this+2, *to = that+2;
+ int ff = strcspn(from, " ]"), tt = strcspn(to, " ]");
+
+ if (from[ff] == ']' && to[tt] == ']') {
+ try = xmprintf("[-%.*s%.*s] ", ff, from, tt, to);
+ qsort(try+2, ff+tt, 1, (void *)charsort);
+ this = trim(this+ff+3);
+ that = trim(that+tt+3);
+ }
+ }
+
+ // Add new collated line (and whitespace).
+ dlist_add(&catch->help, xmprintf("%*cusage: %.*s %s%s%s%s",
+ catch->help_indent, ' ', len, name, try ? try : "",
+ this, *this ? " " : "", that));
+ dlist_add(&catch->help, strdup(""));
+ catch->help = catch->help->prev->prev;
+
+ free(cusage);
+ free(tusage);
+ free(try);
- // Suck help text out of throw into catch.
throw->enabled = 0;
// splice together circularly linked lists
@@ -141,19 +224,16 @@ int main(int argc, char *argv[])
throw->help->prev = catch->help->prev;
catch->help->prev->next = throw->help;
catch->help->prev = bang;
+
throw->help = 0;
throw = catch;
+ this = throw->help->data + throw->help_indent + 8 + len;
}
}
// Did we find one?
if (!throw) break;
-
- // Collate first [-abc] option block?
-
-// if (*s == '[' && s[1] == '-' && s[2] != '-') {
-// }
}
// Print out help #defines
@@ -161,19 +241,16 @@ int main(int argc, char *argv[])
struct double_list *dd;
if (sym->help) {
- int i, padlen = 0;
+ int i;
char *s = xstrdup(sym->name);
for (i = 0; s[i]; i++) s[i] = tolower(s[i]);
printf("#define help_%s \"", s);
free(s);
- // Measure leading whitespace of first line
dd = sym->help;
- while (isspace(dd->data[padlen])) padlen++;
-
for (;;) {
- i = padlen;
+ i = sym->help_indent;
// Trim leading whitespace
s = dd->data;