2013-08-15 08:54:25 +02:00
|
|
|
// vim: set ts=4 sw=4 tw=99 noet:
|
2008-07-11 10:18:43 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "engine2.h"
|
|
|
|
#include "x86/jit_x86.h"
|
|
|
|
#include "zlib/zlib.h"
|
|
|
|
#include "BaseRuntime.h"
|
|
|
|
#include "sp_vm_engine.h"
|
2013-08-15 08:54:25 +02:00
|
|
|
#include "watchdog_timer.h"
|
2013-12-30 23:51:00 +01:00
|
|
|
#include <sourcemod_version.h>
|
2008-07-11 10:18:43 +02:00
|
|
|
|
|
|
|
using namespace SourcePawn;
|
|
|
|
|
|
|
|
SourcePawnEngine2::SourcePawnEngine2()
|
|
|
|
{
|
2014-06-24 10:04:13 +02:00
|
|
|
profiler_ = NULL;
|
2013-09-01 09:23:44 +02:00
|
|
|
jit_enabled_ = true;
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
IPluginRuntime *SourcePawnEngine2::LoadPlugin(ICompilation *co, const char *file, int *err)
|
|
|
|
{
|
|
|
|
sp_file_hdr_t hdr;
|
|
|
|
uint8_t *base;
|
|
|
|
int z_result;
|
|
|
|
int error;
|
|
|
|
BaseRuntime *pRuntime;
|
|
|
|
|
|
|
|
FILE *fp = fopen(file, "rb");
|
|
|
|
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
error = SP_ERROR_NOT_FOUND;
|
|
|
|
goto return_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Rewind for safety */
|
|
|
|
fread(&hdr, sizeof(sp_file_hdr_t), 1, fp);
|
|
|
|
|
|
|
|
if (hdr.magic != SPFILE_MAGIC)
|
|
|
|
{
|
|
|
|
error = SP_ERROR_FILE_FORMAT;
|
|
|
|
goto return_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (hdr.compression)
|
|
|
|
{
|
|
|
|
case SPFILE_COMPRESSION_GZ:
|
|
|
|
{
|
|
|
|
uint32_t uncompsize = hdr.imagesize - hdr.dataoffs;
|
|
|
|
uint32_t compsize = hdr.disksize - hdr.dataoffs;
|
|
|
|
uint32_t sectsize = hdr.dataoffs - sizeof(sp_file_hdr_t);
|
|
|
|
uLongf destlen = uncompsize;
|
|
|
|
|
|
|
|
char *tempbuf = (char *)malloc(compsize);
|
|
|
|
void *uncompdata = malloc(uncompsize);
|
|
|
|
void *sectheader = malloc(sectsize);
|
|
|
|
|
|
|
|
fread(sectheader, sectsize, 1, fp);
|
|
|
|
fread(tempbuf, compsize, 1, fp);
|
|
|
|
|
|
|
|
z_result = uncompress((Bytef *)uncompdata, &destlen, (Bytef *)tempbuf, compsize);
|
|
|
|
free(tempbuf);
|
|
|
|
if (z_result != Z_OK)
|
|
|
|
{
|
|
|
|
free(sectheader);
|
|
|
|
free(uncompdata);
|
|
|
|
error = SP_ERROR_DECOMPRESSOR;
|
|
|
|
goto return_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
base = (uint8_t *)malloc(hdr.imagesize);
|
|
|
|
memcpy(base, &hdr, sizeof(sp_file_hdr_t));
|
|
|
|
memcpy(base + sizeof(sp_file_hdr_t), sectheader, sectsize);
|
|
|
|
free(sectheader);
|
|
|
|
memcpy(base + hdr.dataoffs, uncompdata, uncompsize);
|
|
|
|
free(uncompdata);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SPFILE_COMPRESSION_NONE:
|
|
|
|
{
|
|
|
|
base = (uint8_t *)malloc(hdr.imagesize);
|
|
|
|
rewind(fp);
|
|
|
|
fread(base, hdr.imagesize, 1, fp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
error = SP_ERROR_DECOMPRESSOR;
|
|
|
|
goto return_error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-08-15 07:22:26 +02:00
|
|
|
pRuntime = new BaseRuntime();
|
|
|
|
if ((error = pRuntime->CreateFromMemory(&hdr, base)) != SP_ERROR_NONE)
|
2008-07-11 10:18:43 +02:00
|
|
|
{
|
2008-08-15 07:22:26 +02:00
|
|
|
delete pRuntime;
|
|
|
|
goto return_error;
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
2008-08-15 07:22:26 +02:00
|
|
|
size_t len;
|
|
|
|
|
|
|
|
len = strlen(file);
|
2011-04-24 04:53:53 +02:00
|
|
|
for (size_t i = len - 1; i < len; i--)
|
2008-07-11 10:18:43 +02:00
|
|
|
{
|
2008-08-15 07:22:26 +02:00
|
|
|
if (file[i] == '/'
|
|
|
|
#if defined WIN32
|
|
|
|
|| file[i] == '\\'
|
|
|
|
#endif
|
|
|
|
)
|
|
|
|
{
|
2013-08-08 18:41:24 +02:00
|
|
|
pRuntime->SetName(&file[i+1]);
|
2008-08-15 07:22:26 +02:00
|
|
|
break;
|
|
|
|
}
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
2013-08-08 18:41:24 +02:00
|
|
|
if (!pRuntime->plugin()->name)
|
2008-07-11 10:18:43 +02:00
|
|
|
{
|
2013-08-08 18:41:24 +02:00
|
|
|
pRuntime->SetName(file);
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
2008-08-15 07:22:26 +02:00
|
|
|
pRuntime->ApplyCompilationOptions(co);
|
|
|
|
|
2013-08-08 18:41:24 +02:00
|
|
|
fclose(fp);
|
2008-08-28 18:59:41 +02:00
|
|
|
|
2008-07-11 10:18:43 +02:00
|
|
|
return pRuntime;
|
|
|
|
|
|
|
|
return_error:
|
|
|
|
*err = error;
|
2008-10-27 05:40:24 +01:00
|
|
|
if (fp != NULL)
|
|
|
|
{
|
|
|
|
fclose(fp);
|
|
|
|
}
|
2008-07-11 10:18:43 +02:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
SPVM_NATIVE_FUNC SourcePawnEngine2::CreateFakeNative(SPVM_FAKENATIVE_FUNC callback, void *pData)
|
|
|
|
{
|
2008-08-15 07:22:26 +02:00
|
|
|
return g_Jit.CreateFakeNative(callback, pData);
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SourcePawnEngine2::DestroyFakeNative(SPVM_NATIVE_FUNC func)
|
|
|
|
{
|
2008-08-15 07:22:26 +02:00
|
|
|
g_Jit.DestroyFakeNative(func);
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *SourcePawnEngine2::GetEngineName()
|
|
|
|
{
|
2014-06-24 10:04:13 +02:00
|
|
|
return "SourcePawn 1.3, jit-x86";
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *SourcePawnEngine2::GetVersionString()
|
|
|
|
{
|
2013-12-30 23:51:00 +01:00
|
|
|
return SOURCEMOD_VERSION;
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
IDebugListener *SourcePawnEngine2::SetDebugListener(IDebugListener *listener)
|
|
|
|
{
|
|
|
|
return g_engine1.SetDebugListener(listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int SourcePawnEngine2::GetAPIVersion()
|
|
|
|
{
|
|
|
|
return SOURCEPAWN_ENGINE2_API_VERSION;
|
|
|
|
}
|
|
|
|
|
|
|
|
ICompilation *SourcePawnEngine2::StartCompilation()
|
|
|
|
{
|
2008-08-15 07:22:26 +02:00
|
|
|
return g_Jit.StartCompilation();
|
2008-07-11 10:18:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *SourcePawnEngine2::GetErrorString(int err)
|
|
|
|
{
|
|
|
|
return g_engine1.GetErrorString(err);
|
|
|
|
}
|
2008-08-15 07:22:26 +02:00
|
|
|
|
|
|
|
bool SourcePawnEngine2::Initialize()
|
|
|
|
{
|
|
|
|
return g_Jit.InitializeJIT();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SourcePawnEngine2::Shutdown()
|
|
|
|
{
|
2013-09-24 04:29:28 +02:00
|
|
|
g_WatchdogTimer.Shutdown();
|
2008-08-15 07:22:26 +02:00
|
|
|
g_Jit.ShutdownJIT();
|
|
|
|
}
|
2009-02-01 08:03:03 +01:00
|
|
|
|
|
|
|
IPluginRuntime *SourcePawnEngine2::CreateEmptyRuntime(const char *name, uint32_t memory)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
BaseRuntime *rt;
|
|
|
|
|
|
|
|
rt = new BaseRuntime();
|
|
|
|
if ((err = rt->CreateBlank(memory)) != SP_ERROR_NONE)
|
|
|
|
{
|
|
|
|
delete rt;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-08-08 18:41:24 +02:00
|
|
|
rt->SetName(name != NULL ? name : "<anonymous>");
|
2009-02-01 08:03:03 +01:00
|
|
|
|
|
|
|
rt->ApplyCompilationOptions(NULL);
|
|
|
|
|
|
|
|
return rt;
|
|
|
|
}
|
2013-08-15 08:54:25 +02:00
|
|
|
|
|
|
|
bool SourcePawnEngine2::InstallWatchdogTimer(size_t timeout_ms)
|
|
|
|
{
|
|
|
|
return g_WatchdogTimer.Initialize(timeout_ms);
|
|
|
|
}
|
|
|
|
|