blob: f829203b717c07f25dd1ae227202b614e72c4307 [file] [log] [blame]
// std::messages implementation details, GNU version -*- C++ -*-
// Copyright (C) 2001-2021 Free Software Foundation, Inc.
//
// This file is part of the GNU ISO C++ Library. This library is free
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// any later version.
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// Under Section 7 of GPL version 3, you are granted additional
// permissions described in the GCC Runtime Library Exception, version
// 3.1, as published by the Free Software Foundation.
// You should have received a copy of the GNU General Public License and
// a copy of the GCC Runtime Library Exception along with this program;
// see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
// <http://www.gnu.org/licenses/>.
//
// ISO C++ 14882: 22.2.7.1.2 messages virtual functions
//
// Written by Benjamin Kosnik <bkoz@redhat.com>
#include <locale>
#include <bits/c++locale_internal.h>
#include <cstdlib> // std::free
#include <string.h> // ::strdup
namespace
{
using namespace std;
const char*
get_glibc_msg(__c_locale __locale_messages __attribute__((unused)),
const char* __name_messages __attribute__((unused)),
const char* __domainname,
const char* __dfault)
{
#if __GLIBC__ > 2 || (__GLIBC__ == 2 && __GLIBC_MINOR__ > 2)
std::__c_locale __old = __uselocale(__locale_messages);
const char* __msg = dgettext(__domainname, __dfault);
__uselocale(__old);
return __msg;
#else
if (char* __sav = strdup(setlocale(LC_ALL, 0)))
{
setlocale(LC_ALL, __name_messages);
const char* __msg = dgettext(__domainname, __dfault);
setlocale(LC_ALL, __sav);
free(__sav);
return __msg;
}
return __dfault;
#endif
}
}
namespace std _GLIBCXX_VISIBILITY(default)
{
_GLIBCXX_BEGIN_NAMESPACE_VERSION
// Specializations.
template<>
typename messages<char>::catalog
messages<char>::do_open(const basic_string<char>& __s,
const locale& __l) const
{
typedef codecvt<char, char, mbstate_t> __codecvt_t;
const __codecvt_t& __codecvt = use_facet<__codecvt_t>(__l);
bind_textdomain_codeset(__s.c_str(),
__nl_langinfo_l(CODESET, __codecvt._M_c_locale_codecvt));
return get_catalogs()._M_add(__s.c_str(), __l);
}
template<>
void
messages<char>::do_close(catalog __c) const
{ get_catalogs()._M_erase(__c); }
template<>
string
messages<char>::do_get(catalog __c, int, int,
const string& __dfault) const
{
if (__c < 0 || __dfault.empty())
return __dfault;
const Catalog_info* __cat_info = get_catalogs()._M_get(__c);
if (!__cat_info)
return __dfault;
return get_glibc_msg(_M_c_locale_messages, _M_name_messages,
__cat_info->_M_domain,
__dfault.c_str());
}
#ifdef _GLIBCXX_USE_WCHAR_T
template<>
typename messages<wchar_t>::catalog
messages<wchar_t>::do_open(const basic_string<char>& __s,
const locale& __l) const
{
typedef codecvt<wchar_t, char, mbstate_t> __codecvt_t;
const __codecvt_t& __codecvt = use_facet<__codecvt_t>(__l);
bind_textdomain_codeset(__s.c_str(),
__nl_langinfo_l(CODESET, __codecvt._M_c_locale_codecvt));
return get_catalogs()._M_add(__s.c_str(), __l);
}
template<>
void
messages<wchar_t>::do_close(catalog __c) const
{ get_catalogs()._M_erase(__c); }
template<>
wstring
messages<wchar_t>::do_get(catalog __c, int, int,
const wstring& __wdfault) const
{
if (__c < 0 || __wdfault.empty())
return __wdfault;
const Catalog_info* __cat_info = get_catalogs()._M_get(__c);
if (!__cat_info)
return __wdfault;
typedef codecvt<wchar_t, char, mbstate_t> __codecvt_t;
const __codecvt_t& __conv =
use_facet<__codecvt_t>(__cat_info->_M_locale);
const char* __translation;
mbstate_t __state;
__builtin_memset(&__state, 0, sizeof(mbstate_t));
{
const wchar_t* __wdfault_next;
size_t __mb_size = __wdfault.size() * __conv.max_length();
char* __dfault =
static_cast<char*>(__builtin_alloca(sizeof(char) * (__mb_size + 1)));
char* __dfault_next;
__conv.out(__state,
__wdfault.data(), __wdfault.data() + __wdfault.size(),
__wdfault_next,
__dfault, __dfault + __mb_size, __dfault_next);
// Make sure string passed to dgettext is \0 terminated.
*__dfault_next = '\0';
__translation = get_glibc_msg(_M_c_locale_messages, _M_name_messages,
__cat_info->_M_domain, __dfault);
// If we end up getting default value back we can simply return original
// default value.
if (__translation == __dfault)
return __wdfault;
}
__builtin_memset(&__state, 0, sizeof(mbstate_t));
size_t __size = __builtin_strlen(__translation);
const char* __translation_next;
wchar_t* __wtranslation =
static_cast<wchar_t*>(__builtin_alloca(sizeof(wchar_t) * (__size + 1)));
wchar_t* __wtranslation_next;
__conv.in(__state, __translation, __translation + __size,
__translation_next,
__wtranslation, __wtranslation + __size,
__wtranslation_next);
return wstring(__wtranslation, __wtranslation_next);
}
#endif
_GLIBCXX_END_NAMESPACE_VERSION
} // namespace