unicode.c
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:9k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * unicode.c: Unicode <-> locale functions
  3.  *****************************************************************************
  4.  * Copyright (C) 2005-2006 the VideoLAN team
  5.  * Copyright © 2005-2008 Rémi Denis-Courmont
  6.  *
  7.  * Authors: Rémi Denis-Courmont <rem # videolan.org>
  8.  *
  9.  * This program is free software; you can redistribute it and/or modify
  10.  * it under the terms of the GNU General Public License as published by
  11.  * the Free Software Foundation; either version 2 of the License, or
  12.  * (at your option) any later version.
  13.  *
  14.  * This program is distributed in the hope that it will be useful,
  15.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  17.  * GNU General Public License for more details.
  18.  *
  19.  * You should have received a copy of the GNU General Public License
  20.  * along with this program; if not, write to the Free Software
  21.  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  22.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #ifdef HAVE_CONFIG_H
  27. # include "config.h"
  28. #endif
  29. #include <vlc_common.h>
  30. #include <vlc_charset.h>
  31. #include <assert.h>
  32. #include <stdio.h>
  33. #include <stdarg.h>
  34. #include <stdlib.h>
  35. #include <sys/types.h>
  36. #ifdef UNDER_CE
  37. #  include <tchar.h>
  38. #endif
  39. #ifdef __APPLE__
  40. /* Define this if the OS always use UTF-8 internally */
  41. # define ASSUME_UTF8 1
  42. #endif
  43. #if defined (ASSUME_UTF8)
  44. /* Cool */
  45. #elif defined (WIN32) || defined (UNDER_CE)
  46. # define USE_MB2MB 1
  47. #elif defined (HAVE_ICONV)
  48. # define USE_ICONV 1
  49. #else
  50. # error No UTF8 charset conversion implemented on this platform!
  51. #endif
  52. #if defined (USE_ICONV)
  53. # include <langinfo.h>
  54. static char charset[sizeof ("CSISO11SWEDISHFORNAMES")] = "";
  55. static void find_charset_once (void)
  56. {
  57.     strlcpy (charset, nl_langinfo (CODESET), sizeof (charset));
  58.     if (!strcasecmp (charset, "ASCII")
  59.      || !strcasecmp (charset, "ANSI_X3.4-1968"))
  60.         strcpy (charset, "UTF-8"); /* superset... */
  61. }
  62. static int find_charset (void)
  63. {
  64.     static pthread_once_t once = PTHREAD_ONCE_INIT;
  65.     pthread_once (&once, find_charset_once);
  66.     return !strcasecmp (charset, "UTF-8");
  67. }
  68. #endif
  69. static char *locale_fast (const char *string, bool from)
  70. {
  71.     if( string == NULL )
  72.         return NULL;
  73. #if defined (USE_ICONV)
  74.     if (find_charset ())
  75.         return (char *)string;
  76.     vlc_iconv_t hd = vlc_iconv_open (from ? "UTF-8" : charset,
  77.                                      from ? charset : "UTF-8");
  78.     if (hd == (vlc_iconv_t)(-1))
  79.         return NULL; /* Uho! */
  80.     const char *iptr = string;
  81.     size_t inb = strlen (string);
  82.     size_t outb = inb * 6 + 1;
  83.     char output[outb], *optr = output;
  84.     while (vlc_iconv (hd, &iptr, &inb, &optr, &outb) == (size_t)(-1))
  85.     {
  86.         *optr++ = '?';
  87.         outb--;
  88.         iptr++;
  89.         inb--;
  90.         vlc_iconv (hd, NULL, NULL, NULL, NULL); /* reset */
  91.     }
  92.     *optr = '';
  93.     vlc_iconv_close (hd);
  94.     assert (inb == 0);
  95.     assert (*iptr == '');
  96.     assert (*optr == '');
  97.     assert (strlen (output) == (size_t)(optr - output));
  98.     return strdup (output);
  99. #elif defined (USE_MB2MB)
  100.     char *out;
  101.     int len;
  102.     len = 1 + MultiByteToWideChar (from ? CP_ACP : CP_UTF8,
  103.                                    0, string, -1, NULL, 0);
  104.     wchar_t *wide = malloc (len * sizeof (wchar_t));
  105.     if (wide == NULL)
  106.         return NULL;
  107.     MultiByteToWideChar (from ? CP_ACP : CP_UTF8, 0, string, -1, wide, len);
  108.     len = 1 + WideCharToMultiByte (from ? CP_UTF8 : CP_ACP, 0, wide, -1,
  109.                                    NULL, 0, NULL, NULL);
  110.     out = malloc (len);
  111.     if (out != NULL)
  112.         WideCharToMultiByte (from ? CP_UTF8 : CP_ACP, 0, wide, -1, out, len,
  113.                              NULL, NULL);
  114.     free (wide);
  115.     return out;
  116. #else
  117.     (void)from;
  118.     return (char *)string;
  119. #endif
  120. }
  121. static inline char *locale_dup (const char *string, bool from)
  122. {
  123.     assert( string );
  124. #if defined (USE_ICONV)
  125.     if (find_charset ())
  126.         return strdup (string);
  127.     return locale_fast (string, from);
  128. #elif defined (USE_MB2MB)
  129.     return locale_fast (string, from);
  130. #else
  131.     (void)from;
  132.     return strdup (string);
  133. #endif
  134. }
  135. /**
  136.  * Releases (if needed) a localized or uniformized string.
  137.  * @param str non-NULL return value from FromLocale() or ToLocale().
  138.  */
  139. void LocaleFree (const char *str)
  140. {
  141. #if defined (USE_ICONV)
  142.     if (!find_charset ())
  143.         free ((char *)str);
  144. #elif defined (USE_MB2MB)
  145.     free ((char *)str);
  146. #else
  147.     (void)str;
  148. #endif
  149. }
  150. /**
  151.  * Converts a string from the system locale character encoding to UTF-8.
  152.  *
  153.  * @param locale nul-terminated string to convert
  154.  *
  155.  * @return a nul-terminated UTF-8 string, or NULL in case of error.
  156.  * To avoid memory leak, you have to pass the result to LocaleFree()
  157.  * when it is no longer needed.
  158.  */
  159. char *FromLocale (const char *locale)
  160. {
  161.     return locale_fast (locale, true);
  162. }
  163. /**
  164.  * converts a string from the system locale character encoding to utf-8,
  165.  * the result is always allocated on the heap.
  166.  *
  167.  * @param locale nul-terminated string to convert
  168.  *
  169.  * @return a nul-terminated utf-8 string, or null in case of error.
  170.  * The result must be freed using free() - as with the strdup() function.
  171.  */
  172. char *FromLocaleDup (const char *locale)
  173. {
  174.     return locale_dup (locale, true);
  175. }
  176. /**
  177.  * ToLocale: converts an UTF-8 string to local system encoding.
  178.  *
  179.  * @param utf8 nul-terminated string to be converted
  180.  *
  181.  * @return a nul-terminated string, or NULL in case of error.
  182.  * To avoid memory leak, you have to pass the result to LocaleFree()
  183.  * when it is no longer needed.
  184.  */
  185. char *ToLocale (const char *utf8)
  186. {
  187.     return locale_fast (utf8, false);
  188. }
  189. /**
  190.  * converts a string from UTF-8 to the system locale character encoding,
  191.  * the result is always allocated on the heap.
  192.  *
  193.  * @param utf8 nul-terminated string to convert
  194.  *
  195.  * @return a nul-terminated string, or null in case of error.
  196.  * The result must be freed using free() - as with the strdup() function.
  197.  */
  198. char *ToLocaleDup (const char *utf8)
  199. {
  200.     return locale_dup (utf8, false);
  201. }
  202. /**
  203.  * Formats an UTF-8 string as vasprintf(), then print it to stdout, with
  204.  * appropriate conversion to local encoding.
  205.  */
  206. static int utf8_vasprintf( char **str, const char *fmt, va_list ap )
  207. {
  208.     char *utf8;
  209.     int res = vasprintf( &utf8, fmt, ap );
  210.     if( res == -1 )
  211.         return -1;
  212.     *str = ToLocaleDup( utf8 );
  213.     free( utf8 );
  214.     return res;
  215. }
  216. /**
  217.  * Formats an UTF-8 string as vfprintf(), then print it, with
  218.  * appropriate conversion to local encoding.
  219.  */
  220. int utf8_vfprintf( FILE *stream, const char *fmt, va_list ap )
  221. {
  222.     char *str;
  223.     int res = utf8_vasprintf( &str, fmt, ap );
  224.     if( res == -1 )
  225.         return -1;
  226.     fputs( str, stream );
  227.     free( str );
  228.     return res;
  229. }
  230. /**
  231.  * Formats an UTF-8 string as fprintf(), then print it, with
  232.  * appropriate conversion to local encoding.
  233.  */
  234. int utf8_fprintf( FILE *stream, const char *fmt, ... )
  235. {
  236.     va_list ap;
  237.     int res;
  238.     va_start( ap, fmt );
  239.     res = utf8_vfprintf( stream, fmt, ap );
  240.     va_end( ap );
  241.     return res;
  242. }
  243. static char *CheckUTF8( char *str, char rep )
  244. {
  245.     uint8_t *ptr = (uint8_t *)str;
  246.     assert (str != NULL);
  247.     for (;;)
  248.     {
  249.         uint8_t c = ptr[0];
  250.         int charlen = -1;
  251.         if (c == '')
  252.             break;
  253.         for (int i = 0; i < 7; i++)
  254.             if ((c >> (7 - i)) == ((0xff >> (7 - i)) ^ 1))
  255.             {
  256.                 charlen = i;
  257.                 break;
  258.             }
  259.         switch (charlen)
  260.         {
  261.             case 0: // 7-bit ASCII character -> OK
  262.                 ptr++;
  263.                 continue;
  264.             case -1: // 1111111x -> error
  265.             case 1: // continuation byte -> error
  266.                 goto error;
  267.         }
  268.         assert (charlen >= 2);
  269.         uint32_t cp = c & ~((0xff >> (7 - charlen)) << (7 - charlen));
  270.         for (int i = 1; i < charlen; i++)
  271.         {
  272.             assert (cp < (1 << 26));
  273.             c = ptr[i];
  274.             if ((c == '') // unexpected end of string
  275.              || ((c >> 6) != 2)) // not a continuation byte
  276.                 goto error;
  277.             cp = (cp << 6) | (ptr[i] & 0x3f);
  278.         }
  279.         if (cp < 128) // overlong (special case for ASCII)
  280.             goto error;
  281.         if (cp < (1u << (5 * charlen - 3))) // overlong
  282.             goto error;
  283.         ptr += charlen;
  284.         continue;
  285.     error:
  286.         if (rep == 0)
  287.             return NULL;
  288.         *ptr++ = rep;
  289.         str = NULL;
  290.     }
  291.     return str;
  292. }
  293. /**
  294.  * Replaces invalid/overlong UTF-8 sequences with question marks.
  295.  * Note that it is not possible to convert from Latin-1 to UTF-8 on the fly,
  296.  * so we don't try that, even though it would be less disruptive.
  297.  *
  298.  * @return str if it was valid UTF-8, NULL if not.
  299.  */
  300. char *EnsureUTF8( char *str )
  301. {
  302.     return CheckUTF8( str, '?' );
  303. }
  304. /**
  305.  * Checks whether a string is a valid UTF-8 byte sequence.
  306.  *
  307.  * @param str nul-terminated string to be checked
  308.  *
  309.  * @return str if it was valid UTF-8, NULL if not.
  310.  */
  311. const char *IsUTF8( const char *str )
  312. {
  313.     return CheckUTF8( (char *)str, 0 );
  314. }