213 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			213 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
| /*
 | |
|  * Copyright (c) 2010 Ævar Arnfjörð Bjarmason
 | |
|  */
 | |
| 
 | |
| #include "cache.h"
 | |
| #include "exec-cmd.h"
 | |
| #include "gettext.h"
 | |
| #include "strbuf.h"
 | |
| #include "utf8.h"
 | |
| #include "config.h"
 | |
| 
 | |
| #ifndef NO_GETTEXT
 | |
| #	include <locale.h>
 | |
| #	include <libintl.h>
 | |
| #	ifdef HAVE_LIBCHARSET_H
 | |
| #		include <libcharset.h>
 | |
| #	else
 | |
| #		include <langinfo.h>
 | |
| #		define locale_charset() nl_langinfo(CODESET)
 | |
| #	endif
 | |
| #endif
 | |
| 
 | |
| static const char *charset;
 | |
| 
 | |
| /*
 | |
|  * Guess the user's preferred languages from the value in LANGUAGE environment
 | |
|  * variable and LC_MESSAGES locale category if NO_GETTEXT is not defined.
 | |
|  *
 | |
|  * The result can be a colon-separated list like "ko:ja:en".
 | |
|  */
 | |
| const char *get_preferred_languages(void)
 | |
| {
 | |
| 	const char *retval;
 | |
| 
 | |
| 	retval = getenv("LANGUAGE");
 | |
| 	if (retval && *retval)
 | |
| 		return retval;
 | |
| 
 | |
| #ifndef NO_GETTEXT
 | |
| 	retval = setlocale(LC_MESSAGES, NULL);
 | |
| 	if (retval && *retval &&
 | |
| 		strcmp(retval, "C") &&
 | |
| 		strcmp(retval, "POSIX"))
 | |
| 		return retval;
 | |
| #endif
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| int use_gettext_poison(void)
 | |
| {
 | |
| 	static int poison_requested = -1;
 | |
| 	if (poison_requested == -1) {
 | |
| 		const char *v = getenv("GIT_TEST_GETTEXT_POISON");
 | |
| 		poison_requested = v && strlen(v) ? 1 : 0;
 | |
| 	}
 | |
| 	return poison_requested;
 | |
| }
 | |
| 
 | |
| #ifndef NO_GETTEXT
 | |
| static int test_vsnprintf(const char *fmt, ...)
 | |
| {
 | |
| 	char buf[26];
 | |
| 	int ret;
 | |
| 	va_list ap;
 | |
| 	va_start(ap, fmt);
 | |
| 	ret = vsnprintf(buf, sizeof(buf), fmt, ap);
 | |
| 	va_end(ap);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static void init_gettext_charset(const char *domain)
 | |
| {
 | |
| 	/*
 | |
| 	   This trick arranges for messages to be emitted in the user's
 | |
| 	   requested encoding, but avoids setting LC_CTYPE from the
 | |
| 	   environment for the whole program.
 | |
| 
 | |
| 	   This primarily done to avoid a bug in vsnprintf in the GNU C
 | |
| 	   Library [1]. which triggered a "your vsnprintf is broken" error
 | |
| 	   on Git's own repository when inspecting v0.99.6~1 under a UTF-8
 | |
| 	   locale.
 | |
| 
 | |
| 	   That commit contains a ISO-8859-1 encoded author name, which
 | |
| 	   the locale aware vsnprintf(3) won't interpolate in the format
 | |
| 	   argument, due to mismatch between the data encoding and the
 | |
| 	   locale.
 | |
| 
 | |
| 	   Even if it wasn't for that bug we wouldn't want to use LC_CTYPE at
 | |
| 	   this point, because it'd require auditing all the code that uses C
 | |
| 	   functions whose semantics are modified by LC_CTYPE.
 | |
| 
 | |
| 	   But only setting LC_MESSAGES as we do creates a problem, since
 | |
| 	   we declare the encoding of our PO files[2] the gettext
 | |
| 	   implementation will try to recode it to the user's locale, but
 | |
| 	   without LC_CTYPE it'll emit something like this on 'git init'
 | |
| 	   under the Icelandic locale:
 | |
| 
 | |
| 	       Bj? til t?ma Git lind ? /hlagh/.git/
 | |
| 
 | |
| 	   Gettext knows about the encoding of our PO file, but we haven't
 | |
| 	   told it about the user's encoding, so all the non-US-ASCII
 | |
| 	   characters get encoded to question marks.
 | |
| 
 | |
| 	   But we're in luck! We can set LC_CTYPE from the environment
 | |
| 	   only while we call nl_langinfo and
 | |
| 	   bind_textdomain_codeset. That suffices to tell gettext what
 | |
| 	   encoding it should emit in, so it'll now say:
 | |
| 
 | |
| 	       Bjó til tóma Git lind í /hlagh/.git/
 | |
| 
 | |
| 	   And the equivalent ISO-8859-1 string will be emitted under a
 | |
| 	   ISO-8859-1 locale.
 | |
| 
 | |
| 	   With this change way we get the advantages of setting LC_CTYPE
 | |
| 	   (talk to the user in his language/encoding), without the major
 | |
| 	   drawbacks (changed semantics for C functions we rely on).
 | |
| 
 | |
| 	   However foreign functions using other message catalogs that
 | |
| 	   aren't using our neat trick will still have a problem, e.g. if
 | |
| 	   we have to call perror(3):
 | |
| 
 | |
| 	   #include <stdio.h>
 | |
| 	   #include <locale.h>
 | |
| 	   #include <errno.h>
 | |
| 
 | |
| 	   int main(void)
 | |
| 	   {
 | |
| 		   setlocale(LC_MESSAGES, "");
 | |
| 		   setlocale(LC_CTYPE, "C");
 | |
| 		   errno = ENODEV;
 | |
| 		   perror("test");
 | |
| 		   return 0;
 | |
| 	   }
 | |
| 
 | |
| 	   Running that will give you a message with question marks:
 | |
| 
 | |
| 	   $ LANGUAGE= LANG=de_DE.utf8 ./test
 | |
| 	   test: Kein passendes Ger?t gefunden
 | |
| 
 | |
| 	   The vsnprintf bug has been fixed since glibc 2.17.
 | |
| 
 | |
| 	   Then we could simply set LC_CTYPE from the environment, which would
 | |
| 	   make things like the external perror(3) messages work.
 | |
| 
 | |
| 	   See t/t0203-gettext-setlocale-sanity.sh's "gettext.c" tests for
 | |
| 	   regression tests.
 | |
| 
 | |
| 	   1. http://sourceware.org/bugzilla/show_bug.cgi?id=6530
 | |
| 	   2. E.g. "Content-Type: text/plain; charset=UTF-8\n" in po/is.po
 | |
| 	*/
 | |
| 	setlocale(LC_CTYPE, "");
 | |
| 	charset = locale_charset();
 | |
| 	bind_textdomain_codeset(domain, charset);
 | |
| 	/* the string is taken from v0.99.6~1 */
 | |
| 	if (test_vsnprintf("%.*s", 13, "David_K\345gedal") < 0)
 | |
| 		setlocale(LC_CTYPE, "C");
 | |
| }
 | |
| 
 | |
| void git_setup_gettext(void)
 | |
| {
 | |
| 	const char *podir = getenv(GIT_TEXT_DOMAIN_DIR_ENVIRONMENT);
 | |
| 	char *p = NULL;
 | |
| 
 | |
| 	if (!podir)
 | |
| 		podir = p = system_path(GIT_LOCALE_PATH);
 | |
| 
 | |
| 	use_gettext_poison(); /* getenv() reentrancy paranoia */
 | |
| 
 | |
| 	if (!is_directory(podir)) {
 | |
| 		free(p);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	bindtextdomain("git", podir);
 | |
| 	setlocale(LC_MESSAGES, "");
 | |
| 	setlocale(LC_TIME, "");
 | |
| 	init_gettext_charset("git");
 | |
| 	textdomain("git");
 | |
| 
 | |
| 	free(p);
 | |
| }
 | |
| 
 | |
| /* return the number of columns of string 's' in current locale */
 | |
| int gettext_width(const char *s)
 | |
| {
 | |
| 	static int is_utf8 = -1;
 | |
| 	if (is_utf8 == -1)
 | |
| 		is_utf8 = is_utf8_locale();
 | |
| 
 | |
| 	return is_utf8 ? utf8_strwidth(s) : strlen(s);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| int is_utf8_locale(void)
 | |
| {
 | |
| #ifdef NO_GETTEXT
 | |
| 	if (!charset) {
 | |
| 		const char *env = getenv("LC_ALL");
 | |
| 		if (!env || !*env)
 | |
| 			env = getenv("LC_CTYPE");
 | |
| 		if (!env || !*env)
 | |
| 			env = getenv("LANG");
 | |
| 		if (!env)
 | |
| 			env = "";
 | |
| 		if (strchr(env, '.'))
 | |
| 			env = strchr(env, '.') + 1;
 | |
| 		charset = xstrdup(env);
 | |
| 	}
 | |
| #endif
 | |
| 	return is_encoding_utf8(charset);
 | |
| }
 |