+/* vi: set sw=4 ts=4: */
/*
* Mini sort implementation for busybox
*
*
- * Copyright (C) 1999 by Lineo, inc.
- * Written by John Beppu <beppu@lineo.com>
+ * Copyright (C) 2000 by Matt Kraai <kraai@alumni.carnegiemellon.edu>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
*
*/
-#include "internal.h"
-#include <sys/types.h>
-#include <fcntl.h>
-#include <dirent.h>
-#include <stdio.h>
-#include <errno.h>
+#include "busybox.h"
+#include <getopt.h>
+#include <stdlib.h>
-static const char sort_usage[] =
-"Usage: sort [OPTION]... [FILE]...\n\n"
-;
-
-/* typedefs _______________________________________________________________ */
-
-/* line node */
-typedef struct Line {
- char *data; /* line data */
- struct Line *next; /* pointer to next line node */
-} Line;
-
-/* singly-linked list of lines */
-typedef struct {
- int len; /* number of Lines */
- Line **sorted; /* array fed to qsort */
-
- Line *head; /* head of List */
- Line *current; /* current Line */
-} List;
-
-/* comparison function */
-typedef int (Compare)(const void *, const void *);
-
-
-/* methods ________________________________________________________________ */
-
-static const int max = 1024;
-
-/* mallocate Line */
-static Line *
-line_alloc()
-{
- Line *self;
- self = malloc(1 * sizeof(Line));
- return self;
-}
-
-/* Initialize Line with string */
-static Line *
-line_init(Line *self, const char *string)
-{
- self->data = malloc((strlen(string) + 1) * sizeof(char));
- if (self->data == NULL) { return NULL; }
- strcpy(self->data, string);
- self->next = NULL;
- return self;
-}
-
-/* Construct Line from FILE* */
-static Line *
-line_newFromFile(FILE *src)
-{
- char buffer[max];
- Line *self;
-
- if (fgets(buffer, max, src)) {
- self = line_alloc();
- if (self == NULL) { return NULL; }
- line_init(self, buffer);
- return self;
- }
- return NULL;
-}
-
-/* Line destructor */
-static Line *
-line_release(Line *self)
-{
- if (self->data) {
- free(self->data);
- free(self);
- }
- return self;
-}
-
-
-/* Comparison */
-
-/* ascii order */
-static int
-compare_ascii(const void *a, const void *b)
-{
- Line **doh;
- Line *x, *y;
-
- doh = (Line **) a;
- x = (Line *) *doh;
- doh = (Line **) b;
- y = (Line *) *doh;
-
- // fprintf(stdout, "> %p: %s< %p: %s", x, x->data, y, y->data);
- return strcmp(x->data, y->data);
-}
-
-/* numeric order */
-static int
-compare_numeric(const void *a, const void *b)
-{
- return 0;
-}
-
-
-/* List */
-
-/* */
-static List *
-list_init(List *self)
-{
- self->len = 0;
- self->sorted = NULL;
- self->head = NULL;
- self->current = NULL;
- return self;
-}
-
-/* for simplicity, the List gains ownership of the line */
-static List *
-list_insert(List *self, Line *line)
-{
- if (line == NULL) { return NULL; }
-
- /* first insertion */
- if (self->head == NULL) {
- self->head = line;
- self->current = line;
-
- /* all subsequent insertions */
- } else {
- self->current->next = line;
- self->current = line;
- }
- self->len++;
- return self;
-}
-
-/* order the list according to compare() */
-static List *
-list_sort(List *self, Compare *compare)
+int compare_ascii(const void *x, const void *y)
{
- int i;
- Line *line;
-
- /* mallocate array of Line*s */
- self->sorted = (Line **) malloc(self->len * sizeof(Line*));
- if (self->sorted == NULL) { return NULL; }
-
- /* fill array w/ List's contents */
- i = 0;
- line = self->head;
- while (line) {
- self->sorted[i++] = line;
- line = line->next;
- }
-
- /* apply qsort */
- qsort(self->sorted, self->len, sizeof(Line*), compare);
- return self;
+ return strcmp(*(char **)x, *(char **)y);
}
-/* precondition: list must be sorted */
-static List *
-list_writeToFile(List *self, FILE* dst)
+int compare_numeric(const void *x, const void *y)
{
- int i;
- Line **line = self->sorted;
-
- if (self->sorted == NULL) { return NULL; }
- for (i = 0; i < self->len; i++) {
- fprintf(dst, "%s", line[i]->data);
- }
- return self;
+ return atoi(*(char **)x) - atoi(*(char **)y);
}
-/* deallocate */
-static void
-list_release(List *self)
+int sort_main(int argc, char **argv)
{
- Line *i;
- Line *die;
-
- i = self->head;
- while (i) {
- die = i;
- i = die->next;
- line_release(die);
- }
-}
-
-
-/*
- * I need a list
- * to insert lines into
- * then I need to sort this list
- * and finally print it
- */
-
-int
-sort_main(int argc, char **argv)
-{
- int i;
- char opt;
- List list;
- Line *l;
-
- /* default behaviour */
-
- /* parse argv[] */
- for (i = 1; i < argc; i++) {
- if (argv[i][0] == '-') {
- opt = argv[i][1];
- switch (opt) {
- case 'h':
- usage(sort_usage);
- break;
- default:
- fprintf(stderr, "sort: invalid option -- %c\n", opt);
- usage(sort_usage);
- }
- } else {
- break;
+ FILE *fp;
+ char *line, **lines = NULL;
+ int i, opt, nlines = 0;
+ int (*compare)(const void *, const void *) = compare_ascii;
+#ifdef BB_FEATURE_SORT_REVERSE
+ int reverse = FALSE;
+#endif
+
+ while ((opt = getopt(argc, argv, "nr")) != -1) {
+ switch (opt) {
+ case 'n':
+ compare = compare_numeric;
+ break;
+#ifdef BB_FEATURE_SORT_REVERSE
+ case 'r':
+ reverse = TRUE;
+ break;
+#endif
+ default:
+ usage(sort_usage);
+ }
}
- }
-
- /* initialize list */
- list_init(&list);
- /* go through remaining args (if any) */
- if (i >= argc) {
- while ( (l = line_newFromFile(stdin))) {
- list_insert(&list, l);
+ /* read the input */
+ for (i = optind; i == optind || i < argc; i++) {
+ if (argv[i] == NULL)
+ fp = stdin;
+ else
+ fp = xfopen(argv[i], "r");
+
+ while ((line = get_line_from_file(fp)) != NULL) {
+ lines = xrealloc(lines, sizeof(char *) * (nlines + 1));
+ lines[nlines++] = line;
+ }
}
- list_sort(&list, compare_ascii);
- list_writeToFile(&list, stdout);
- list_release(&list);
- } else {
- for ( ; i < argc; i++) {
- }
- }
- exit(0);
+ /* sort it */
+ qsort(lines, nlines, sizeof(char *), compare);
+
+ /* print it */
+#ifdef BB_FEATURE_SORT_REVERSE
+ if (reverse)
+ for (i = nlines - 1; 0 <= i; i--)
+ fputs(lines[i], stdout);
+ else
+#endif
+ for (i = 0; i < nlines; i++)
+ fputs(lines[i], stdout);
+ return EXIT_SUCCESS;
}
-
-/* $Id: sort.c,v 1.6 1999/12/22 23:02:12 beppu Exp $ */
-/*
- * $Log: sort.c,v $
- * Revision 1.6 1999/12/22 23:02:12 beppu
- * oops.. qsort(2) misunderstanding on my part.
- * it's ok, now.
- *
- * Revision 1.5 1999/12/22 22:27:01 beppu
- * playing w/ $Log: sort.c,v $
- * playing w/ Revision 1.6 1999/12/22 23:02:12 beppu
- * playing w/ oops.. qsort(2) misunderstanding on my part.
- * playing w/ it's ok, now.
- * playing w/
- *
- */