2012-05-23 19:26:15 +00:00
|
|
|
/***************************************************************************
|
|
|
|
|
|
|
|
gbx_jit.c
|
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
(c) 2018 Benoît Minisini <g4mba5@gmail.com>
|
2012-05-23 19:26:15 +00:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
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, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
|
|
MA 02110-1301, USA.
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#define __GBX_JIT_C
|
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
#include "gb_common.h"
|
|
|
|
#include "gb_common_buffer.h"
|
|
|
|
#include "gb_common_case.h"
|
|
|
|
#include "gbx_component.h"
|
2018-05-28 03:18:44 +02:00
|
|
|
#include "gbx_exec.h"
|
|
|
|
#include "gbx_object.h"
|
2012-05-23 19:26:15 +00:00
|
|
|
#include "gbx_api.h"
|
2018-05-26 16:50:00 +02:00
|
|
|
#include "gbx_jit.h"
|
|
|
|
|
2018-06-01 03:50:42 +02:00
|
|
|
typedef
|
|
|
|
struct {
|
|
|
|
JIT_FUNC addr;
|
|
|
|
PCODE *code;
|
|
|
|
}
|
|
|
|
JIT_FUNCTION;
|
|
|
|
|
2018-08-11 07:53:02 +02:00
|
|
|
bool JIT_disabled = FALSE;
|
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
static bool _component_loaded = FALSE;
|
2018-06-01 03:50:42 +02:00
|
|
|
static GB_FUNCTION _jit_compile_func;
|
2012-05-23 19:26:15 +00:00
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
static bool _jit_compiling = FALSE;
|
2018-05-26 16:50:00 +02:00
|
|
|
static void *_jit_library = NULL;
|
2012-05-23 19:26:15 +00:00
|
|
|
|
2018-06-01 03:50:42 +02:00
|
|
|
static JIT_FUNCTION *_jit_func = NULL;
|
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
static bool _debug = FALSE;
|
|
|
|
|
2018-06-01 03:50:42 +02:00
|
|
|
void JIT_exit(void)
|
|
|
|
{
|
|
|
|
ARRAY_delete(&_jit_func);
|
|
|
|
}
|
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
bool JIT_can_compile(ARCHIVE *arch)
|
|
|
|
{
|
|
|
|
return arch ? !arch->jit_compiling : !_jit_compiling;
|
|
|
|
}
|
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
bool JIT_compile(ARCHIVE *arch)
|
2012-05-23 23:39:02 +00:00
|
|
|
{
|
2018-05-26 16:50:00 +02:00
|
|
|
GB_VALUE *ret;
|
|
|
|
char *path;
|
|
|
|
void *lib;
|
|
|
|
void **iface;
|
2018-06-09 22:42:35 +02:00
|
|
|
COMPONENT *current;
|
2012-05-23 19:26:15 +00:00
|
|
|
|
2018-08-11 07:53:02 +02:00
|
|
|
if (JIT_disabled)
|
2012-05-23 19:26:15 +00:00
|
|
|
return TRUE;
|
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
if (arch)
|
|
|
|
{
|
|
|
|
if (arch->jit_library)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (_jit_library)
|
|
|
|
return FALSE;
|
|
|
|
}
|
2012-05-23 19:26:15 +00:00
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
if (!_component_loaded)
|
2012-05-23 19:26:15 +00:00
|
|
|
{
|
2018-06-09 22:42:35 +02:00
|
|
|
char *var;
|
2012-05-25 22:24:13 +00:00
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
_component_loaded = TRUE;
|
2012-05-23 19:26:15 +00:00
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
var = getenv("GB_NO_JIT");
|
2018-05-26 16:50:00 +02:00
|
|
|
if (var && var[0] && !(var[0] == '0' && var[1] == 0))
|
|
|
|
{
|
2018-08-11 07:53:02 +02:00
|
|
|
JIT_disabled = TRUE;
|
2018-05-26 16:50:00 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
2012-07-02 22:38:29 +00:00
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
var = getenv("GB_JIT_DEBUG");
|
|
|
|
if (var && var[0] && !(var[0] == '0' && var[1] == 0))
|
|
|
|
_debug = TRUE;
|
|
|
|
|
|
|
|
if (_debug)
|
|
|
|
fprintf(stderr, "gbx3: loading gb.jit component\n");
|
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
COMPONENT_load(COMPONENT_create("gb.jit"));
|
2018-07-03 00:52:19 +02:00
|
|
|
if (GB_GetFunction(&_jit_compile_func, CLASS_find_global("Jit"), "_Compile", "s", "s"))
|
|
|
|
ERROR_panic("Unable to find JIT compilation method");
|
2012-05-23 19:26:15 +00:00
|
|
|
}
|
2012-06-01 00:18:38 +00:00
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
arch ? (arch->jit_compiling = TRUE) : (_jit_compiling = TRUE);
|
|
|
|
|
|
|
|
current = COMPONENT_current;
|
|
|
|
COMPONENT_current = NULL;
|
|
|
|
|
|
|
|
GB_Push(1, T_STRING, arch ? arch->name : "", -1);
|
|
|
|
ret = GB_Call(&_jit_compile_func, 1, FALSE);
|
2018-05-26 16:50:00 +02:00
|
|
|
path = GB_ToZeroString((GB_STRING *)ret);
|
2018-06-09 22:42:35 +02:00
|
|
|
|
|
|
|
COMPONENT_current = current;
|
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
if (!*path)
|
2018-07-03 00:52:19 +02:00
|
|
|
ERROR_panic("Unable to compile JIT source file");
|
2018-05-26 16:50:00 +02:00
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
arch ? (arch->jit_compiling = FALSE) : (_jit_compiling = FALSE);
|
|
|
|
|
2018-05-28 03:18:44 +02:00
|
|
|
//fprintf(stderr, "gbx3: shared jit library is: %s\n", path);
|
2018-05-26 16:50:00 +02:00
|
|
|
|
|
|
|
lib = dlopen(path, RTLD_NOW);
|
|
|
|
if (!lib)
|
2018-07-03 00:52:19 +02:00
|
|
|
ERROR_panic("Unable to load JIT library: %s", dlerror());
|
2018-05-26 16:50:00 +02:00
|
|
|
|
|
|
|
if (arch)
|
|
|
|
arch->jit_library = lib;
|
|
|
|
else
|
|
|
|
_jit_library = lib;
|
|
|
|
|
2018-05-28 03:18:44 +02:00
|
|
|
iface = dlsym(lib, "GB_PTR");
|
|
|
|
if (iface) *((void **)iface) = &GAMBAS_Api;
|
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
iface = dlsym(lib, "JIT_PTR");
|
2018-05-28 03:18:44 +02:00
|
|
|
if (iface) *((void **)iface) = &GAMBAS_JitApi;
|
2018-05-26 16:50:00 +02:00
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
static bool create_function(CLASS *class, int index)
|
2018-05-26 16:50:00 +02:00
|
|
|
{
|
2018-06-09 22:42:35 +02:00
|
|
|
ARCHIVE *arch;
|
2018-06-01 03:50:42 +02:00
|
|
|
FUNCTION *func;
|
|
|
|
JIT_FUNCTION *jit;
|
2018-05-26 16:50:00 +02:00
|
|
|
void *lib;
|
|
|
|
void *addr;
|
|
|
|
int i;
|
|
|
|
int len;
|
2018-06-09 22:42:35 +02:00
|
|
|
char *name;
|
|
|
|
|
|
|
|
arch = class->component ? class->component->archive : NULL;
|
2018-05-26 16:50:00 +02:00
|
|
|
|
2018-06-01 03:50:42 +02:00
|
|
|
func = &class->load->func[index];
|
2018-06-09 22:42:35 +02:00
|
|
|
func->fast_linked = TRUE;
|
2018-06-01 03:50:42 +02:00
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
if (!arch)
|
|
|
|
lib = _jit_library;
|
|
|
|
else
|
|
|
|
lib = arch->jit_library;
|
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
name = class->name;
|
|
|
|
while (*name == '^')
|
|
|
|
name++;
|
|
|
|
|
|
|
|
len = sprintf(COMMON_buffer, "jit_%s_%d", name, index);
|
2018-05-26 16:50:00 +02:00
|
|
|
|
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
COMMON_buffer[i] = tolower(COMMON_buffer[i]);
|
2012-05-23 19:26:15 +00:00
|
|
|
|
2018-05-26 16:50:00 +02:00
|
|
|
addr = dlsym(lib, COMMON_buffer);
|
2018-06-01 03:50:42 +02:00
|
|
|
if (!addr)
|
|
|
|
{
|
|
|
|
func->fast = FALSE;
|
2018-06-09 22:42:35 +02:00
|
|
|
return TRUE;
|
2018-06-01 03:50:42 +02:00
|
|
|
}
|
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
if (_debug && func->debug)
|
2018-06-01 03:50:42 +02:00
|
|
|
fprintf(stderr, "gbx3: loading jit function: %s.%s\n", class->name, func->debug->name);
|
|
|
|
|
|
|
|
if (!_jit_func)
|
|
|
|
ARRAY_create(&_jit_func);
|
|
|
|
|
|
|
|
jit = (JIT_FUNCTION *)ARRAY_add(&_jit_func);
|
|
|
|
|
|
|
|
jit->addr = addr;
|
|
|
|
jit->code = func->code;
|
|
|
|
|
|
|
|
func->code = (PCODE *)jit;
|
2018-06-09 22:42:35 +02:00
|
|
|
|
|
|
|
return FALSE;
|
2012-05-23 19:26:15 +00:00
|
|
|
}
|
|
|
|
|
2018-05-28 03:18:44 +02:00
|
|
|
|
2018-06-05 11:43:39 +02:00
|
|
|
void JIT_exec(bool ret_on_stack)
|
2018-05-28 03:18:44 +02:00
|
|
|
{
|
|
|
|
VALUE *sp = SP;
|
2018-06-01 03:50:42 +02:00
|
|
|
JIT_FUNCTION *jit;
|
2018-06-09 22:42:35 +02:00
|
|
|
CLASS *class = EXEC.class;
|
2018-06-01 14:40:07 +02:00
|
|
|
char nparam = EXEC.nparam;
|
2018-06-02 17:59:47 +02:00
|
|
|
VALUE ret;
|
|
|
|
FUNCTION *func = EXEC.func;
|
|
|
|
|
2018-10-31 01:56:35 +01:00
|
|
|
if (UNLIKELY(nparam < func->npmin))
|
|
|
|
THROW(E_NEPARAM);
|
|
|
|
else if (UNLIKELY(nparam > func->n_param && !func->vararg))
|
|
|
|
THROW(E_TMPARAM);
|
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
if (!func->fast_linked)
|
|
|
|
{
|
|
|
|
if (create_function(class, EXEC.index))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-06-02 17:59:47 +02:00
|
|
|
STACK_push_frame(&EXEC_current, func->stack_usage);
|
2018-05-28 03:18:44 +02:00
|
|
|
|
2018-06-09 22:42:35 +02:00
|
|
|
CP = class;
|
2018-05-28 03:18:44 +02:00
|
|
|
OP = (void *)EXEC.object;
|
2018-06-12 16:30:11 +02:00
|
|
|
FP = func;
|
2018-11-29 09:10:56 +01:00
|
|
|
EC = NULL;
|
2018-05-28 03:18:44 +02:00
|
|
|
|
2018-06-02 17:59:47 +02:00
|
|
|
jit = (JIT_FUNCTION *)(func->code);
|
2018-06-01 03:50:42 +02:00
|
|
|
|
2018-06-12 16:30:11 +02:00
|
|
|
PROFILE_ENTER_FUNCTION();
|
|
|
|
|
2018-06-22 04:47:44 +02:00
|
|
|
TRY
|
|
|
|
{
|
|
|
|
(*(jit->addr))(nparam);
|
|
|
|
}
|
|
|
|
CATCH
|
|
|
|
{
|
|
|
|
PROFILE_LEAVE_FUNCTION();
|
2018-11-27 02:59:21 +01:00
|
|
|
if (SP != sp)
|
2018-12-07 01:03:47 +01:00
|
|
|
ERROR_panic("Stack mismatch in JIT function (SP %+ld)\n", SP - sp);
|
2018-06-22 04:47:44 +02:00
|
|
|
RELEASE_MANY(SP, nparam);
|
|
|
|
STACK_pop_frame(&EXEC_current);
|
|
|
|
PROPAGATE();
|
|
|
|
}
|
|
|
|
END_TRY
|
2018-05-28 03:18:44 +02:00
|
|
|
|
2018-06-12 16:30:11 +02:00
|
|
|
PROFILE_LEAVE_FUNCTION();
|
|
|
|
|
2018-05-28 03:18:44 +02:00
|
|
|
if (SP != sp)
|
2018-12-07 01:03:47 +01:00
|
|
|
ERROR_panic("Stack mismatch in JIT function (SP %+ld)\n", SP - sp);
|
2018-05-28 03:18:44 +02:00
|
|
|
|
2018-06-02 17:59:47 +02:00
|
|
|
if (func->type != T_VOID)
|
|
|
|
{
|
|
|
|
ret = TEMP;
|
|
|
|
BORROW(&ret);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ret.type = T_VOID;
|
|
|
|
|
2018-06-01 14:40:07 +02:00
|
|
|
RELEASE_MANY(SP, nparam);
|
|
|
|
|
2018-06-05 11:43:39 +02:00
|
|
|
RET = ret;
|
2018-06-02 17:59:47 +02:00
|
|
|
|
|
|
|
STACK_pop_frame(&EXEC_current);
|
2018-06-05 11:43:39 +02:00
|
|
|
|
|
|
|
if (ret_on_stack)
|
|
|
|
{
|
|
|
|
if (SP[-1].type == T_FUNCTION)
|
|
|
|
{
|
|
|
|
SP--;
|
|
|
|
OBJECT_UNREF(SP->_function.object);
|
|
|
|
}
|
|
|
|
|
|
|
|
*SP++ = ret;
|
|
|
|
ret.type = T_VOID;
|
|
|
|
}
|
2018-06-02 17:59:47 +02:00
|
|
|
}
|
|
|
|
|
2018-07-02 19:37:37 +02:00
|
|
|
PCODE *JIT_get_code(FUNCTION *func)
|
2018-05-28 03:18:44 +02:00
|
|
|
{
|
2018-06-09 22:42:35 +02:00
|
|
|
if (func->fast_linked)
|
|
|
|
return ((JIT_FUNCTION *)(func->code))->code;
|
|
|
|
else
|
|
|
|
return func->code;
|
2018-05-28 03:18:44 +02:00
|
|
|
}
|
|
|
|
|
2018-05-29 23:25:57 +02:00
|
|
|
void *JIT_get_class_ref(int index)
|
|
|
|
{
|
|
|
|
return CP->load->class_ref[index];
|
|
|
|
}
|
|
|
|
|
2018-05-28 03:18:44 +02:00
|
|
|
CLASS_CONST *JIT_get_constant(int index)
|
2012-05-23 19:26:15 +00:00
|
|
|
{
|
2018-05-28 03:18:44 +02:00
|
|
|
return &CP->load->cst[index];
|
2012-05-23 19:26:15 +00:00
|
|
|
}
|
2018-05-29 03:43:23 +02:00
|
|
|
|
|
|
|
void JIT_debug(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
va_start(args, fmt);
|
|
|
|
vfprintf(stderr, fmt, args);
|
|
|
|
va_end(args);
|
|
|
|
}
|
2018-06-02 17:59:47 +02:00
|
|
|
|
2018-08-22 09:30:27 +02:00
|
|
|
typedef
|
|
|
|
struct {
|
|
|
|
PCODE *pc;
|
|
|
|
VALUE **psp;
|
|
|
|
PCODE code;
|
|
|
|
}
|
|
|
|
JIT_call_unknown_ERROR;
|
|
|
|
|
|
|
|
static void error_JIT_call_unknown(JIT_call_unknown_ERROR *save)
|
2018-06-22 04:47:44 +02:00
|
|
|
{
|
2018-08-22 09:30:27 +02:00
|
|
|
save->pc[1] = (PCODE)save->code;
|
|
|
|
*save->psp = SP;
|
2018-06-22 04:47:44 +02:00
|
|
|
}
|
|
|
|
|
2018-08-22 09:30:27 +02:00
|
|
|
void JIT_call_unknown(PCODE *pc, VALUE **psp)
|
2018-06-02 17:59:47 +02:00
|
|
|
{
|
2018-08-22 09:30:27 +02:00
|
|
|
JIT_call_unknown_ERROR save;
|
2018-06-02 17:59:47 +02:00
|
|
|
|
|
|
|
PC = pc;
|
2018-08-22 09:30:27 +02:00
|
|
|
SP = *psp;
|
|
|
|
|
|
|
|
save.pc = pc;
|
|
|
|
save.psp = psp;
|
|
|
|
save.code = pc[1];
|
2018-06-02 17:59:47 +02:00
|
|
|
|
|
|
|
pc[1] = 0x140B;
|
|
|
|
|
2018-08-22 09:30:27 +02:00
|
|
|
ON_ERROR_1(error_JIT_call_unknown, &save)
|
2018-06-22 04:47:44 +02:00
|
|
|
{
|
2018-09-10 20:11:59 +02:00
|
|
|
EXEC_function_loop();
|
2018-06-22 04:47:44 +02:00
|
|
|
}
|
|
|
|
END_ERROR
|
2018-08-22 09:30:27 +02:00
|
|
|
|
|
|
|
error_JIT_call_unknown(&save);
|
2018-06-02 17:59:47 +02:00
|
|
|
}
|
2018-08-18 15:28:19 +02:00
|
|
|
|
|
|
|
void JIT_load_class(CLASS *class)
|
|
|
|
{
|
|
|
|
CLASS_load(class);
|
|
|
|
}
|