// vim: set ts=4 sw=4 tw=99 noet: // // AMX Mod X, based on AMX Mod by Aleksander Naszko ("OLO"). // Copyright (C) The AMX Mod X Development Team. // // This software is licensed under the GNU General Public License, version 3 or higher. // Additional exceptions apply. For full license details, see LICENSE.txt or visit: // https://alliedmods.net/amxmodx-license #include "amxmodx.h" #include "debugger.h" #include "binlog.h" CForward::CForward(const char *name, ForwardExecType et, int numParams, const ForwardParam *paramTypes) { m_FuncName = name; m_ExecType = et; m_NumParams = numParams; memcpy((void *)m_ParamTypes, paramTypes, numParams * sizeof(ForwardParam)); // find funcs int func; m_Funcs.clear(); for (CPluginMngr::iterator iter = g_plugins.begin(); iter; ++iter) { if ((*iter).isValid() && amx_FindPublic((*iter).getAMX(), name, &func) == AMX_ERR_NONE) { AMXForward tmp; tmp.pPlugin = &(*iter); tmp.func = func; m_Funcs.append(tmp); } } m_Name = name; } cell CForward::execute(cell *params, ForwardPreparedArray *preparedArrays) { cell realParams[FORWARD_MAX_PARAMS]; cell *physAddrs[FORWARD_MAX_PARAMS]; const int STRINGEX_MAXLENGTH = 128; cell globRetVal = 0; for (size_t i = 0; i < m_Funcs.length(); ++i) { auto iter = &m_Funcs[i]; if (iter->pPlugin->isExecutable(iter->func)) { // Get debug info AMX *amx = iter->pPlugin->getAMX(); Debugger *pDebugger = (Debugger *)amx->userdata[UD_DEBUGGER]; if (pDebugger) pDebugger->BeginExec(); // handle strings & arrays int i; for (i = 0; i < m_NumParams; ++i) { if (m_ParamTypes[i] == FP_STRING || m_ParamTypes[i] == FP_STRINGEX) { const char *str = reinterpret_cast(params[i]); cell *tmp; if (!str) str = ""; amx_Allot(iter->pPlugin->getAMX(), (m_ParamTypes[i] == FP_STRING) ? strlen(str) + 1 : STRINGEX_MAXLENGTH, &realParams[i], &tmp); amx_SetStringOld(tmp, str, 0, 0); physAddrs[i] = tmp; } else if (m_ParamTypes[i] == FP_ARRAY) { cell *tmp; amx_Allot(amx, preparedArrays[params[i]].size, &realParams[i], &tmp); physAddrs[i] = tmp; if (preparedArrays[params[i]].type == Type_Cell) { memcpy(tmp, preparedArrays[params[i]].ptr, preparedArrays[params[i]].size * sizeof(cell)); } else { char *data = (char*)preparedArrays[params[i]].ptr; for (unsigned int j = 0; j < preparedArrays[params[i]].size; ++j) *tmp++ = (static_cast(*data++)) & 0xFF; } } else { realParams[i] = params[i]; } } //Push the parameters in reverse order. Weird, unfriendly part of Small 3.0! for (i = m_NumParams-1; i >= 0; i--) { amx_Push(amx, realParams[i]); } // exec cell retVal = 0; #if defined BINLOG_ENABLED g_BinLog.WriteOp(BinLog_CallPubFunc, iter->pPlugin->getId(), iter->func); #endif int err = amx_Exec(amx, &retVal, iter->func); // log runtime error, if any if (err != AMX_ERR_NONE) { //Did something else set an error? if (pDebugger && pDebugger->ErrorExists()) { //we don't care, something else logged the error. } else if (err != -1) { //nothing logged the error so spit it out anyway LogError(amx, err, NULL); } } amx->error = AMX_ERR_NONE; if (pDebugger) pDebugger->EndExec(); // cleanup strings & arrays for (i = 0; i < m_NumParams; ++i) { if (m_ParamTypes[i] == FP_STRING) { amx_Release(iter->pPlugin->getAMX(), realParams[i]); } else if (m_ParamTypes[i] == FP_STRINGEX) { // copy back amx_GetStringOld(reinterpret_cast(params[i]), physAddrs[i], 0); amx_Release(iter->pPlugin->getAMX(), realParams[i]); } else if (m_ParamTypes[i] == FP_ARRAY) { // copy back if (preparedArrays[params[i]].copyBack) { cell *tmp = physAddrs[i]; if (preparedArrays[params[i]].type == Type_Cell) { memcpy(preparedArrays[params[i]].ptr, tmp, preparedArrays[params[i]].size * sizeof(cell)); } else { char *data = (char*)preparedArrays[params[i]].ptr; for (unsigned int j = 0; j < preparedArrays[params[i]].size; ++j) *data++ = static_cast(*tmp++ & 0xFF); } } amx_Release(iter->pPlugin->getAMX(), realParams[i]); } } // decide what to do (based on exectype and retval) switch (m_ExecType) { case ET_IGNORE: break; case ET_STOP: if (retVal > 0) return retVal; case ET_STOP2: if (retVal == 1) return 1; else if (retVal > globRetVal) globRetVal = retVal; break; case ET_CONTINUE: if (retVal > globRetVal) globRetVal = retVal; break; } } } return globRetVal; } void CSPForward::Set(int func, AMX *amx, int numParams, const ForwardParam *paramTypes) { char name[sNAMEMAX]; m_Func = func; m_Amx = amx; m_NumParams = numParams; memcpy((void *)m_ParamTypes, paramTypes, numParams * sizeof(ForwardParam)); m_HasFunc = true; isFree = false; name[0] = '\0'; amx_GetPublic(amx, func, name); m_Name = name; m_ToDelete = false; m_InExec = false; } void CSPForward::Set(const char *funcName, AMX *amx, int numParams, const ForwardParam *paramTypes) { m_Amx = amx; m_NumParams = numParams; memcpy((void *)m_ParamTypes, paramTypes, numParams * sizeof(ForwardParam)); m_HasFunc = (amx_FindPublic(amx, funcName, &m_Func) == AMX_ERR_NONE); isFree = false; m_Name = funcName; m_ToDelete = false; m_InExec = false; } cell CSPForward::execute(cell *params, ForwardPreparedArray *preparedArrays) { if (isFree) return 0; const int STRINGEX_MAXLENGTH = 128; cell realParams[FORWARD_MAX_PARAMS]; cell *physAddrs[FORWARD_MAX_PARAMS]; if (!m_HasFunc || m_ToDelete) return 0; CPluginMngr::CPlugin *pPlugin = g_plugins.findPluginFast(m_Amx); if (!pPlugin->isExecutable(m_Func)) return 0; m_InExec = true; Debugger *pDebugger = (Debugger *)m_Amx->userdata[UD_DEBUGGER]; if (pDebugger) pDebugger->BeginExec(); // handle strings & arrays int i; for (i = 0; i < m_NumParams; ++i) { if (m_ParamTypes[i] == FP_STRING || m_ParamTypes[i] == FP_STRINGEX) { const char *str = reinterpret_cast(params[i]); if (!str) str = ""; cell *tmp; amx_Allot(m_Amx, (m_ParamTypes[i] == FP_STRING) ? strlen(str) + 1 : STRINGEX_MAXLENGTH, &realParams[i], &tmp); amx_SetStringOld(tmp, str, 0, 0); physAddrs[i] = tmp; } else if (m_ParamTypes[i] == FP_ARRAY) { cell *tmp; amx_Allot(m_Amx, preparedArrays[params[i]].size, &realParams[i], &tmp); physAddrs[i] = tmp; if (preparedArrays[params[i]].type == Type_Cell) { memcpy(tmp, preparedArrays[params[i]].ptr, preparedArrays[params[i]].size * sizeof(cell)); } else { char *data = (char*)preparedArrays[params[i]].ptr; for (unsigned int j = 0; j < preparedArrays[params[i]].size; ++j) *tmp++ = (static_cast(*data++)) & 0xFF; } } else { realParams[i] = params[i]; } } for (i = m_NumParams - 1; i >= 0; i--) amx_Push(m_Amx, realParams[i]); // exec cell retVal; #if defined BINLOG_ENABLED g_BinLog.WriteOp(BinLog_CallPubFunc, pPlugin->getId(), m_Func); #endif int err = amx_Exec(m_Amx, &retVal, m_Func); if (err != AMX_ERR_NONE) { //Did something else set an error? if (pDebugger && pDebugger->ErrorExists()) { //we don't care, something else logged the error. } else if (err != -1) { //nothing logged the error so spit it out anyway LogError(m_Amx, err, NULL); } } if (pDebugger) pDebugger->EndExec(); m_Amx->error = AMX_ERR_NONE; // cleanup strings & arrays for (i = 0; i < m_NumParams; ++i) { if (m_ParamTypes[i] == FP_STRING) { amx_Release(m_Amx, realParams[i]); } else if (m_ParamTypes[i] == FP_STRINGEX) { // copy back amx_GetStringOld(reinterpret_cast(params[i]), physAddrs[i], 0); amx_Release(m_Amx, realParams[i]); } else if (m_ParamTypes[i] == FP_ARRAY) { // copy back if (preparedArrays[params[i]].copyBack) { cell *tmp = physAddrs[i]; if (preparedArrays[params[i]].type == Type_Cell) { memcpy(preparedArrays[params[i]].ptr, tmp, preparedArrays[params[i]].size * sizeof(cell)); } else { char *data = (char*)preparedArrays[params[i]].ptr; for (unsigned int j = 0; j < preparedArrays[params[i]].size; ++j) *data++ = static_cast(*tmp++ & 0xFF); } } amx_Release(m_Amx, realParams[i]); } } m_InExec = false; return retVal; } int CForwardMngr::registerForward(const char *funcName, ForwardExecType et, int numParams, const ForwardParam * paramTypes) { int retVal = m_Forwards.length() << 1; CForward *tmp = new CForward(funcName, et, numParams, paramTypes); if (!tmp) { return -1; // should be invalid } m_Forwards.append(tmp); return retVal; } int CForwardMngr::registerSPForward(int func, AMX *amx, int numParams, const ForwardParam *paramTypes) { int retVal = -1; CSPForward *pForward; if (!m_FreeSPForwards.empty()) { retVal = m_FreeSPForwards.front(); pForward = m_SPForwards[retVal >> 1]; pForward->Set(func, amx, numParams, paramTypes); if (pForward->getFuncsNum() == 0) return -1; m_FreeSPForwards.pop(); } else { retVal = (m_SPForwards.length() << 1) | 1; pForward = new CSPForward(); if (!pForward) return -1; pForward->Set(func, amx, numParams, paramTypes); if (pForward->getFuncsNum() == 0) { return -1; delete pForward; } m_SPForwards.append(pForward); } return retVal; } int CForwardMngr::registerSPForward(const char *funcName, AMX *amx, int numParams, const ForwardParam *paramTypes) { int retVal = (m_SPForwards.length() << 1) | 1; CSPForward *pForward; if (!m_FreeSPForwards.empty()) { retVal = m_FreeSPForwards.front(); pForward = m_SPForwards[retVal>>1]; // >>1 because unregisterSPForward pushes the id which contains the sp flag pForward->Set(funcName, amx, numParams, paramTypes); if (pForward->getFuncsNum() == 0) return -1; m_FreeSPForwards.pop(); } else { pForward = new CSPForward(); if (!pForward) return -1; pForward->Set(funcName, amx, numParams, paramTypes); if (pForward->getFuncsNum() == 0) { delete pForward; return -1; } m_SPForwards.append(pForward); } return retVal; } bool CForwardMngr::isIdValid(int id) const { return (id >= 0) && ((id & 1) ? (static_cast(id >> 1) < m_SPForwards.length()) : (static_cast(id >> 1) < m_Forwards.length())); } cell CForwardMngr::executeForwards(int id, cell *params) { int retVal; if (id & 1) { CSPForward *fwd = m_SPForwards[id >> 1]; retVal = fwd->execute(params, m_TmpArrays); if (fwd->m_ToDelete) { fwd->m_ToDelete = false; unregisterSPForward(id); } } else { retVal = m_Forwards[id >> 1]->execute(params, m_TmpArrays); } m_TmpArraysNum = 0; return retVal; } const char *CForwardMngr::getFuncName(int id) const { if (!isIdValid(id)) { return ""; } return (id & 1) ? m_SPForwards[id >> 1]->getFuncName() : m_Forwards[id >> 1]->getFuncName(); } int CForwardMngr::getFuncsNum(int id) const { if (!isIdValid(id)) { return 0; } return (id & 1) ? m_SPForwards[id >> 1]->getFuncsNum() : m_Forwards[id >> 1]->getFuncsNum(); } int CForwardMngr::getParamsNum(int id) const { return (id & 1) ? m_SPForwards[id >> 1]->getParamsNum() : m_Forwards[id >> 1]->getParamsNum(); } ForwardParam CForwardMngr::getParamType(int id, int paramNum) const { if (!isIdValid(id)) { return FP_DONE; } return (id & 1) ? m_SPForwards[id >> 1]->getParamType(paramNum) : m_Forwards[id >> 1]->getParamType(paramNum); } void CForwardMngr::clear() { size_t i; for (i = 0; i < m_Forwards.length(); ++i) { delete m_Forwards[i]; } for (i = 0; i < m_SPForwards.length(); ++i) { delete m_SPForwards[i]; } m_Forwards.clear(); m_SPForwards.clear(); while (!m_FreeSPForwards.empty()) { m_FreeSPForwards.pop(); } m_TmpArraysNum = 0; } bool CForwardMngr::isSPForward(int id) const { return ((id & 1) == 0) ? false : true; } void CForwardMngr::unregisterSPForward(int id) { //make sure the id is valid if (!isIdValid(id) || m_SPForwards.at(id >> 1)->isFree) { return; } CSPForward *fwd = m_SPForwards.at(id >> 1); if (fwd->m_InExec) { fwd->m_ToDelete = true; } else { fwd->isFree = true; m_FreeSPForwards.push(id); } } int CForwardMngr::duplicateSPForward(int id) { if (!isIdValid(id) || m_SPForwards.at(id >> 1)->isFree) { return -1; } CSPForward *fwd = m_SPForwards.at(id >> 1); return registerSPForward(fwd->m_Func, fwd->m_Amx, fwd->m_NumParams, fwd->m_ParamTypes); } int CForwardMngr::isSameSPForward(int id1, int id2) { if (!isIdValid(id1) || !isIdValid(id2)) { return false; } CSPForward *fwd1 = m_SPForwards.at(id1 >> 1); CSPForward *fwd2 = m_SPForwards.at(id2 >> 1); if (fwd1->isFree || fwd2->isFree) { return false; } return ((fwd1->m_Amx == fwd2->m_Amx) && (fwd1->m_Func == fwd2->m_Func) && (fwd1->m_NumParams == fwd2->m_NumParams)); } int registerForwardC(const char *funcName, ForwardExecType et, cell *list, size_t num) { ForwardParam params[FORWARD_MAX_PARAMS]; for (size_t i=0; i(list[i]); } return g_forwards.registerForward(funcName, et, num, params); } int registerForward(const char *funcName, ForwardExecType et, ...) { int curParam = 0; va_list argptr; va_start(argptr, et); ForwardParam params[FORWARD_MAX_PARAMS]; ForwardParam tmp; while (true) { if (curParam == FORWARD_MAX_PARAMS) break; tmp = (ForwardParam)va_arg(argptr, int); if (tmp == FP_DONE) break; params[curParam] = tmp; ++curParam; } va_end(argptr); return g_forwards.registerForward(funcName, et, curParam, params); } int registerSPForwardByNameC(AMX *amx, const char *funcName, cell *list, size_t num) { ForwardParam params[FORWARD_MAX_PARAMS]; for (size_t i=0; i(list[i]); return g_forwards.registerSPForward(funcName, amx, num, params); } int registerSPForwardByName(AMX *amx, const char *funcName, ...) { int curParam = 0; va_list argptr; va_start(argptr, funcName); ForwardParam params[FORWARD_MAX_PARAMS]; ForwardParam tmp; while (true) { if (curParam == FORWARD_MAX_PARAMS) break; tmp = (ForwardParam)va_arg(argptr, int); if (tmp == FP_DONE) break; params[curParam] = tmp; ++curParam; } va_end(argptr); return g_forwards.registerSPForward(funcName, amx, curParam, params); } int registerSPForward(AMX *amx, int func, ...) { int curParam = 0; va_list argptr; va_start(argptr, func); ForwardParam params[FORWARD_MAX_PARAMS]; ForwardParam tmp; while (true) { if (curParam == FORWARD_MAX_PARAMS) break; tmp = (ForwardParam)va_arg(argptr, int); if (tmp == FP_DONE) break; params[curParam] = tmp; ++curParam; } va_end(argptr); return g_forwards.registerSPForward(func, amx, curParam, params); } cell executeForwards(int id, ...) { if (!g_forwards.isIdValid(id)) return -1; cell params[FORWARD_MAX_PARAMS]; int paramsNum = g_forwards.getParamsNum(id); va_list argptr; va_start(argptr, id); for (int i = 0; i < paramsNum && i < FORWARD_MAX_PARAMS; ++i) { if (g_forwards.getParamType(id, i) == FP_FLOAT) { REAL tmp = (REAL)va_arg(argptr, double); // floats get converted to doubles params[i] = *(cell*)&tmp; } else params[i] = (cell)va_arg(argptr, cell); } va_end(argptr); return g_forwards.executeForwards(id, params); } cell CForwardMngr::prepareArray(void *ptr, unsigned int size, ForwardArrayElemType type, bool copyBack) { if (m_TmpArraysNum >= FORWARD_MAX_PARAMS) { #ifdef MEMORY_TEST m_validateAllAllocUnits(); #endif // MEMORY_TEST AMXXLOG_Log("[AMXX] Forwards with more than 32 parameters are not supported (tried to prepare array # %d).", m_TmpArraysNum + 1); m_TmpArraysNum = 0; return -1; } m_TmpArrays[m_TmpArraysNum].ptr = ptr; m_TmpArrays[m_TmpArraysNum].size = size; m_TmpArrays[m_TmpArraysNum].type = type; m_TmpArrays[m_TmpArraysNum].copyBack = copyBack; return m_TmpArraysNum++; } cell prepareCellArray(cell *ptr, unsigned int size, bool copyBack) { return g_forwards.prepareArray((void*)ptr, size, Type_Cell, copyBack); } cell prepareCharArray(char *ptr, unsigned int size, bool copyBack) { return g_forwards.prepareArray((void*)ptr, size, Type_Char, copyBack); } void unregisterSPForward(int id) { g_forwards.unregisterSPForward(id); }