- Handle ACS string escape sequences at load time rather than print time.

- Fixed: FBehavior::FindStringInChunk() was not big-endian safe.
- Fixed: FBehavior::LookupString() was not big-endian safe.

SVN r3406 (trunk)
This commit is contained in:
Randy Heit 2012-03-08 21:37:34 +00:00
parent 5f88d2c9f0
commit e5e1ee00a6
3 changed files with 65 additions and 10 deletions

View file

@ -550,7 +550,7 @@ exists: if (p != NULL)
//========================================================================== //==========================================================================
// //
// strbin1 -- In-place version // strbin -- In-place version
// //
// [RH] Replaces the escape sequences in a string with actual escaped characters. // [RH] Replaces the escape sequences in a string with actual escaped characters.
// This operation is done in-place. The result is the new length of the string. // This operation is done in-place. The result is the new length of the string.
@ -650,7 +650,7 @@ int strbin (char *str)
// strbin1 -- String-creating version // strbin1 -- String-creating version
// //
// [RH] Replaces the escape sequences in a string with actual escaped characters. // [RH] Replaces the escape sequences in a string with actual escaped characters.
// This operation is done in-place. // The result is a new string.
// //
//========================================================================== //==========================================================================

View file

@ -1128,6 +1128,7 @@ FBehavior::FBehavior (int lumpnum, FileReader * fr, int len)
{ {
StringTable = LittleLong(((DWORD *)Data)[1]); StringTable = LittleLong(((DWORD *)Data)[1]);
StringTable += LittleLong(((DWORD *)(Data + StringTable))[0]) * 12 + 4; StringTable += LittleLong(((DWORD *)(Data + StringTable))[0]) * 12 + 4;
UnescapeStringTable(Data + StringTable, false);
} }
else else
{ {
@ -1136,6 +1137,7 @@ FBehavior::FBehavior (int lumpnum, FileReader * fr, int len)
if (strings != NULL) if (strings != NULL)
{ {
StringTable = DWORD(strings - Data + 8); StringTable = DWORD(strings - Data + 8);
UnescapeStringTable(strings, true);
} }
else else
{ {
@ -1263,6 +1265,7 @@ FBehavior::FBehavior (int lumpnum, FileReader * fr, int len)
} }
} }
// Load required libraries.
if (NULL != (chunk = (DWORD *)FindChunk (MAKE_ID('L','O','A','D')))) if (NULL != (chunk = (DWORD *)FindChunk (MAKE_ID('L','O','A','D'))))
{ {
const char *const parse = (char *)&chunk[2]; const char *const parse = (char *)&chunk[2];
@ -1322,7 +1325,7 @@ FBehavior::FBehavior (int lumpnum, FileReader * fr, int len)
ModuleName, func->ArgCount); ModuleName, func->ArgCount);
Format = ACS_Unknown; Format = ACS_Unknown;
} }
// The next two properties do not effect code compatibility, so it is // The next two properties do not affect code compatibility, so it is
// okay for them to be different in the imported module than they are // okay for them to be different in the imported module than they are
// in this one, as long as we make sure to use the real values. // in this one, as long as we make sure to use the real values.
func->LocalCount = realfunc->LocalCount; func->LocalCount = realfunc->LocalCount;
@ -1577,6 +1580,7 @@ void FBehavior::LoadScriptsDirectory ()
scripts.b = FindChunk(MAKE_ID('S','N','A','M')); scripts.b = FindChunk(MAKE_ID('S','N','A','M'));
if (scripts.dw != NULL) if (scripts.dw != NULL)
{ {
UnescapeStringTable(scripts.b, false);
for (i = 0; i < NumScripts; ++i) for (i = 0; i < NumScripts; ++i)
{ {
// ACC stores script names as an index into the SNAM chunk, with the first index as // ACC stores script names as an index into the SNAM chunk, with the first index as
@ -1602,15 +1606,23 @@ int STACK_ARGS FBehavior::SortScripts (const void *a, const void *b)
return ptr1->Number - ptr2->Number; return ptr1->Number - ptr2->Number;
} }
//============================================================================
//
// FBehavior :: UnencryptStrings
//
// Descrambles strings in a STRE chunk to transform it into a STRL chunk.
//
//============================================================================
void FBehavior::UnencryptStrings () void FBehavior::UnencryptStrings ()
{ {
DWORD *prevchunk = NULL; DWORD *prevchunk = NULL;
DWORD *chunk = (DWORD *)FindChunk(MAKE_ID('S','T','R','E')); DWORD *chunk = (DWORD *)FindChunk(MAKE_ID('S','T','R','E'));
while (chunk != NULL) while (chunk != NULL)
{ {
for (DWORD strnum = 0; strnum < chunk[3]; ++strnum) for (DWORD strnum = 0; strnum < LittleLong(chunk[3]); ++strnum)
{ {
int ofs = chunk[5+strnum]; int ofs = LittleLong(chunk[5+strnum]);
BYTE *data = (BYTE *)chunk + ofs + 8, last; BYTE *data = (BYTE *)chunk + ofs + 8, last;
int p = (BYTE)(ofs*157135); int p = (BYTE)(ofs*157135);
int i = 0; int i = 0;
@ -1630,6 +1642,49 @@ void FBehavior::UnencryptStrings ()
} }
} }
//============================================================================
//
// FBehavior :: UnescapeStringTable
//
// Processes escape sequences for every string in a string table.
//
//============================================================================
void FBehavior::UnescapeStringTable(BYTE *chunkstart, bool has_padding)
{
assert(chunkstart != NULL);
DWORD *chunk = (DWORD *)chunkstart;
chunk += 2;
if (!has_padding)
{
chunk[0] = LittleLong(chunk[0]);
for (DWORD strnum = 0; strnum < chunk[0]; ++strnum)
{
int ofs = LittleLong(chunk[1 + strnum]); // Byte swap offset, if needed.
chunk[1 + strnum] = ofs;
strbin((char *)chunk + ofs);
}
}
else
{
chunk[1] = LittleLong(chunk[1]);
for (DWORD strnum = 0; strnum < chunk[1]; ++strnum)
{
int ofs = LittleLong(chunk[3 + strnum]); // Byte swap offset, if needed.
chunk[3 + strnum] = ofs;
strbin((char *)chunk + ofs);
}
}
}
//============================================================================
//
// FBehavior :: IsGood
//
//============================================================================
bool FBehavior::IsGood () bool FBehavior::IsGood ()
{ {
bool bad; bool bad;
@ -1741,9 +1796,9 @@ int FBehavior::FindStringInChunk (DWORD *names, const char *varname) const
{ {
DWORD i; DWORD i;
for (i = 0; i < names[2]; ++i) for (i = 0; i < LittleLong(names[2]); ++i)
{ {
if (stricmp (varname, (char *)(names + 2) + names[3+i]) == 0) if (stricmp (varname, (char *)(names + 2) + LittleLong(names[3+i])) == 0)
{ {
return (int)i; return (int)i;
} }
@ -1851,7 +1906,7 @@ const char *FBehavior::LookupString (DWORD index) const
if (index >= list[0]) if (index >= list[0])
return NULL; // Out of range for this list; return NULL; // Out of range for this list;
return (const char *)(Data + LittleLong(list[1+index])); return (const char *)(Data + list[1+index]);
} }
else else
{ {
@ -5454,7 +5509,6 @@ int DLevelScript::RunScript ()
case PCD_ENDPRINTBOLD: case PCD_ENDPRINTBOLD:
case PCD_MOREHUDMESSAGE: case PCD_MOREHUDMESSAGE:
case PCD_ENDLOG: case PCD_ENDLOG:
work = strbin1 (work);
if (pcd == PCD_ENDLOG) if (pcd == PCD_ENDLOG)
{ {
Printf ("%s\n", work.GetChars()); Printf ("%s\n", work.GetChars());
@ -7016,7 +7070,7 @@ int DLevelScript::RunScript ()
case PCD_SAVESTRING: case PCD_SAVESTRING:
// Saves the string // Saves the string
{ {
unsigned int str_otf = ACS_StringsOnTheFly.Push(strbin1(work)); unsigned int str_otf = ACS_StringsOnTheFly.Push(work);
if (str_otf > 0xffff) if (str_otf > 0xffff)
{ {
PushToStack(-1); PushToStack(-1);

View file

@ -220,6 +220,7 @@ private:
static int STACK_ARGS SortScripts (const void *a, const void *b); static int STACK_ARGS SortScripts (const void *a, const void *b);
void UnencryptStrings (); void UnencryptStrings ();
void UnescapeStringTable(BYTE *chunkstart, bool haspadding);
int FindStringInChunk (DWORD *chunk, const char *varname) const; int FindStringInChunk (DWORD *chunk, const char *varname) const;
const char *LookupString (DWORD index) const; const char *LookupString (DWORD index) const;