Commit c44eb6d6 authored by Alexandre Julliard's avatar Alexandre Julliard

wmc: Add support for named locales.

parent 4e89a259
...@@ -2781,6 +2781,7 @@ static void output_source_mc( struct makefile *make, struct incl_file *source, c ...@@ -2781,6 +2781,7 @@ static void output_source_mc( struct makefile *make, struct incl_file *source, c
output( "%s.pot %s.res: %s", obj_path, obj_path, source->filename ); output( "%s.pot %s.res: %s", obj_path, obj_path, source->filename );
output_filename( tools_path( make, "wmc" )); output_filename( tools_path( make, "wmc" ));
output_filenames( source->dependencies ); output_filenames( source->dependencies );
if (make->src_dir) output_filename( "nls/locale.nls" );
output( "\n" ); output( "\n" );
output( "\t%s%s -u -o $@ %s", cmd_prefix( "WMC" ), tools_path( make, "wmc" ), source->filename ); output( "\t%s%s -u -o $@ %s", cmd_prefix( "WMC" ), tools_path( make, "wmc" ), source->filename );
if (linguas.count) if (linguas.count)
......
...@@ -28,6 +28,7 @@ ...@@ -28,6 +28,7 @@
#include <ctype.h> #include <ctype.h>
#include "wmc.h" #include "wmc.h"
#include "winternl.h"
#include "winnls.h" #include "winnls.h"
#include "utils.h" #include "utils.h"
...@@ -327,6 +328,14 @@ WCHAR *codepage_to_unicode( int codepage, const char *src, int srclen, int *dstl ...@@ -327,6 +328,14 @@ WCHAR *codepage_to_unicode( int codepage, const char *src, int srclen, int *dstl
return dst; return dst;
} }
unsigned int get_language_from_name( const char *name )
{
WCHAR nameW[LOCALE_NAME_MAX_LENGTH];
MultiByteToWideChar( 1252, 0, name, -1, nameW, ARRAY_SIZE(nameW) );
return LocaleNameToLCID( nameW, LOCALE_ALLOW_NEUTRAL_NAMES );
}
#else /* _WIN32 */ #else /* _WIN32 */
struct nls_info struct nls_info
...@@ -354,29 +363,40 @@ static void init_nls_info( struct nls_info *info, unsigned short *ptr ) ...@@ -354,29 +363,40 @@ static void init_nls_info( struct nls_info *info, unsigned short *ptr )
info->dbcs_offsets = *ptr ? ptr + 1 : NULL; info->dbcs_offsets = *ptr ? ptr + 1 : NULL;
} }
static void *load_nls_file( const char *name )
{
unsigned int i;
void *data;
size_t size;
for (i = 0; nlsdirs[i]; i++)
{
char *path = strmake( "%s/%s", nlsdirs[i], name );
if ((data = read_file( path, &size )))
{
free( path );
return data;
}
free( path );
}
return NULL;
}
static const struct nls_info *get_nls_info( unsigned int codepage ) static const struct nls_info *get_nls_info( unsigned int codepage )
{ {
unsigned short *data; unsigned short *data;
char *path;
unsigned int i; unsigned int i;
size_t size;
for (i = 0; i < ARRAY_SIZE(nlsinfo) && nlsinfo[i].codepage; i++) for (i = 0; i < ARRAY_SIZE(nlsinfo) && nlsinfo[i].codepage; i++)
if (nlsinfo[i].codepage == codepage) return &nlsinfo[i]; if (nlsinfo[i].codepage == codepage) return &nlsinfo[i];
assert( i < ARRAY_SIZE(nlsinfo) ); assert( i < ARRAY_SIZE(nlsinfo) );
for (i = 0; nlsdirs[i]; i++) if ((data = load_nls_file( strmake( "c_%03u.nls", codepage ))))
{
path = strmake( "%s/c_%03u.nls", nlsdirs[i], codepage );
if ((data = read_file( path, &size )))
{ {
free( path );
init_nls_info( &nlsinfo[i], data ); init_nls_info( &nlsinfo[i], data );
return &nlsinfo[i]; return &nlsinfo[i];
} }
free( path );
}
return NULL; return NULL;
} }
...@@ -427,6 +447,76 @@ WCHAR *codepage_to_unicode( int codepage, const char *src, int srclen, int *dstl ...@@ -427,6 +447,76 @@ WCHAR *codepage_to_unicode( int codepage, const char *src, int srclen, int *dstl
return dst; return dst;
} }
static const NLS_LOCALE_LCID_INDEX *lcids_index;
static const NLS_LOCALE_HEADER *locale_table;
static const NLS_LOCALE_LCNAME_INDEX *lcnames_index;
static const WCHAR *locale_strings;
static void load_locale_nls(void)
{
struct
{
unsigned int ctypes;
unsigned int unknown1;
unsigned int unknown2;
unsigned int unknown3;
unsigned int locales;
unsigned int charmaps;
unsigned int geoids;
unsigned int scripts;
} *header;
if (!(header = load_nls_file( "locale.nls" ))) error( "unable to load locale.nls\n" );
locale_table = (const NLS_LOCALE_HEADER *)((char *)header + header->locales);
lcids_index = (const NLS_LOCALE_LCID_INDEX *)((char *)locale_table + locale_table->lcids_offset);
lcnames_index = (const NLS_LOCALE_LCNAME_INDEX *)((char *)locale_table + locale_table->lcnames_offset);
locale_strings = (const WCHAR *)((char *)locale_table + locale_table->strings_offset);
}
static int compare_locale_names( const char *n1, const WCHAR *n2 )
{
for (;;)
{
WCHAR ch1 = (unsigned char)*n1++;
WCHAR ch2 = *n2++;
if (ch1 >= 'a' && ch1 <= 'z') ch1 -= 'a' - 'A';
if (ch2 >= 'a' && ch2 <= 'z') ch2 -= 'a' - 'A';
if (!ch1 || ch1 != ch2) return ch1 - ch2;
}
}
static const NLS_LOCALE_LCNAME_INDEX *find_lcname_entry( const char *name )
{
int min = 0, max = locale_table->nb_lcnames - 1;
if (!name) return NULL;
while (min <= max)
{
int res, pos = (min + max) / 2;
const WCHAR *str = locale_strings + lcnames_index[pos].name;
res = compare_locale_names( name, str + 1 );
if (res < 0) max = pos - 1;
else if (res > 0) min = pos + 1;
else return &lcnames_index[pos];
}
return NULL;
}
static const NLS_LOCALE_DATA *get_locale_data( UINT idx )
{
ULONG offset = locale_table->locales_offset + idx * locale_table->locale_size;
return (const NLS_LOCALE_DATA *)((const char *)locale_table + offset);
}
unsigned int get_language_from_name( const char *name )
{
const NLS_LOCALE_LCNAME_INDEX *entry;
if (!locale_table) load_locale_nls();
if (!(entry = find_lcname_entry( name ))) return 0;
return get_locale_data( entry->idx )->unique_lcid;
}
#endif /* _WIN32 */ #endif /* _WIN32 */
unsigned char *output_buffer; unsigned char *output_buffer;
......
...@@ -40,5 +40,6 @@ WCHAR *utf8_to_unicode( const char *src, int srclen, int *dstlen ); ...@@ -40,5 +40,6 @@ WCHAR *utf8_to_unicode( const char *src, int srclen, int *dstlen );
char *unicode_to_utf8( const WCHAR *src, int srclen, int *dstlen ); char *unicode_to_utf8( const WCHAR *src, int srclen, int *dstlen );
int is_valid_codepage(int id); int is_valid_codepage(int id);
WCHAR *codepage_to_unicode( int codepage, const char *src, int srclen, int *dstlen ); WCHAR *codepage_to_unicode( int codepage, const char *src, int srclen, int *dstlen );
unsigned int get_language_from_name( const char *name );
#endif #endif
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment