d02c0bade7
uselocale now attempts to create a backend and a databrige. If the attempt fails due to a missing libroot-addon-icu, uselocale does nothing (to support applications calling uselocale during startup to enforce the C locale). Else, uselocale will fail with ENOMEM. LocaleBackend::CreateBackend() has been modified to return a status_t that indicates whether NULL is returned due to out of memory (B_NO_MEMORY) or due to being unable to load the ICU addon (B_MISSING_LIBRARY). Change-Id: I0f62ebde5890364c64e6694ec58d38de43ec6841 Reviewed-on: https://review.haiku-os.org/c/haiku/+/5505 Tested-by: Commit checker robot <no-reply+buildbot@haiku-os.org> Reviewed-by: waddlesplash <waddlesplash@gmail.com>
214 lines
5.2 KiB
C++
214 lines
5.2 KiB
C++
/*
|
|
* Copyright 2010-2011, Oliver Tappe, zooey@hirschkaefer.de.
|
|
* Distributed under the terms of the MIT License.
|
|
*/
|
|
#ifndef _LOCALE_BACKEND_H
|
|
#define _LOCALE_BACKEND_H
|
|
|
|
|
|
#include <SupportDefs.h>
|
|
|
|
#include <locale.h>
|
|
#include <time.h>
|
|
#include <wctype.h>
|
|
|
|
|
|
struct lc_time_t;
|
|
struct locale_data; // glibc
|
|
|
|
|
|
namespace BPrivate {
|
|
namespace Libroot {
|
|
|
|
|
|
struct LocaleCtypeDataBridge {
|
|
private:
|
|
const unsigned short* localClassInfoTable;
|
|
const int* localToLowerTable;
|
|
const int* localToUpperTable;
|
|
|
|
public:
|
|
const unsigned short** addrOfClassInfoTable;
|
|
const int** addrOfToLowerTable;
|
|
const int** addrOfToUpperTable;
|
|
|
|
const unsigned short* const posixClassInfo;
|
|
const int* const posixToLowerMap;
|
|
const int* const posixToUpperMap;
|
|
|
|
bool isGlobal;
|
|
|
|
LocaleCtypeDataBridge(bool isGlobal);
|
|
|
|
void setMbCurMax(unsigned short mbCurMax);
|
|
void ApplyToCurrentThread();
|
|
};
|
|
|
|
|
|
struct LocaleMessagesDataBridge {
|
|
const char** const posixLanginfo;
|
|
|
|
LocaleMessagesDataBridge();
|
|
};
|
|
|
|
|
|
struct LocaleMonetaryDataBridge {
|
|
const struct lconv* const posixLocaleConv;
|
|
|
|
LocaleMonetaryDataBridge();
|
|
};
|
|
|
|
|
|
struct LocaleNumericDataBridge {
|
|
private:
|
|
// struct used by glibc to store numeric locale data
|
|
struct GlibcNumericLocale {
|
|
const char* name;
|
|
const char* filedata;
|
|
off_t filesize;
|
|
int mmaped;
|
|
unsigned int usage_count;
|
|
int use_translit;
|
|
const char *options;
|
|
unsigned int nstrings;
|
|
union locale_data_value
|
|
{
|
|
const uint32_t* wstr;
|
|
const char* string;
|
|
unsigned int word;
|
|
}
|
|
values[6];
|
|
};
|
|
locale_data* originalGlibcLocale;
|
|
GlibcNumericLocale glibcNumericLocaleData;
|
|
|
|
public:
|
|
const struct lconv* const posixLocaleConv;
|
|
GlibcNumericLocale* glibcNumericLocale;
|
|
bool isGlobal;
|
|
|
|
LocaleNumericDataBridge(bool isGlobal);
|
|
~LocaleNumericDataBridge();
|
|
|
|
void ApplyToCurrentThread();
|
|
};
|
|
|
|
|
|
struct LocaleTimeDataBridge {
|
|
const struct lc_time_t* const posixLCTimeInfo;
|
|
|
|
LocaleTimeDataBridge();
|
|
};
|
|
|
|
|
|
struct TimeConversionDataBridge {
|
|
private:
|
|
int localDaylight;
|
|
long localTimezone;
|
|
char* localTZName[2];
|
|
char localTZName0[64];
|
|
char localTZName1[64];
|
|
|
|
public:
|
|
int* addrOfDaylight;
|
|
long* addrOfTimezone;
|
|
char** addrOfTZName;
|
|
bool isGlobal;
|
|
|
|
TimeConversionDataBridge(bool isGlobal);
|
|
};
|
|
|
|
|
|
struct LocaleDataBridge {
|
|
LocaleCtypeDataBridge ctypeDataBridge;
|
|
LocaleMessagesDataBridge messagesDataBridge;
|
|
LocaleMonetaryDataBridge monetaryDataBridge;
|
|
LocaleNumericDataBridge numericDataBridge;
|
|
LocaleTimeDataBridge timeDataBridge;
|
|
TimeConversionDataBridge timeConversionDataBridge;
|
|
const char** const posixLanginfo;
|
|
bool isGlobal;
|
|
|
|
LocaleDataBridge(bool isGlobal);
|
|
|
|
void ApplyToCurrentThread();
|
|
};
|
|
|
|
|
|
class LocaleBackend {
|
|
public:
|
|
LocaleBackend();
|
|
virtual ~LocaleBackend();
|
|
|
|
virtual const char* SetLocale(int category, const char* locale) = 0;
|
|
virtual const struct lconv* LocaleConv() = 0;
|
|
virtual const struct lc_time_t* LCTimeInfo() = 0;
|
|
|
|
virtual int IsWCType(wint_t wc, wctype_t charClass) = 0;
|
|
virtual status_t ToWCTrans(wint_t wc, wctrans_t transition,
|
|
wint_t& result) = 0;
|
|
|
|
virtual status_t MultibyteToWchar(wchar_t* wcOut, const char* mb,
|
|
size_t mbLength, mbstate_t* mbState,
|
|
size_t& lengthOut) = 0;
|
|
virtual status_t MultibyteStringToWchar(wchar_t* wcDest,
|
|
size_t wcDestLength, const char** mbSource,
|
|
size_t mbSourceLength, mbstate_t* mbState,
|
|
size_t& lengthOut) = 0;
|
|
virtual status_t WcharToMultibyte(char* mbOut, wchar_t wc,
|
|
mbstate_t* mbState, size_t& lengthOut) = 0;
|
|
virtual status_t WcharStringToMultibyte(char* mbDest,
|
|
size_t mbDestLength,
|
|
const wchar_t** wcSource,
|
|
size_t wcSourceLength, mbstate_t* mbState,
|
|
size_t& lengthOut) = 0;
|
|
|
|
virtual const char* GetLanginfo(int index) = 0;
|
|
|
|
virtual status_t Strcoll(const char* a, const char* b,
|
|
int& out) = 0;
|
|
virtual status_t Strxfrm(char* out, const char* in, size_t size,
|
|
size_t& outSize) = 0;
|
|
virtual status_t Wcscoll(const wchar_t* a, const wchar_t* b,
|
|
int& out) = 0;
|
|
virtual status_t Wcsxfrm(wchar_t* out, const wchar_t* in,
|
|
size_t outSize, size_t& requiredSize) = 0;
|
|
|
|
virtual status_t TZSet(const char* timeZoneID,
|
|
const char* tz) = 0;
|
|
virtual status_t Localtime(const time_t* inTime,
|
|
struct tm* tmOut) = 0;
|
|
virtual status_t Gmtime(const time_t* inTime,
|
|
struct tm* tmOut) = 0;
|
|
virtual status_t Mktime(struct tm* inOutTm, time_t& timeOut) = 0;
|
|
|
|
virtual status_t Timegm(struct tm* inOutTm, time_t& timeOut) = 0;
|
|
|
|
virtual void Initialize(LocaleDataBridge* dataBridge) = 0;
|
|
|
|
static status_t LoadBackend();
|
|
static status_t CreateBackend(LocaleBackend*& backendOut);
|
|
static void DestroyBackend(LocaleBackend* instance);
|
|
};
|
|
|
|
|
|
// The real struct behind locale_t
|
|
struct LocaleBackendData {
|
|
int magic;
|
|
LocaleBackend* backend;
|
|
LocaleDataBridge* databridge;
|
|
};
|
|
|
|
|
|
LocaleBackendData* GetCurrentLocaleInfo();
|
|
void SetCurrentLocaleInfo(LocaleBackendData* newLocale);
|
|
LocaleBackend* GetCurrentLocaleBackend();
|
|
extern LocaleBackend* gGlobalLocaleBackend;
|
|
extern LocaleDataBridge gGlobalLocaleDataBridge;
|
|
|
|
} // namespace Libroot
|
|
} // namespace BPrivate
|
|
|
|
|
|
#endif // _LOCALE_BACKEND_H
|