2008-08-08 02:17:37 +02:00
|
|
|
/**
|
|
|
|
* vim: set ts=4 :
|
|
|
|
* =============================================================================
|
|
|
|
* SourceMod Sample Extension
|
|
|
|
* Copyright (C) 2004-2008 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* 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 <http://www.sourcemod.net/license.php>.
|
|
|
|
*
|
|
|
|
* Version: $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "extension.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file extension.cpp
|
|
|
|
* @brief Implement extension code here.
|
|
|
|
*/
|
|
|
|
|
|
|
|
Structs g_Structs; /**< Global singleton for extension's main interface */
|
|
|
|
|
|
|
|
SMEXT_LINK(&g_Structs);
|
|
|
|
|
|
|
|
HandleType_t g_StructHandle = 0;
|
|
|
|
StructHandler g_StructHandler;
|
|
|
|
|
|
|
|
IGameConfig *conf;
|
|
|
|
|
|
|
|
size_t UTIL_Format(char *buffer, size_t maxlength, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
size_t len = vsnprintf(buffer, maxlength, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
if (len >= maxlength)
|
|
|
|
{
|
|
|
|
buffer[maxlength - 1] = '\0';
|
|
|
|
return (maxlength - 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Structs::SDK_OnAllLoaded()
|
|
|
|
{
|
|
|
|
g_StructHandle = handlesys->CreateType("Struct",
|
|
|
|
&g_StructHandler,
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
myself->GetIdentity(),
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
gameconfs->AddUserConfigHook("Structs", this);
|
|
|
|
|
|
|
|
sharesys->AddNatives(myself, MyNatives);
|
|
|
|
|
|
|
|
|
|
|
|
char error[100];
|
2008-11-01 06:54:19 +01:00
|
|
|
if (!gameconfs->LoadGameConfigFile("structs.gamedata", &conf, error, sizeof(error)))
|
2008-08-08 02:17:37 +02:00
|
|
|
{
|
|
|
|
g_pSM->LogError(myself, "Parsing Failed!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Structs::SDK_OnUnload()
|
|
|
|
{
|
|
|
|
handlesys->RemoveType(g_StructHandle, myself->GetIdentity());
|
2008-11-01 06:47:13 +01:00
|
|
|
gameconfs->CloseGameConfigFile(conf);
|
2008-08-08 02:17:37 +02:00
|
|
|
gameconfs->RemoveUserConfigHook("Structs", this);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Structs::SDK_OnLoad( char *error, size_t maxlength, bool late )
|
|
|
|
{
|
|
|
|
sharesys->AddInterface(myself, &g_StructManager);
|
|
|
|
|
|
|
|
m_typeLookup.insert("int", Member_Int32);
|
|
|
|
m_typeLookup.insert("int*", Member_Int32Pointer);
|
|
|
|
m_typeLookup.insert("float", Member_Float);
|
|
|
|
m_typeLookup.insert("float*", Member_FloatPointer);
|
|
|
|
m_typeLookup.insert("char", Member_Char);
|
|
|
|
m_typeLookup.insert("char*", Member_CharPointer);
|
|
|
|
m_typeLookup.insert("Vector", Member_Vector);
|
|
|
|
m_typeLookup.insert("Vector*", Member_VectorPointer);
|
|
|
|
m_typeLookup.insert("ent", Member_EHandle);
|
|
|
|
m_typeLookup.insert("ent*", Member_EHandlePointer);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Structs::ReadSMC_ParseStart()
|
|
|
|
{
|
|
|
|
m_bInStruct = false;
|
|
|
|
m_currentStruct = NULL;
|
|
|
|
m_bInMember = false;
|
|
|
|
m_currentMember = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceMod::SMCResult Structs::ReadSMC_NewSection( const SMCStates *states, const char *name )
|
|
|
|
{
|
|
|
|
if (!m_bInStruct)
|
|
|
|
{
|
|
|
|
m_currentStruct = new StructInfo();
|
|
|
|
UTIL_Format(m_currentStruct->name, sizeof(m_currentStruct->name), "%s", name);
|
|
|
|
m_bInStruct = true;
|
|
|
|
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_bInMember)
|
|
|
|
{
|
|
|
|
m_bInMember = true;
|
|
|
|
m_currentMember = new MemberInfo();
|
|
|
|
UTIL_Format(m_currentMember->name, sizeof(m_currentMember->name), "%s", name);
|
|
|
|
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_pSM->LogMessage(myself, "Cannot nest within a member: line: %i col: %i", states->line, states->col);
|
|
|
|
|
|
|
|
return SMCResult_HaltFail;
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceMod::SMCResult Structs::ReadSMC_KeyValue( const SMCStates *states, const char *key, const char *value )
|
|
|
|
{
|
|
|
|
if (!m_bInStruct)
|
|
|
|
{
|
|
|
|
//hrm..
|
|
|
|
g_pSM->LogMessage(myself, "Unknown value %s: line: %i col: %i", key, states->line, states->col);
|
|
|
|
return SMCResult_HaltFail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_bInMember)
|
|
|
|
{
|
|
|
|
if (strcmp(key, "type") == 0)
|
|
|
|
{
|
|
|
|
MemberType *pType = m_typeLookup.retrieve(value);
|
|
|
|
|
|
|
|
if (pType == NULL)
|
|
|
|
{
|
|
|
|
//invalid type
|
|
|
|
g_pSM->LogMessage(myself, "Invalid Type: line: %i col: %i", states->line, states->col);
|
|
|
|
return SMCResult_HaltFail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Note all 'int' types are assumed to be 32bit for the moment */
|
|
|
|
m_currentMember->type = *pType;
|
|
|
|
}
|
|
|
|
else if (strcmp(key, "size") == 0)
|
|
|
|
{
|
|
|
|
m_currentMember->size = atoi(value);
|
|
|
|
}
|
|
|
|
#if defined WIN32
|
|
|
|
else if (strcmp(key, "windows") == 0)
|
|
|
|
#else
|
|
|
|
else if (strcmp(key, "linux") == 0)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
m_currentMember->offset = atoi(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (strcmp(key, "size") == 0)
|
|
|
|
{
|
|
|
|
m_currentStruct->size = atoi(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceMod::SMCResult Structs::ReadSMC_LeavingSection( const SMCStates *states )
|
|
|
|
{
|
|
|
|
if (m_bInMember)
|
|
|
|
{
|
|
|
|
/** Check for missing or invalid parameters */
|
|
|
|
if (m_currentMember->type == Member_Unknown || m_currentMember->offset == -1)
|
|
|
|
{
|
|
|
|
g_pSM->LogMessage(myself, "Missing offset or type: line: %i col: %i", states->line, states->col);
|
|
|
|
return SMCResult_HaltFail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_currentMember->size == -1)
|
|
|
|
{
|
|
|
|
switch (m_currentMember->type)
|
|
|
|
{
|
|
|
|
case Member_Float:
|
|
|
|
case Member_FloatPointer:
|
|
|
|
case Member_Vector:
|
|
|
|
case Member_VectorPointer:
|
|
|
|
case Member_EHandle:
|
|
|
|
case Member_EHandlePointer:
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
g_pSM->LogMessage(myself, "Missing size: line: %i col: %i", states->line, states->col);
|
|
|
|
return SMCResult_HaltFail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Work out the int size */
|
|
|
|
if (m_currentMember->type == Member_Int32)
|
|
|
|
{
|
|
|
|
if (m_currentMember->size == 1)
|
|
|
|
{
|
|
|
|
m_currentMember->type = Member_Int8;
|
|
|
|
}
|
|
|
|
else if (m_currentMember->size == 2)
|
|
|
|
{
|
|
|
|
m_currentMember->type = Member_Int16;
|
|
|
|
}
|
|
|
|
else if (m_currentMember->size == 4)
|
|
|
|
{
|
|
|
|
m_currentMember->type = Member_Int32;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_pSM->LogMessage(myself, "Invalid int size %i: line: %i col: %i", m_currentMember->size, states->line, states->col);
|
|
|
|
return SMCResult_HaltFail;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (m_currentMember->type == Member_Int32)
|
|
|
|
{
|
|
|
|
if (m_currentMember->size == 1)
|
|
|
|
{
|
|
|
|
m_currentMember->type = Member_Int8Pointer;
|
|
|
|
}
|
|
|
|
else if (m_currentMember->size == 2)
|
|
|
|
{
|
|
|
|
m_currentMember->type = Member_Int16Pointer;
|
|
|
|
}
|
|
|
|
else if (m_currentMember->size == 4)
|
|
|
|
{
|
|
|
|
m_currentMember->type = Member_Int32Pointer;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_pSM->LogMessage(myself, "Invalid int size %i: line: %i col: %i", m_currentMember->size, states->line, states->col);
|
|
|
|
return SMCResult_HaltFail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_currentStruct->AddMember(m_currentMember->name, m_currentMember);
|
|
|
|
|
|
|
|
m_bInMember = false;
|
|
|
|
m_currentMember = NULL;
|
|
|
|
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_StructManager.AddStruct(m_currentStruct->name, m_currentStruct);
|
|
|
|
|
|
|
|
m_bInStruct = false;
|
|
|
|
m_currentStruct = NULL;
|
|
|
|
|
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|