X-Git-Url: https://pintos-os.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Flibpspp%2Fi18n.c;h=59665bee63b6a44ba6910eea288d3ca8b1ae7b12;hb=6eeee3ec27d9e75cba2f390fc7dceef7ff38c2f3;hp=6cb721a993df60ecd9fc3d4707a9d7b2d19dbeb4;hpb=c555358da9c0b1747b9b236ba28ec55efdc89852;p=pspp diff --git a/src/libpspp/i18n.c b/src/libpspp/i18n.c index 6cb721a993..59665bee63 100644 --- a/src/libpspp/i18n.c +++ b/src/libpspp/i18n.c @@ -32,6 +32,7 @@ #include "libpspp/hmapx.h" #include "libpspp/hash-functions.h" #include "libpspp/pool.h" +#include "libpspp/str.h" #include "libpspp/version.h" #include "gl/localcharset.h" @@ -99,14 +100,16 @@ recode_string (const char *to, const char *from, /* Uses CONV to convert the INBYTES starting at IP into the OUTBYTES starting at OP, and appends a null terminator to the output. - Returns true if successful, false if the output buffer is too small. */ -static bool + Returns the output length if successful, -1 if the output buffer is too + small. */ +static ssize_t try_recode (iconv_t conv, const char *ip, size_t inbytes, - char *op, size_t outbytes) + char *op_, size_t outbytes) { /* FIXME: Need to ensure that this char is valid in the target encoding */ const char fallbackchar = '?'; + char *op = op_; /* Put the converter into the initial shift state, in case there was any state information left over from its last usage. */ @@ -118,14 +121,14 @@ try_recode (iconv_t conv, { case EINVAL: if (outbytes < 2) - return false; + return -1; *op++ = fallbackchar; - *op++ = '\0'; - return true; + *op = '\0'; + return op - op_; case EILSEQ: if (outbytes == 0) - return false; + return -1; *op++ = fallbackchar; outbytes--; ip++; @@ -133,7 +136,7 @@ try_recode (iconv_t conv, break; case E2BIG: - return false; + return -1; default: /* should never happen */ @@ -143,10 +146,10 @@ try_recode (iconv_t conv, } if (outbytes == 0) - return false; + return -1; *op = '\0'; - return true; + return op - op_; } /* Converts the string TEXT, which should be encoded in FROM-encoding, to a @@ -165,14 +168,34 @@ char * recode_string_pool (const char *to, const char *from, const char *text, int length, struct pool *pool) { - size_t outbufferlength; - iconv_t conv ; + struct substring out; if ( text == NULL ) return NULL; if ( length == -1 ) - length = strlen(text); + length = strlen (text); + + out = recode_substring_pool (to, from, ss_buffer (text, length), pool); + return out.string; +} + +/* Converts the string TEXT, which should be encoded in FROM-encoding, to a + dynamically allocated string in TO-encoding. Any characters which cannot be + converted will be represented by '?'. + + The returned string will be null-terminated and allocated on POOL. + + This function's behaviour differs from that of g_convert_with_fallback + provided by GLib. The GLib function will fail (returns NULL) if any part of + the input string is not valid in the declared input encoding. This function + however perseveres even in the presence of badly encoded input. */ +struct substring +recode_substring_pool (const char *to, const char *from, + struct substring text, struct pool *pool) +{ + size_t outbufferlength; + iconv_t conv ; if (to == NULL) to = default_encoding; @@ -183,14 +206,20 @@ recode_string_pool (const char *to, const char *from, conv = create_iconv (to, from); if ( (iconv_t) -1 == conv ) - return xstrdup (text); + { + struct substring out; + ss_alloc_substring_pool (&out, text, pool); + return out; + } for ( outbufferlength = 1 ; outbufferlength != 0; outbufferlength <<= 1 ) - if ( outbufferlength > length) + if ( outbufferlength > text.length) { char *output = pool_malloc (pool, outbufferlength); - if (try_recode (conv, text, length, output, outbufferlength)) - return output; + ssize_t output_len = try_recode (conv, text.string, text.length, + output, outbufferlength); + if (output_len >= 0) + return ss_buffer (output, output_len); pool_free (pool, output); } @@ -200,17 +229,13 @@ recode_string_pool (const char *to, const char *from, void i18n_init (void) { -#if ENABLE_NLS setlocale (LC_CTYPE, ""); -#ifdef LC_MESSAGES setlocale (LC_MESSAGES, ""); -#endif #if HAVE_LC_PAPER setlocale (LC_PAPER, ""); #endif bindtextdomain (PACKAGE, relocate(locale_dir)); textdomain (PACKAGE); -#endif /* ENABLE_NLS */ assert (default_encoding == NULL); default_encoding = xstrdup (locale_charset ()); @@ -300,7 +325,7 @@ i18n_done (void) bool valid_encoding (const char *enc) { - iconv_t conv = iconv_open ("UTF8", enc); + iconv_t conv = iconv_open (UTF8, enc); if ( conv == (iconv_t) -1) return false;