2013-06-13 23:33:15 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Original Name: src/common/intl.cpp
|
|
|
|
// Original Purpose: Internationalization and localisation for wxWidgets
|
|
|
|
// Original Author: Vadim Zeitlin
|
|
|
|
// Modified by: Charlie Fenton for BOINC 13 June, 2013
|
|
|
|
// Created: 29/01/98
|
|
|
|
// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
|
|
|
|
// Licence: wxWindows licence
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2013-06-12 11:03:02 +00:00
|
|
|
// This file is part of BOINC.
|
|
|
|
// http://boinc.berkeley.edu
|
|
|
|
//
|
2013-06-13 23:33:15 +00:00
|
|
|
// Simplified string localization code for use in BOINC adapted from
|
2013-06-12 11:03:02 +00:00
|
|
|
// wxWidgets src/common/intl.cpp.
|
|
|
|
//
|
|
|
|
// For sample code to get preferred system language, see
|
|
|
|
// wxLocale::GetSystemLanguage() in wxWidgets src/common/intl.cpp
|
2013-06-13 23:33:15 +00:00
|
|
|
//
|
2023-05-05 18:05:20 +00:00
|
|
|
// Those portions of this code which are taken from wxWidgets are
|
|
|
|
// Copyright: (c) 1998 Vadim Zeitlin
|
|
|
|
// and are covered by the wxWidgets license which can be found here:
|
2013-06-13 23:33:15 +00:00
|
|
|
// <http://www.wxwidgets.org/about/licence3.txt>
|
|
|
|
//
|
|
|
|
// This code assumes all catalogs are encoded UTF-8
|
|
|
|
// (Note: wxstd.mo files are encoded ISO 8859-1 not UTF-8.)
|
|
|
|
//
|
|
|
|
// We recommended that you first call BOINCTranslationAddCatalog()
|
2013-08-17 04:48:18 +00:00
|
|
|
// for each desired catalog (*.mo) file for the user's preferred
|
|
|
|
// language, then for each desired catalog (*.mo) file for the
|
|
|
|
// user's second preferred language and (optionally) then for each
|
|
|
|
// desired catalog (*.mo) file for the user's third preferred
|
2023-05-05 18:05:20 +00:00
|
|
|
// language. This will make it more likely that a translation
|
2013-08-17 04:48:18 +00:00
|
|
|
// will be found in some language useful to the user.
|
2013-06-13 23:33:15 +00:00
|
|
|
//
|
2013-06-12 11:03:02 +00:00
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <cstring>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <sys/param.h> // for MAXPATHLEN
|
|
|
|
#include <sys/stat.h> // For stat()
|
2013-08-16 12:11:32 +00:00
|
|
|
#include <filesys.h>
|
2013-06-12 11:03:02 +00:00
|
|
|
|
|
|
|
#include "translate.h"
|
|
|
|
|
2013-06-14 11:50:06 +00:00
|
|
|
static const uint32_t MSGCATALOG_MAGIC = 0x950412de;
|
|
|
|
static const uint32_t MSGCATALOG_MAGIC_SW = 0xde120495;
|
2013-06-12 11:03:02 +00:00
|
|
|
|
2013-06-13 23:33:15 +00:00
|
|
|
#define MAXCATALOGS 20
|
2013-08-17 04:48:18 +00:00
|
|
|
#define VERBOSE true
|
2013-06-12 11:03:02 +00:00
|
|
|
|
|
|
|
// an entry in the string table
|
|
|
|
struct MsgTableEntry {
|
|
|
|
uint32_t nLen; // length of the string
|
|
|
|
uint32_t ofsString; // pointer to the string
|
|
|
|
};
|
|
|
|
|
|
|
|
// header of a .mo file
|
|
|
|
struct MsgCatalogHeader {
|
|
|
|
uint32_t magic, // offset +00: magic id
|
|
|
|
revision, // +04: revision
|
|
|
|
numStrings; // +08: number of strings in the file
|
|
|
|
uint32_t ofsOrigTable, // +0C: start of original string table
|
|
|
|
ofsTransTable; // +10: start of translated string table
|
|
|
|
uint32_t nHashSize, // +14: hash table size
|
|
|
|
ofsHashTable; // +18: offset of hash table start
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct MsgCatalogData {
|
2013-08-16 12:11:32 +00:00
|
|
|
uint8_t *pData; // Pointer to catalog data
|
|
|
|
uint32_t nSize; // Amount of memory pointed to by pData.
|
|
|
|
uint32_t NumStrings; // number of strings in this domain
|
|
|
|
bool bSwapped; // wrong endianness?
|
|
|
|
MsgTableEntry *pOrigTable; // pointer to original strings
|
|
|
|
MsgTableEntry *pTransTable; // pointer to translated strings
|
|
|
|
char languageCode[32]; // language code (e.g., "it_IT")
|
|
|
|
char catalogName[128]; // catalog name (e.g., "BOINC-Setup")
|
2013-06-12 11:03:02 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct MsgCatalogData theCatalogData[MAXCATALOGS];
|
|
|
|
|
|
|
|
static uint32_t numLoadedCatalogs = 0;
|
|
|
|
|
|
|
|
// swap the 2 halves of 32 bit integer if needed
|
|
|
|
static uint32_t Swap(uint32_t ui, bool bSwapped) {
|
|
|
|
return bSwapped ? (ui << 24) | ((ui & 0xff00) << 8) |
|
|
|
|
((ui >> 8) & 0xff00) | (ui >> 24)
|
|
|
|
: ui;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// load a catalog from disk
|
|
|
|
// open disk file and read in it's contents
|
|
|
|
//
|
|
|
|
// catalogsDir is the path to the locale directory in
|
|
|
|
// the BOINC Data directory (ending in "locale/").
|
|
|
|
//
|
|
|
|
// language code is the standad language code such
|
|
|
|
// as "fr" or "zh_CN".
|
|
|
|
//
|
|
|
|
// catalogName is the domain (the file name of the
|
|
|
|
// *.mo file without the .mo such as "BOINC-Manager").
|
|
|
|
//
|
|
|
|
// Returns true if successful.
|
|
|
|
//
|
2013-06-13 23:33:15 +00:00
|
|
|
static bool LoadCatalog(const char * catalogsDir,
|
2013-06-12 11:03:02 +00:00
|
|
|
const char *languageCode,
|
|
|
|
const char *catalogName
|
|
|
|
) {
|
2013-08-16 12:11:32 +00:00
|
|
|
unsigned int j;
|
2013-06-12 11:03:02 +00:00
|
|
|
char searchPath[MAXPATHLEN];
|
|
|
|
struct stat sbuf;
|
|
|
|
char temp[32];
|
|
|
|
char *underscore;
|
|
|
|
FILE * f;
|
|
|
|
uint8_t *pData;
|
|
|
|
uint32_t nSize;
|
|
|
|
uint32_t NumStrings;
|
|
|
|
bool bSwapped;
|
|
|
|
MsgTableEntry *pOrigTable;
|
|
|
|
MsgTableEntry *pTransTable;
|
|
|
|
MsgCatalogData *pCatalog;
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-08-17 04:48:18 +00:00
|
|
|
#if VERBOSE
|
|
|
|
fprintf(stderr, "Attempting to load catalog %s for language code %s\n",
|
|
|
|
catalogName, languageCode);
|
|
|
|
#endif
|
2013-08-16 12:11:32 +00:00
|
|
|
for (j=0; j<numLoadedCatalogs; ++j) {
|
|
|
|
pCatalog = &(theCatalogData[j]);
|
2013-08-17 04:48:18 +00:00
|
|
|
if (!strcmp(pCatalog->catalogName, catalogName)) {
|
|
|
|
if (!strcmp(pCatalog->languageCode, languageCode)) {
|
|
|
|
// Don't load a catalog twice for the same language
|
|
|
|
#if VERBOSE
|
|
|
|
fprintf(stderr, "Ignoring Catalog %s for language code %s; it was already loaded\n",
|
|
|
|
catalogName, languageCode);
|
|
|
|
#endif
|
|
|
|
return true; // Already loaded
|
|
|
|
}
|
|
|
|
// Don't load more languages for this catalog
|
|
|
|
// if we've already "loaded" it for English
|
|
|
|
if (!strcmp(pCatalog->languageCode, "en")) {
|
|
|
|
#if VERBOSE
|
|
|
|
fprintf(stderr, "Ignoring Catalog %s for language code %s; after English for same catalog\n",
|
|
|
|
catalogName, languageCode);
|
|
|
|
#endif
|
2013-08-16 12:11:32 +00:00
|
|
|
return true; // Already loaded
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-05-05 18:05:20 +00:00
|
|
|
|
|
|
|
// Since the original strings are English, no
|
2013-08-16 12:11:32 +00:00
|
|
|
// translation is needed for language en.
|
2013-06-12 11:03:02 +00:00
|
|
|
if (!strcmp("en", languageCode)) {
|
2013-08-16 12:11:32 +00:00
|
|
|
pCatalog = &(theCatalogData[numLoadedCatalogs++]);
|
|
|
|
strlcpy(pCatalog->languageCode, languageCode, sizeof(pCatalog->languageCode));
|
|
|
|
strlcpy(pCatalog->catalogName, catalogName, sizeof(pCatalog->catalogName));
|
2013-06-12 11:03:02 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
strlcpy(searchPath, catalogsDir, sizeof(searchPath));
|
|
|
|
strlcat(searchPath, languageCode, sizeof(searchPath));
|
|
|
|
strlcat(searchPath, "/", sizeof(searchPath));
|
|
|
|
strlcat(searchPath, catalogName, sizeof(searchPath));
|
|
|
|
strlcat(searchPath, ".mo", sizeof(searchPath));
|
|
|
|
if (stat(searchPath, &sbuf) != 0) {
|
|
|
|
// Try just base locale name: for things like "fr_BE" (belgium
|
2023-05-05 18:05:20 +00:00
|
|
|
// french) we should use "fr" if no belgium specific message
|
2013-06-12 11:03:02 +00:00
|
|
|
// catalogs exist
|
|
|
|
strlcpy(temp, languageCode, sizeof(temp));
|
|
|
|
underscore = strchr(temp, (int)'_');
|
|
|
|
if (underscore == NULL) return false;
|
|
|
|
*underscore = '\0';
|
|
|
|
strlcpy(searchPath, catalogsDir, sizeof(searchPath));
|
|
|
|
strlcat(searchPath, temp, sizeof(searchPath));
|
|
|
|
strlcat(searchPath, "/", sizeof(searchPath));
|
|
|
|
strlcat(searchPath, catalogName, sizeof(searchPath));
|
|
|
|
strlcat(searchPath, ".mo", sizeof(searchPath));
|
|
|
|
if (stat(searchPath, &sbuf) != 0) return false;
|
|
|
|
}
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
|
|
|
|
pData = (uint8_t*)malloc(sbuf.st_size);
|
|
|
|
if (pData == NULL) return false;
|
|
|
|
f = fopen(searchPath, "r");
|
|
|
|
if (f == NULL) {
|
|
|
|
free(pData);
|
|
|
|
pData = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
nSize = fread(pData, 1, sbuf.st_size, f);
|
|
|
|
fclose(f);
|
|
|
|
if (nSize != sbuf.st_size) {
|
|
|
|
free(pData);
|
|
|
|
pData = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
// examine header
|
|
|
|
bool bValid = nSize + (size_t)0 > sizeof(MsgCatalogHeader);
|
|
|
|
|
|
|
|
MsgCatalogHeader *pHeader = (MsgCatalogHeader *)pData;
|
|
|
|
if ( bValid ) {
|
|
|
|
// we'll have to swap all the integers if it's true
|
|
|
|
bSwapped = pHeader->magic == MSGCATALOG_MAGIC_SW;
|
|
|
|
|
|
|
|
// check the magic number
|
|
|
|
bValid = bSwapped || pHeader->magic == MSGCATALOG_MAGIC;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !bValid ) {
|
|
|
|
// it's either too short or has incorrect magic number
|
|
|
|
free(pData);
|
|
|
|
pData = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// initialize
|
|
|
|
NumStrings = Swap(pHeader->numStrings, bSwapped);
|
2013-06-13 23:33:15 +00:00
|
|
|
if (NumStrings <= 1) {
|
2023-05-05 18:05:20 +00:00
|
|
|
// This file has no translations (is effectively
|
2013-06-13 23:33:15 +00:00
|
|
|
// empty) so don't load it for better efficiency.
|
2013-08-17 04:48:18 +00:00
|
|
|
#if VERBOSE
|
2013-06-14 11:50:06 +00:00
|
|
|
fprintf(stderr, "File %s contains no translated strings!\n", searchPath);
|
2013-08-17 04:48:18 +00:00
|
|
|
#endif
|
2013-06-13 23:33:15 +00:00
|
|
|
free(pData);
|
|
|
|
pData = NULL;
|
|
|
|
return true; // Not an error
|
|
|
|
}
|
2013-06-12 11:03:02 +00:00
|
|
|
pOrigTable = (MsgTableEntry *)(pData +
|
|
|
|
Swap(pHeader->ofsOrigTable, bSwapped));
|
|
|
|
pTransTable = (MsgTableEntry *)(pData +
|
|
|
|
Swap(pHeader->ofsTransTable, bSwapped));
|
|
|
|
|
|
|
|
|
|
|
|
// now parse catalog's header and try to extract catalog charset
|
|
|
|
uint32_t ofsString = Swap(pOrigTable->ofsString, bSwapped);
|
|
|
|
// this check could fail for a corrupt message catalog
|
|
|
|
if ( ofsString + Swap(pOrigTable->nLen, bSwapped) <= nSize) {
|
2013-06-13 23:33:15 +00:00
|
|
|
char *begin = strstr((char *)(pData + Swap(pTransTable->ofsString, bSwapped)),
|
2013-06-12 11:03:02 +00:00
|
|
|
"Content-Type: text/plain; charset="
|
|
|
|
);
|
|
|
|
if (begin != NULL) {
|
2013-06-13 23:33:15 +00:00
|
|
|
begin += 34; //strlen("Content-Type: text/plain; charset=")
|
2013-06-12 11:03:02 +00:00
|
|
|
if (strncasecmp(begin, "utf-8", 5)) {
|
2013-08-17 04:48:18 +00:00
|
|
|
#if VERBOSE
|
2013-06-12 11:03:02 +00:00
|
|
|
fprintf(stderr, "File %s is not utf-8!\n", searchPath);
|
2013-08-17 04:48:18 +00:00
|
|
|
#endif
|
2013-06-12 11:03:02 +00:00
|
|
|
free(pData);
|
|
|
|
pData = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pCatalog = &(theCatalogData[numLoadedCatalogs++]);
|
|
|
|
pCatalog->pData = pData;
|
|
|
|
pCatalog->nSize = nSize;
|
|
|
|
pCatalog->NumStrings = NumStrings;
|
|
|
|
pCatalog->bSwapped = bSwapped;
|
|
|
|
pCatalog->pOrigTable = pOrigTable;
|
|
|
|
pCatalog->pTransTable = pTransTable;
|
2013-08-16 12:11:32 +00:00
|
|
|
strlcpy(pCatalog->languageCode, languageCode, sizeof(pCatalog->languageCode));
|
|
|
|
strlcpy(pCatalog->catalogName, catalogName, sizeof(pCatalog->catalogName));
|
2013-08-17 04:48:18 +00:00
|
|
|
#if VERBOSE
|
|
|
|
fprintf(stderr, "Successfully loaded catalog %s for language code %s\n",
|
|
|
|
catalogName, languageCode);
|
|
|
|
#endif
|
2013-06-12 11:03:02 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Searches through the catalogs in the order they were added
|
|
|
|
// until it finds a translation for the src string, and
|
2021-10-01 07:36:00 +00:00
|
|
|
// returns a pointer to the UTF-8 encoded localized string.
|
2013-06-12 11:03:02 +00:00
|
|
|
// Returns a pointer to the original string if no translation
|
|
|
|
// was found.
|
|
|
|
//
|
|
|
|
uint8_t * _(char *src) {
|
2013-06-14 11:50:06 +00:00
|
|
|
unsigned int i, j;
|
2013-06-12 11:03:02 +00:00
|
|
|
MsgCatalogData *pCatalog;
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
for (j=0; j<numLoadedCatalogs; ++j) {
|
|
|
|
pCatalog = &(theCatalogData[j]);
|
2023-05-05 18:05:20 +00:00
|
|
|
|
|
|
|
// Since the original strings are English, no
|
2013-06-12 11:03:02 +00:00
|
|
|
// translation is needed for language en.
|
2013-08-16 12:11:32 +00:00
|
|
|
if (!strcmp("en", pCatalog->languageCode)) {
|
2013-08-17 04:48:18 +00:00
|
|
|
continue; // Try next catalog
|
2013-06-12 11:03:02 +00:00
|
|
|
}
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
if (pCatalog->pData == NULL) continue; // Should never happen
|
2013-06-13 23:33:15 +00:00
|
|
|
for (i=0; i<pCatalog->NumStrings; ++i) {
|
2013-06-12 11:03:02 +00:00
|
|
|
if (!strcmp((char *)pCatalog->pData + pCatalog->pOrigTable[i].ofsString, src)) {
|
|
|
|
return (pCatalog->pData + pCatalog->pTransTable[i].ofsString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (uint8_t *)src;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// catalogsDir is the path to the locale directory in
|
|
|
|
// the BOINC Data directory (ending in "locale/").
|
|
|
|
//
|
|
|
|
// language code is the standad language code such
|
|
|
|
// as "fr" or "zh_CN".
|
|
|
|
//
|
|
|
|
// catalogName is the domain (the file name of the
|
|
|
|
// *.mo file without the .mo such as "BOINC-Manager").
|
|
|
|
//
|
|
|
|
// Returns true if successful.
|
|
|
|
//
|
|
|
|
bool BOINCTranslationAddCatalog(const char * catalogsDir,
|
|
|
|
const char *languageCode,
|
|
|
|
const char * catalogName
|
|
|
|
) {
|
2013-08-16 12:11:32 +00:00
|
|
|
bool success = false;
|
|
|
|
DIRREF dirp;
|
|
|
|
char filename[64];
|
|
|
|
int retval;
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
if (numLoadedCatalogs >= (MAXCATALOGS)) {
|
2013-08-17 04:48:18 +00:00
|
|
|
#if VERBOSE
|
2013-06-12 11:03:02 +00:00
|
|
|
fprintf(stderr, "Trying to load too many catalogs\n");
|
2013-08-17 04:48:18 +00:00
|
|
|
#endif
|
2013-06-12 11:03:02 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-08-16 12:11:32 +00:00
|
|
|
// Add a catalog for this exact language and region code
|
|
|
|
success = LoadCatalog(catalogsDir, languageCode, catalogName);
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-08-16 12:11:32 +00:00
|
|
|
// Add catalogs for the same language code but different region codes
|
|
|
|
dirp = dir_open(catalogsDir);
|
|
|
|
if (dirp) {
|
|
|
|
while (true) {
|
|
|
|
retval = dir_scan(filename, dirp, sizeof(filename));
|
|
|
|
if (retval) break;
|
|
|
|
if (!strcmp(languageCode, filename)) continue; // Already added above
|
|
|
|
if (!strncmp(languageCode, filename, 2)) { // First 2 characters match
|
|
|
|
if (LoadCatalog(catalogsDir, filename, catalogName)) {
|
|
|
|
success = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-08-16 12:11:32 +00:00
|
|
|
dir_close(dirp);
|
2013-06-12 11:03:02 +00:00
|
|
|
}
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-08-16 12:11:32 +00:00
|
|
|
return success;
|
2013-06-12 11:03:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BOINCTranslationInit() {
|
|
|
|
int i;
|
|
|
|
MsgCatalogData *pCatalog;
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
numLoadedCatalogs = 0;
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
for (i=0; i<MAXCATALOGS; ++i) {
|
|
|
|
pCatalog = &(theCatalogData[i]);
|
|
|
|
pCatalog->pData = NULL;
|
|
|
|
pCatalog->nSize = 0;
|
|
|
|
pCatalog->NumStrings = 0;
|
|
|
|
pCatalog->bSwapped = false;
|
|
|
|
pCatalog->pOrigTable = NULL;
|
|
|
|
pCatalog->pTransTable = NULL;
|
2013-08-16 12:11:32 +00:00
|
|
|
pCatalog->languageCode[0] = '\0';
|
|
|
|
pCatalog->catalogName[0] = '\0';
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BOINCTranslationCleanup() {
|
|
|
|
int i;
|
|
|
|
MsgCatalogData *pCatalog;
|
2023-05-05 18:05:20 +00:00
|
|
|
|
2013-06-12 11:03:02 +00:00
|
|
|
for (i=0; i<MAXCATALOGS; ++i) {
|
|
|
|
pCatalog = &(theCatalogData[i]);
|
|
|
|
if (pCatalog->pData) free(pCatalog->pData);
|
|
|
|
pCatalog->pData = NULL;
|
|
|
|
pCatalog->nSize = 0;
|
|
|
|
pCatalog->NumStrings = 0;
|
|
|
|
pCatalog->bSwapped = false;
|
|
|
|
pCatalog->pOrigTable = NULL;
|
|
|
|
pCatalog->pTransTable = NULL;
|
2013-08-16 12:11:32 +00:00
|
|
|
pCatalog->languageCode[0] = '\0';
|
|
|
|
pCatalog->catalogName[0] = '\0';
|
2013-06-12 11:03:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
numLoadedCatalogs = 0;
|
|
|
|
}
|