diff options
Diffstat (limited to 'src/skaembutils/s6-sort.c')
| -rw-r--r-- | src/skaembutils/s6-sort.c | 59 |
1 files changed, 35 insertions, 24 deletions
diff --git a/src/skaembutils/s6-sort.c b/src/skaembutils/s6-sort.c index f7a7721..66955fe 100644 --- a/src/skaembutils/s6-sort.c +++ b/src/skaembutils/s6-sort.c @@ -4,6 +4,7 @@ #include <strings.h> #include <stdlib.h> #include <errno.h> + #include <skalibs/allreadwrite.h> #include <skalibs/sgetopt.h> #include <skalibs/buffer.h> @@ -15,26 +16,34 @@ #define USAGE "s6-sort [ -bcfru0 ]" -typedef int strncmp_t (char const *, char const *, size_t) ; -typedef strncmp_t *strncmp_t_ref ; -typedef int qsortcmp_t (void const *, void const *) ; -typedef qsortcmp_t *qsortcmp_t_ref ; +typedef int strncmp_func (char const *, char const *, size_t) ; +typedef strncmp_func *strncmp_func_ref ; +typedef int qsortcmp_func (void const *, void const *) ; +typedef qsortcmp_func *qsortcmp_func_ref ; -static int flagnoblanks = 0, flagreverse = 0, flaguniq = 0 ; +typedef struct sort_global_s sort_global, *sort_global_ref ; +struct sort_global_s +{ + strncmp_func_ref comp ; + unsigned char flagnoblanks : 1 ; + unsigned char flagreverse : 1 ; + unsigned char flaguniq : 1 ; +} ; +#define SORT_GLOBAL_ZERO { .flagnoblanks = 0, .flagreverse = 0, .flaguniq = 0, .comp = &strncmp } -static strncmp_t_ref comp = &strncmp ; +static sort_global_ref sort_G ; static int compit (char const *s1, size_t n1, char const *s2, size_t n2) { int r ; - if (flagnoblanks) + if (sort_G->flagnoblanks) { while ((*s1 == ' ') || (*s1 == '\t')) (s1++, n1--) ; while ((*s2 == ' ') || (*s2 == '\t')) (s2++, n2--) ; } - r = (*comp)(s1, s2, n1 < n2 ? n1 : n2) ; + r = (*sort_G->comp)(s1, s2, n1 < n2 ? n1 : n2) ; if (!r) r = n1 - n2 ; - return flagreverse ? -r : r ; + return sort_G->flagreverse ? -r : r ; } static int sacmp (stralloc const *a, stralloc const *b) @@ -42,7 +51,7 @@ static int sacmp (stralloc const *a, stralloc const *b) return compit(a->s, a->len - 1, b->s, b->len - 1) ; } -static ssize_t slurplines (genalloc *lines, char sep) +static ssize_t sort_slurplines (genalloc *lines, char sep) { ssize_t i = 0 ; for (;; i++) @@ -58,7 +67,7 @@ static ssize_t slurplines (genalloc *lines, char sep) return i ; } -static void uniq (genalloc *lines) +static void sort_uniq (genalloc *lines) { size_t len = genalloc_len(stralloc, lines) ; stralloc *s = genalloc_s(stralloc, lines) ; @@ -67,7 +76,7 @@ static void uniq (genalloc *lines) if (!sacmp(s+i-1, s+i)) stralloc_free(s+i-1) ; } -static ssize_t outputlines (stralloc const *s, size_t len) +static ssize_t sort_outputlines (stralloc const *s, size_t len) { size_t i = 0 ; for (; i < len ; i++) @@ -75,19 +84,21 @@ static ssize_t outputlines (stralloc const *s, size_t len) return buffer_flush(buffer_1) ; } -static int check (stralloc const *s, size_t len) +static int sort_check (stralloc const *s, size_t len) { size_t i = 1 ; for (; i < len ; i++) - if (sacmp(s+i-1, s+i) >= !flaguniq) return 0 ; + if (sacmp(s+i-1, s+i) >= !sort_G->flaguniq) return 0 ; return 1 ; } int main (int argc, char const *const *argv) { genalloc lines = GENALLOC_ZERO ; /* array of stralloc */ - char sep = '\n' ; int flagcheck = 0 ; + char sep = '\n' ; + sort_global globals = SORT_GLOBAL_ZERO ; + sort_G = &globals ; PROG = "s6-sort" ; { subgetopt l = SUBGETOPT_ZERO ; @@ -97,11 +108,11 @@ int main (int argc, char const *const *argv) if (opt == -1) break ; switch (opt) { - case 'b' : flagnoblanks = 1 ; break ; + case 'b' : sort_G->flagnoblanks = 1 ; break ; case 'c' : flagcheck = 1 ; break ; - case 'f' : comp = &strncasecmp ; break ; - case 'r' : flagreverse = 1 ; break ; - case 'u' : flaguniq = 1 ; break ; + case 'f' : sort_G->comp = &strncasecmp ; break ; + case 'r' : sort_G->flagreverse = 1 ; break ; + case 'u' : sort_G->flaguniq = 1 ; break ; case '0' : sep = '\0' ; break ; default : strerr_dieusage(100, USAGE) ; } @@ -109,11 +120,11 @@ int main (int argc, char const *const *argv) argc -= l.ind ; argv += l.ind ; } - if (slurplines(&lines, sep) < 0) strerr_diefu1sys(111, "read from stdin") ; - if (flagcheck) return !check(genalloc_s(stralloc, &lines), genalloc_len(stralloc, &lines)) ; - qsort(genalloc_s(stralloc, &lines), genalloc_len(stralloc, &lines), sizeof(stralloc), (qsortcmp_t_ref)&sacmp) ; - if (flaguniq) uniq(&lines) ; - if (!outputlines(genalloc_s(stralloc, &lines), genalloc_len(stralloc, &lines))) + if (sort_slurplines(&lines, sep) < 0) strerr_diefu1sys(111, "read from stdin") ; + if (flagcheck) return !sort_check(genalloc_s(stralloc, &lines), genalloc_len(stralloc, &lines)) ; + qsort(genalloc_s(stralloc, &lines), genalloc_len(stralloc, &lines), sizeof(stralloc), (qsortcmp_func_ref)&sacmp) ; + if (sort_G->flaguniq) sort_uniq(&lines) ; + if (!sort_outputlines(genalloc_s(stralloc, &lines), genalloc_len(stralloc, &lines))) strerr_diefu1sys(111, "write to stdout") ; return 0 ; } |
