busybox/util-linux/rev.c
Denys Vlasenko c484846c44 introduce and use exitcode_t
function                                             old     new   delta
strings_main                                         422     420      -2
setfattr_main                                        175     173      -2
brctl_main                                          1548    1546      -2
makedevs_main                                        979     975      -4
rev_main                                             337     332      -5
getfattr_main                                        307     302      -5
cut_main                                            1201    1196      -5
cksum_main                                           398     393      -5
umount_main                                          573     565      -8
ln_main                                              516     508      -8
expand_main                                          660     652      -8
df_main                                             1068    1060      -8
renice_main                                          346     332     -14
------------------------------------------------------------------------------
(add/remove: 0/0 grow/shrink: 0/13 up/down: 0/-76)            Total: -76 bytes

Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2023-07-17 17:29:36 +02:00

121 lines
2.3 KiB
C

/*
* rev implementation for busybox
*
* Copyright (C) 2010 Marek Polacek <mmpolacek@gmail.com>
*
* Licensed under GPLv2, see file LICENSE in this source tree.
*/
//config:config REV
//config: bool "rev (4.6 kb)"
//config: default y
//config: help
//config: Reverse lines of a file or files.
//applet:IF_REV(APPLET(rev, BB_DIR_BIN, BB_SUID_DROP))
//kbuild:lib-$(CONFIG_REV) += rev.o
//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)
{
exitcode_t 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_stdout(buf);
pos = 0;
}
fclose(fp);
} while (*argv);
if (ENABLE_FEATURE_CLEAN_UP)
free(buf);
fflush_stdout_and_exit(retval);
}