rev: new applet
text data bss dec hexfilename 377 0 0 377 179rev.o Signed-off-by: Marek Polacek <mmpolacek@gmail.com> Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
parent
6d9c88ad98
commit
fc6f6e933c
121
util-linux/rev.c
Normal file
121
util-linux/rev.c
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
/*
|
||||||
|
* 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);
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user