/** * vim: set ts=4 sw=4 tw=99 noet : * ============================================================================= * SourceMod * Copyright (C) 2004-2017 AlliedModders LLC. All rights reserved. * ============================================================================= * * This program is free software; you can redistribute it and/or modify it under * the terms of the GNU General Public License, version 3.0, as published by the * Free Software Foundation. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more * details. * * You should have received a copy of the GNU General Public License along with * this program. If not, see . * * As a special exception, AlliedModders LLC gives you permission to link the * code of this program (as well as its derivative works) to "Half-Life 2," the * "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software * by the Valve Corporation. You must obey the GNU General Public License in * all respects for all other code used. Additionally, AlliedModders LLC grants * this exception to all derivative works. AlliedModders LLC defines further * exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007), * or . * * Version: $Id$ */ #include #include #include #include "common_logic.h" #include "Logger.h" #include #include #include #include #include #include #if defined PLATFORM_WINDOWS #include #elif defined PLATFORM_POSIX #include #include #include #endif #include #include #include #include #include using namespace SourceMod; using namespace SourcePawn; HandleType_t g_PlIter; HandleType_t g_FrameIter; IForward *g_OnLogAction = NULL; static ConVar *sm_datetime_format = NULL; class CoreNativeHelpers : public SMGlobalClass, public IHandleTypeDispatch { public: void OnSourceModAllInitialized() { HandleAccess hacc; handlesys->InitAccessDefaults(NULL, &hacc); hacc.access[HandleAccess_Clone] = HANDLE_RESTRICT_IDENTITY|HANDLE_RESTRICT_OWNER; g_PlIter = handlesys->CreateType("PluginIterator", this, 0, NULL, NULL, g_pCoreIdent, NULL); g_FrameIter = handlesys->CreateType("FrameIterator", this, 0, NULL, NULL, g_pCoreIdent, NULL); g_OnLogAction = forwardsys->CreateForward("OnLogAction", ET_Hook, 5, NULL, Param_Cell, Param_Cell, Param_Cell, Param_Cell, Param_String); sm_datetime_format = bridge->FindConVar("sm_datetime_format"); } void OnHandleDestroy(HandleType_t type, void *object) { if (type == g_FrameIter) { delete (SafeFrameIterator *) object; } else if (type == g_PlIter) { IPluginIterator *iter = (IPluginIterator *)object; iter->Release(); } } void OnSourceModShutdown() { forwardsys->ReleaseForward(g_OnLogAction); handlesys->RemoveType(g_PlIter, g_pCoreIdent); handlesys->RemoveType(g_FrameIter, g_pCoreIdent); } } g_CoreNativeHelpers; void LogAction(Handle_t hndl, int type, int client, int target, const char *message) { if (g_OnLogAction->GetFunctionCount()) { cell_t result = 0; g_OnLogAction->PushCell(hndl); g_OnLogAction->PushCell(type); g_OnLogAction->PushCell(client); g_OnLogAction->PushCell(target); g_OnLogAction->PushString(message); g_OnLogAction->Execute(&result); if (result >= (ResultType)Pl_Handled) { return; } } const char *logtag = "SM"; if (type == 2) { HandleError err; IPlugin *pPlugin = scripts->FindPluginByHandle(hndl, &err); if (pPlugin) { logtag = pPlugin->GetFilename(); } } g_Logger.LogMessage("[%s] %s", logtag, message); } static cell_t ThrowError(IPluginContext *pContext, const cell_t *params) { char buffer[512]; g_pSM->SetGlobalTarget(SOURCEMOD_SERVER_LANGUAGE); { DetectExceptions eh(pContext); g_pSM->FormatString(buffer, sizeof(buffer), pContext, params, 1); if (eh.HasException()) return 0; } pContext->ReportError("%s", buffer); return 0; } static cell_t GetTime(IPluginContext *pContext, const cell_t *params) { time_t t = g_pSM->GetAdjustedTime(); cell_t *addr; pContext->LocalToPhysAddr(params[1], &addr); *(time_t *)addr = t; return static_cast(t); } #if defined SUBPLATFORM_SECURECRT void _ignore_invalid_parameter( const wchar_t * expression, const wchar_t * function, const wchar_t * file, unsigned int line, uintptr_t pReserved ) { /* Wow we don't care, thanks Microsoft. */ } #endif static cell_t FormatTime(IPluginContext *pContext, const cell_t *params) { char *format, *buffer; pContext->LocalToString(params[1], &buffer); pContext->LocalToStringNULL(params[3], &format); if (format == NULL) { format = const_cast(bridge->GetCvarString(sm_datetime_format)); } #if defined SUBPLATFORM_SECURECRT _invalid_parameter_handler handler = _set_invalid_parameter_handler(_ignore_invalid_parameter); #endif time_t t = (params[4] == -1) ? g_pSM->GetAdjustedTime() : (time_t)params[4]; size_t written = strftime(buffer, params[2], format, localtime(&t)); #if defined SUBPLATFORM_SECURECRT _set_invalid_parameter_handler(handler); #endif if (params[2] && format[0] != '\0' && !written) { pContext->ThrowNativeError("Invalid time format or buffer too small"); return 0; } return 1; } static cell_t GetPluginIterator(IPluginContext *pContext, const cell_t *params) { IPluginIterator *iter = scripts->GetPluginIterator(); Handle_t hndl = handlesys->CreateHandle(g_PlIter, iter, pContext->GetIdentity(), g_pCoreIdent, NULL); if (hndl == BAD_HANDLE) { iter->Release(); } return hndl; } static cell_t MorePlugins(IPluginContext *pContext, const cell_t *params) { Handle_t hndl = (Handle_t)params[1]; HandleError err; IPluginIterator *pIter; HandleSecurity sec; sec.pIdentity = g_pCoreIdent; sec.pOwner = pContext->GetIdentity(); if ((err=handlesys->ReadHandle(hndl, g_PlIter, &sec, (void **)&pIter)) != HandleError_None) { return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err); } return pIter->MorePlugins() ? 1 : 0; } static cell_t ReadPlugin(IPluginContext *pContext, const cell_t *params) { Handle_t hndl = (Handle_t)params[1]; HandleError err; IPluginIterator *pIter; HandleSecurity sec; sec.pIdentity = g_pCoreIdent; sec.pOwner = pContext->GetIdentity(); if ((err=handlesys->ReadHandle(hndl, g_PlIter, &sec, (void **)&pIter)) != HandleError_None) { return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err); } IPlugin *pPlugin = pIter->GetPlugin(); if (!pPlugin) { return BAD_HANDLE; } pIter->NextPlugin(); return pPlugin->GetMyHandle(); } IPlugin *GetPluginFromHandle(IPluginContext *pContext, Handle_t hndl) { if (hndl == BAD_HANDLE) { return scripts->FindPluginByContext(pContext->GetContext()); } else { HandleError err; IPlugin *pPlugin = scripts->FindPluginByHandle(hndl, &err); if (!pPlugin) { pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err); } return pPlugin; } } static cell_t GetPluginStatus(IPluginContext *pContext, const cell_t *params) { IPlugin *pPlugin = GetPluginFromHandle(pContext, params[1]); if (!pPlugin) { return 0; } return pPlugin->GetStatus(); } static cell_t GetPluginFilename(IPluginContext *pContext, const cell_t *params) { IPlugin *pPlugin = GetPluginFromHandle(pContext, params[1]); if (!pPlugin) { return 0; } pContext->StringToLocalUTF8(params[2], params[3], pPlugin->GetFilename(), NULL); return 1; } static cell_t IsPluginDebugging(IPluginContext *pContext, const cell_t *params) { IPlugin *pPlugin = GetPluginFromHandle(pContext, params[1]); if (!pPlugin) { return 0; } return pPlugin->IsDebugging() ? 1 : 0; } /* Local to plugins only */ enum PluginInfo { PlInfo_Name, /**< Plugin name */ PlInfo_Author, /**< Plugin author */ PlInfo_Description, /**< Plugin description */ PlInfo_Version, /**< Plugin verison */ PlInfo_URL, /**< Plugin URL */ }; static cell_t GetPluginInfo(IPluginContext *pContext, const cell_t *params) { IPlugin *pPlugin = GetPluginFromHandle(pContext, params[1]); if (!pPlugin) { return 0; } const sm_plugininfo_t *info = pPlugin->GetPublicInfo(); if (!info) { return 0; } const char *str = NULL; switch ((PluginInfo)params[2]) { case PlInfo_Name: { str = info->name; break; } case PlInfo_Author: { str = info->author; break; } case PlInfo_Description: { str = info->description; break; } case PlInfo_Version: { str = info->version; break; } case PlInfo_URL: { str = info->url; break; } } if (!str || str[0] == '\0') { return 0; } pContext->StringToLocalUTF8(params[3], params[4], str, NULL); return 1; } static cell_t SetFailState(IPluginContext *pContext, const cell_t *params) { char *str; SMPlugin *pPlugin; pContext->LocalToString(params[1], &str); pPlugin = scripts->FindPluginByContext(pContext->GetContext()); if (params[0] == 1) { pPlugin->EvictWithError(Plugin_Failed, "%s", str); return pContext->ThrowNativeErrorEx(SP_ERROR_ABORTED, "%s", str); } else { char buffer[2048]; { DetectExceptions eh(pContext); g_pSM->FormatString(buffer, sizeof(buffer), pContext, params, 1); if (eh.HasException()) { pPlugin->EvictWithError(Plugin_Failed, "%s", str); return 0; } pPlugin->EvictWithError(Plugin_Failed, "%s", buffer); pContext->ReportFatalError("%s", buffer); return 0; } } return 0; } static cell_t GetSysTickCount(IPluginContext *pContext, const cell_t *params) { #if defined PLATFORM_WINDOWS return (cell_t)GetTickCount(); #elif defined PLATFORM_POSIX tms tm; clock_t ticks = times(&tm); long ticks_per_sec = sysconf(_SC_CLK_TCK); double fticks = (double)ticks / (double)ticks_per_sec; fticks *= 1000.0f; if (fticks > INT_MAX) { double r = (int)(fticks / INT_MAX) * (double)INT_MAX; fticks -= r; } return (cell_t)fticks; #endif } static cell_t AutoExecConfig(IPluginContext *pContext, const cell_t *params) { SMPlugin *plugin = scripts->FindPluginByContext(pContext->GetContext()); char *cfg, *folder; pContext->LocalToString(params[2], &cfg); pContext->LocalToString(params[3], &folder); if (cfg[0] == '\0') { static char temp_str[255]; static char temp_file[PLATFORM_MAX_PATH]; char *ptr; libsys->GetFileFromPath(temp_str, sizeof(temp_str), plugin->GetFilename()); if ((ptr = strstr(temp_str, ".smx")) != NULL) { *ptr = '\0'; } /* We have the raw filename! */ g_pSM->Format(temp_file, sizeof(temp_file), "plugin.%s", temp_str); cfg = temp_file; } plugin->AddConfig(params[1] ? true : false, cfg, folder); return 1; } static cell_t MarkNativeAsOptional(IPluginContext *pContext, const cell_t *params) { char *name; uint32_t idx; pContext->LocalToString(params[1], &name); if (pContext->FindNativeByName(name, &idx) != SP_ERROR_NONE) { /* Oops! This HAS to silently fail! */ return 0; } pContext->GetRuntime()->UpdateNativeBinding(idx, nullptr, SP_NTVFLAG_OPTIONAL, nullptr); return 1; } static cell_t RegPluginLibrary(IPluginContext *pContext, const cell_t *params) { char *name; SMPlugin *pl = scripts->FindPluginByContext(pContext->GetContext()); pContext->LocalToString(params[1], &name); pl->AddLibrary(name); return 1; } static cell_t LibraryExists(IPluginContext *pContext, const cell_t *params) { char *str; pContext->LocalToString(params[1], &str); if (strcmp(str, "__CanTestFeatures__") == 0) { return 1; } if (scripts->LibraryExists(str)) { return 1; } if (extsys->LibraryExists(str)) { return 1; } return 0; } static cell_t sm_LogAction(IPluginContext *pContext, const cell_t *params) { char buffer[2048]; g_pSM->SetGlobalTarget(SOURCEMOD_SERVER_LANGUAGE); { DetectExceptions eh(pContext); g_pSM->FormatString(buffer, sizeof(buffer), pContext, params, 3); if (eh.HasException()) return 0; } IPlugin *pPlugin = scripts->FindPluginByContext(pContext->GetContext()); LogAction(pPlugin->GetMyHandle(), 2, params[1], params[2], buffer); return 1; } static cell_t LogToFile(IPluginContext *pContext, const cell_t *params) { char *file; pContext->LocalToString(params[1], &file); char path[PLATFORM_MAX_PATH]; g_pSM->BuildPath(Path_Game, path, sizeof(path), "%s", file); FILE *fp = fopen(path, "at"); if (!fp) { return pContext->ThrowNativeError("Could not open file \"%s\"", path); } char buffer[2048]; { DetectExceptions eh(pContext); g_pSM->SetGlobalTarget(SOURCEMOD_SERVER_LANGUAGE); g_pSM->FormatString(buffer, sizeof(buffer), pContext, params, 2); if (eh.HasException()) { fclose(fp); return 0; } } IPlugin *pPlugin = scripts->FindPluginByContext(pContext->GetContext()); g_Logger.LogToOpenFile(fp, "[%s] %s", pPlugin->GetFilename(), buffer); fclose(fp); return 1; } static cell_t LogToFileEx(IPluginContext *pContext, const cell_t *params) { char *file; pContext->LocalToString(params[1], &file); char path[PLATFORM_MAX_PATH]; g_pSM->BuildPath(Path_Game, path, sizeof(path), "%s", file); FILE *fp = fopen(path, "at"); if (!fp) { return pContext->ThrowNativeError("Could not open file \"%s\"", path); } char buffer[2048]; { DetectExceptions eh(pContext); g_pSM->SetGlobalTarget(SOURCEMOD_SERVER_LANGUAGE); g_pSM->FormatString(buffer, sizeof(buffer), pContext, params, 2); if (eh.HasException()) { fclose(fp); return 0; } } g_Logger.LogToOpenFile(fp, "%s", buffer); fclose(fp); return 1; } static cell_t GetExtensionFileStatus(IPluginContext *pContext, const cell_t *params) { char *str; pContext->LocalToString(params[1], &str); IExtension *pExtension = extsys->FindExtensionByFile(str); if (!pExtension) { return -2; } if (!pExtension->IsLoaded()) { return -1; } char *error; pContext->LocalToString(params[2], &error); if (!pExtension->IsRunning(error, params[3])) { return 0; } return 1; } static cell_t FindPluginByNumber(IPluginContext *pContext, const cell_t *params) { IPlugin *pPlugin = scripts->FindPluginByOrder(params[1]); if (pPlugin == NULL) { return BAD_HANDLE; } return pPlugin->GetMyHandle(); } static cell_t VerifyCoreVersion(IPluginContext *pContext, const cell_t *params) { return 4; } static cell_t GetFeatureStatus(IPluginContext *pContext, const cell_t *params) { FeatureType type = (FeatureType)params[1]; char *name; pContext->LocalToString(params[2], &name); return sharesys->TestFeature(pContext->GetRuntime(), type, name); } static cell_t RequireFeature(IPluginContext *pContext, const cell_t *params) { FeatureType type = (FeatureType)params[1]; char *name; pContext->LocalToString(params[2], &name); if (sharesys->TestFeature(pContext->GetRuntime(), type, name) != FeatureStatus_Available) { char buffer[255]; char *msg = buffer; char default_message[255]; SMPlugin *pPlugin = scripts->FindPluginByContext(pContext->GetContext()); DetectExceptions eh(pContext); g_pSM->FormatString(buffer, sizeof(buffer), pContext, params, 3); if (eh.HasException()) buffer[0] = '\0'; if (buffer[0] == '\0') { g_pSM->Format(default_message, sizeof(default_message), "Feature \"%s\" not available", name); msg = default_message; } pPlugin->EvictWithError(Plugin_Error, "%s", msg); if (!eh.HasException()) pContext->ReportFatalError("%s", msg); return 0; } return 1; } enum NumberType { NumberType_Int8, NumberType_Int16, NumberType_Int32 }; //memory addresses below 0x10000 are automatically considered invalid for dereferencing #define VALID_MINIMUM_MEMORY_ADDRESS 0x10000 static cell_t LoadFromAddress(IPluginContext *pContext, const cell_t *params) { #ifdef PLATFORM_X86 void *addr = reinterpret_cast(params[1]); #else void *addr = pseudoAddr.FromPseudoAddress(params[1]); #endif if (addr == NULL) { return pContext->ThrowNativeError("Address cannot be null"); } else if (reinterpret_cast(addr) < VALID_MINIMUM_MEMORY_ADDRESS) { return pContext->ThrowNativeError("Invalid address 0x%x is pointing to reserved memory.", addr); } NumberType size = static_cast(params[2]); switch(size) { case NumberType_Int8: return *reinterpret_cast(addr); case NumberType_Int16: return *reinterpret_cast(addr); case NumberType_Int32: return *reinterpret_cast(addr); default: return pContext->ThrowNativeError("Invalid number types %d", size); } } static cell_t StoreToAddress(IPluginContext *pContext, const cell_t *params) { #ifdef PLATFORM_X86 void *addr = reinterpret_cast(params[1]); #else void *addr = pseudoAddr.FromPseudoAddress(params[1]); #endif if (addr == NULL) { return pContext->ThrowNativeError("Address cannot be null"); } else if (reinterpret_cast(addr) < VALID_MINIMUM_MEMORY_ADDRESS) { return pContext->ThrowNativeError("Invalid address 0x%x is pointing to reserved memory.", addr); } cell_t data = params[2]; NumberType size = static_cast(params[3]); switch(size) { case NumberType_Int8: SourceHook::SetMemAccess(addr, sizeof(uint8_t), SH_MEM_READ|SH_MEM_WRITE|SH_MEM_EXEC); *reinterpret_cast(addr) = data; break; case NumberType_Int16: SourceHook::SetMemAccess(addr, sizeof(uint16_t), SH_MEM_READ|SH_MEM_WRITE|SH_MEM_EXEC); *reinterpret_cast(addr) = data; break; case NumberType_Int32: SourceHook::SetMemAccess(addr, sizeof(uint32_t), SH_MEM_READ|SH_MEM_WRITE|SH_MEM_EXEC); *reinterpret_cast(addr) = data; break; default: return pContext->ThrowNativeError("Invalid number types %d", size); } return 0; } static cell_t IsNullVector(IPluginContext *pContext, const cell_t *params) { cell_t *pNullVec = pContext->GetNullRef(SP_NULL_VECTOR); if (!pNullVec) return 0; cell_t *addr; pContext->LocalToPhysAddr(params[1], &addr); return addr == pNullVec; } static cell_t IsNullString(IPluginContext *pContext, const cell_t *params) { char *str; if (pContext->LocalToStringNULL(params[1], &str) != SP_ERROR_NONE) return 0; return str == nullptr; } static cell_t FrameIterator_Create(IPluginContext *pContext, const cell_t *params) { IFrameIterator *it = pContext->CreateFrameIterator(); SafeFrameIterator *iterator = new SafeFrameIterator(it); pContext->DestroyFrameIterator(it); Handle_t handle = handlesys->CreateHandle(g_FrameIter, iterator, pContext->GetIdentity(), g_pCoreIdent, NULL); if (handle == BAD_HANDLE) { delete iterator; return BAD_HANDLE; } return handle; } static cell_t FrameIterator_Next(IPluginContext *pContext, const cell_t *params) { Handle_t hndl = (Handle_t)params[1]; HandleError err; SafeFrameIterator *iterator; HandleSecurity sec; sec.pIdentity = g_pCoreIdent; sec.pOwner = pContext->GetIdentity(); if ((err=handlesys->ReadHandle(hndl, g_FrameIter, &sec, (void **)&iterator)) != HandleError_None) { return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err); } return iterator->Next(); } static cell_t FrameIterator_Reset(IPluginContext *pContext, const cell_t *params) { Handle_t hndl = (Handle_t)params[1]; HandleError err; SafeFrameIterator *iterator; HandleSecurity sec; sec.pIdentity = g_pCoreIdent; sec.pOwner = pContext->GetIdentity(); if ((err=handlesys->ReadHandle(hndl, g_FrameIter, &sec, (void **)&iterator)) != HandleError_None) { return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err); } iterator->Reset(); return 0; } static cell_t FrameIterator_LineNumber(IPluginContext *pContext, const cell_t *params) { Handle_t hndl = (Handle_t)params[1]; HandleError err; SafeFrameIterator *iterator; HandleSecurity sec; sec.pIdentity = g_pCoreIdent; sec.pOwner = pContext->GetIdentity(); if ((err=handlesys->ReadHandle(hndl, g_FrameIter, &sec, (void **)&iterator)) != HandleError_None) { return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err); } int lineNum = iterator->LineNumber(); if (lineNum < 0) { return pContext->ThrowNativeError("Iterator out of bounds. Check return value of FrameIterator.Next"); } return lineNum; } static cell_t FrameIterator_GetFunctionName(IPluginContext *pContext, const cell_t *params) { Handle_t hndl = (Handle_t)params[1]; HandleError err; SafeFrameIterator *iterator; HandleSecurity sec; sec.pIdentity = g_pCoreIdent; sec.pOwner = pContext->GetIdentity(); if ((err=handlesys->ReadHandle(hndl, g_FrameIter, &sec, (void **)&iterator)) != HandleError_None) { return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err); } const char* functionName = iterator->FunctionName(); if (!functionName) { return pContext->ThrowNativeError("Iterator out of bounds. Check return value of FrameIterator.Next"); } char* buffer; pContext->LocalToString(params[2], &buffer); size_t size = static_cast(params[3]); ke::SafeStrcpy(buffer, size, functionName); return 0; } static cell_t FrameIterator_GetFilePath(IPluginContext *pContext, const cell_t *params) { Handle_t hndl = (Handle_t)params[1]; HandleError err; SafeFrameIterator *iterator; HandleSecurity sec; sec.pIdentity = g_pCoreIdent; sec.pOwner = pContext->GetIdentity(); if ((err=handlesys->ReadHandle(hndl, g_FrameIter, &sec, (void **)&iterator)) != HandleError_None) { return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err); } const char* filePath = iterator->FilePath(); if (!filePath) { return pContext->ThrowNativeError("Iterator out of bounds. Check return value of FrameIterator.Next"); } char* buffer; pContext->LocalToString(params[2], &buffer); size_t size = static_cast(params[3]); ke::SafeStrcpy(buffer, size, filePath); return 0; } REGISTER_NATIVES(coreNatives) { {"ThrowError", ThrowError}, {"GetTime", GetTime}, {"FormatTime", FormatTime}, {"GetPluginIterator", GetPluginIterator}, {"MorePlugins", MorePlugins}, {"ReadPlugin", ReadPlugin}, {"GetPluginStatus", GetPluginStatus}, {"GetPluginFilename", GetPluginFilename}, {"IsPluginDebugging", IsPluginDebugging}, {"GetPluginInfo", GetPluginInfo}, {"SetFailState", SetFailState}, {"GetSysTickCount", GetSysTickCount}, {"AutoExecConfig", AutoExecConfig}, {"MarkNativeAsOptional", MarkNativeAsOptional}, {"RegPluginLibrary", RegPluginLibrary}, {"LibraryExists", LibraryExists}, {"LogAction", sm_LogAction}, {"LogToFile", LogToFile}, {"LogToFileEx", LogToFileEx}, {"GetExtensionFileStatus", GetExtensionFileStatus}, {"FindPluginByNumber", FindPluginByNumber}, {"VerifyCoreVersion", VerifyCoreVersion}, {"GetFeatureStatus", GetFeatureStatus}, {"RequireFeature", RequireFeature}, {"LoadFromAddress", LoadFromAddress}, {"StoreToAddress", StoreToAddress}, {"IsNullVector", IsNullVector}, {"IsNullString", IsNullString}, {"FrameIterator.FrameIterator", FrameIterator_Create}, {"FrameIterator.Next", FrameIterator_Next}, {"FrameIterator.Reset", FrameIterator_Reset}, {"FrameIterator.LineNumber.get", FrameIterator_LineNumber}, {"FrameIterator.GetFunctionName", FrameIterator_GetFunctionName}, {"FrameIterator.GetFilePath", FrameIterator_GetFilePath}, {NULL, NULL}, };