#if defined _INC_y_amx #endinput #endif #define _INC_y_amx /** * *
* Description *
* Allows a script access to information about itself, such as function names. * This can be used for a range of things, including automatic callback hooking * and testing. *
* Version *
* 1.0 *
*//** *//* Legal: Version: MPL 1.1 The contents of this file are subject to the Mozilla Public License Version 1.1 the "License"; you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.mozilla.org/MPL/ Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the specific language governing rights and limitations under the License. The Original Code is the YSI framework. The Initial Developer of the Original Code is Alex "Y_Less" Cole. Portions created by the Initial Developer are Copyright C 2011 the Initial Developer. All Rights Reserved. Contributors: Y_Less koolk JoeBullet/Google63 g_aSlice/Slice Misiur samphunter tianmeta maddinat0r spacemud Crayder Dayvison Ahmad45123 Zeex irinel1996 Yiin- Chaprnks Konstantinos Masterchen09 Southclaws PatchwerkQWER m0k1 paulommu udan111 Thanks: JoeBullet/Google63 - Handy arbitrary ASM jump code using SCTRL. ZeeX - Very productive conversations. koolk - IsPlayerinAreaEx code. TheAlpha - Danish translation. breadfish - German translation. Fireburn - Dutch translation. yom - French translation. 50p - Polish translation. Zamaroht - Spanish translation. Los - Portuguese translation. Dracoblue, sintax, mabako, Xtreme, other coders - Producing other modes for me to strive to better. Pixels^ - Running XScripters where the idea was born. Matite - Pestering me to release it and using it. Very special thanks to: Thiadmer - PAWN, whose limits continue to amaze me! Kye/Kalcor - SA:MP. SA:MP Team past, present and future - SA:MP. Optional plugins: Gamer_Z - GPS. Incognito - Streamer. Me - sscanf2, fixes2, Whirlpool. */ // I debated porting this over to use ZeeX's advanced AMX system, but there's no // point - this is stable and works well. Doing so would mean testing that all // the code that uses this file still operates corectly given how differently // the two systems do things. I fully admit that his is FAR more general but // there are legacy libraries to consider. #include "..\YSI_Internal\y_version" #include "..\YSI_Server\y_scriptinit" #include "..\YSI_Core\y_debug" #include "..\YSI_Internal\y_natives" #include "..\YSI_Internal\y_thirdpartyinclude" #define AMX_FastString(%1,%2,%3,%4) \ (((%1) << 0) | ((%2) << 8) | ((%3) << 16) | ((%4) << 24)) #define AMX_MEMORY_TRACE_0 0xAABBCCDD #define AMX_MEMORY_TRACE_1 0xDDCCBBAA #define AMX_MEMORY_TRACE_2 0x12345678 #define AMX_MEMORY_TRACE_3 0x87654321 stock AMX_HEADER_SIZE, AMX_HEADER_MAGIC, AMX_HEADER_FILE_VERSION, AMX_HEADER_AMX_VERSION, AMX_HEADER_FLAGS, AMX_HEADER_DEFSIZE, // These are not as they appear in the AMX - they are relative to the dat // pointer so that they can be directly manipulated. AMX_HEADER_COD, AMX_HEADER_DAT, AMX_HEADER_HEA, AMX_HEADER_STP, AMX_HEADER_CIP, AMX_HEADER_PUBLICS, AMX_HEADER_NATIVES, AMX_HEADER_LIBRARIES, AMX_HEADER_PUBVARS, AMX_HEADER_TAGS, AMX_HEADER_NAMETABLE, //E_AMX_HEADER_OVERLAYS, AMX_REAL_ADDRESS, AMX_BASE_ADDRESS, AMX_REAL_DATA; enum E_AMX_TABLE { AMX_TABLE_PUBLICS, AMX_TABLE_NATIVES, AMX_TABLE_LIBRARIES, AMX_TABLE_PUBVARS, AMX_TABLE_TAGS } // This is based on the AMX version used in SA:MP - it DOES NOT match the code // found in the PAWN documentation as that's for a later version. public OnScriptInit() { new addr, data; #emit LCTRL 1 #emit STOR.S.pri addr // Invert to get the prefix offset relative to the data. // Get all the script data. addr = -addr; // Now read in all the data. #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_SIZE = data; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_MAGIC = data & 0xFFFF; AMX_HEADER_FILE_VERSION = data >>> 16 & 0xFF; AMX_HEADER_AMX_VERSION = data >>> 24; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_FLAGS = data & 0xFFFF; AMX_HEADER_DEFSIZE = data >>> 16; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_COD = data; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data //dat = data; AMX_HEADER_DAT = 0; AMX_BASE_ADDRESS = -data; AMX_HEADER_COD += AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_HEA = data + AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_STP = data + AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_CIP = data + AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_PUBLICS = data + AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_NATIVES = data + AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_LIBRARIES = data + AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_PUBVARS = data + AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_TAGS = data + AMX_BASE_ADDRESS; addr += 4; #emit LREF.S.pri addr #emit STOR.S.pri data AMX_HEADER_NAMETABLE = data + AMX_BASE_ADDRESS; // Now find the AMX's base address in global memory. This is VERY handy to // have for more advanced functionality (none of which actually exists yet). AMX_REAL_ADDRESS = AMX_GetGlobal(); AMX_REAL_DATA = AMX_REAL_ADDRESS - AMX_BASE_ADDRESS; // Call next ALS callback. #if defined YSI_LOCK_MODE GetServerVarAsString(YSI_gLockData, YSI_gLockData[5], sizeof (YSI_gLockData) - 5); #endif #if defined AMX_OnScriptInit AMX_OnScriptInit(); #endif return 1; } #undef OnScriptInit #define OnScriptInit AMX_OnScriptInit #if defined AMX_OnScriptInit forward AMX_OnScriptInit(); #endif stock AMX_GetGlobalAddress(...) { new addr; // addr = numargs(); #emit LOAD.S.pri 8 #emit STOR.S.pri addr if (addr >= 4) { // getargptr(0); #emit LOAD.S.pri 12 #emit LOAD.alt AMX_REAL_DATA #emit ADD #emit STACK 4 #emit RETN } return 0; } stock AMX_GetRelativeAddress(...) { new addr; // addr = numargs(); #emit LOAD.S.pri 8 #emit STOR.S.pri addr if (addr >= 4) { // getargptr(0); #emit LOAD.S.pri 12 #emit STACK 4 #emit RETN } return 0; } static AMX_DoNothing() { return 0; } static AMX_GetGlobal() { new addr = -1; // Call dummy function and read its (absolute) address from code. AMX_DoNothing(); #emit LCTRL 6 #emit CONST.alt 12 #emit SUB #emit LOAD.alt AMX_HEADER_COD #emit ADD #emit STOR.S.pri addr #emit LREF.S.pri addr // Get difference between absolute and relative addresses. #emit SUB #emit CONST.alt AMX_DoNothing #emit SUB #emit MOVE.alt #emit LCTRL 1 #emit XCHG #emit SUB // Return #emit STACK 4 #emit RETN return 0; } stock AMX_TraceCode(pattern[], &addrRet, &dataRet, size = sizeof (pattern)) { new addr = AMX_HEADER_COD + addrRet, data, i; while (addr < AMX_HEADER_DAT) { #emit LREF.S.pri addr #emit STOR.S.pri data if (data == pattern[i]) { ++i; addr += 4; if (i == size) { addrRet = addr - i * 4 - AMX_HEADER_COD; #emit LREF.S.pri addr #emit STOR.S.pri data dataRet = data; return 1; } } else if (i) { addr -= i * 4 - 4; i = 0; } else { addr += 4; } } return 0; } stock AMX_TraceMemory(pattern[], &addrRet, &dataRet, size = sizeof (pattern)) { new addr = AMX_HEADER_DAT + addrRet, data, i; while (addr < AMX_HEADER_HEA) { #emit LREF.S.pri addr #emit STOR.S.pri data if (data == pattern[i]) { ++i; addr += 4; if (i == size) { addrRet = addr - i * 4 - AMX_HEADER_DAT; #emit LREF.S.pri addr #emit STOR.S.pri data dataRet = data; return 1; } } else if (i) { addr -= i * 4 - 4; i = 0; } else { addr += 4; } } return 0; } stock AMX_GetBaseCount(E_AMX_TABLE:table, &base, &count) { P:7("AMX_GetBaseCount called: %i, %i, %i", _:table, base, count); switch (table) { case AMX_TABLE_PUBLICS: { base = AMX_HEADER_PUBLICS; count = (AMX_HEADER_NATIVES - AMX_HEADER_PUBLICS) / 8; } case AMX_TABLE_NATIVES: { base = AMX_HEADER_NATIVES; count = (AMX_HEADER_LIBRARIES - AMX_HEADER_NATIVES) / 8; } case AMX_TABLE_LIBRARIES: { base = AMX_HEADER_LIBRARIES; count = (AMX_HEADER_PUBVARS - AMX_HEADER_LIBRARIES) / 8; } case AMX_TABLE_PUBVARS: { base = AMX_HEADER_PUBVARS; count = (AMX_HEADER_TAGS - AMX_HEADER_PUBVARS) / 8; } case AMX_TABLE_TAGS: { base = AMX_HEADER_TAGS; count = (AMX_HEADER_NAMETABLE - AMX_HEADER_TAGS) / 8; } default: { base = 0; count = 0; } } } #define AMX_GetPublicEntry( AMX_GetEntry(AMX_TABLE_PUBLICS, #define AMX_GetNativeEntry( AMX_GetEntry(AMX_TABLE_NATIVES, #define AMX_GetLibraryEntry( AMX_GetEntry(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarEntry( AMX_GetEntry(AMX_TABLE_PUBVARS, #define AMX_GetTagEntry( AMX_GetEntry(AMX_TABLE_TAGS, stock AMX_GetEntry(E_AMX_TABLE:table, idx, &buffer, const pattern[] = "", const bool:exact = false) { P:7("AMX_GetEntry called: %i, %i, %i, \"%s\", %i", _:table, idx, buffer, pattern, exact); new base, count; AMX_GetBaseCount(table, base, count); if (idx < count) { if (pattern[0] == '\0') { buffer = idx * 8 + base; return idx + 1; } else { new addr, pos = idx * 8 + base + 4, str[32]; do { ++idx; #emit LREF.S.pri pos #emit STOR.S.pri addr if (addr) { AMX_ReadString(AMX_BASE_ADDRESS + addr, str); if (exact ? (str{0} && !strcmp(str, pattern)) : (strfind(str, pattern) != -1)) { buffer = pos - 4; return idx; } } pos += 8; } while (idx < count); } } return 0; } #define AMX_GetPublicEntryPrefix( AMX_GetEntryPrefix(AMX_TABLE_PUBLICS, #define AMX_GetNativeEntryPrefix( AMX_GetEntryPrefix(AMX_TABLE_NATIVES, #define AMX_GetLibraryEntryPrefix( AMX_GetEntryPrefix(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarEntryPrefix( AMX_GetEntryPrefix(AMX_TABLE_PUBVARS, #define AMX_GetTagEntryPrefix( AMX_GetEntryPrefix(AMX_TABLE_TAGS, stock AMX_GetEntryPrefix(E_AMX_TABLE:table, idx, &buffer, pattern) { P:7("AMX_GetEntryPrefix called: %i, %i, %i, %i", _:table, idx, buffer, pattern); new base, count, addr; AMX_GetBaseCount(table, base, count); new pos = idx * 8 + base + 4; while (idx < count) { ++idx; // Get the address of the string. #emit LREF.S.pri pos #emit STOR.S.pri addr if (addr) { // Get the 4 bytes 5 bytes before this position. addr += AMX_BASE_ADDRESS; #emit LREF.S.pri addr #emit STOR.S.pri addr if (addr == pattern) { buffer = pos - 4; return idx; } } pos += 8; } return 0; } #define AMX_GetPublicEntrySuffix( AMX_GetEntrySuffix(AMX_TABLE_PUBLICS, #define AMX_GetNativeEntrySuffix( AMX_GetEntrySuffix(AMX_TABLE_NATIVES, #define AMX_GetLibraryEntrySuffix( AMX_GetEntrySuffix(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarEntrySuffix( AMX_GetEntrySuffix(AMX_TABLE_PUBVARS, #define AMX_GetTagEntrySuffix( AMX_GetEntrySuffix(AMX_TABLE_TAGS, stock AMX_GetEntrySuffix(E_AMX_TABLE:table, idx, &buffer, pattern) { P:7("AMX_GetEntrySuffix called: %i, %i, %i, %i", _:table, idx, buffer, pattern); new base, count, addr, ch; AMX_GetBaseCount(table, base, count); new pos = idx * 8 + base + 4; while (idx < count) { ++idx; // Get the address of the string. #emit LREF.S.pri pos #emit STOR.S.pri addr if (addr) { addr += AMX_BASE_ADDRESS; for ( ; ; ) { // Find the end of the string. #emit LREF.S.pri addr #emit STOR.S.pri ch if (ch & 0x000000FF) { if (ch & 0x0000FF00) { if (ch & 0x00FF0000) { if (ch & 0xFF000000) { addr += 4; continue; } else addr -= 1; } else addr -= 2; } else addr -= 3; } else addr -= 4; break; } #emit LREF.S.pri addr #emit STOR.S.pri ch if (ch == pattern) { buffer = pos - 4; return idx; } } pos += 8; } return 0; } #define AMX_GetPublicName( AMX_GetName(AMX_TABLE_PUBLICS, #define AMX_GetNativeName( AMX_GetName(AMX_TABLE_NATIVES, #define AMX_GetLibraryName( AMX_GetName(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarName( AMX_GetName(AMX_TABLE_PUBVARS, #define AMX_GetTagName( AMX_GetName(AMX_TABLE_TAGS, stock AMX_GetName(E_AMX_TABLE:table, idx, buffer[32], const pattern[] = "", const bool:exact = false) { P:7("AMX_GetName called: %i, %i, \"%s\", \"%s\"", _:table, idx, buffer, pattern); new base, count, addr; AMX_GetBaseCount(table, base, count); new pos = idx * 8 + base + 4; while (idx < count) { ++idx; #emit LREF.S.pri pos #emit STOR.S.pri addr if (addr) { AMX_ReadString(AMX_BASE_ADDRESS + addr, buffer); if ((pattern[0] == '\0') || (exact ? (!strcmp(buffer, pattern)) : (strfind(buffer, pattern) != -1))) { return idx; } } pos += 8; } return 0; } #define AMX_GetPublicNamePrefix( AMX_GetNamePrefix(AMX_TABLE_PUBLICS, #define AMX_GetNativeNamePrefix( AMX_GetNamePrefix(AMX_TABLE_NATIVES, #define AMX_GetLibraryNamePrefix( AMX_GetNamePrefix(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarNamePrefix( AMX_GetNamePrefix(AMX_TABLE_PUBVARS, #define AMX_GetTagNamePrefix( AMX_GetNamePrefix(AMX_TABLE_TAGS, stock AMX_GetNamePrefix(E_AMX_TABLE:table, idx, buffer[32], pattern) { P:7("AMX_GetNamePrefix called: %i, %i, \"%s\", %i", _:table, idx, buffer, pattern); new base, count, addr; AMX_GetBaseCount(table, base, count); new pos = idx * 8 + base + 4; while (idx < count) { ++idx; // Get the address of the string. #emit LREF.S.pri pos #emit STOR.S.pri addr if (addr) { // Get the 4 bytes 5 bytes before this position. addr += AMX_BASE_ADDRESS; #emit LREF.S.pri addr #emit STOR.S.pri addr if (addr == pattern) { #emit LREF.S.pri pos #emit STOR.S.pri addr AMX_ReadString(AMX_BASE_ADDRESS + addr, buffer); return idx; } } pos += 8; } return 0; } #define AMX_GetPublicNameSuffix( AMX_GetNameSuffix(AMX_TABLE_PUBLICS, #define AMX_GetNativeNameSuffix( AMX_GetNameSuffix(AMX_TABLE_NATIVES, #define AMX_GetLibraryNameSuffix( AMX_GetNameSuffix(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarNameSuffix( AMX_GetNameSuffix(AMX_TABLE_PUBVARS, #define AMX_GetTagNameSuffix( AMX_GetNameSuffix(AMX_TABLE_TAGS, stock AMX_GetNameSuffix(E_AMX_TABLE:table, idx, buffer[32], pattern) { P:7("AMX_GetNameSuffix called: %i, %i, \"%s\", %i", _:table, idx, buffer, pattern); new base, count, addr, ch; AMX_GetBaseCount(table, base, count); new pos = idx * 8 + base + 4; while (idx < count) { ++idx; #emit LREF.S.pri pos #emit STOR.S.pri addr if (addr) { // We used to do this by using the next entry in the table. // However, we have since started mangling the table at run-time, so // we can no longer use the shortcut as a) the next entries may not // exist (bypassable) and b) the strings are no longer in order (an // unfixable problem). addr += AMX_BASE_ADDRESS; for ( ; ; ) { // Find the end of the string. #emit LREF.S.pri addr #emit STOR.S.pri ch if (ch & 0x000000FF) { if (ch & 0x0000FF00) { if (ch & 0x00FF0000) { if (ch & 0xFF000000) { addr += 4; continue; } else addr -= 1; } else addr -= 2; } else addr -= 3; } else addr -= 4; break; } #emit LREF.S.pri addr #emit STOR.S.pri ch if (ch == pattern) { #emit LREF.S.pri pos #emit STOR.S.pri addr AMX_ReadString(AMX_BASE_ADDRESS + addr, buffer); return idx; } } pos += 8; } return 0; } #define AMX_GetPublicPointer( AMX_GetPointer(AMX_TABLE_PUBLICS, #define AMX_GetNativePointer( AMX_GetPointer(AMX_TABLE_NATIVES, #define AMX_GetLibraryPointer( AMX_GetPointer(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarPointer( AMX_GetPointer(AMX_TABLE_PUBVARS, #define AMX_GetTagPointer( AMX_GetPointer(AMX_TABLE_TAGS, stock AMX_GetPointer(E_AMX_TABLE:table, idx, &buffer, const pattern[] = "", const bool:exact = false) { P:7("AMX_GetPointer called: %i, %i, %i, \"%s\", %i", _:table, idx, buffer, pattern, exact); new pointer; idx = AMX_GetEntry(table, idx, pointer, pattern, exact); if (idx) { #emit LREF.S.pri pointer #emit SREF.S.pri buffer } return idx; } #define AMX_GetPublicPointerPrefix( AMX_GetPointerPrefix(AMX_TABLE_PUBLICS, #define AMX_GetNativePointerPrefix( AMX_GetPointerPrefix(AMX_TABLE_NATIVES, #define AMX_GetLibraryPointerPrefix( AMX_GetPointerPrefix(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarPointerPrefix( AMX_GetPointerPrefix(AMX_TABLE_PUBVARS, #define AMX_GetTagPointerPrefix( AMX_GetPointerPrefix(AMX_TABLE_TAGS, stock AMX_GetPointerPrefix(E_AMX_TABLE:table, idx, &buffer, pattern) { P:7("AMX_GetPointerPrefix called: %i, %i, %i, %i", _:table, idx, buffer, pattern); new pointer; idx = AMX_GetEntryPrefix(table, idx, pointer, pattern); if (idx) { #emit LREF.S.pri pointer #emit SREF.S.pri buffer } return idx; } #define AMX_GetPublicPointerSuffix( AMX_GetPointerSuffix(AMX_TABLE_PUBLICS, #define AMX_GetNativePointerSuffix( AMX_GetPointerSuffix(AMX_TABLE_NATIVES, #define AMX_GetLibraryPointerSuffix( AMX_GetPointerSuffix(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarPointerSuffix( AMX_GetPointerSuffix(AMX_TABLE_PUBVARS, #define AMX_GetTagPointerSuffix( AMX_GetPointerSuffix(AMX_TABLE_TAGS, stock AMX_GetPointerSuffix(E_AMX_TABLE:table, idx, &buffer, pattern) { P:7("AMX_GetPointerSuffix called: %i, %i, %i, %i", _:table, idx, buffer, pattern); new pointer; idx = AMX_GetEntrySuffix(table, idx, pointer, pattern); if (idx) { #emit LREF.S.pri pointer #emit SREF.S.pri buffer } return idx; } #define AMX_GetPublicValue( AMX_GetValue(AMX_TABLE_PUBLICS, #define AMX_GetNativeValue( AMX_GetValue(AMX_TABLE_NATIVES, #define AMX_GetLibraryValue( AMX_GetValue(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarValue( AMX_GetValue(AMX_TABLE_PUBVARS, #define AMX_GetTagValue( AMX_GetValue(AMX_TABLE_TAGS, stock AMX_GetValue(E_AMX_TABLE:table, idx, &buffer, const pattern[] = "", const bool:exact = false) { P:7("AMX_GetValue called: %i, %i, %i, \"%s\", %i", _:table, idx, buffer, pattern, exact); new pointer; idx = AMX_GetPointer(table, idx, pointer, pattern, exact); if (idx) { #emit LREF.S.pri pointer #emit SREF.S.pri buffer } return idx; } #define AMX_GetPublicValuePrefix( AMX_GetValuePrefix(AMX_TABLE_PUBLICS, #define AMX_GetNativeValuePrefix( AMX_GetValuePrefix(AMX_TABLE_NATIVES, #define AMX_GetLibraryValuePrefix( AMX_GetValuePrefix(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarValuePrefix( AMX_GetValuePrefix(AMX_TABLE_PUBVARS, #define AMX_GetTagValuePrefix( AMX_GetValuePrefix(AMX_TABLE_TAGS, stock AMX_GetValuePrefix(E_AMX_TABLE:table, idx, &buffer, pattern) { P:7("AMX_GetValuePrefix called: %i, %i, %i, %i", _:table, idx, buffer, pattern); new pointer; idx = AMX_GetPointerPrefix(table, idx, pointer, pattern); if (idx) { #emit LREF.S.pri pointer #emit SREF.S.pri buffer } return idx; } #define AMX_GetPublicValueSuffix( AMX_GetValueSuffix(AMX_TABLE_PUBLICS, #define AMX_GetNativeValueSuffix( AMX_GetValueSuffix(AMX_TABLE_NATIVES, #define AMX_GetLibraryValueSuffix( AMX_GetValueSuffix(AMX_TABLE_LIBRARIES, #define AMX_GetPubvarValueSuffix( AMX_GetValueSuffix(AMX_TABLE_PUBVARS, #define AMX_GetTagValueSuffix( AMX_GetValueSuffix(AMX_TABLE_TAGS, stock AMX_GetValueSuffix(E_AMX_TABLE:table, idx, &buffer, pattern) { P:7("AMX_GetValueSuffix called: %i, %i, %i, %i", _:table, idx, buffer, pattern); new pointer; idx = AMX_GetPointerSuffix(table, idx, pointer, pattern); if (idx) { #emit LREF.S.pri pointer #emit SREF.S.pri buffer } return idx; } /*enum e_AMX_HEADER_TYPE (<<= 1) { e_AMX_HEADER_TYPE_NONE = 0, e_AMX_HEADER_TYPE_PUBLIC = 1, e_AMX_HEADER_TYPE_NATIVE, e_AMX_HEADER_TYPE_LIBRARY, e_AMX_HEADER_TYPE_PUBVAR, e_AMX_HEADER_TYPE_TAG } enum E_AMX_HEADER { e_AMX_HEADER_TYPE:E_AMX_HEADER_TYPE, // Public, native, tag, etc. E_AMX_HEADER_ENTRY, E_AMX_HEADER_PTR, // For variables, this is their value. For publics and natives, this is // their table index (used by "funcidx" and "SYSREQ.C" respectively). E_AMX_HEADER_VALUE, E_AMX_HEADER_NAME[32 char] } stock AMX_GetEntryData(entry, ret[E_AMX_HEADER]) { if ((ret[e_AMX_HEADER_TYPE] = AMX_GetEntryType(entry))) { } }*/ stock AMX_GetEntryPointer(entry) { #emit LREF.S.pri entry #emit RETN return 0; // Make the compiler happy. } /*stock bool:AMX_GetData(const name[], ret[E_AMX_HEADER], e_AMX_HEADER_TYPE:type = e_AMX_HEADER_TYPE_NONE) { new entry; switch (type) { case e_AMX_HEADER_TYPE_NONE : { if (AMX_GetEntry(AMX_TABLE_PUBLICS, 0, entry, name)) type |= e_AMX_HEADER_TYPE_PUBLIC ; if (AMX_GetEntry(AMX_TABLE_NATIVES, 0, entry, name)) type |= e_AMX_HEADER_TYPE_NATIVE ; if (AMX_GetEntry(AMX_TABLE_LIBRARIES, 0, entry, name)) type |= e_AMX_HEADER_TYPE_LIBRARY; if (AMX_GetEntry(AMX_TABLE_PUBVARS, 0, entry, name)) type |= e_AMX_HEADER_TYPE_PUBVAR ; if (AMX_GetEntry(AMX_TABLE_TAGS, 0, entry, name)) type |= e_AMX_HEADER_TYPE_TAG ; } case e_AMX_HEADER_TYPE_PUBLIC : AMX_GetEntry(AMX_TABLE_PUBLICS, 0, entry, name); case e_AMX_HEADER_TYPE_NATIVE : AMX_GetEntry(AMX_TABLE_NATIVES, 0, entry, name); case e_AMX_HEADER_TYPE_LIBRARY: AMX_GetEntry(AMX_TABLE_LIBRARIES, 0, entry, name); case e_AMX_HEADER_TYPE_PUBVAR : AMX_GetEntry(AMX_TABLE_PUBVARS, 0, entry, name); case e_AMX_HEADER_TYPE_TAG : AMX_GetEntry(AMX_TABLE_TAGS, 0, entry, name); } strpack(ret[E_AMX_HEADER_NAME], name), ret[E_AMX_HEADER_TYPE] = type; new ptr; #emit LREF.S.pri entry #emit SREF.S.pri ptr switch (type) { case e_AMX_HEADER_TYPE_PUBLIC : { // Has a pointer. return ret[E_AMX_HEADER_ENTRY] = entry, ret[E_AMX_HEADER_PTR] = ptr, ret[E_AMX_HEADER_VALUE] = AMX_GetPublicIndexFromEntry(entry); } case e_AMX_HEADER_TYPE_NATIVE : { // Has a pointer. return ret[E_AMX_HEADER_ENTRY] = entry, ret[E_AMX_HEADER_PTR] = ptr, ret[E_AMX_HEADER_VALUE] = AMX_GetNativeIndexFromEntry(entry); } case e_AMX_HEADER_TYPE_LIBRARY: { } case e_AMX_HEADER_TYPE_PUBVAR : { } case e_AMX_HEADER_TYPE_TAG : { } } // Multiple matches found, or none. return ret[E_AMX_HEADER_INDEX] = -1, ret[E_AMX_HEADER_PTR] = 0, ret[E_AMX_HEADER_VALUE] = cellmin; }*/ stock AMX_GetEntryFromNativeIndex(index) { return (index * 8) + AMX_HEADER_NATIVES; } stock AMX_GetEntryFromPublicIndex(index) { // An index is an offset in to their own table. return (index * 8) + AMX_HEADER_PUBLICS; } stock AMX_GetNativeIndexFromEntry(entry) { return (entry - AMX_HEADER_NATIVES) / 8; } stock AMX_GetPublicIndexFromEntry(entry) { return (entry - AMX_HEADER_PUBLICS) / 8; } stock AMX_GetStringFromEntry(entry, str[], size = sizeof (str)) { entry += 4; #emit LREF.S.pri entry #emit STOR.S.pri entry AMX_ReadString(AMX_BASE_ADDRESS + entry, str, size); } /*stock AMX_GetEntryType(entry) { if (AMX_HEADER_PUBLICS <= entry < AMX_HEADER_NATIVES) return e_AMX_HEADER_TYPE_PUBLIC; else if (AMX_HEADER_NATIVES <= entry < AMX_HEADER_LIBRARIES) return e_AMX_HEADER_TYPE_NATIVE; else if (AMX_HEADER_LIBRARIES <= entry < AMX_HEADER_PUBVARS) return e_AMX_HEADER_TYPE_LIBRARY; else if (AMX_HEADER_PUBVARS <= entry < AMX_HEADER_TAGS) return e_AMX_HEADER_TYPE_PUBVAR; else if (AMX_HEADER_TAGS <= entry < AMX_HEADER_NAMETABLE) return e_AMX_HEADER_TYPE_TAG; return e_AMX_HEADER_TYPE_NONE; } stock e_AMX_HEADER_TYPE:AMX_GetType(const name[]) { new e_AMX_HEADER_TYPE:type = e_AMX_HEADER_TYPE_NONE, entry; if (AMX_GetEntry(AMX_TABLE_PUBLICS, 0, entry, name)) type |= e_AMX_HEADER_TYPE_PUBLIC ; if (AMX_GetEntry(AMX_TABLE_NATIVES, 0, entry, name)) type |= e_AMX_HEADER_TYPE_NATIVE ; if (AMX_GetEntry(AMX_TABLE_LIBRARIES, 0, entry, name)) type |= e_AMX_HEADER_TYPE_LIBRARY; if (AMX_GetEntry(AMX_TABLE_PUBVARS, 0, entry, name)) type |= e_AMX_HEADER_TYPE_PUBVAR ; if (AMX_GetEntry(AMX_TABLE_TAGS, 0, entry, name)) type |= e_AMX_HEADER_TYPE_TAG ; return type; }*/ stock AMX_ReadString(addr, str[], len = sizeof (str)) { new buffer, idx; do { // Read 4 bytes. #emit LREF.S.pri addr #emit STOR.S.pri buffer // Write PACKED strings. buffer = (buffer >>> 24) | (buffer >> 8 & 0x0000FF00) | (buffer << 8 & 0x00FF0000) | (buffer << 24); str[idx] = buffer; if (!(buffer & 0x000000FF && buffer & 0x0000FF00 && buffer & 0x00FF0000 && buffer & 0xFF000000)) { return; } addr += 4; } while (++idx < len); } stock AMX_ReadUnpackedString(addr, str[], len = sizeof (str)) { new buffer = 1; while (buffer && len--); { // Read 4 bytes. #emit LREF.S.pri addr #emit STOR.S.pri buffer #emit SREF.S.pri str #emit LOAD.S.pri str #emit ADD.C 4 #emit STOR.S.pri str addr += 4; } } stock AMX_WriteString(addr, const str[], len = sizeof (str)) { new old, buffer, idx; do { buffer = str[idx]; P:7("AMX_WriteString: Writing %04x%04x", buffer >>> 16, buffer & 0xFFFF); if (!(buffer & 0xFF000000)) { #emit LREF.S.pri addr #emit CONST.alt 0xFFFFFF00 #emit AND #emit SREF.S.pri addr return; } else if (!(buffer & 0x00FF0000)) { // This: // '\0', '1'. // Becomes: // '1', '\0', ???, ???. #emit LREF.S.pri addr #emit STOR.S.pri old buffer = (old & 0xFFFF0000) | (buffer >>> 24); #emit LOAD.S.pri buffer #emit SREF.S.pri addr return; } else if (!(buffer & 0x0000FF00)) { // This: // '\0', '2', '1'. // Becomes: // '1', '2', '\0', ???. #emit LREF.S.pri addr #emit STOR.S.pri old buffer = (old & 0xFF000000) | (buffer >>> 24) | (buffer >> 8 & 0x0000FF00); #emit LOAD.S.pri buffer #emit SREF.S.pri addr return; } else if (!(buffer & 0x000000FF)) { // This: // '\0', '3', '2', '1'. // Becomes: // '1', '2', '3', '\0'. // Write 3 bytes. buffer = (buffer >>> 24) | (buffer >> 8 & 0x0000FF00) | (buffer << 8 & 0x00FF0000); #emit LOAD.S.pri buffer #emit SREF.S.pri addr return; } else { // Write 4 bytes. buffer = (buffer >>> 24) | (buffer >> 8 & 0x0000FF00) | (buffer << 8 & 0x00FF0000) | (buffer << 24); #emit LOAD.S.pri buffer #emit SREF.S.pri addr addr += 4; } } while (++idx < len); } stock AMX_Read(addr) { // This is now, AFAIK, the first ever self-inlining function! static sFrame; // Get the current stack pointer. #emit LCTRL 4 #emit ADD.C 12 // Get the previous frame pointer. #emit LOAD.S.alt 0 #emit SUB #emit STOR.pri sFrame P:7("AMX_Read called: %i %d", addr, sFrame); new start = GetCurrentFrameReturn() - 16, ctx2[DisasmContext]; DisasmInit(ctx2, start, start + 16); if (!DisasmDecodeInsn(ctx2) || DisasmGetOpcode(ctx2) != OP_PUSH_C || DisasmGetOperand(ctx2) != 4) { P:W("Couldn't rewrite \"AMX_Read\" call"); return AMX_RawRead(addr); } new ctx[AsmContext]; AsmInitPtr(ctx, start + AMX_HEADER_COD, 16); // This is the actual code. The standard function call puts the variable to // read from on to the stack, so use that as a frame offset and call "LREF" // on that stack value. Then remove it from the stack and skip the next // cell - since we are replacing 4 cells with just 3. @emit LREF.S.pri sFrame @emit POP.alt @emit NOP return AMX_RawRead(addr); } stock AMX_Write(addr, value) { P:7("AMX_Write called: %i, %i", addr, value); #emit LOAD.S.pri value #emit SREF.S.pri addr } stock AMX_RawRead(addr) { P:7("AMX_RawRead called: %i", addr); #emit LREF.S.pri addr #emit RETN return 0; } stock AMX_RawWrite(addr, value) { P:7("AMX_RawWrite called: %i, %i", addr, value); #emit LOAD.S.pri value #emit SREF.S.pri addr } stock AMX_ReadArray(addr, dest[], len = sizeof (dest)) { // I tried to use memcpy, I couldn't get it to work, even when exactly // replicating compiler generated code... while (len--) { // Load the address possibly outside "dat". Can't be done using only // "pri"/"alt" as it relies on "LREF.S" explicitly. #emit LREF.S.pri addr #emit SREF.S.pri dest #emit LOAD.S.pri addr #emit ADD.C 4 #emit STOR.S.pri addr #emit LOAD.S.pri dest #emit ADD.C 4 #emit STOR.S.pri dest } } stock AMX_WriteArray(addr, const src[], len = sizeof (src)) { while (len--) { #emit DEC.pri // Read the data. #emit LREF.S.pri dest #emit SREF.S.pri addr #emit LOAD.S.pri addr #emit ADD.C 4 #emit STOR.S.pri addr #emit LOAD.S.pri dest #emit ADD.C 4 #emit STOR.S.pri dest } } //stock AMX_CodToDat(ptr) //{ // // Convert a pointer in to the code segment in to a data offset. // return ptr + AMX_HEADER_COD; //} #define AMX_CodToDat(%0) ((%0) + AMX_HEADER_COD) #define AMX_DatToCod(%0) ((%0) - AMX_HEADER_COD) #define _A<%0> (_:H_Re(%0,0)) // Do the next character test as this one failed. #define H_Se(%0,%1,%3) H_Ne%1(%0,%3) // End of string test failed. #define H_Ee(%0,%3) @E_:H_Se(%0,_,%3) // Do the single addition. #define H_De(%0,%1,%3) (_:H_Re(%0,%3+8))|%1<<%3 // Recurse through the string. #define H_Re(%0,%3) he:H_Ee(%0,%3) // Test for the end of a string (4 characters only). #define he:H_Ee(%0,0+8+8+8+8) 0 // Test for the current character. #define @E_:H_Se(_%0,%1,%3) H_De(%0,95,%3) #define @E@:H_Se(@%0,%1,%3) H_De(%0,64,%3) #define @Ey:H_Se(y%0,%1,%3) H_De(%0,121,%3) #define @Ea:H_Se(a%0,%1,%3) H_De(%0,97,%3) #define @Eb:H_Se(b%0,%1,%3) H_De(%0,98,%3) #define @Ec:H_Se(c%0,%1,%3) H_De(%0,99,%3) #define @Ed:H_Se(d%0,%1,%3) H_De(%0,100,%3) #define @Ee:H_Se(e%0,%1,%3) H_De(%0,101,%3) #define @Ef:H_Se(f%0,%1,%3) H_De(%0,102,%3) #define @Eg:H_Se(g%0,%1,%3) H_De(%0,103,%3) #define @Eh:H_Se(h%0,%1,%3) H_De(%0,104,%3) #define @Ei:H_Se(i%0,%1,%3) H_De(%0,105,%3) #define @Ej:H_Se(j%0,%1,%3) H_De(%0,106,%3) #define @Ek:H_Se(k%0,%1,%3) H_De(%0,107,%3) #define @El:H_Se(l%0,%1,%3) H_De(%0,108,%3) #define @Em:H_Se(m%0,%1,%3) H_De(%0,109,%3) #define @En:H_Se(n%0,%1,%3) H_De(%0,110,%3) #define @Eo:H_Se(o%0,%1,%3) H_De(%0,111,%3) #define @Ep:H_Se(p%0,%1,%3) H_De(%0,112,%3) #define @Eq:H_Se(q%0,%1,%3) H_De(%0,113,%3) #define @Er:H_Se(r%0,%1,%3) H_De(%0,114,%3) #define @Es:H_Se(s%0,%1,%3) H_De(%0,115,%3) #define @Et:H_Se(t%0,%1,%3) H_De(%0,116,%3) #define @Eu:H_Se(u%0,%1,%3) H_De(%0,117,%3) #define @Ev:H_Se(v%0,%1,%3) H_De(%0,118,%3) #define @Ew:H_Se(w%0,%1,%3) H_De(%0,119,%3) #define @Ex:H_Se(x%0,%1,%3) H_De(%0,120,%3) #define @Ez:H_Se(z%0,%1,%3) H_De(%0,122,%3) #define @EA:H_Se(A%0,%1,%3) H_De(%0,65,%3) #define @EB:H_Se(B%0,%1,%3) H_De(%0,66,%3) #define @EC:H_Se(C%0,%1,%3) H_De(%0,67,%3) #define @ED:H_Se(D%0,%1,%3) H_De(%0,68,%3) #define @EE:H_Se(E%0,%1,%3) H_De(%0,69,%3) #define @EF:H_Se(F%0,%1,%3) H_De(%0,70,%3) #define @EG:H_Se(G%0,%1,%3) H_De(%0,71,%3) #define @EH:H_Se(H%0,%1,%3) H_De(%0,72,%3) #define @EI:H_Se(I%0,%1,%3) H_De(%0,73,%3) #define @EJ:H_Se(J%0,%1,%3) H_De(%0,74,%3) #define @EK:H_Se(K%0,%1,%3) H_De(%0,75,%3) #define @EL:H_Se(L%0,%1,%3) H_De(%0,76,%3) #define @EM:H_Se(M%0,%1,%3) H_De(%0,77,%3) #define @EN:H_Se(N%0,%1,%3) H_De(%0,78,%3) #define @EO:H_Se(O%0,%1,%3) H_De(%0,79,%3) #define @EP:H_Se(P%0,%1,%3) H_De(%0,80,%3) #define @EQ:H_Se(Q%0,%1,%3) H_De(%0,81,%3) #define @ER:H_Se(R%0,%1,%3) H_De(%0,82,%3) #define @ES:H_Se(S%0,%1,%3) H_De(%0,83,%3) #define @ET:H_Se(T%0,%1,%3) H_De(%0,84,%3) #define @EU:H_Se(U%0,%1,%3) H_De(%0,85,%3) #define @EV:H_Se(V%0,%1,%3) H_De(%0,86,%3) #define @EW:H_Se(W%0,%1,%3) H_De(%0,87,%3) #define @EX:H_Se(X%0,%1,%3) H_De(%0,88,%3) #define @EY:H_Se(Y%0,%1,%3) H_De(%0,89,%3) #define @EZ:H_Se(Z%0,%1,%3) H_De(%0,90,%3) #define @E0:H_Se(0%0,%1,%3) H_De(%0,48,%3) #define @E1:H_Se(1%0,%1,%3) H_De(%0,49,%3) #define @E2:H_Se(2%0,%1,%3) H_De(%0,50,%3) #define @E3:H_Se(3%0,%1,%3) H_De(%0,51,%3) #define @E4:H_Se(4%0,%1,%3) H_De(%0,52,%3) #define @E5:H_Se(5%0,%1,%3) H_De(%0,53,%3) #define @E6:H_Se(6%0,%1,%3) H_De(%0,54,%3) #define @E7:H_Se(7%0,%1,%3) H_De(%0,55,%3) #define @E8:H_Se(8%0,%1,%3) H_De(%0,56,%3) #define @E9:H_Se(9%0,%1,%3) H_De(%0,57,%3) // Find the next character to test. #define H_Ne_(%0,%3) @E@:H_Se(%0,@,%3) #define H_Ne@(%0,%3) @Ey:H_Se(%0,y,%3) #define H_Ney(%0,%3) @Ea:H_Se(%0,a,%3) #define H_Nea(%0,%3) @Eb:H_Se(%0,b,%3) #define H_Neb(%0,%3) @Ec:H_Se(%0,c,%3) #define H_Nec(%0,%3) @Ed:H_Se(%0,d,%3) #define H_Ned(%0,%3) @Ee:H_Se(%0,e,%3) #define H_Nee(%0,%3) @Ef:H_Se(%0,f,%3) #define H_Nef(%0,%3) @Eg:H_Se(%0,g,%3) #define H_Neg(%0,%3) @Eh:H_Se(%0,h,%3) #define H_Neh(%0,%3) @Ei:H_Se(%0,i,%3) #define H_Nei(%0,%3) @Ej:H_Se(%0,j,%3) #define H_Nej(%0,%3) @Ek:H_Se(%0,k,%3) #define H_Nek(%0,%3) @El:H_Se(%0,l,%3) #define H_Nel(%0,%3) @Em:H_Se(%0,m,%3) #define H_Nem(%0,%3) @En:H_Se(%0,n,%3) #define H_Nen(%0,%3) @Eo:H_Se(%0,o,%3) #define H_Neo(%0,%3) @Ep:H_Se(%0,p,%3) #define H_Nep(%0,%3) @Eq:H_Se(%0,q,%3) #define H_Neq(%0,%3) @Er:H_Se(%0,r,%3) #define H_Ner(%0,%3) @Es:H_Se(%0,s,%3) #define H_Nes(%0,%3) @Et:H_Se(%0,t,%3) #define H_Net(%0,%3) @Eu:H_Se(%0,u,%3) #define H_Neu(%0,%3) @Ev:H_Se(%0,v,%3) #define H_Nev(%0,%3) @Ew:H_Se(%0,w,%3) #define H_New(%0,%3) @Ex:H_Se(%0,x,%3) #define H_Nex(%0,%3) @Ez:H_Se(%0,z,%3) #define H_Nez(%0,%3) @EA:H_Se(%0,A,%3) #define H_NeA(%0,%3) @EB:H_Se(%0,B,%3) #define H_NeB(%0,%3) @EC:H_Se(%0,C,%3) #define H_NeC(%0,%3) @ED:H_Se(%0,D,%3) #define H_NeD(%0,%3) @EE:H_Se(%0,E,%3) #define H_NeE(%0,%3) @EF:H_Se(%0,F,%3) #define H_NeF(%0,%3) @EG:H_Se(%0,G,%3) #define H_NeG(%0,%3) @EH:H_Se(%0,H,%3) #define H_NeH(%0,%3) @EI:H_Se(%0,I,%3) #define H_NeI(%0,%3) @EJ:H_Se(%0,J,%3) #define H_NeJ(%0,%3) @EK:H_Se(%0,K,%3) #define H_NeK(%0,%3) @EL:H_Se(%0,L,%3) #define H_NeL(%0,%3) @EM:H_Se(%0,M,%3) #define H_NeM(%0,%3) @EN:H_Se(%0,N,%3) #define H_NeN(%0,%3) @EO:H_Se(%0,O,%3) #define H_NeO(%0,%3) @EP:H_Se(%0,P,%3) #define H_NeP(%0,%3) @EQ:H_Se(%0,Q,%3) #define H_NeQ(%0,%3) @ER:H_Se(%0,R,%3) #define H_NeR(%0,%3) @ES:H_Se(%0,S,%3) #define H_NeS(%0,%3) @ET:H_Se(%0,T,%3) #define H_NeT(%0,%3) @EU:H_Se(%0,U,%3) #define H_NeU(%0,%3) @EV:H_Se(%0,V,%3) #define H_NeV(%0,%3) @EW:H_Se(%0,W,%3) #define H_NeW(%0,%3) @EX:H_Se(%0,X,%3) #define H_NeX(%0,%3) @EY:H_Se(%0,Y,%3) #define H_NeY(%0,%3) @EZ:H_Se(%0,Z,%3) #define H_NeZ(%0,%3) @E0:H_Se(%0,0,%3) #define H_Ne0(%0,%3) @E1:H_Se(%0,1,%3) #define H_Ne1(%0,%3) @E2:H_Se(%0,2,%3) #define H_Ne2(%0,%3) @E3:H_Se(%0,3,%3) #define H_Ne3(%0,%3) @E4:H_Se(%0,4,%3) #define H_Ne4(%0,%3) @E5:H_Se(%0,5,%3) #define H_Ne5(%0,%3) @E6:H_Se(%0,6,%3) #define H_Ne6(%0,%3) @E7:H_Se(%0,7,%3) #define H_Ne7(%0,%3) @E8:H_Se(%0,8,%3) #define H_Ne8(%0,%3) @E9:H_Se(%0,9,%3) #define H_Ne9(%0,%3) () #define _C<%0> (_:H_Rc(%0,24)) // Do the next character test as this one failed. #define H_Sc(%0,%1,%3) H_Nc%1(%0,%3) // End of string test failed. #define H_Ec(%0,%3) @C_:H_Sc(%0,_,%3) // Do the single addition. #define H_Dc(%0,%1,%3) (_:H_Rc(%0,%3-8))|%1<<%3 // Recurse through the string. #define H_Rc(%0,%3) hc:H_Ec(%0,%3) // Test for the end of a string (4 characters only). #define hc:H_Ec(%0,24-8-8-8-8) 0 // Test for the current character. #define @C_:H_Sc(_%0,%1,%3) H_Dc(%0,95,%3) #define @C@:H_Sc(@%0,%1,%3) H_Dc(%0,64,%3) #define @Cy:H_Sc(y%0,%1,%3) H_Dc(%0,121,%3) #define @Ca:H_Sc(a%0,%1,%3) H_Dc(%0,97,%3) #define @Cb:H_Sc(b%0,%1,%3) H_Dc(%0,98,%3) #define @Cc:H_Sc(c%0,%1,%3) H_Dc(%0,99,%3) #define @Cd:H_Sc(d%0,%1,%3) H_Dc(%0,100,%3) #define @Ce:H_Sc(e%0,%1,%3) H_Dc(%0,101,%3) #define @Cf:H_Sc(f%0,%1,%3) H_Dc(%0,102,%3) #define @Cg:H_Sc(g%0,%1,%3) H_Dc(%0,103,%3) #define @Ch:H_Sc(h%0,%1,%3) H_Dc(%0,104,%3) #define @Ci:H_Sc(i%0,%1,%3) H_Dc(%0,105,%3) #define @Cj:H_Sc(j%0,%1,%3) H_Dc(%0,106,%3) #define @Ck:H_Sc(k%0,%1,%3) H_Dc(%0,107,%3) #define @Cl:H_Sc(l%0,%1,%3) H_Dc(%0,108,%3) #define @Cm:H_Sc(m%0,%1,%3) H_Dc(%0,109,%3) #define @Cn:H_Sc(n%0,%1,%3) H_Dc(%0,110,%3) #define @Co:H_Sc(o%0,%1,%3) H_Dc(%0,111,%3) #define @Cp:H_Sc(p%0,%1,%3) H_Dc(%0,112,%3) #define @Cq:H_Sc(q%0,%1,%3) H_Dc(%0,113,%3) #define @Cr:H_Sc(r%0,%1,%3) H_Dc(%0,114,%3) #define @Cs:H_Sc(s%0,%1,%3) H_Dc(%0,115,%3) #define @Ct:H_Sc(t%0,%1,%3) H_Dc(%0,116,%3) #define @Cu:H_Sc(u%0,%1,%3) H_Dc(%0,117,%3) #define @Cv:H_Sc(v%0,%1,%3) H_Dc(%0,118,%3) #define @Cw:H_Sc(w%0,%1,%3) H_Dc(%0,119,%3) #define @Cx:H_Sc(x%0,%1,%3) H_Dc(%0,120,%3) #define @Cz:H_Sc(z%0,%1,%3) H_Dc(%0,122,%3) #define @CA:H_Sc(A%0,%1,%3) H_Dc(%0,65,%3) #define @CB:H_Sc(B%0,%1,%3) H_Dc(%0,66,%3) #define @CC:H_Sc(C%0,%1,%3) H_Dc(%0,67,%3) #define @CD:H_Sc(D%0,%1,%3) H_Dc(%0,68,%3) #define @CE:H_Sc(E%0,%1,%3) H_Dc(%0,69,%3) #define @CF:H_Sc(F%0,%1,%3) H_Dc(%0,70,%3) #define @CG:H_Sc(G%0,%1,%3) H_Dc(%0,71,%3) #define @CH:H_Sc(H%0,%1,%3) H_Dc(%0,72,%3) #define @CI:H_Sc(I%0,%1,%3) H_Dc(%0,73,%3) #define @CJ:H_Sc(J%0,%1,%3) H_Dc(%0,74,%3) #define @CK:H_Sc(K%0,%1,%3) H_Dc(%0,75,%3) #define @CL:H_Sc(L%0,%1,%3) H_Dc(%0,76,%3) #define @CM:H_Sc(M%0,%1,%3) H_Dc(%0,77,%3) #define @CN:H_Sc(N%0,%1,%3) H_Dc(%0,78,%3) #define @CO:H_Sc(O%0,%1,%3) H_Dc(%0,79,%3) #define @CP:H_Sc(P%0,%1,%3) H_Dc(%0,80,%3) #define @CQ:H_Sc(Q%0,%1,%3) H_Dc(%0,81,%3) #define @CR:H_Sc(R%0,%1,%3) H_Dc(%0,82,%3) #define @CS:H_Sc(S%0,%1,%3) H_Dc(%0,83,%3) #define @CT:H_Sc(T%0,%1,%3) H_Dc(%0,84,%3) #define @CU:H_Sc(U%0,%1,%3) H_Dc(%0,85,%3) #define @CV:H_Sc(V%0,%1,%3) H_Dc(%0,86,%3) #define @CW:H_Sc(W%0,%1,%3) H_Dc(%0,87,%3) #define @CX:H_Sc(X%0,%1,%3) H_Dc(%0,88,%3) #define @CY:H_Sc(Y%0,%1,%3) H_Dc(%0,89,%3) #define @CZ:H_Sc(Z%0,%1,%3) H_Dc(%0,90,%3) #define @C0:H_Sc(0%0,%1,%3) H_Dc(%0,48,%3) #define @C1:H_Sc(1%0,%1,%3) H_Dc(%0,49,%3) #define @C2:H_Sc(2%0,%1,%3) H_Dc(%0,50,%3) #define @C3:H_Sc(3%0,%1,%3) H_Dc(%0,51,%3) #define @C4:H_Sc(4%0,%1,%3) H_Dc(%0,52,%3) #define @C5:H_Sc(5%0,%1,%3) H_Dc(%0,53,%3) #define @C6:H_Sc(6%0,%1,%3) H_Dc(%0,54,%3) #define @C7:H_Sc(7%0,%1,%3) H_Dc(%0,55,%3) #define @C8:H_Sc(8%0,%1,%3) H_Dc(%0,56,%3) #define @C9:H_Sc(9%0,%1,%3) H_Dc(%0,57,%3) // Find the next character to test. #define H_Nc_(%0,%3) @C@:H_Sc(%0,@,%3) #define H_Nc@(%0,%3) @Cy:H_Sc(%0,y,%3) #define H_Ncy(%0,%3) @Ca:H_Sc(%0,a,%3) #define H_Nca(%0,%3) @Cb:H_Sc(%0,b,%3) #define H_Ncb(%0,%3) @Cc:H_Sc(%0,c,%3) #define H_Ncc(%0,%3) @Cd:H_Sc(%0,d,%3) #define H_Ncd(%0,%3) @Ce:H_Sc(%0,e,%3) #define H_Nce(%0,%3) @Cf:H_Sc(%0,f,%3) #define H_Ncf(%0,%3) @Cg:H_Sc(%0,g,%3) #define H_Ncg(%0,%3) @Ch:H_Sc(%0,h,%3) #define H_Nch(%0,%3) @Ci:H_Sc(%0,i,%3) #define H_Nci(%0,%3) @Cj:H_Sc(%0,j,%3) #define H_Ncj(%0,%3) @Ck:H_Sc(%0,k,%3) #define H_Nck(%0,%3) @Cl:H_Sc(%0,l,%3) #define H_Ncl(%0,%3) @Cm:H_Sc(%0,m,%3) #define H_Ncm(%0,%3) @Cn:H_Sc(%0,n,%3) #define H_Ncn(%0,%3) @Co:H_Sc(%0,o,%3) #define H_Nco(%0,%3) @Cp:H_Sc(%0,p,%3) #define H_Ncp(%0,%3) @Cq:H_Sc(%0,q,%3) #define H_Ncq(%0,%3) @Cr:H_Sc(%0,r,%3) #define H_Ncr(%0,%3) @Cs:H_Sc(%0,s,%3) #define H_Ncs(%0,%3) @Ct:H_Sc(%0,t,%3) #define H_Nct(%0,%3) @Cu:H_Sc(%0,u,%3) #define H_Ncu(%0,%3) @Cv:H_Sc(%0,v,%3) #define H_Ncv(%0,%3) @Cw:H_Sc(%0,w,%3) #define H_Ncw(%0,%3) @Cx:H_Sc(%0,x,%3) #define H_Ncx(%0,%3) @Cz:H_Sc(%0,z,%3) #define H_Ncz(%0,%3) @CA:H_Sc(%0,A,%3) #define H_NcA(%0,%3) @CB:H_Sc(%0,B,%3) #define H_NcB(%0,%3) @CC:H_Sc(%0,C,%3) #define H_NcC(%0,%3) @CD:H_Sc(%0,D,%3) #define H_NcD(%0,%3) @CE:H_Sc(%0,E,%3) #define H_NcE(%0,%3) @CF:H_Sc(%0,F,%3) #define H_NcF(%0,%3) @CG:H_Sc(%0,G,%3) #define H_NcG(%0,%3) @CH:H_Sc(%0,H,%3) #define H_NcH(%0,%3) @CI:H_Sc(%0,I,%3) #define H_NcI(%0,%3) @CJ:H_Sc(%0,J,%3) #define H_NcJ(%0,%3) @CK:H_Sc(%0,K,%3) #define H_NcK(%0,%3) @CL:H_Sc(%0,L,%3) #define H_NcL(%0,%3) @CM:H_Sc(%0,M,%3) #define H_NcM(%0,%3) @CN:H_Sc(%0,N,%3) #define H_NcN(%0,%3) @CO:H_Sc(%0,O,%3) #define H_NcO(%0,%3) @CP:H_Sc(%0,P,%3) #define H_NcP(%0,%3) @CQ:H_Sc(%0,Q,%3) #define H_NcQ(%0,%3) @CR:H_Sc(%0,R,%3) #define H_NcR(%0,%3) @CS:H_Sc(%0,S,%3) #define H_NcS(%0,%3) @CT:H_Sc(%0,T,%3) #define H_NcT(%0,%3) @CU:H_Sc(%0,U,%3) #define H_NcU(%0,%3) @CV:H_Sc(%0,V,%3) #define H_NcV(%0,%3) @CW:H_Sc(%0,W,%3) #define H_NcW(%0,%3) @CX:H_Sc(%0,X,%3) #define H_NcX(%0,%3) @CY:H_Sc(%0,Y,%3) #define H_NcY(%0,%3) @CZ:H_Sc(%0,Z,%3) #define H_NcZ(%0,%3) @C0:H_Sc(%0,0,%3) #define H_Nc0(%0,%3) @C1:H_Sc(%0,1,%3) #define H_Nc1(%0,%3) @C2:H_Sc(%0,2,%3) #define H_Nc2(%0,%3) @C3:H_Sc(%0,3,%3) #define H_Nc3(%0,%3) @C4:H_Sc(%0,4,%3) #define H_Nc4(%0,%3) @C5:H_Sc(%0,5,%3) #define H_Nc5(%0,%3) @C6:H_Sc(%0,6,%3) #define H_Nc6(%0,%3) @C7:H_Sc(%0,7,%3) #define H_Nc7(%0,%3) @C8:H_Sc(%0,8,%3) #define H_Nc8(%0,%3) @C9:H_Sc(%0,9,%3) #define H_Nc9(%0,%3) ()