This includes StartMessage(), StartMessageEx() from Core as well as EmitSound(), EmitSentence(), and TE_Send() from SDKTools (Warning: This may potentially cause a minor compatibility problem with plugins that don't check client validity before passing to these natives) --HG-- extra : convert_revision : svn%3A39bc706e-5318-0410-9160-8a85361fbb7c/trunk%402252
		
			
				
	
	
		
			562 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			562 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**
 | |
|  * vim: set ts=4 :
 | |
|  * =============================================================================
 | |
|  * SourceMod SDKTools 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 "tempents.h"
 | |
| #include "CellRecipientFilter.h"
 | |
| #include <IForwardSys.h>
 | |
| 
 | |
| SH_DECL_HOOK5_void(IVEngineServer, PlaybackTempEntity, SH_NOATTRIB, 0, IRecipientFilter &, float, const void *, const SendTable *, int);
 | |
| 
 | |
| CellRecipientFilter g_TERecFilter;
 | |
| TempEntityInfo *g_CurrentTE = NULL;
 | |
| int g_TEPlayers[256];
 | |
| bool tenatives_initialized = false;
 | |
| 
 | |
| /*************************
 | |
| *                        *
 | |
| * Temp Entity Hook Class *
 | |
| *                        *
 | |
| **************************/
 | |
| 
 | |
| void TempEntHooks::Initialize()
 | |
| {
 | |
| 	m_TEHooks = adtfactory->CreateBasicTrie();
 | |
| 	plsys->AddPluginsListener(this);
 | |
| 	tenatives_initialized = true;
 | |
| }
 | |
| 
 | |
| void TempEntHooks::Shutdown()
 | |
| {
 | |
| 	if (!tenatives_initialized)
 | |
| 	{
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	plsys->RemovePluginsListener(this);
 | |
| 	SourceHook::List<TEHookInfo *>::iterator iter;
 | |
| 	for (iter=m_HookInfo.begin(); iter!=m_HookInfo.end(); iter++)
 | |
| 	{
 | |
| 		delete (*iter);
 | |
| 	}
 | |
| 	if (m_HookCount)
 | |
| 	{
 | |
| 		m_HookCount = 1;
 | |
| 		_DecRefCounter();
 | |
| 	}
 | |
| 	m_TEHooks->Destroy();
 | |
| 	tenatives_initialized = false;
 | |
| }
 | |
| 
 | |
| void TempEntHooks::OnPluginUnloaded(IPlugin *plugin)
 | |
| {
 | |
| 	SourceHook::List<TEHookInfo *>::iterator iter = m_HookInfo.begin();
 | |
| 	IPluginContext *pContext = plugin->GetBaseContext();
 | |
| 
 | |
| 	/* For each hook list... */
 | |
| 	while (iter != m_HookInfo.end())
 | |
| 	{
 | |
| 		SourceHook::List<IPluginFunction *>::iterator f_iter = (*iter)->lst.begin();
 | |
| 
 | |
| 		/* Find the hooks on the given temp entity */
 | |
| 		while (f_iter != (*iter)->lst.end())
 | |
| 		{
 | |
| 			/* If it matches, remove it and dec the ref count */
 | |
| 			if ((*f_iter)->GetParentContext() == pContext)
 | |
| 			{
 | |
| 				f_iter = (*iter)->lst.erase(f_iter);
 | |
| 				_DecRefCounter();
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				f_iter++;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		/* If there are no more hooks left, we can safely 
 | |
| 		 * remove it from the cache and remove its list.
 | |
| 		 */
 | |
| 		if ((*iter)->lst.size() == 0)
 | |
| 		{
 | |
| 			m_TEHooks->Delete((*iter)->te->GetName());
 | |
| 			delete (*iter);
 | |
| 			iter = m_HookInfo.erase(iter);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			iter++;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TempEntHooks::_IncRefCounter()
 | |
| {
 | |
| 	if (m_HookCount++ == 0)
 | |
| 	{
 | |
| 		SH_ADD_HOOK_MEMFUNC(IVEngineServer, PlaybackTempEntity, engine, this, &TempEntHooks::OnPlaybackTempEntity, false);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TempEntHooks::_DecRefCounter()
 | |
| {
 | |
| 	if (--m_HookCount == 0)
 | |
| 	{
 | |
| 		SH_REMOVE_HOOK_MEMFUNC(IVEngineServer, PlaybackTempEntity, engine, this, &TempEntHooks::OnPlaybackTempEntity, false);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| size_t TempEntHooks::_FillInPlayers(int *pl_array, IRecipientFilter *pFilter)
 | |
| {
 | |
| 	size_t size = static_cast<size_t>(pFilter->GetRecipientCount());
 | |
| 
 | |
| 	for (size_t i=0; i<size; i++)
 | |
| 	{
 | |
| 		pl_array[i] = pFilter->GetRecipientIndex(i);
 | |
| 	}
 | |
| 
 | |
| 	return size;
 | |
| }
 | |
| 
 | |
| bool TempEntHooks::AddHook(const char *name, IPluginFunction *pFunc)
 | |
| {
 | |
| 	TEHookInfo *pInfo;
 | |
| 
 | |
| 	if (m_TEHooks->Retrieve(name, reinterpret_cast<void **>(&pInfo)))
 | |
| 	{
 | |
| 		pInfo->lst.push_back(pFunc);
 | |
| 	} else {
 | |
| 		TempEntityInfo *te;
 | |
| 		if (!(te=g_TEManager.GetTempEntityInfo(name)))
 | |
| 		{
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		pInfo = new TEHookInfo;
 | |
| 		pInfo->te = te;
 | |
| 		pInfo->lst.push_back(pFunc);
 | |
| 
 | |
| 		m_TEHooks->Insert(name, reinterpret_cast<void *>(pInfo));
 | |
| 		m_HookInfo.push_back(pInfo);
 | |
| 	}
 | |
| 
 | |
| 	_IncRefCounter();
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool TempEntHooks::RemoveHook(const char *name, IPluginFunction *pFunc)
 | |
| {
 | |
| 	TEHookInfo *pInfo;
 | |
| 
 | |
| 	if (m_TEHooks->Retrieve(name, reinterpret_cast<void **>(&pInfo)))
 | |
| 	{
 | |
| 		SourceHook::List<IPluginFunction *>::iterator iter;
 | |
| 		if ((iter=pInfo->lst.find(pFunc)) != pInfo->lst.end())
 | |
| 		{
 | |
| 			pInfo->lst.erase(iter);
 | |
| 			if (pInfo->lst.empty())
 | |
| 			{
 | |
| 				m_HookInfo.remove(pInfo);
 | |
| 				m_TEHooks->Delete(name);
 | |
| 				delete pInfo;
 | |
| 			}
 | |
| 			_DecRefCounter();
 | |
| 		} else {
 | |
| 			return false;
 | |
| 		}
 | |
| 	} else {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void TempEntHooks::OnPlaybackTempEntity(IRecipientFilter &filter, float delay, const void *pSender, const SendTable *pST, int classID)
 | |
| {
 | |
| 	TEHookInfo *pInfo;
 | |
| 	const char *name = g_TEManager.GetNameFromThisPtr(const_cast<void *>(pSender));
 | |
| 
 | |
| 	if (m_TEHooks->Retrieve(name, reinterpret_cast<void **>(&pInfo)))
 | |
| 	{
 | |
| 		SourceHook::List<IPluginFunction *>::iterator iter;
 | |
| 		IPluginFunction *pFunc;
 | |
| 		size_t size;
 | |
| 		cell_t res = static_cast<ResultType>(Pl_Continue);
 | |
| 
 | |
| 		TempEntityInfo *oldinfo = g_CurrentTE;
 | |
| 		g_CurrentTE = pInfo->te;
 | |
| 		size = _FillInPlayers(g_TEPlayers, &filter);
 | |
| 
 | |
| 		for (iter=pInfo->lst.begin(); iter!=pInfo->lst.end(); iter++)
 | |
| 		{
 | |
| 			pFunc = (*iter);
 | |
| 			pFunc->PushString(name);
 | |
| 			pFunc->PushArray(g_TEPlayers, size);
 | |
| 			pFunc->PushCell(size);
 | |
| 			pFunc->PushFloat(delay);
 | |
| 			pFunc->Execute(&res);
 | |
| 
 | |
| 			if (res != Pl_Continue)
 | |
| 			{
 | |
| 				g_CurrentTE = oldinfo;
 | |
| 				RETURN_META(MRES_SUPERCEDE);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		g_CurrentTE = oldinfo;
 | |
| 		RETURN_META(MRES_IGNORED);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /**********************
 | |
| *                     *
 | |
| * Temp Entity Natives *
 | |
| *                     *
 | |
| ***********************/
 | |
| 
 | |
| TempEntHooks s_TempEntHooks;
 | |
| 
 | |
| static cell_t smn_TEStart(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 
 | |
| 	char *name;
 | |
| 	pContext->LocalToString(params[1], &name);
 | |
| 
 | |
| 	g_CurrentTE = g_TEManager.GetTempEntityInfo(name);
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Invalid TempEntity name: \"%s\"", name);
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static cell_t smn_TEWriteNum(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	char *prop;
 | |
| 	pContext->LocalToString(params[1], &prop);
 | |
| 
 | |
| 	if (!g_CurrentTE->TE_SetEntData(prop, params[2]))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Temp entity property \"%s\" not found", prop);
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static cell_t smn_TEReadNum(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	char *prop;
 | |
| 	int val;
 | |
| 	pContext->LocalToString(params[1], &prop);
 | |
| 
 | |
| 	if (!g_CurrentTE->TE_GetEntData(prop, &val))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Temp entity property \"%s\" not found", prop);
 | |
| 	}
 | |
| 
 | |
| 	return val;
 | |
| }
 | |
| 
 | |
| static cell_t smn_TE_WriteFloat(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	char *prop;
 | |
| 	pContext->LocalToString(params[1], &prop);
 | |
| 
 | |
| 	if (!g_CurrentTE->TE_SetEntDataFloat(prop, sp_ctof(params[2])))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Temp entity property \"%s\" not found", prop);
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static cell_t smn_TE_ReadFloat(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	char *prop;
 | |
| 	float val;
 | |
| 	pContext->LocalToString(params[1], &prop);
 | |
| 
 | |
| 	if (!g_CurrentTE->TE_GetEntDataFloat(prop, &val))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Temp entity property \"%s\" not found", prop);
 | |
| 	}
 | |
| 
 | |
| 	return sp_ftoc(val);
 | |
| }
 | |
| 
 | |
| static cell_t smn_TEWriteVector(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	char *prop;
 | |
| 	pContext->LocalToString(params[1], &prop);
 | |
| 
 | |
| 	cell_t *addr;
 | |
| 	pContext->LocalToPhysAddr(params[2], &addr);
 | |
| 	float vec[3] = {sp_ctof(addr[0]), sp_ctof(addr[1]), sp_ctof(addr[2])};
 | |
| 
 | |
| 	if (!g_CurrentTE->TE_SetEntDataVector(prop, vec))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Temp entity property \"%s\" not found", prop);
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static cell_t smn_TEReadVector(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	char *prop;
 | |
| 	pContext->LocalToString(params[1], &prop);
 | |
| 
 | |
| 	cell_t *addr;
 | |
| 	float vec[3];
 | |
| 	pContext->LocalToPhysAddr(params[2], &addr);
 | |
| 
 | |
| 	if (!g_CurrentTE->TE_GetEntDataVector(prop, vec))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Temp entity property \"%s\" not found", prop);
 | |
| 	}
 | |
| 
 | |
| 	addr[0] = sp_ftoc(vec[0]);
 | |
| 	addr[1] = sp_ftoc(vec[1]);
 | |
| 	addr[2] = sp_ftoc(vec[2]);
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static cell_t smn_TEWriteFloatArray(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	char *prop;
 | |
| 	pContext->LocalToString(params[1], &prop);
 | |
| 
 | |
| 	cell_t *addr;
 | |
| 	pContext->LocalToPhysAddr(params[2], &addr);
 | |
| 
 | |
| 	if (!g_CurrentTE->TE_SetEntDataFloatArray(prop, addr, params[3]))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Temp entity property \"%s\" not found", prop);
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static cell_t smn_TESend(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	cell_t *cl_array;
 | |
| 	unsigned int numClients;
 | |
| 	int client;
 | |
| 	IGamePlayer *pPlayer = NULL;
 | |
| 
 | |
| 	pContext->LocalToPhysAddr(params[1], &cl_array);
 | |
| 	numClients = params[2];
 | |
| 
 | |
| 	/* Client validation */
 | |
| 	for (unsigned int i = 0; i < numClients; i++)
 | |
| 	{
 | |
| 		client = cl_array[i];
 | |
| 		pPlayer = playerhelpers->GetGamePlayer(client);
 | |
| 
 | |
| 		if (!pPlayer)
 | |
| 		{
 | |
| 			return pContext->ThrowNativeError("Client index %d is invalid", client);
 | |
| 		} else if (!pPlayer->IsInGame()) {
 | |
| 			return pContext->ThrowNativeError("Client %d is not connected", client);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	g_TERecFilter.Reset();
 | |
| 	g_TERecFilter.Initialize(cl_array, numClients);
 | |
| 
 | |
| 	g_CurrentTE->Send(g_TERecFilter, sp_ctof(params[3]));
 | |
| 	g_CurrentTE = NULL;
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static cell_t smn_TEIsValidProp(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 	if (!g_CurrentTE)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("No TempEntity call is in progress");
 | |
| 	}
 | |
| 
 | |
| 	char *prop;
 | |
| 	pContext->LocalToString(params[1], &prop);
 | |
| 
 | |
| 	return g_CurrentTE->IsValidProp(prop) ? 1 : 0;
 | |
| }
 | |
| 
 | |
| static cell_t smn_AddTempEntHook(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	char *name;
 | |
| 	IPluginFunction *pFunc;
 | |
| 
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 
 | |
| 	pContext->LocalToString(params[1], &name);
 | |
| 	pFunc = pContext->GetFunctionById(params[2]);
 | |
| 	if (!pFunc)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Invalid function id (%X)", params[2]);
 | |
| 	}
 | |
| 
 | |
| 	if (!s_TempEntHooks.AddHook(name, pFunc))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Invalid TempEntity name: \"%s\"", name);
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static cell_t smn_RemoveTempEntHook(IPluginContext *pContext, const cell_t *params)
 | |
| {
 | |
| 	char *name;
 | |
| 	IPluginFunction *pFunc;
 | |
| 
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("TempEntity System unsupported or not available, file a bug report");
 | |
| 	}
 | |
| 
 | |
| 	pContext->LocalToString(params[1], &name);
 | |
| 	pFunc = pContext->GetFunctionById(params[2]);
 | |
| 	if (!pFunc)
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Invalid function id (%X)", params[2]);
 | |
| 	}
 | |
| 
 | |
| 	if (!s_TempEntHooks.RemoveHook(name, pFunc))
 | |
| 	{
 | |
| 		return pContext->ThrowNativeError("Invalid hooked TempEntity name or function");
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| sp_nativeinfo_t g_TENatives[] = 
 | |
| {
 | |
| 	{"TE_Start",				smn_TEStart},
 | |
| 	{"TE_WriteNum",				smn_TEWriteNum},
 | |
| 	{"TE_ReadNum",				smn_TEReadNum},
 | |
| 	{"TE_WriteFloat",			smn_TE_WriteFloat},
 | |
| 	{"TE_ReadFloat",			smn_TE_ReadFloat},
 | |
| 	{"TE_WriteVector",			smn_TEWriteVector},
 | |
| 	{"TE_ReadVector",			smn_TEReadVector},
 | |
| 	{"TE_WriteAngles",			smn_TEWriteVector},
 | |
| 	{"TE_Send",					smn_TESend},
 | |
| 	{"TE_IsValidProp",			smn_TEIsValidProp},
 | |
| 	{"TE_WriteFloatArray",		smn_TEWriteFloatArray},
 | |
| 	{"AddTempEntHook",			smn_AddTempEntHook},
 | |
| 	{"RemoveTempEntHook",		smn_RemoveTempEntHook},
 | |
| 	{NULL,						NULL}
 | |
| };
 |