2007-06-16 10:36:34 +02:00
|
|
|
#include "smsdk_ext.h"
|
|
|
|
#include "extension.h"
|
|
|
|
#include "vdecoder.h"
|
2007-06-20 10:03:43 +02:00
|
|
|
#include "vcallbuilder.h"
|
2007-06-16 10:36:34 +02:00
|
|
|
|
|
|
|
using namespace SourceMod;
|
|
|
|
using namespace SourcePawn;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* For object pointers, the data looks like this instead:
|
2007-06-20 10:03:43 +02:00
|
|
|
* 4 bytes: POINTER TO LATER
|
2007-06-16 10:36:34 +02:00
|
|
|
* + bytes: Object internal data
|
|
|
|
*
|
|
|
|
* We use the virtual stack as extra fake stack space and create a temp object.
|
|
|
|
* If these objects had destructors, we'd need to fake destroy toom of course.
|
|
|
|
* Of course, BinTools only reads the first four bytes and passes the pointer.
|
|
|
|
*/
|
|
|
|
|
|
|
|
size_t ValveParamToBinParam(ValveType type,
|
|
|
|
PassType pass,
|
|
|
|
unsigned int flags,
|
2007-06-20 10:03:43 +02:00
|
|
|
PassInfo *info,
|
|
|
|
bool &needs_extra)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
2007-06-20 10:03:43 +02:00
|
|
|
needs_extra = false;
|
2007-06-16 10:36:34 +02:00
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case Valve_Vector:
|
|
|
|
{
|
|
|
|
size_t mySize = sizeof(Vector *);
|
|
|
|
if (pass == PassType_Basic)
|
|
|
|
{
|
2007-06-17 22:43:21 +02:00
|
|
|
if (flags & PASSFLAG_BYREF)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
info->type = PassType_Basic;
|
|
|
|
info->flags = flags;
|
|
|
|
info->size = sizeof(Vector *);
|
2007-06-20 10:03:43 +02:00
|
|
|
mySize = sizeof(Vector);
|
|
|
|
needs_extra = true;
|
2007-06-16 10:36:34 +02:00
|
|
|
} else if (pass == PassType_Object) {
|
|
|
|
info->type = PassType_Object;
|
|
|
|
info->flags = flags | PASSFLAG_OASSIGNOP | PASSFLAG_OCTOR;
|
|
|
|
info->size = sizeof(Vector);
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return mySize;
|
|
|
|
}
|
|
|
|
case Valve_QAngle:
|
|
|
|
{
|
|
|
|
size_t mySize = sizeof(QAngle *);
|
|
|
|
if (pass == PassType_Basic)
|
|
|
|
{
|
2007-06-17 22:43:21 +02:00
|
|
|
if (flags & PASSFLAG_BYREF)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
info->type = PassType_Basic;
|
|
|
|
info->flags = flags;
|
|
|
|
info->size = sizeof(QAngle *);
|
2007-06-20 10:03:43 +02:00
|
|
|
mySize = sizeof(QAngle);
|
|
|
|
needs_extra = true;
|
2007-06-16 10:36:34 +02:00
|
|
|
} else if (pass == PassType_Object) {
|
|
|
|
info->type = PassType_Object;
|
|
|
|
info->flags = flags | PASSFLAG_OASSIGNOP | PASSFLAG_OCTOR;
|
|
|
|
info->size = sizeof(QAngle);
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return mySize;
|
|
|
|
}
|
|
|
|
case Valve_CBaseEntity:
|
|
|
|
case Valve_CBasePlayer:
|
|
|
|
case Valve_Edict:
|
|
|
|
case Valve_String:
|
|
|
|
{
|
2007-06-17 22:43:21 +02:00
|
|
|
if (pass != PassType_Basic || (flags & PASSFLAG_BYREF))
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
info->type = PassType_Basic;
|
|
|
|
info->flags = flags;
|
|
|
|
info->size = sizeof(void *);
|
|
|
|
return sizeof(void *);
|
|
|
|
}
|
2007-06-16 20:03:05 +02:00
|
|
|
case Valve_POD:
|
|
|
|
{
|
|
|
|
info->type = PassType_Basic;
|
|
|
|
info->flags = flags;
|
|
|
|
if (flags & PASSFLAG_ASPOINTER)
|
|
|
|
{
|
|
|
|
info->size = sizeof(int *);
|
|
|
|
return sizeof(int *) + sizeof(int);
|
|
|
|
} else {
|
|
|
|
info->size = sizeof(int);
|
|
|
|
return sizeof(int);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case Valve_Bool:
|
|
|
|
{
|
|
|
|
info->type = PassType_Basic;
|
|
|
|
info->flags = flags;
|
|
|
|
if (flags & PASSFLAG_ASPOINTER)
|
|
|
|
{
|
|
|
|
info->size = sizeof(bool *);
|
|
|
|
return sizeof(bool *) + sizeof(bool);
|
|
|
|
} else {
|
|
|
|
info->size = sizeof(bool);
|
|
|
|
return sizeof(bool);
|
|
|
|
}
|
|
|
|
}
|
2007-06-16 10:36:34 +02:00
|
|
|
case Valve_Float:
|
|
|
|
{
|
|
|
|
info->type = PassType_Float;
|
|
|
|
info->flags = flags;
|
2007-06-16 20:03:05 +02:00
|
|
|
if (flags & PASSFLAG_ASPOINTER)
|
|
|
|
{
|
|
|
|
info->size = sizeof(float *);
|
|
|
|
return sizeof(float *) + sizeof(float);
|
|
|
|
} else {
|
|
|
|
info->size = sizeof(float);
|
|
|
|
return sizeof(float);
|
|
|
|
}
|
2007-06-16 10:36:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
DataStatus EncodeValveParam(IPluginContext *pContext,
|
|
|
|
cell_t param,
|
2007-06-20 10:03:43 +02:00
|
|
|
const ValveCall *pCall,
|
2007-06-16 20:03:05 +02:00
|
|
|
const ValvePassInfo *data,
|
2007-06-20 10:03:43 +02:00
|
|
|
const void *_buffer)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
2007-06-20 10:03:43 +02:00
|
|
|
const void *buffer = (const unsigned char *)_buffer + data->offset;
|
2007-06-16 20:03:05 +02:00
|
|
|
switch (data->vtype)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
case Valve_Vector:
|
|
|
|
{
|
|
|
|
Vector *v = NULL;
|
|
|
|
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->type == PassType_Basic)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
v = *(Vector **)((unsigned char *)buffer + sizeof(Vector *));
|
2007-06-16 20:03:05 +02:00
|
|
|
} else if (data->type == PassType_Object) {
|
2007-06-16 10:36:34 +02:00
|
|
|
v = (Vector *)buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
|
2007-06-16 20:33:21 +02:00
|
|
|
addr[0] = sp_ftoc(v->x);
|
|
|
|
addr[1] = sp_ftoc(v->y);
|
|
|
|
addr[2] = sp_ftoc(v->z);
|
2007-06-16 10:36:34 +02:00
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_QAngle:
|
|
|
|
{
|
|
|
|
QAngle *q = NULL;
|
|
|
|
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->type == PassType_Basic)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
q = *(QAngle **)((unsigned char *)buffer + sizeof(QAngle *));
|
2007-06-16 20:03:05 +02:00
|
|
|
} else if (data->type == PassType_Object) {
|
2007-06-16 10:36:34 +02:00
|
|
|
q = (QAngle *)buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
|
2007-06-16 20:33:21 +02:00
|
|
|
addr[0] = sp_ftoc(q->x);
|
|
|
|
addr[1] = sp_ftoc(q->y);
|
|
|
|
addr[2] = sp_ftoc(q->z);
|
2007-06-16 10:36:34 +02:00
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_CBaseEntity:
|
|
|
|
case Valve_CBasePlayer:
|
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
|
|
|
|
CBaseEntity *pEntity = *(CBaseEntity **)buffer;
|
|
|
|
if (pEntity)
|
|
|
|
{
|
|
|
|
edict_t *pEdict = gameents->BaseEntityToEdict(pEntity);
|
|
|
|
*addr = engine->IndexOfEdict(pEdict);
|
|
|
|
} else {
|
|
|
|
*addr = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_Edict:
|
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
|
|
|
|
edict_t *pEdict = *(edict_t **)buffer;
|
|
|
|
if (pEdict)
|
|
|
|
{
|
|
|
|
*addr = engine->IndexOfEdict(pEdict);
|
|
|
|
} else {
|
|
|
|
*addr = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_POD:
|
|
|
|
case Valve_Float:
|
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->flags & PASSFLAG_ASPOINTER)
|
|
|
|
{
|
|
|
|
buffer = (char *)buffer + sizeof(void *);
|
|
|
|
}
|
|
|
|
|
2007-06-16 10:36:34 +02:00
|
|
|
*addr = *(cell_t *)buffer;
|
|
|
|
|
2007-06-16 20:03:05 +02:00
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_Bool:
|
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
|
|
|
|
if (data->flags & PASSFLAG_ASPOINTER)
|
|
|
|
{
|
|
|
|
buffer = (char *)buffer + sizeof(bool *);
|
|
|
|
}
|
|
|
|
|
|
|
|
*addr = *(bool *)buffer ? 1 : 0;
|
|
|
|
|
2007-06-16 10:36:34 +02:00
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
DataStatus DecodeValveParam(IPluginContext *pContext,
|
|
|
|
cell_t param,
|
2007-06-20 10:03:43 +02:00
|
|
|
const ValveCall *pCall,
|
2007-06-16 20:03:05 +02:00
|
|
|
const ValvePassInfo *data,
|
2007-06-20 10:03:43 +02:00
|
|
|
void *_buffer)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
2007-06-20 10:03:43 +02:00
|
|
|
void *buffer = (unsigned char *)_buffer + data->offset;
|
2007-06-16 20:03:05 +02:00
|
|
|
switch (data->vtype)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
case Valve_Vector:
|
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
int err;
|
|
|
|
err = pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
|
|
|
|
unsigned char *mem = (unsigned char *)buffer;
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->type == PassType_Basic)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
/* Store the object in the next N bytes, and store
|
|
|
|
* a pointer to that object right beforehand.
|
|
|
|
*/
|
|
|
|
Vector **realPtr = (Vector **)buffer;
|
|
|
|
|
|
|
|
if (addr == pContext->GetNullRef(SP_NULL_VECTOR))
|
|
|
|
{
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_ALLOWNULL)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
*realPtr = NULL;
|
|
|
|
return Data_Okay;
|
|
|
|
} else {
|
|
|
|
pContext->ThrowNativeError("NULL not allowed");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
} else {
|
2007-06-20 10:03:43 +02:00
|
|
|
mem = (unsigned char *)_buffer + pCall->stackEnd + data->obj_offset;
|
2007-06-16 10:36:34 +02:00
|
|
|
*realPtr = (Vector *)mem;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err != SP_ERROR_NONE)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeErrorEx(err, "Could not read plugin data");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Use placement new to initialize the object cleanly
|
|
|
|
* This has no destructor so we don't need to do
|
|
|
|
* DestroyValveParam() or something :]
|
|
|
|
*/
|
|
|
|
Vector *v = new (mem) Vector(
|
|
|
|
sp_ctof(addr[0]),
|
|
|
|
sp_ctof(addr[1]),
|
|
|
|
sp_ctof(addr[2]));
|
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_QAngle:
|
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
int err;
|
|
|
|
err = pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
|
|
|
|
unsigned char *mem = (unsigned char *)buffer;
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->type == PassType_Basic)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
/* Store the object in the next N bytes, and store
|
2007-06-16 17:40:22 +02:00
|
|
|
* a pointer to that object right beforehand.
|
|
|
|
*/
|
2007-06-16 10:36:34 +02:00
|
|
|
QAngle **realPtr = (QAngle **)buffer;
|
|
|
|
|
|
|
|
if (addr == pContext->GetNullRef(SP_NULL_VECTOR))
|
|
|
|
{
|
2007-06-16 20:03:05 +02:00
|
|
|
if (!(data->decflags & VDECODE_FLAG_ALLOWNULL))
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("NULL not allowed");
|
|
|
|
return Data_Fail;
|
|
|
|
} else {
|
|
|
|
*realPtr = NULL;
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
} else {
|
2007-06-20 10:03:43 +02:00
|
|
|
mem = (unsigned char *)_buffer + pCall->stackEnd + data->obj_offset;
|
2007-06-16 10:36:34 +02:00
|
|
|
*realPtr = (QAngle *)mem;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err != SP_ERROR_NONE)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeErrorEx(err, "Could not read plugin data");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Use placement new to initialize the object cleanly
|
2007-06-16 17:40:22 +02:00
|
|
|
* This has no destructor so we don't need to do
|
|
|
|
* DestroyValveParam() or something :]
|
|
|
|
*/
|
2007-06-16 10:36:34 +02:00
|
|
|
QAngle *v = new (mem) QAngle(
|
|
|
|
sp_ctof(addr[0]),
|
|
|
|
sp_ctof(addr[1]),
|
|
|
|
sp_ctof(addr[2]));
|
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_CBasePlayer:
|
|
|
|
{
|
|
|
|
edict_t *pEdict;
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_BYREF)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
param = *addr;
|
|
|
|
}
|
|
|
|
if (param >= 1 && param <= playerhelpers->GetMaxClients())
|
|
|
|
{
|
|
|
|
IGamePlayer *player = playerhelpers->GetGamePlayer(param);
|
2007-06-16 20:03:05 +02:00
|
|
|
if ((data->decflags & VDECODE_FLAG_ALLOWNOTINGAME)
|
2007-06-16 10:36:34 +02:00
|
|
|
&& !player->IsConnected())
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Client %d is not connected", param);
|
|
|
|
return Data_Fail;
|
|
|
|
} else if (!player->IsInGame()) {
|
|
|
|
pContext->ThrowNativeError("Client %d is not in game", param);
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
pEdict = player->GetEdict();
|
|
|
|
} else if (param == -1) {
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_ALLOWNULL)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
pEdict = NULL;
|
|
|
|
} else {
|
|
|
|
pContext->ThrowNativeError("NULL not allowed");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
} else if (param == 0) {
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_ALLOWWORLD)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
pEdict = engine->PEntityOfEntIndex(0);
|
|
|
|
} else {
|
|
|
|
pContext->ThrowNativeError("World not allowed");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pContext->ThrowNativeError("Entity index %d is not a valid client", param);
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
2007-06-16 20:33:21 +02:00
|
|
|
CBaseEntity *pEntity = NULL;
|
2007-06-16 10:36:34 +02:00
|
|
|
if (pEdict)
|
|
|
|
{
|
|
|
|
IServerUnknown *pUnknown = pEdict->GetUnknown();
|
|
|
|
if (!pUnknown)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Entity %d is a not an IServerUnknown");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
pEntity = pUnknown->GetBaseEntity();
|
|
|
|
if (!pEntity)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Entity %d is not a CBaseEntity");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CBaseEntity **ebuf = (CBaseEntity **)buffer;
|
|
|
|
*ebuf = pEntity;
|
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_CBaseEntity:
|
|
|
|
{
|
|
|
|
edict_t *pEdict;
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_BYREF)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
param = *addr;
|
|
|
|
}
|
|
|
|
if (param >= 1 && param <= playerhelpers->GetMaxClients())
|
|
|
|
{
|
|
|
|
IGamePlayer *player = playerhelpers->GetGamePlayer(param);
|
2007-06-16 20:03:05 +02:00
|
|
|
if ((data->decflags & VDECODE_FLAG_ALLOWNOTINGAME)
|
2007-06-16 10:36:34 +02:00
|
|
|
&& !player->IsConnected())
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Client %d is not connected", param);
|
|
|
|
return Data_Fail;
|
|
|
|
} else if (!player->IsInGame()) {
|
|
|
|
pContext->ThrowNativeError("Client %d is not in game", param);
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
pEdict = player->GetEdict();
|
|
|
|
} else if (param == -1) {
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_ALLOWNULL)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
pEdict = NULL;
|
|
|
|
} else {
|
|
|
|
pContext->ThrowNativeError("NULL not allowed");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
} else if (param == 0) {
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_ALLOWWORLD)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
pEdict = engine->PEntityOfEntIndex(0);
|
|
|
|
} else {
|
|
|
|
pContext->ThrowNativeError("World not allowed");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pEdict = engine->PEntityOfEntIndex(param);
|
|
|
|
if (!pEdict || pEdict->IsFree())
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Entity %d is not valid or is freed", param);
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
}
|
2007-06-16 20:33:21 +02:00
|
|
|
CBaseEntity *pEntity = NULL;
|
2007-06-16 10:36:34 +02:00
|
|
|
if (pEdict)
|
|
|
|
{
|
|
|
|
IServerUnknown *pUnknown = pEdict->GetUnknown();
|
|
|
|
if (!pUnknown)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Entity %d is a not an IServerUnknown");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
pEntity = pUnknown->GetBaseEntity();
|
|
|
|
if (!pEntity)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Entity %d is not a CBaseEntity");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CBaseEntity **ebuf = (CBaseEntity **)buffer;
|
|
|
|
*ebuf = pEntity;
|
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_Edict:
|
|
|
|
{
|
|
|
|
edict_t *pEdict;
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_BYREF)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
param = *addr;
|
|
|
|
}
|
|
|
|
if (param >= 1 && param <= playerhelpers->GetMaxClients())
|
|
|
|
{
|
|
|
|
IGamePlayer *player = playerhelpers->GetGamePlayer(param);
|
2007-06-16 20:03:05 +02:00
|
|
|
if ((data->decflags & VDECODE_FLAG_ALLOWNOTINGAME)
|
2007-06-16 10:36:34 +02:00
|
|
|
&& !player->IsConnected())
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Client %d is not connected", param);
|
|
|
|
return Data_Fail;
|
|
|
|
} else if (!player->IsInGame()) {
|
|
|
|
pContext->ThrowNativeError("Client %d is not in game", param);
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
pEdict = player->GetEdict();
|
|
|
|
} else if (param == -1) {
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_ALLOWNULL)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
pEdict = NULL;
|
|
|
|
} else {
|
|
|
|
pContext->ThrowNativeError("NULL not allowed");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
} else if (param == 0) {
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_ALLOWWORLD)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
pEdict = engine->PEntityOfEntIndex(0);
|
|
|
|
} else {
|
|
|
|
pContext->ThrowNativeError("World not allowed");
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pEdict = engine->PEntityOfEntIndex(param);
|
|
|
|
if (!pEdict || pEdict->IsFree())
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Entity %d is not valid or is freed", param);
|
|
|
|
return Data_Fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
edict_t **ebuf = (edict_t **)buffer;
|
|
|
|
*ebuf = pEdict;
|
|
|
|
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
case Valve_POD:
|
|
|
|
case Valve_Float:
|
|
|
|
{
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->decflags & VDECODE_FLAG_BYREF)
|
2007-06-16 10:36:34 +02:00
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
param = *addr;
|
|
|
|
}
|
2007-06-16 20:03:05 +02:00
|
|
|
if (data->flags & PASSFLAG_ASPOINTER)
|
|
|
|
{
|
|
|
|
*(void **)buffer = (char *)buffer + sizeof(void *);
|
|
|
|
buffer = *(void **)buffer;
|
|
|
|
}
|
2007-06-16 10:36:34 +02:00
|
|
|
*(cell_t *)buffer = param;
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
2007-06-16 20:03:05 +02:00
|
|
|
case Valve_Bool:
|
|
|
|
{
|
|
|
|
if (data->decflags & VDECODE_FLAG_BYREF)
|
|
|
|
{
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(param, &addr);
|
|
|
|
param = *addr;
|
|
|
|
}
|
|
|
|
if (data->flags & PASSFLAG_ASPOINTER)
|
|
|
|
{
|
|
|
|
*(bool **)buffer = (bool *)((char *)buffer + sizeof(bool *));
|
|
|
|
buffer = *(bool **)buffer;
|
|
|
|
}
|
|
|
|
*(bool *)buffer = param ? true : false;
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
2007-06-16 10:36:34 +02:00
|
|
|
case Valve_String:
|
|
|
|
{
|
|
|
|
char *addr;
|
|
|
|
pContext->LocalToString(param, &addr);
|
|
|
|
*(char **)buffer = addr;
|
|
|
|
return Data_Okay;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Data_Fail;
|
|
|
|
}
|