fold: unicode support. Based on a patch by Tomas Heinrich <heinrich.tomas@gmail.com>

General Unicode support is tweaked to expose unicode_status.

function                                             old     new   delta
init_unicode                                           -      77     +77
write2stdout                                           -      19     +19
adjust_column                                         68      71      +3
unicode_status                                         -       1      +1
unicode_is_enabled                                     1       -      -1
grep_main                                            780     773      -7
fold_main                                            619     552     -67
check_unicode_in_env                                  77       -     -77
------------------------------------------------------------------------------
(add/remove: 3/2 grow/shrink: 1/2 up/down: 100/-152)          Total: -52 bytes

Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
Denys Vlasenko 2010-01-04 20:49:58 +01:00
parent d2b1ba6fde
commit 28055028a7
11 changed files with 185 additions and 107 deletions

View File

@ -58,7 +58,7 @@ int df_main(int argc UNUSED_PARAM, char **argv)
const char *disp_units_hdr = NULL; const char *disp_units_hdr = NULL;
char *chp; char *chp;
check_unicode_in_env(); init_unicode();
#if ENABLE_FEATURE_HUMAN_READABLE && ENABLE_FEATURE_DF_FANCY #if ENABLE_FEATURE_HUMAN_READABLE && ENABLE_FEATURE_DF_FANCY
opt_complementary = "k-mB:m-Bk:B-km"; opt_complementary = "k-mB:m-Bk:B-km";

View File

