380 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			380 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/**
 | 
						|
 * vim: set ts=4 :
 | 
						|
 * =============================================================================
 | 
						|
 * SourcePawn
 | 
						|
 * Copyright (C) 2004-2009 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 "NativeInvoker.h"
 | 
						|
#include "ShareSys.h"
 | 
						|
 | 
						|
NativeInterface g_NInvoke;
 | 
						|
 | 
						|
NativeInvoker::NativeInvoker()
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
NativeInvoker::~NativeInvoker()
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
const char *NativeInterface::GetInterfaceName()
 | 
						|
{
 | 
						|
	return SMINTERFACE_NINVOKE_NAME;
 | 
						|
}
 | 
						|
 | 
						|
unsigned int NativeInterface::GetInterfaceVersion()
 | 
						|
{
 | 
						|
	return SMINTERFACE_NINVOKE_VERSION;
 | 
						|
}
 | 
						|
 | 
						|
void NativeInterface::OnSourceModAllInitialized()
 | 
						|
{
 | 
						|
	g_ShareSys.AddInterface(NULL, &g_NInvoke);
 | 
						|
}
 | 
						|
 | 
						|
IPluginRuntime *NativeInterface::CreateRuntime(const char *name, size_t bytes)
 | 
						|
{
 | 
						|
	return g_pSourcePawn2->CreateEmptyRuntime(name, bytes);
 | 
						|
}
 | 
						|
 | 
						|
INativeInvoker *NativeInterface::CreateInvoker()
 | 
						|
{
 | 
						|
	return new NativeInvoker();
 | 
						|
}
 | 
						|
 | 
						|
bool NativeInvoker::Start(IPluginContext *pContext, const char *name)
 | 
						|
{
 | 
						|
	NativeEntry *entry;
 | 
						|
	
 | 
						|
	entry = g_ShareSys.FindNative(name);
 | 
						|
	if (entry == NULL)
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	native = NULL;
 | 
						|
	if (entry->replacement.owner != NULL)
 | 
						|
	{
 | 
						|
		native = entry->replacement.func;
 | 
						|
	}
 | 
						|
	else if (entry->owner != NULL)
 | 
						|
	{
 | 
						|
		native = entry->func;
 | 
						|
	}
 | 
						|
 | 
						|
	if (native == NULL)
 | 
						|
	{
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	this->pContext = pContext;
 | 
						|
 | 
						|
	m_curparam = 0;
 | 
						|
	m_errorstate = SP_ERROR_NONE;
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
cell_t NativeInvoker::PushCell(cell_t cell)
 | 
						|
{
 | 
						|
	if (m_curparam >= SP_MAX_EXEC_PARAMS)
 | 
						|
	{
 | 
						|
		return SetError(SP_ERROR_PARAMS_MAX);
 | 
						|
	}
 | 
						|
 | 
						|
	m_info[m_curparam].marked = false;
 | 
						|
	m_params[m_curparam] = cell;
 | 
						|
	m_curparam++;
 | 
						|
 | 
						|
	return SP_ERROR_NONE;
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::PushCellByRef(cell_t *cell, int flags)
 | 
						|
{
 | 
						|
	return PushArray(cell, 1, flags);
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::PushFloat(float number)
 | 
						|
{
 | 
						|
	cell_t val = *(cell_t *)&number;
 | 
						|
 | 
						|
	return PushCell(val);
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::PushFloatByRef(float *number, int flags)
 | 
						|
{
 | 
						|
	return PushCellByRef((cell_t *)number, flags);
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::PushArray(cell_t *inarray, unsigned int cells, int copyback)
 | 
						|
{
 | 
						|
	if (m_curparam >= SP_MAX_EXEC_PARAMS)
 | 
						|
	{
 | 
						|
		return SetError(SP_ERROR_PARAMS_MAX);
 | 
						|
	}
 | 
						|
 | 
						|
	ParamInfo *info = &m_info[m_curparam];
 | 
						|
 | 
						|
	info->flags = inarray ? copyback : 0;
 | 
						|
	info->marked = true;
 | 
						|
	info->size = cells;
 | 
						|
	info->str.is_sz = false;
 | 
						|
	info->orig_addr = inarray;
 | 
						|
 | 
						|
	m_curparam++;
 | 
						|
 | 
						|
	return SP_ERROR_NONE;
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::PushString(const char *string)
 | 
						|
{
 | 
						|
	return _PushString(string, SM_PARAM_STRING_COPY, 0, strlen(string)+1);
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::PushStringEx(char *buffer, size_t length, int sz_flags, int cp_flags)
 | 
						|
{
 | 
						|
	return _PushString(buffer, sz_flags, cp_flags, length);
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::_PushString(const char *string, int sz_flags, int cp_flags, size_t len)
 | 
						|
{
 | 
						|
	if (m_curparam >= SP_MAX_EXEC_PARAMS)
 | 
						|
	{
 | 
						|
		return SetError(SP_ERROR_PARAMS_MAX);
 | 
						|
	}
 | 
						|
 | 
						|
	ParamInfo *info = &m_info[m_curparam];
 | 
						|
 | 
						|
	info->marked = true;
 | 
						|
	info->orig_addr = (cell_t *)string;
 | 
						|
	info->flags = cp_flags;
 | 
						|
	info->size = len;
 | 
						|
	info->str.sz_flags = sz_flags;
 | 
						|
	info->str.is_sz = true;
 | 
						|
 | 
						|
	m_curparam++;
 | 
						|
 | 
						|
	return SP_ERROR_NONE;
 | 
						|
}
 | 
						|
 | 
						|
void NativeInvoker::Cancel()
 | 
						|
{
 | 
						|
	if (pContext == NULL)
 | 
						|
	{
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	m_errorstate = SP_ERROR_NONE;
 | 
						|
	m_curparam = 0;
 | 
						|
	pContext = NULL;
 | 
						|
	native = NULL;
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::SetError(int err)
 | 
						|
{
 | 
						|
	m_errorstate = err;
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
int NativeInvoker::Invoke(cell_t *result)
 | 
						|
{
 | 
						|
	int err = SP_ERROR_NONE;
 | 
						|
 | 
						|
	if (pContext == NULL)
 | 
						|
	{
 | 
						|
		return SP_ERROR_INVALID_NATIVE;
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_errorstate != SP_ERROR_NONE)
 | 
						|
	{
 | 
						|
		err = m_errorstate;
 | 
						|
		Cancel();
 | 
						|
		return err;
 | 
						|
	}
 | 
						|
 | 
						|
	cell_t tresult;
 | 
						|
 | 
						|
	if (result == NULL)
 | 
						|
	{
 | 
						|
		result = &tresult;
 | 
						|
	}
 | 
						|
 | 
						|
	//This is for re-entrancy!
 | 
						|
	IPluginContext *ctx = pContext;
 | 
						|
	cell_t _temp_params[SP_MAX_EXEC_PARAMS + 1];
 | 
						|
	cell_t *temp_params = &_temp_params[1];
 | 
						|
	ParamInfo temp_info[SP_MAX_EXEC_PARAMS];
 | 
						|
	unsigned int numparams = m_curparam;
 | 
						|
	unsigned int i;
 | 
						|
	bool docopies = true;
 | 
						|
 | 
						|
	if (numparams)
 | 
						|
	{
 | 
						|
		//Save the info locally, then reset it for re-entrant calls.
 | 
						|
		memcpy(temp_info, m_info, numparams * sizeof(ParamInfo));
 | 
						|
	}
 | 
						|
	m_curparam = 0;
 | 
						|
	pContext = NULL;
 | 
						|
 | 
						|
	/* Initialize 0th parameter */
 | 
						|
	_temp_params[0] = numparams;
 | 
						|
 | 
						|
	/* Browse the parameters and build arrays */
 | 
						|
	for (i = 0; i < numparams; i++)
 | 
						|
	{
 | 
						|
		/* Is this marked as an array? */
 | 
						|
		if (temp_info[i].marked)
 | 
						|
		{
 | 
						|
			if (!temp_info[i].str.is_sz)
 | 
						|
			{
 | 
						|
				/* Allocate a normal/generic array */
 | 
						|
				if ((err = ctx->HeapAlloc(temp_info[i].size, 
 | 
						|
										  &temp_info[i].local_addr,
 | 
						|
										  &temp_info[i].phys_addr))
 | 
						|
					!= SP_ERROR_NONE)
 | 
						|
				{
 | 
						|
					break;
 | 
						|
				}
 | 
						|
				if (temp_info[i].orig_addr)
 | 
						|
				{
 | 
						|
					memcpy(temp_info[i].phys_addr, temp_info[i].orig_addr, sizeof(cell_t) * temp_info[i].size);
 | 
						|
				}
 | 
						|
			}
 | 
						|
			else
 | 
						|
			{
 | 
						|
				/* Calculate cells required for the string */
 | 
						|
				size_t cells = (temp_info[i].size + sizeof(cell_t) - 1) / sizeof(cell_t);
 | 
						|
 | 
						|
				/* Allocate the buffer */
 | 
						|
				if ((err = ctx->HeapAlloc(cells,
 | 
						|
										  &temp_info[i].local_addr,
 | 
						|
										  &temp_info[i].phys_addr))
 | 
						|
					!= SP_ERROR_NONE)
 | 
						|
				{
 | 
						|
					break;
 | 
						|
				}
 | 
						|
				/* Copy original string if necessary */
 | 
						|
				if ((temp_info[i].str.sz_flags & SM_PARAM_STRING_COPY) && (temp_info[i].orig_addr != NULL))
 | 
						|
				{
 | 
						|
					/* Cut off UTF-8 properly */
 | 
						|
					if (temp_info[i].str.sz_flags & SM_PARAM_STRING_UTF8)
 | 
						|
					{
 | 
						|
						if ((err = ctx->StringToLocalUTF8(temp_info[i].local_addr, 
 | 
						|
														  temp_info[i].size, 
 | 
						|
														  (const char *)temp_info[i].orig_addr,
 | 
						|
														  NULL))
 | 
						|
							!= SP_ERROR_NONE)
 | 
						|
						{
 | 
						|
							break;
 | 
						|
						}
 | 
						|
					}
 | 
						|
					/* Copy a binary blob */
 | 
						|
					else if (temp_info[i].str.sz_flags & SM_PARAM_STRING_BINARY)
 | 
						|
					{
 | 
						|
						memmove(temp_info[i].phys_addr, temp_info[i].orig_addr, temp_info[i].size);
 | 
						|
					}
 | 
						|
					/* Copy ASCII characters */
 | 
						|
					else
 | 
						|
					{
 | 
						|
						if ((err = ctx->StringToLocal(temp_info[i].local_addr,
 | 
						|
													  temp_info[i].size,
 | 
						|
													  (const char *)temp_info[i].orig_addr))
 | 
						|
							!= SP_ERROR_NONE)
 | 
						|
						{
 | 
						|
							break;
 | 
						|
						}
 | 
						|
					}
 | 
						|
				}
 | 
						|
			} /* End array/string calculation */
 | 
						|
			/* Update the pushed parameter with the byref local address */
 | 
						|
			temp_params[i] = temp_info[i].local_addr;
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			/* Just copy the value normally */
 | 
						|
			temp_params[i] = m_params[i];
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/* Make the call if we can */
 | 
						|
	if (err == SP_ERROR_NONE)
 | 
						|
	{
 | 
						|
		*result = native(ctx, _temp_params);
 | 
						|
		if (ctx->GetLastNativeError() != SP_ERROR_NONE)
 | 
						|
		{
 | 
						|
			docopies = false;
 | 
						|
			ctx->ClearLastNativeError();
 | 
						|
		}
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		docopies = false;
 | 
						|
	}
 | 
						|
 | 
						|
	/* i should be equal to the last valid parameter + 1 */
 | 
						|
	while (i--)
 | 
						|
	{
 | 
						|
		if (!temp_info[i].marked)
 | 
						|
		{
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		if (docopies && (temp_info[i].flags & SM_PARAM_COPYBACK))
 | 
						|
		{
 | 
						|
			if (temp_info[i].orig_addr)
 | 
						|
			{
 | 
						|
				if (temp_info[i].str.is_sz)
 | 
						|
				{
 | 
						|
					memcpy(temp_info[i].orig_addr, temp_info[i].phys_addr, temp_info[i].size);
 | 
						|
 | 
						|
				}
 | 
						|
				else
 | 
						|
				{
 | 
						|
					if (temp_info[i].size == 1)
 | 
						|
					{
 | 
						|
						*temp_info[i].orig_addr = *(temp_info[i].phys_addr);
 | 
						|
					}
 | 
						|
					else
 | 
						|
					{
 | 
						|
						memcpy(temp_info[i].orig_addr, 
 | 
						|
							temp_info[i].phys_addr, 
 | 
						|
							temp_info[i].size * sizeof(cell_t));
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if ((err = ctx->HeapPop(temp_info[i].local_addr)) != SP_ERROR_NONE)
 | 
						|
		{
 | 
						|
			return err;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return err;
 | 
						|
}
 |