mirror of
https://github.com/reactos/reactos.git
synced 2025-01-01 12:04:51 +00:00
Use Wine's language matching algorithm when searching for resources.
Fixes bug 261. svn path=/trunk/; revision=10823
This commit is contained in:
parent
14399d738b
commit
44d3d874aa
2 changed files with 130 additions and 31 deletions
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: res.c,v 1.21 2004/06/13 20:04:56 navaraf Exp $
|
/* $Id: res.c,v 1.22 2004/09/11 17:06:33 gvg Exp $
|
||||||
*
|
*
|
||||||
* COPYRIGHT: See COPYING in the top level directory
|
* COPYRIGHT: See COPYING in the top level directory
|
||||||
* PROJECT : ReactOS user mode libraries
|
* PROJECT : ReactOS user mode libraries
|
||||||
|
@ -132,12 +132,6 @@ FindResourceExW (
|
||||||
ResourceInfo.Type = (ULONG)lpType;
|
ResourceInfo.Type = (ULONG)lpType;
|
||||||
ResourceInfo.Name = (ULONG)lpName;
|
ResourceInfo.Name = (ULONG)lpName;
|
||||||
ResourceInfo.Language = (ULONG)wLanguage;
|
ResourceInfo.Language = (ULONG)wLanguage;
|
||||||
if (ResourceInfo.Language == (ULONG) MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL)) {
|
|
||||||
ResourceInfo.Language = (ULONG) GetUserDefaultLangID();
|
|
||||||
if (ResourceInfo.Language == (ULONG) MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL)) {
|
|
||||||
ResourceInfo.Language = (ULONG) MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Status = LdrFindResource_U (hModule,
|
Status = LdrFindResource_U (hModule,
|
||||||
&ResourceInfo,
|
&ResourceInfo,
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: res.c,v 1.6 2003/12/20 21:43:27 navaraf Exp $
|
/* $Id: res.c,v 1.7 2004/09/11 17:06:33 gvg Exp $
|
||||||
*
|
*
|
||||||
* COPYRIGHT: See COPYING in the top level directory
|
* COPYRIGHT: See COPYING in the top level directory
|
||||||
* PROJECT: ReactOS kernel
|
* PROJECT: ReactOS kernel
|
||||||
|
@ -7,6 +7,13 @@
|
||||||
* PROGRAMMERS: Jean Michault
|
* PROGRAMMERS: Jean Michault
|
||||||
* Rex Jolliff (rex@lvcablemodem.com)
|
* Rex Jolliff (rex@lvcablemodem.com)
|
||||||
* Robert Dickenson (robd@mok.lvcm.com)
|
* Robert Dickenson (robd@mok.lvcm.com)
|
||||||
|
* NOTES: Parts based on Wine code
|
||||||
|
* Copyright 1995 Thomas Sandford
|
||||||
|
* Copyright 1996 Martin von Loewis
|
||||||
|
* Copyright 2003 Alexandre Julliard
|
||||||
|
* Copyright 1993 Robert J. Amstadt
|
||||||
|
* Copyright 1995 Alexandre Julliard
|
||||||
|
* Copyright 1997 Marcus Meissner
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -33,6 +40,52 @@
|
||||||
|
|
||||||
/* FUNCTIONS *****************************************************************/
|
/* FUNCTIONS *****************************************************************/
|
||||||
|
|
||||||
|
static PIMAGE_RESOURCE_DIRECTORY_ENTRY FASTCALL
|
||||||
|
FindEntryById(PIMAGE_RESOURCE_DIRECTORY ResDir,
|
||||||
|
ULONG Id)
|
||||||
|
{
|
||||||
|
PIMAGE_RESOURCE_DIRECTORY_ENTRY ResEntry;
|
||||||
|
LONG low, high, mid, result;
|
||||||
|
|
||||||
|
/* We use ID number instead of string */
|
||||||
|
ResEntry = (PIMAGE_RESOURCE_DIRECTORY_ENTRY)(ResDir + 1) + ResDir->NumberOfNamedEntries;
|
||||||
|
DPRINT("ResEntry %x - Resource ID number instead of string\n", (ULONG)ResEntry);
|
||||||
|
DPRINT("EntryCount %d\n", (ULONG)ResDir->NumberOfIdEntries);
|
||||||
|
|
||||||
|
low = 0;
|
||||||
|
high = ResDir->NumberOfIdEntries - 1;
|
||||||
|
mid = high/2;
|
||||||
|
while( low <= high ) {
|
||||||
|
result = Id - ResEntry[mid].Name;
|
||||||
|
if(result == 0)
|
||||||
|
return ResEntry + mid;
|
||||||
|
if(result < 0)
|
||||||
|
high = mid - 1;
|
||||||
|
else
|
||||||
|
low = mid + 1;
|
||||||
|
|
||||||
|
mid = (low + high)/2;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int FASTCALL
|
||||||
|
PushLanguage(WORD *list, int pos, WORD lang)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < pos; i++) {
|
||||||
|
if (list[i] == lang) {
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
list[pos++] = lang;
|
||||||
|
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Status = LdrFindResource_U (hModule,
|
Status = LdrFindResource_U (hModule,
|
||||||
&ResourceInfo,
|
&ResourceInfo,
|
||||||
|
@ -53,6 +106,10 @@ LdrFindResource_U(PVOID BaseAddress,
|
||||||
ULONG i;
|
ULONG i;
|
||||||
ULONG Id;
|
ULONG Id;
|
||||||
LONG low, high, mid, result;
|
LONG low, high, mid, result;
|
||||||
|
WORD list[9]; /* list of languages to try */
|
||||||
|
int j, pos = 0;
|
||||||
|
LCID UserLCID, SystemLCID;
|
||||||
|
LANGID UserLangID, SystemLangID;
|
||||||
|
|
||||||
DPRINT("LdrFindResource_U(%08x, %08x, %d, %08x)\n", BaseAddress, ResourceInfo, Level, ResourceDataEntry);
|
DPRINT("LdrFindResource_U(%08x, %08x, %d, %08x)\n", BaseAddress, ResourceInfo, Level, ResourceDataEntry);
|
||||||
|
|
||||||
|
@ -68,13 +125,10 @@ LdrFindResource_U(PVOID BaseAddress,
|
||||||
ResBase = ResDir;
|
ResBase = ResDir;
|
||||||
|
|
||||||
/* Let's go into resource tree */
|
/* Let's go into resource tree */
|
||||||
for (i = 0; i < Level; i++) {
|
for (i = 0; i < (2 < Level ? 2 : Level); i++) {
|
||||||
DPRINT("ResDir: %x Level: %d\n", (ULONG)ResDir, i);
|
DPRINT("ResDir: %x Level: %d\n", (ULONG)ResDir, i);
|
||||||
|
|
||||||
Id = ((PULONG)ResourceInfo)[i];
|
Id = ((PULONG)ResourceInfo)[i];
|
||||||
// ResourceInfo.Type = (ULONG)lpType;
|
|
||||||
// ResourceInfo.Name = (ULONG)lpName;
|
|
||||||
// ResourceInfo.Language = (ULONG)wLanguage;
|
|
||||||
|
|
||||||
if (Id & 0xFFFF0000) {
|
if (Id & 0xFFFF0000) {
|
||||||
/* Resource name is a unicode string */
|
/* Resource name is a unicode string */
|
||||||
|
@ -92,7 +146,10 @@ LdrFindResource_U(PVOID BaseAddress,
|
||||||
/* Need double check for lexical & length */
|
/* Need double check for lexical & length */
|
||||||
if(result == 0) {
|
if(result == 0) {
|
||||||
result = (wcslen((PWCHAR)Id) - (int)*ws);
|
result = (wcslen((PWCHAR)Id) - (int)*ws);
|
||||||
if(result == 0) goto found;
|
if(result == 0) {
|
||||||
|
ResEntry += mid;
|
||||||
|
goto found;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if(result < 0)
|
if(result < 0)
|
||||||
high = mid - 1;
|
high = mid - 1;
|
||||||
|
@ -103,23 +160,8 @@ LdrFindResource_U(PVOID BaseAddress,
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* We use ID number instead of string */
|
/* We use ID number instead of string */
|
||||||
ResEntry = (PIMAGE_RESOURCE_DIRECTORY_ENTRY)(ResDir + 1) + ResDir->NumberOfNamedEntries;
|
ResEntry = FindEntryById(ResDir, Id);
|
||||||
DPRINT("ResEntry %x - Resource ID number instead of string\n", (ULONG)ResEntry);
|
if (NULL != ResEntry) goto found;
|
||||||
DPRINT("EntryCount %d\n", (ULONG)ResDir->NumberOfIdEntries);
|
|
||||||
|
|
||||||
low = 0;
|
|
||||||
high = ResDir->NumberOfIdEntries - 1;
|
|
||||||
mid = high/2;
|
|
||||||
while( low <= high ) {
|
|
||||||
result = Id - ResEntry[mid].Name;
|
|
||||||
if(result == 0) goto found;
|
|
||||||
if(result < 0)
|
|
||||||
high = mid - 1;
|
|
||||||
else
|
|
||||||
low = mid + 1;
|
|
||||||
|
|
||||||
mid = (low + high)/2;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (i) {
|
switch (i) {
|
||||||
|
@ -146,7 +188,70 @@ LdrFindResource_U(PVOID BaseAddress,
|
||||||
}
|
}
|
||||||
found:;
|
found:;
|
||||||
ResDir = (PIMAGE_RESOURCE_DIRECTORY)((ULONG)ResBase +
|
ResDir = (PIMAGE_RESOURCE_DIRECTORY)((ULONG)ResBase +
|
||||||
(ResEntry[mid].OffsetToData & 0x7FFFFFFF));
|
(ResEntry->OffsetToData & 0x7FFFFFFF));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (3 <= Level) {
|
||||||
|
/* 1. specified language */
|
||||||
|
pos = PushLanguage(list, pos, ResourceInfo->Language );
|
||||||
|
|
||||||
|
/* 2. specified language with neutral sublanguage */
|
||||||
|
pos = PushLanguage(list, pos, MAKELANGID(PRIMARYLANGID(ResourceInfo->Language), SUBLANG_NEUTRAL));
|
||||||
|
|
||||||
|
/* 3. neutral language with neutral sublanguage */
|
||||||
|
pos = PushLanguage(list, pos, MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL));
|
||||||
|
|
||||||
|
/* if no explicitly specified language, try some defaults */
|
||||||
|
if (LANG_NEUTRAL == PRIMARYLANGID(ResourceInfo->Language)) {
|
||||||
|
/* user defaults, unless SYS_DEFAULT sublanguage specified */
|
||||||
|
if (SUBLANG_SYS_DEFAULT != SUBLANGID(ResourceInfo->Language)) {
|
||||||
|
NtQueryDefaultLocale(TRUE, &UserLCID);
|
||||||
|
UserLangID = LANGIDFROMLCID(UserLCID);
|
||||||
|
|
||||||
|
/* 4. current thread locale language */
|
||||||
|
pos = PushLanguage(list, pos, LANGIDFROMLCID(NtCurrentTeb()->CurrentLocale));
|
||||||
|
|
||||||
|
/* 5. user locale language */
|
||||||
|
pos = PushLanguage(list, pos, UserLangID);
|
||||||
|
|
||||||
|
/* 6. user locale language with neutral sublanguage */
|
||||||
|
pos = PushLanguage(list, pos, MAKELANGID(PRIMARYLANGID(UserLangID),
|
||||||
|
SUBLANG_NEUTRAL));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* now system defaults */
|
||||||
|
NtQueryDefaultLocale(FALSE, &SystemLCID);
|
||||||
|
SystemLangID = LANGIDFROMLCID(SystemLCID);
|
||||||
|
|
||||||
|
/* 7. system locale language */
|
||||||
|
pos = PushLanguage(list, pos, SystemLangID);
|
||||||
|
|
||||||
|
/* 8. system locale language with neutral sublanguage */
|
||||||
|
pos = PushLanguage(list, pos, MAKELANGID(PRIMARYLANGID(SystemLangID),
|
||||||
|
SUBLANG_NEUTRAL));
|
||||||
|
|
||||||
|
/* 9. English */
|
||||||
|
pos = PushLanguage(list, pos, MAKELANGID(LANG_ENGLISH, SUBLANG_DEFAULT));
|
||||||
|
}
|
||||||
|
|
||||||
|
ResEntry = NULL;
|
||||||
|
for (j = 0; NULL == ResEntry && j < pos; j++)
|
||||||
|
ResEntry = FindEntryById(ResDir, list[j]);
|
||||||
|
if (NULL == ResEntry) {
|
||||||
|
if (ResDir->NumberOfNamedEntries || ResDir->NumberOfIdEntries) {
|
||||||
|
/* Use the first available language */
|
||||||
|
ResEntry = (IMAGE_RESOURCE_DIRECTORY_ENTRY*)(ResDir + 1);
|
||||||
|
} else {
|
||||||
|
DPRINT("Error - STATUS_RESOURCE_LANG_NOT_FOUND\n", i);
|
||||||
|
return STATUS_RESOURCE_LANG_NOT_FOUND;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ResDir = (PIMAGE_RESOURCE_DIRECTORY)((ULONG)ResBase +
|
||||||
|
(ResEntry->OffsetToData & 0x7FFFFFFF));
|
||||||
|
if (3 < Level) {
|
||||||
|
DPRINT("Error - STATUS_INVALID_PARAMETER\n", i);
|
||||||
|
return STATUS_INVALID_PARAMETER;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
DPRINT("ResourceDataEntry: %x\n", (ULONG)ResDir);
|
DPRINT("ResourceDataEntry: %x\n", (ULONG)ResDir);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue