--- /dev/null
+/*
+ * rev implementation for busybox
+ *
+ * Copyright (C) 2010 Marek Polacek <mmpolacek@gmail.com>
+ *
+ * Licensed under GPLv2, see file License in this tarball for details.
+ */
+
+//applet:IF_REV(APPLET(rev, _BB_DIR_BIN, _BB_SUID_DROP))
+
+//kbuild:lib-$(CONFIG_REV) += rev.o
+
+//config:config REV
+//config: bool "rev"
+//config: default y
+//config: help
+//config: Reverse lines of a file or files.
+
+//usage:#define rev_trivial_usage
+//usage: "[FILE]..."
+//usage:#define rev_full_usage "\n\n"
+//usage: "Reverse lines of FILE"
+
+#include "libbb.h"
+#include "unicode.h"
+
+#undef CHAR_T
+#if ENABLE_UNICODE_SUPPORT
+# define CHAR_T wchar_t
+#else
+# define CHAR_T char
+#endif
+
+/* In-place invert */
+static void strrev(CHAR_T *s, int len)
+{
+ int i;
+
+ if (len != 0) {
+ len--;
+ if (len != 0 && s[len] == '\n')
+ len--;
+ }
+
+ for (i = 0; i < len; i++, len--) {
+ CHAR_T c = s[i];
+ s[i] = s[len];
+ s[len] = c;
+ }
+}
+
+int rev_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
+int rev_main(int argc UNUSED_PARAM, char **argv)
+{
+ int retval;
+ size_t bufsize;
+ char *buf;
+
+ init_unicode();
+
+ getopt32(argv, "");
+ argv += optind;
+ if (!argv[0])
+ argv = (char **)&bb_argv_dash;
+
+ retval = EXIT_SUCCESS;
+ bufsize = 256;
+ buf = xmalloc(bufsize);
+ do {
+ size_t pos;
+ FILE *fp;
+
+ fp = fopen_or_warn_stdin(*argv++);
+ if (!fp) {
+ retval = EXIT_FAILURE;
+ continue;
+ }
+
+ pos = 0;
+ while (1) {
+ /* Read one line */
+ buf[bufsize - 1] = 1; /* not 0 */
+ if (!fgets(buf + pos, bufsize - pos, fp))
+ break; /* EOF/error */
+ if (buf[bufsize - 1] == '\0' /* fgets filled entire buffer */
+ && buf[bufsize - 2] != '\n' /* and did not read '\n' */
+ && !feof(fp)
+ ) {
+ /* Line is too long, extend buffer */
+ pos = bufsize - 1;
+ bufsize += 64 + bufsize / 8;
+ buf = xrealloc(buf, bufsize);
+ continue;
+ }
+
+ /* Process and print it */
+#if ENABLE_UNICODE_SUPPORT
+ {
+ wchar_t *tmp = xmalloc(bufsize * sizeof(wchar_t));
+ /* Convert to wchar_t (might error out!) */
+ int len = mbstowcs(tmp, buf, bufsize);
+ if (len >= 0) {
+ strrev(tmp, len);
+ /* Convert back to char */
+ wcstombs(buf, tmp, bufsize);
+ }
+ free(tmp);
+ }
+#else
+ strrev(buf, strlen(buf));
+#endif
+ fputs(buf, stdout);
+ }
+ fclose(fp);
+ } while (*argv);
+
+ if (ENABLE_FEATURE_CLEAN_UP)
+ free(buf);
+
+ fflush_stdout_and_exit(retval);
+}