@ -144,7 +144,7 @@ int expand_main(int argc UNUSED_PARAM, char **argv)
"all\0" No_argument "a" "all\0" No_argument "a"
; ;
#endif #endif
check_unicode_in_env(); init_unicode();
if (ENABLE_EXPAND && (!ENABLE_UNEXPAND || applet_name[0] == 'e')) { if (ENABLE_EXPAND && (!ENABLE_UNEXPAND || applet_name[0] == 'e')) {
IF_FEATURE_EXPAND_LONG_OPTIONS(applet_long_options = expand_longopts); IF_FEATURE_EXPAND_LONG_OPTIONS(applet_long_options = expand_longopts);

View File

@ -9,8 +9,8 @@
Licensed under the GPL v2 or later, see the file LICENSE in this tarball. Licensed under the GPL v2 or later, see the file LICENSE in this tarball.
*/ */
#include "libbb.h" #include "libbb.h"
#include "unicode.h"
/* Must match getopt32 call */ /* Must match getopt32 call */
#define FLAG_COUNT_BYTES 1 #define FLAG_COUNT_BYTES 1
@ -20,39 +20,53 @@
/* Assuming the current column is COLUMN, return the column that /* Assuming the current column is COLUMN, return the column that
printing C will move the cursor to. printing C will move the cursor to.
The first column is 0. */ The first column is 0. */
static int adjust_column(int column, char c) static int adjust_column(unsigned column, char c)
{ {
if (!(option_mask32 & FLAG_COUNT_BYTES)) { if (option_mask32 & FLAG_COUNT_BYTES)
return ++column;
if (c == '\t')
return column + 8 - column % 8;
if (c == '\b') { if (c == '\b') {
if (column > 0) if ((int)--column < 0)
column--;
} else if (c == '\r')
column = 0; column = 0;
else if (c == '\t') }
column = column + 8 - column % 8; else if (c == '\r')
else /* if (isprint(c)) */ column = 0;
column++; else { /* just a printable char */
} else if (unicode_status != UNICODE_ON /* every byte is a new char */
|| (c & 0xc0) != 0x80 /* it isn't a 2nd+ byte of a Unicode char */
) {
column++; column++;
}
}
return column; return column;
} }
/* Note that this function can write NULs, unlike fputs etc. */
static void write2stdout(const void *buf, unsigned size)
{
fwrite(buf, 1, size, stdout);
}
int fold_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; int fold_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
int fold_main(int argc UNUSED_PARAM, char **argv) int fold_main(int argc UNUSED_PARAM, char **argv)
{ {
char *line_out = NULL; char *line_out = NULL;
int allocated_out = 0; const char *w_opt = "80";
char *w_opt; unsigned width;
int width = 80; smallint exitcode = EXIT_SUCCESS;
int i;
int errs = 0; init_unicode();
if (ENABLE_INCLUDE_SUSv2) { if (ENABLE_INCLUDE_SUSv2) {
/* Turn any numeric options into -w options. */ /* Turn any numeric options into -w options. */
int i;
for (i = 1; argv[i]; i++) { for (i = 1; argv[i]; i++) {
char const *a = argv[i]; const char *a = argv[i];
if (*a == '-') {
if (*a++ == '-') { a++;
if (*a == '-' && !a[1]) /* "--" */ if (*a == '-' && !a[1]) /* "--" */
break; break;
if (isdigit(*a)) if (isdigit(*a))
@ -62,8 +76,7 @@ int fold_main(int argc UNUSED_PARAM, char **argv)
} }
getopt32(argv, "bsw:", &w_opt); getopt32(argv, "bsw:", &w_opt);
if (option_mask32 & FLAG_WIDTH) width = xatou_range(w_opt, 1, 10000);
width = xatoul_range(w_opt, 1, 10000);
argv += optind; argv += optind;
if (!*argv) if (!*argv)
@ -72,49 +85,56 @@ int fold_main(int argc UNUSED_PARAM, char **argv)
do { do {
FILE *istream = fopen_or_warn_stdin(*argv); FILE *istream = fopen_or_warn_stdin(*argv);
int c; int c;
int column = 0; /* Screen column where next char will go. */ unsigned column = 0; /* Screen column where next char will go */
int offset_out = 0; /* Index in 'line_out' for next char. */ unsigned offset_out = 0; /* Index in 'line_out' for next char */
if (istream == NULL) { if (istream == NULL) {
errs |= EXIT_FAILURE; exitcode = EXIT_FAILURE;
continue; continue;
} }
while ((c = getc(istream)) != EOF) { while ((c = getc(istream)) != EOF) {
if (offset_out + 1 >= allocated_out) { /* We grow line_out in chunks of 0x1000 bytes */
allocated_out += 1024; if ((offset_out & 0xfff) == 0) {
line_out = xrealloc(line_out, allocated_out); line_out = xrealloc(line_out, offset_out + 0x1000);
} }
rescan:
line_out[offset_out] = c;
if (c == '\n') { if (c == '\n') {
line_out[offset_out++] = c; write2stdout(line_out, offset_out + 1);
fwrite(line_out, sizeof(char), (size_t) offset_out, stdout);
column = offset_out = 0; column = offset_out = 0;
continue; continue;
} }
rescan:
column = adjust_column(column, c); column = adjust_column(column, c);
if (column <= width || offset_out == 0) {
/* offset_out == 0 case happens
* with small width (say, 1) and tabs.
* The very first tab already goes to column 8,
* but we must not wrap it */
offset_out++;
continue;
}
if (column > width) {
/* This character would make the line too long. /* This character would make the line too long.
Print the line plus a newline, and make this character * Print the line plus a newline, and make this character
start the next line. */ * start the next line */
if (option_mask32 & FLAG_BREAK_SPACES) { if (option_mask32 & FLAG_BREAK_SPACES) {
/* Look for the last blank. */ unsigned i;
int logical_end; unsigned logical_end;
for (logical_end = offset_out - 1; logical_end >= 0; logical_end--) { /* Look for the last blank. */
if (isblank(line_out[logical_end])) { for (logical_end = offset_out - 1; (int)logical_end >= 0; logical_end--) {
break; if (!isblank(line_out[logical_end]))
} continue;
}
if (logical_end >= 0) { /* Found a space or tab.
/* Found a blank. Don't output the part after it. */ * Output up to and including it, and start a new line */
logical_end++; logical_end++;
fwrite(line_out, sizeof(char), (size_t) logical_end, stdout); /*line_out[logical_end] = '\n'; - NO! this nukes one buffered character */
bb_putchar('\n'); write2stdout(line_out, logical_end);
putchar('\n');
/* Move the remainder to the beginning of the next line. /* Move the remainder to the beginning of the next line.
The areas being copied here might overlap. */ * The areas being copied here might overlap. */
memmove(line_out, line_out + logical_end, offset_out - logical_end); memmove(line_out, line_out + logical_end, offset_out - logical_end);
offset_out -= logical_end; offset_out -= logical_end;
for (column = i = 0; i < offset_out; i++) { for (column = i = 0; i < offset_out; i++) {
@ -122,29 +142,24 @@ int fold_main(int argc UNUSED_PARAM, char **argv)
} }
goto rescan; goto rescan;
} }
/* No blank found, wrap will split the overlong word */
} }
if (offset_out == 0) { /* Output what we accumulated up to now, and start a new line */
line_out[offset_out++] = c; line_out[offset_out] = '\n';
continue; write2stdout(line_out, offset_out + 1);
}
line_out[offset_out++] = '\n';
fwrite(line_out, sizeof(char), (size_t) offset_out, stdout);
column = offset_out = 0; column = offset_out = 0;
goto rescan; goto rescan;
} } /* while (not EOF) */
line_out[offset_out++] = c;
}
if (offset_out) { if (offset_out) {
fwrite(line_out, sizeof(char), (size_t) offset_out, stdout); write2stdout(line_out, offset_out);
} }
if (fclose_if_not_stdin(istream)) { if (fclose_if_not_stdin(istream)) {
bb_simple_perror_msg(*argv); /* Avoid multibyte problems. */ bb_simple_perror_msg(*argv);
errs |= EXIT_FAILURE; exitcode = EXIT_FAILURE;
} }
} while (*++argv); } while (*++argv);
fflush_stdout_and_exit(errs); fflush_stdout_and_exit(exitcode);
} }

View File

@ -945,7 +945,7 @@ int ls_main(int argc UNUSED_PARAM, char **argv)
INIT_G(); INIT_G();
check_unicode_in_env(); init_unicode();
all_fmt = LIST_SHORT | all_fmt = LIST_SHORT |
(ENABLE_FEATURE_LS_SORTFILES * (SORT_NAME | SORT_FORWARD)); (ENABLE_FEATURE_LS_SORTFILES * (SORT_NAME | SORT_FORWARD));

View File

@ -5,10 +5,17 @@
#ifndef UNICODE_H #ifndef UNICODE_H
#define UNICODE_H 1 #define UNICODE_H 1
enum {
UNICODE_UNKNOWN = 0,
UNICODE_OFF = 1,
UNICODE_ON = 2,
};
#if !ENABLE_FEATURE_ASSUME_UNICODE #if !ENABLE_FEATURE_ASSUME_UNICODE
# define bb_mbstrlen(string) strlen(string) # define bb_mbstrlen(string) strlen(string)
# define check_unicode_in_env() ((void)0) # define unicode_status UNICODE_OFF
# define init_unicode() ((void)0)
#else #else
@ -18,16 +25,19 @@ size_t bb_mbstrlen(const char *string) FAST_FUNC;
# include <wchar.h> # include <wchar.h>
# include <wctype.h> # include <wctype.h>
# define check_unicode_in_env() ((void)0) extern uint8_t unicode_status;
void init_unicode(void) FAST_FUNC;
# else # else
/* Crude "locale support" which knows only C and Unicode locales */ /* Crude "locale support" which knows only C and Unicode locales */
# if !ENABLE_FEATURE_CHECK_UNICODE_IN_ENV # if !ENABLE_FEATURE_CHECK_UNICODE_IN_ENV
# define check_unicode_in_env() ((void)0) # define unicode_status UNICODE_ON
# define init_unicode() ((void)0)
# else # else
void check_unicode_in_env(void) FAST_FUNC; extern uint8_t unicode_status;
void init_unicode(void) FAST_FUNC;
# endif # endif
# undef MB_CUR_MAX # undef MB_CUR_MAX

View File

@ -1763,7 +1763,7 @@ int FAST_FUNC read_line_input(const char *prompt, char *command, int maxsize, li
return len; return len;
} }
check_unicode_in_env(); init_unicode();
// FIXME: audit & improve this // FIXME: audit & improve this
if (maxsize > MAX_LINELEN) if (maxsize > MAX_LINELEN)

View File

@ -78,7 +78,7 @@ void FAST_FUNC bb_progress_update(bb_progress_t *p,
} }
#if ENABLE_FEATURE_ASSUME_UNICODE #if ENABLE_FEATURE_ASSUME_UNICODE
check_unicode_in_env(); init_unicode();
/* libbb candidate? */ /* libbb candidate? */
{ {
wchar_t wbuf21[21]; wchar_t wbuf21[21];

View File

@ -7,7 +7,9 @@
* Licensed under GPL version 2, see file LICENSE in this tarball for details. * Licensed under GPL version 2, see file LICENSE in this tarball for details.
*/ */
#include "libbb.h" #include "libbb.h"
# include "unicode.h" #include "unicode.h"
uint8_t unicode_status;
size_t FAST_FUNC bb_mbstrlen(const char *string) size_t FAST_FUNC bb_mbstrlen(const char *string)
{ {
@ -17,32 +19,38 @@ size_t FAST_FUNC bb_mbstrlen(const char *string)
return width; return width;
} }
#if !ENABLE_LOCALE_SUPPORT #if ENABLE_LOCALE_SUPPORT
/* Unicode support using libc */
void FAST_FUNC init_unicode(void)
{
/* In unicode, this is a one character string */
static const char unicode_0x394[] = { 0xce, 0x94, 0 };
if (unicode_status != UNICODE_UNKNOWN)
return;
unicode_status = bb_mbstrlen(unicode_0x394) == 1 ? UNICODE_ON : UNICODE_OFF;
}
#else
/* Crude "locale support" which knows only C and Unicode locales */ /* Crude "locale support" which knows only C and Unicode locales */
/* unicode_is_enabled: # if ENABLE_FEATURE_CHECK_UNICODE_IN_ENV
* 0: not known yet, void FAST_FUNC init_unicode(void)
* 1: not unicode (IOW: assuming one char == one byte)
* 2: unicode
*/
# if !ENABLE_FEATURE_CHECK_UNICODE_IN_ENV
# define unicode_is_enabled 2
# else
static smallint unicode_is_enabled;
void FAST_FUNC check_unicode_in_env(void)
{ {
char *lang; char *lang;
if (unicode_is_enabled) if (unicode_status != UNICODE_UNKNOWN)
return; return;
unicode_is_enabled = 1;
unicode_status = UNICODE_OFF;
lang = getenv("LANG"); lang = getenv("LANG");
if (!lang || !(strstr(lang, ".utf") || strstr(lang, ".UTF"))) if (!lang || !(strstr(lang, ".utf") || strstr(lang, ".UTF")))
return; return;
unicode_status = UNICODE_ON;
unicode_is_enabled = 2;
} }
# endif # endif
@ -85,7 +93,7 @@ static size_t wcrtomb_internal(char *s, wchar_t wc)
size_t FAST_FUNC wcrtomb(char *s, wchar_t wc, mbstate_t *ps UNUSED_PARAM) size_t FAST_FUNC wcrtomb(char *s, wchar_t wc, mbstate_t *ps UNUSED_PARAM)
{ {
if (unicode_is_enabled != 2) { if (unicode_status != UNICODE_ON) {
*s = wc; *s = wc;
return 1; return 1;
} }
@ -97,7 +105,7 @@ size_t FAST_FUNC wcstombs(char *dest, const wchar_t *src, size_t n)
{ {
size_t org_n = n; size_t org_n = n;
if (unicode_is_enabled != 2) { if (unicode_status != UNICODE_ON) {
while (n) { while (n) {
wchar_t c = *src++; wchar_t c = *src++;
*dest++ = c; *dest++ = c;
@ -137,7 +145,7 @@ size_t FAST_FUNC mbstowcs(wchar_t *dest, const char *src, size_t n)
{ {
size_t org_n = n; size_t org_n = n;
if (unicode_is_enabled != 2) { if (unicode_status != UNICODE_ON) {
while (n) { while (n) {
unsigned char c = *src++; unsigned char c = *src++;

View File

@ -49,7 +49,7 @@ int lsmod_main(int argc UNUSED_PARAM, char **argv UNUSED_PARAM)
# if ENABLE_FEATURE_ASSUME_UNICODE # if ENABLE_FEATURE_ASSUME_UNICODE
size_t name_len; size_t name_len;
# endif # endif
check_unicode_in_env(); init_unicode();
printf("%-24sSize Used by", "Module"); printf("%-24sSize Used by", "Module");
check_tainted(); check_tainted();

View File

@ -43,7 +43,7 @@ int dumpleases_main(int argc UNUSED_PARAM, char **argv)
applet_long_options = dumpleases_longopts; applet_long_options = dumpleases_longopts;
#endif #endif
check_unicode_in_env(); init_unicode();
opt_complementary = "=0:a--r:r--a"; opt_complementary = "=0:a--r:r--a";
opt = getopt32(argv, "arf:", &file); opt = getopt32(argv, "arf:", &file);

View File

@ -11,4 +11,49 @@ testing "fold -s" "fold -w 7 -s" \
"" \ "" \
"123456\tasdf" \ "123456\tasdf" \
testing "fold -w1" "fold -w1" \
"q\nq\n \nw\n \ne\ne\ne\n \nr\n \nt\nt\nt\nt\n \ny" \
"" \
"qq w eee r tttt y" \
testing "fold with NULs" "fold -sw22" \
"\
The NUL is here:>\0< \n\
and another one is \n\
here:>\0< - they must \n\
be preserved
" \
"" \
"The NUL is here:>\0< and another one \
is here:>\0< - they must be preserved
" \
# The text was taken from English and Ukrainian wikipedia pages
testing "fold -sw66 with unicode input" "fold -sw66" \
"\
The Andromeda Galaxy (pronounced /ænˈdrɒmədə/, also known as \n\
Messier 31, M31, or NGC224; often referred to as the Great \n\
Andromeda Nebula in older texts) is a spiral galaxy approximately \n\
2,500,000 light-years (1.58×10^11 AU) away in the constellation \n\
Andromeda. It is the nearest spiral galaxy to our own, the Milky \n\
Way.\n\
Галактика або Туманність Андромеди (також відома як M31 за \n\
каталогом Мессьє та NGC224 за Новим загальним каталогом) — \n\
спіральна галактика, що знаходиться на відстані приблизно у 2,5 \n\
мільйони світлових років від нашої планети у сузір'ї Андромеди. \n\
На початку ХХІ ст. в центрі галактики виявлено чорну дірку." \
"" \
"\
The Andromeda Galaxy (pronounced /ænˈdrɒmədə/, also known as \
Messier 31, M31, or NGC224; often referred to as the Great \
Andromeda Nebula in older texts) is a spiral galaxy approximately \
2,500,000 light-years (1.58×10^11 AU) away in the constellation \
Andromeda. It is the nearest spiral galaxy to our own, the Milky \
Way.
Галактика або Туманність Андромеди (також відома як M31 за \
каталогом Мессьє та NGC224 за Новим загальним каталогом) — \
спіральна галактика, що знаходиться на відстані приблизно у 2,5 \
мільйони світлових років від нашої планети у сузір'ї Андромеди. \
На початку ХХІ ст. в центрі галактики виявлено чорну дірку." \
exit $FAILCOUNT exit $FAILCOUNT