fixed heap corruption
improved memory usage removed useless stringpooler
This commit is contained in:
parent
cef005317b
commit
24cdcde968
@ -408,6 +408,11 @@ static cell AMX_NATIVE_CALL SQL_SetAffinity(AMX *amx, cell *params)
|
|||||||
int len;
|
int len;
|
||||||
char *str = MF_GetAmxString(amx, params[1], 0, &len);
|
char *str = MF_GetAmxString(amx, params[1], 0, &len);
|
||||||
|
|
||||||
|
if (!str[0])
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
if (stricmp(str, g_Mysql.NameString()) == 0)
|
if (stricmp(str, g_Mysql.NameString()) == 0)
|
||||||
{
|
{
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -25,8 +25,7 @@ void OnAmxxAttach()
|
|||||||
MF_AddNatives(g_ThreadSqlNatives);
|
MF_AddNatives(g_ThreadSqlNatives);
|
||||||
g_MysqlFuncs.prev = (SqlFunctions *)MF_RegisterFunctionEx(&g_MysqlFuncs, SQL_DRIVER_FUNC);
|
g_MysqlFuncs.prev = (SqlFunctions *)MF_RegisterFunctionEx(&g_MysqlFuncs, SQL_DRIVER_FUNC);
|
||||||
if (!MF_RequestFunction("GetDbDriver")
|
if (!MF_RequestFunction("GetDbDriver")
|
||||||
&& !MF_FindLibrary("SQLITE", LibType_Library)
|
&& !MF_FindLibrary("SQLITE", LibType_Library))
|
||||||
&& !MF_FindLibrary("sqlitex", LibType_Library))
|
|
||||||
{
|
{
|
||||||
MF_AddNatives(g_OldCompatNatives);
|
MF_AddNatives(g_OldCompatNatives);
|
||||||
MF_AddLibraries("dbi", LibType_Class, &g_ident);
|
MF_AddLibraries("dbi", LibType_Class, &g_ident);
|
||||||
|
@ -33,7 +33,7 @@
|
|||||||
<Tool
|
<Tool
|
||||||
Name="VCLinkerTool"
|
Name="VCLinkerTool"
|
||||||
AdditionalDependencies="wsock32.lib mysqlclient.lib"
|
AdditionalDependencies="wsock32.lib mysqlclient.lib"
|
||||||
OutputFile="$(OutDir)/mysqlx_amxx.dll"
|
OutputFile="$(OutDir)/mysql_amxx.dll"
|
||||||
LinkIncremental="2"
|
LinkIncremental="2"
|
||||||
IgnoreDefaultLibraryNames="LIBCMT"
|
IgnoreDefaultLibraryNames="LIBCMT"
|
||||||
GenerateDebugInformation="TRUE"
|
GenerateDebugInformation="TRUE"
|
||||||
@ -82,7 +82,7 @@
|
|||||||
<Tool
|
<Tool
|
||||||
Name="VCLinkerTool"
|
Name="VCLinkerTool"
|
||||||
AdditionalDependencies="wsock32.lib mysqlclient.lib"
|
AdditionalDependencies="wsock32.lib mysqlclient.lib"
|
||||||
OutputFile="$(OutDir)/mysqlx_amxx.dll"
|
OutputFile="$(OutDir)/mysql_amxx.dll"
|
||||||
LinkIncremental="1"
|
LinkIncremental="1"
|
||||||
IgnoreDefaultLibraryNames=""
|
IgnoreDefaultLibraryNames=""
|
||||||
GenerateDebugInformation="TRUE"
|
GenerateDebugInformation="TRUE"
|
||||||
|
@ -14,12 +14,12 @@
|
|||||||
* -For both library and library class, you can use a comma
|
* -For both library and library class, you can use a comma
|
||||||
* to add multiple entries.
|
* to add multiple entries.
|
||||||
*/
|
*/
|
||||||
#define MODULE_NAME "MySQLX"
|
#define MODULE_NAME "MySQL"
|
||||||
#define MODULE_VERSION "1.75"
|
#define MODULE_VERSION "1.75"
|
||||||
#define MODULE_AUTHOR "AMX Mod X Dev Team"
|
#define MODULE_AUTHOR "AMX Mod X Dev Team"
|
||||||
#define MODULE_URL "http://www.amxmodx.org/"
|
#define MODULE_URL "http://www.amxmodx.org/"
|
||||||
#define MODULE_LOGTAG "MySQLX"
|
#define MODULE_LOGTAG "MySQL"
|
||||||
#define MODULE_LIBRARY "mysqlx"
|
#define MODULE_LIBRARY "mysql"
|
||||||
#define MODULE_LIBCLASS "sqlx"
|
#define MODULE_LIBCLASS "sqlx"
|
||||||
// If you want the module not to be reloaded on mapchange, remove / comment out the next line
|
// If you want the module not to be reloaded on mapchange, remove / comment out the next line
|
||||||
#define MODULE_RELOAD_ON_MAPCHANGE
|
#define MODULE_RELOAD_ON_MAPCHANGE
|
||||||
|
@ -8,7 +8,6 @@ using namespace SourceHook;
|
|||||||
MainThreader g_Threader;
|
MainThreader g_Threader;
|
||||||
ThreadWorker *g_pWorker = NULL;
|
ThreadWorker *g_pWorker = NULL;
|
||||||
extern DLL_FUNCTIONS *g_pFunctionTable;
|
extern DLL_FUNCTIONS *g_pFunctionTable;
|
||||||
StringPool g_StringPool;
|
|
||||||
IMutex *g_QueueLock = NULL;
|
IMutex *g_QueueLock = NULL;
|
||||||
CStack<MysqlThread *> g_ThreadQueue;
|
CStack<MysqlThread *> g_ThreadQueue;
|
||||||
CStack<MysqlThread *> g_FreeThreads;
|
CStack<MysqlThread *> g_FreeThreads;
|
||||||
@ -264,11 +263,11 @@ void OnPluginsLoaded()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!g_StringPool.IsThreadable())
|
if (!g_QueueLock)
|
||||||
{
|
{
|
||||||
g_StringPool.SetMutex(g_Threader.MakeMutex());
|
|
||||||
g_QueueLock = g_Threader.MakeMutex();
|
g_QueueLock = g_Threader.MakeMutex();
|
||||||
}
|
}
|
||||||
|
|
||||||
g_pWorker = new ThreadWorker(&g_Threader, 250);
|
g_pWorker = new ThreadWorker(&g_Threader, 250);
|
||||||
if (!g_pWorker->Start())
|
if (!g_pWorker->Start())
|
||||||
{
|
{
|
||||||
@ -327,11 +326,10 @@ void OnPluginsUnloading()
|
|||||||
AtomicResult::AtomicResult()
|
AtomicResult::AtomicResult()
|
||||||
{
|
{
|
||||||
m_IsFree = true;
|
m_IsFree = true;
|
||||||
m_CurRow = 0;
|
m_CurRow = 1;
|
||||||
m_AllocFields = 0;
|
m_RowCount = NULL;
|
||||||
m_AllocRows = 0;
|
m_Table = NULL;
|
||||||
m_Rows = NULL;
|
m_AllocSize = 0;
|
||||||
m_Fields = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
AtomicResult::~AtomicResult()
|
AtomicResult::~AtomicResult()
|
||||||
@ -341,20 +339,15 @@ AtomicResult::~AtomicResult()
|
|||||||
FreeHandle();
|
FreeHandle();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_AllocFields)
|
for (size_t i=0; i<=m_AllocSize; i++)
|
||||||
{
|
{
|
||||||
delete [] m_Fields;
|
delete m_Table[i];
|
||||||
m_AllocFields = 0;
|
|
||||||
m_Fields = NULL;
|
|
||||||
}
|
|
||||||
if (m_AllocRows)
|
|
||||||
{
|
|
||||||
for (unsigned int i=0; i<m_AllocRows; i++)
|
|
||||||
delete [] m_Rows[i];
|
|
||||||
delete [] m_Rows;
|
|
||||||
m_AllocRows = 0;
|
|
||||||
m_Rows = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
delete [] m_Table;
|
||||||
|
|
||||||
|
m_Table = NULL;
|
||||||
|
m_IsFree = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int AtomicResult::RowCount()
|
unsigned int AtomicResult::RowCount()
|
||||||
@ -376,14 +369,22 @@ bool AtomicResult::FieldNameToNum(const char *name, unsigned int *columnId)
|
|||||||
{
|
{
|
||||||
for (unsigned int i=0; i<m_FieldCount; i++)
|
for (unsigned int i=0; i<m_FieldCount; i++)
|
||||||
{
|
{
|
||||||
if (strcmp(g_StringPool.GetString(m_Fields[i]), name) == 0)
|
assert(m_Table[i] != NULL);
|
||||||
|
if (strcmp(m_Table[i]->c_str(), name) == 0)
|
||||||
|
{
|
||||||
|
if (columnId)
|
||||||
{
|
{
|
||||||
if (*columnId)
|
|
||||||
*columnId = i;
|
*columnId = i;
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (columnId)
|
||||||
|
{
|
||||||
|
*columnId = -1;
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -392,7 +393,9 @@ const char *AtomicResult::FieldNumToName(unsigned int num)
|
|||||||
if (num >= m_FieldCount)
|
if (num >= m_FieldCount)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
return g_StringPool.GetString(m_Fields[num]);
|
assert(m_Table[num] != NULL);
|
||||||
|
|
||||||
|
return m_Table[num]->c_str();
|
||||||
}
|
}
|
||||||
|
|
||||||
double AtomicResult::GetDouble(unsigned int columnId)
|
double AtomicResult::GetDouble(unsigned int columnId)
|
||||||
@ -429,7 +432,11 @@ const char *AtomicResult::GetString(unsigned int columnId)
|
|||||||
if (columnId >= m_FieldCount)
|
if (columnId >= m_FieldCount)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
return g_StringPool.GetString(m_Rows[m_CurRow][columnId]);
|
size_t idx = (m_CurRow * m_FieldCount) + columnId;
|
||||||
|
|
||||||
|
assert(m_Table[idx] != NULL);
|
||||||
|
|
||||||
|
return m_Table[idx]->c_str();
|
||||||
}
|
}
|
||||||
|
|
||||||
IResultRow *AtomicResult::GetRow()
|
IResultRow *AtomicResult::GetRow()
|
||||||
@ -439,7 +446,7 @@ IResultRow *AtomicResult::GetRow()
|
|||||||
|
|
||||||
bool AtomicResult::IsDone()
|
bool AtomicResult::IsDone()
|
||||||
{
|
{
|
||||||
if (m_CurRow >= m_RowCount)
|
if (m_CurRow > m_RowCount)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
@ -456,19 +463,6 @@ void AtomicResult::_InternalClear()
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
m_IsFree = true;
|
m_IsFree = true;
|
||||||
|
|
||||||
g_StringPool.StartHardLock();
|
|
||||||
|
|
||||||
for (unsigned int i=0; i<m_FieldCount; i++)
|
|
||||||
g_StringPool.FreeString(m_Fields[i]);
|
|
||||||
|
|
||||||
for (unsigned int i=0; i<m_RowCount; i++)
|
|
||||||
{
|
|
||||||
for (unsigned int j=0; j<m_FieldCount; j++)
|
|
||||||
g_StringPool.FreeString(m_Rows[i][j]);
|
|
||||||
}
|
|
||||||
|
|
||||||
g_StringPool.StopHardLock();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void AtomicResult::FreeHandle()
|
void AtomicResult::FreeHandle()
|
||||||
@ -487,176 +481,50 @@ void AtomicResult::CopyFrom(IResultSet *rs)
|
|||||||
|
|
||||||
m_FieldCount = rs->FieldCount();
|
m_FieldCount = rs->FieldCount();
|
||||||
m_RowCount = rs->RowCount();
|
m_RowCount = rs->RowCount();
|
||||||
if (m_RowCount > m_AllocRows)
|
m_CurRow = 1;
|
||||||
{
|
|
||||||
/** allocate new array, zero it */
|
|
||||||
stridx_t **newRows = new stridx_t *[m_RowCount];
|
|
||||||
memset(newRows, 0, m_RowCount * sizeof(stridx_t *));
|
|
||||||
/** if we have a new field count, just delete all the old stuff. */
|
|
||||||
if (m_FieldCount > m_AllocFields)
|
|
||||||
{
|
|
||||||
for (unsigned int i=0; i<m_AllocRows; i++)
|
|
||||||
{
|
|
||||||
delete [] m_Rows[i];
|
|
||||||
newRows[i] = new stridx_t[m_FieldCount];
|
|
||||||
}
|
|
||||||
for (unsigned int i=m_AllocRows; i<m_RowCount; i++)
|
|
||||||
newRows[i] = new stridx_t[m_FieldCount];
|
|
||||||
} else {
|
|
||||||
/** copy the old pointers */
|
|
||||||
memcpy(newRows, m_Rows, m_AllocRows * sizeof(stridx_t *));
|
|
||||||
for (unsigned int i=m_AllocRows; i<m_RowCount; i++)
|
|
||||||
newRows[i] = new stridx_t[m_AllocFields];
|
|
||||||
}
|
|
||||||
delete [] m_Rows;
|
|
||||||
m_Rows = newRows;
|
|
||||||
m_AllocRows = m_RowCount;
|
|
||||||
}
|
|
||||||
if (m_FieldCount > m_AllocFields)
|
|
||||||
{
|
|
||||||
delete [] m_Fields;
|
|
||||||
m_Fields = new stridx_t[m_FieldCount];
|
|
||||||
m_AllocFields = m_FieldCount;
|
|
||||||
}
|
|
||||||
m_CurRow = 0;
|
|
||||||
|
|
||||||
g_StringPool.StartHardLock();
|
size_t newTotal = (m_RowCount * m_FieldCount) + m_FieldCount;
|
||||||
|
if (newTotal > m_AllocSize)
|
||||||
IResultRow *row;
|
|
||||||
unsigned int idx = 0;
|
|
||||||
while (!rs->IsDone())
|
|
||||||
{
|
{
|
||||||
row = rs->GetRow();
|
SourceHook::String **table = new SourceHook::String *[newTotal];
|
||||||
for (size_t i=0; i<m_FieldCount; i++)
|
memset(table, 0, newTotal * sizeof(SourceHook::String *));
|
||||||
m_Rows[idx][i] = g_StringPool.MakeString(row->GetString(i));
|
if (m_Table)
|
||||||
rs->NextRow();
|
{
|
||||||
idx++;
|
memcpy(table, m_Table, m_AllocSize * sizeof(SourceHook::String *));
|
||||||
|
delete [] m_Table;
|
||||||
|
}
|
||||||
|
m_Table = table;
|
||||||
|
m_AllocSize = newTotal;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (unsigned int i=0; i<m_FieldCount; i++)
|
for (unsigned int i=0; i<m_FieldCount; i++)
|
||||||
m_Fields[i] = g_StringPool.MakeString(rs->FieldNumToName(i));
|
|
||||||
|
|
||||||
g_StringPool.StopHardLock();
|
|
||||||
}
|
|
||||||
|
|
||||||
/*********************
|
|
||||||
* STRING POOL STUFF *
|
|
||||||
*********************/
|
|
||||||
|
|
||||||
StringPool::StringPool()
|
|
||||||
{
|
|
||||||
m_mutex = NULL;
|
|
||||||
m_stoplock = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
StringPool::~StringPool()
|
|
||||||
{
|
|
||||||
if (m_stoplock)
|
|
||||||
StopHardLock();
|
|
||||||
|
|
||||||
if (m_mutex)
|
|
||||||
UnsetMutex();
|
|
||||||
|
|
||||||
for (size_t i=0; i<m_Strings.size(); i++)
|
|
||||||
delete m_Strings[i];
|
|
||||||
|
|
||||||
m_Strings.clear();
|
|
||||||
m_UseTable.clear();
|
|
||||||
while (!m_FreeStrings.empty())
|
|
||||||
m_FreeStrings.pop();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool StringPool::IsThreadable()
|
|
||||||
{
|
|
||||||
return (m_mutex != NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
const char *StringPool::GetString(stridx_t idx)
|
|
||||||
{
|
|
||||||
if (idx < 0 || idx >= (int)m_Strings.size() || !m_UseTable[idx])
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
return m_Strings[idx]->c_str();
|
|
||||||
}
|
|
||||||
|
|
||||||
void StringPool::FreeString(stridx_t idx)
|
|
||||||
{
|
|
||||||
if (idx < 0 || idx >= (int)m_Strings.size())
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (!m_stoplock && m_mutex)
|
|
||||||
m_mutex->Lock();
|
|
||||||
|
|
||||||
if (m_UseTable[idx])
|
|
||||||
{
|
{
|
||||||
m_FreeStrings.push(idx);
|
if (m_Table[i])
|
||||||
m_UseTable[idx] = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!m_stoplock && m_mutex)
|
|
||||||
m_mutex->Unlock();
|
|
||||||
}
|
|
||||||
|
|
||||||
stridx_t StringPool::MakeString(const char *str)
|
|
||||||
{
|
|
||||||
if (!str)
|
|
||||||
return StringPool::NullString;
|
|
||||||
|
|
||||||
if (!m_stoplock && m_mutex)
|
|
||||||
m_mutex->Lock();
|
|
||||||
|
|
||||||
stridx_t idx;
|
|
||||||
|
|
||||||
if (m_FreeStrings.empty())
|
|
||||||
{
|
{
|
||||||
idx = static_cast<stridx_t>(m_Strings.size());
|
m_Table[i]->assign(rs->FieldNumToName(i));
|
||||||
SourceHook::String *shString = new SourceHook::String(str);
|
|
||||||
m_Strings.push_back(shString);
|
|
||||||
m_UseTable.push_back(1);
|
|
||||||
} else {
|
} else {
|
||||||
idx = m_FreeStrings.front();
|
m_Table[i] = new SourceHook::String(rs->FieldNumToName(i));
|
||||||
m_FreeStrings.pop();
|
}
|
||||||
m_UseTable[idx] = 1;
|
|
||||||
m_Strings[idx]->assign(str);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!m_stoplock && m_mutex)
|
IResultRow *row;
|
||||||
m_mutex->Unlock();
|
unsigned int idx = m_FieldCount;
|
||||||
|
while (!rs->IsDone())
|
||||||
return idx;
|
|
||||||
}
|
|
||||||
|
|
||||||
void StringPool::SetMutex(IMutex *m)
|
|
||||||
{
|
|
||||||
m_mutex = m;
|
|
||||||
}
|
|
||||||
|
|
||||||
void StringPool::UnsetMutex()
|
|
||||||
{
|
|
||||||
if (m_mutex)
|
|
||||||
{
|
{
|
||||||
m_mutex->DestroyThis();
|
row = rs->GetRow();
|
||||||
m_mutex = NULL;
|
for (unsigned int i=0; i<m_FieldCount; i++,idx++)
|
||||||
|
{
|
||||||
|
if (m_Table[idx])
|
||||||
|
{
|
||||||
|
m_Table[idx]->assign(row->GetString(i));
|
||||||
|
} else {
|
||||||
|
m_Table[idx] = new SourceHook::String(row->GetString(i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
rs->NextRow();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void StringPool::StartHardLock()
|
|
||||||
{
|
|
||||||
if (m_stoplock)
|
|
||||||
return;
|
|
||||||
|
|
||||||
m_mutex->Lock();
|
|
||||||
m_stoplock = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void StringPool::StopHardLock()
|
|
||||||
{
|
|
||||||
if (!m_stoplock)
|
|
||||||
return;
|
|
||||||
|
|
||||||
m_mutex->Unlock();
|
|
||||||
m_stoplock = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
AMX_NATIVE_INFO g_ThreadSqlNatives[] =
|
AMX_NATIVE_INFO g_ThreadSqlNatives[] =
|
||||||
{
|
{
|
||||||
|
@ -14,32 +14,6 @@ struct QueuedResultInfo
|
|||||||
bool query_success;
|
bool query_success;
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef int stridx_t;
|
|
||||||
|
|
||||||
class StringPool
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
StringPool();
|
|
||||||
~StringPool();
|
|
||||||
void SetMutex(IMutex *m);
|
|
||||||
void UnsetMutex();
|
|
||||||
bool IsThreadable();
|
|
||||||
public:
|
|
||||||
stridx_t MakeString(const char *str);
|
|
||||||
void FreeString(stridx_t idx);
|
|
||||||
const char *GetString(stridx_t idx);
|
|
||||||
void StartHardLock();
|
|
||||||
void StopHardLock();
|
|
||||||
public:
|
|
||||||
static const int NullString = -1;
|
|
||||||
private:
|
|
||||||
CStack<stridx_t> m_FreeStrings;
|
|
||||||
CVector<stridx_t> m_UseTable;
|
|
||||||
CVector<SourceHook::String *> m_Strings;
|
|
||||||
IMutex *m_mutex;
|
|
||||||
bool m_stoplock;
|
|
||||||
};
|
|
||||||
|
|
||||||
class AtomicResult :
|
class AtomicResult :
|
||||||
public IResultSet,
|
public IResultSet,
|
||||||
public IResultRow
|
public IResultRow
|
||||||
@ -73,10 +47,8 @@ private:
|
|||||||
private:
|
private:
|
||||||
unsigned int m_RowCount;
|
unsigned int m_RowCount;
|
||||||
unsigned int m_FieldCount;
|
unsigned int m_FieldCount;
|
||||||
unsigned int m_AllocFields;
|
size_t m_AllocSize;
|
||||||
unsigned int m_AllocRows;
|
SourceHook::String **m_Table;
|
||||||
stridx_t *m_Fields;
|
|
||||||
stridx_t **m_Rows;
|
|
||||||
unsigned int m_CurRow;
|
unsigned int m_CurRow;
|
||||||
bool m_IsFree;
|
bool m_IsFree;
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user