1026 lines
30 KiB
C++
1026 lines
30 KiB
C++
#include <stdafx.h>
|
|
#include <dwrite.h>
|
|
#include "ScriptAPI.h"
|
|
#include "N64Image.h"
|
|
|
|
#pragma warning(disable: 4702) // disable unreachable code warning
|
|
|
|
void ScriptAPI::InitEnvironment(duk_context* ctx, CScriptInstance* inst)
|
|
{
|
|
duk_push_global_object(ctx);
|
|
duk_push_string(ctx, "global");
|
|
duk_dup(ctx, -2);
|
|
duk_def_prop(ctx, -3, DUK_DEFPROP_HAVE_VALUE | DUK_DEFPROP_SET_ENUMERABLE);
|
|
duk_pop(ctx);
|
|
|
|
duk_module_duktape_init(ctx);
|
|
duk_get_global_string(ctx, "Duktape");
|
|
duk_push_c_function(ctx, js_Duktape_modSearch, 4);
|
|
duk_put_prop_string(ctx, -2, "modSearch");
|
|
duk_pop(ctx);
|
|
|
|
duk_push_pointer(ctx, inst);
|
|
duk_put_global_string(ctx, HS_gInstancePtr);
|
|
|
|
duk_push_object(ctx); // callbackId => { hookId, callbackId, function }
|
|
duk_put_global_string(ctx, HS_gAppCallbacks);
|
|
|
|
duk_push_object(ctx); // fd => { fp }
|
|
duk_put_global_string(ctx, HS_gOpenFileDescriptors);
|
|
|
|
duk_push_array(ctx); // [{modPtr: hModule}, ...]
|
|
duk_put_global_string(ctx, HS_gNativeModules);
|
|
|
|
duk_push_int(ctx, 0);
|
|
duk_put_global_string(ctx, HS_gNextObjectRefId);
|
|
|
|
duk_push_object(ctx); // { refId: object, ... }
|
|
duk_put_global_string(ctx, HS_gObjectRefs);
|
|
|
|
duk_push_int(ctx, 0);
|
|
duk_put_global_string(ctx, HS_gNextInvervalId);
|
|
|
|
duk_push_object(ctx); // { intervalId: { func, worker }, ... }
|
|
duk_put_global_string(ctx, HS_gIntervals);
|
|
|
|
Define_asm(ctx);
|
|
Define_console(ctx);
|
|
Define_cpu(ctx);
|
|
Define_debug(ctx);
|
|
Define_events(ctx);
|
|
Define_fs(ctx);
|
|
Define_mem(ctx);
|
|
Define_pj64(ctx);
|
|
Define_script(ctx);
|
|
|
|
Define_alert(ctx);
|
|
Define_exec(ctx);
|
|
Define_interval(ctx);
|
|
|
|
Define_AddressRange(ctx);
|
|
Define_N64Image(ctx);
|
|
Define_Server(ctx);
|
|
Define_Socket(ctx);
|
|
|
|
Define_Number_prototype_hex(ctx);
|
|
DefineGlobalConstants(ctx);
|
|
|
|
if(duk_get_top(ctx) > 0)
|
|
{
|
|
inst->System()->ConsoleLog("[SCRIPTSYS]: warning: duktape stack is dirty after API init");
|
|
}
|
|
}
|
|
|
|
void ScriptAPI::DefineGlobalClass(duk_context* ctx, const char* className,
|
|
duk_c_function constructorFunc,
|
|
const DukPropListEntry* prototypeProps,
|
|
const DukPropListEntry* staticProps)
|
|
{
|
|
duk_push_global_object(ctx);
|
|
duk_push_string(ctx, className);
|
|
duk_push_c_function(ctx, constructorFunc, DUK_VARARGS);
|
|
|
|
duk_push_string(ctx, "name");
|
|
duk_push_string(ctx, className);
|
|
duk_def_prop(ctx, -3, DUK_DEFPROP_HAVE_VALUE);
|
|
|
|
if (staticProps != nullptr)
|
|
{
|
|
DukPutPropList(ctx, -1, staticProps);
|
|
}
|
|
|
|
duk_push_object(ctx); // prototype
|
|
duk_push_string(ctx, className);
|
|
duk_put_prop_string(ctx, -2, DUK_WELLKNOWN_SYMBOL("Symbol.toStringTag"));
|
|
|
|
if (prototypeProps != nullptr)
|
|
{
|
|
DukPutPropList(ctx, -1, prototypeProps);
|
|
}
|
|
|
|
duk_freeze(ctx, -1);
|
|
duk_put_prop_string(ctx, -2, "prototype");
|
|
|
|
duk_freeze(ctx, -1);
|
|
duk_def_prop(ctx, -3, DUK_DEFPROP_HAVE_VALUE | DUK_DEFPROP_SET_ENUMERABLE);
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
void ScriptAPI::DefineGlobalInterface(duk_context* ctx, const char* name, const DukPropListEntry* props)
|
|
{
|
|
duk_push_global_object(ctx);
|
|
duk_push_string(ctx, name);
|
|
duk_push_object(ctx);
|
|
DukPutPropList(ctx, -1, props);
|
|
duk_freeze(ctx, -1);
|
|
duk_def_prop(ctx, -3, DUK_DEFPROP_HAVE_VALUE | DUK_DEFPROP_SET_ENUMERABLE);
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
void ScriptAPI::DefineGlobalFunction(duk_context* ctx, const char* name, duk_c_function func)
|
|
{
|
|
duk_push_global_object(ctx);
|
|
duk_push_string(ctx, name);
|
|
duk_push_c_function(ctx, func, DUK_VARARGS);
|
|
duk_freeze(ctx, -1);
|
|
duk_def_prop(ctx, -3, DUK_DEFPROP_HAVE_VALUE | DUK_DEFPROP_SET_ENUMERABLE);
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
void ScriptAPI::DefineGlobalConstants(duk_context* ctx)
|
|
{
|
|
const duk_number_list_entry numbers[] = {
|
|
{ "u8", U8 },
|
|
{ "u16", U16 },
|
|
{ "u32", U32 },
|
|
{ "s8", S8 },
|
|
{ "s16", S16 },
|
|
{ "s32", S32 },
|
|
{ "f32", F32 },
|
|
{ "f64", F64 },
|
|
|
|
{ "s64", S64 },
|
|
{ "u64", U64 },
|
|
|
|
{ "GPR_R0", GPR_R0 },
|
|
{ "GPR_AT", GPR_AT },
|
|
{ "GPR_V0", GPR_V0 },
|
|
{ "GPR_V1", GPR_V1 },
|
|
{ "GPR_A0", GPR_A0 },
|
|
{ "GPR_A1", GPR_A1 },
|
|
{ "GPR_A2", GPR_A2 },
|
|
{ "GPR_A3", GPR_A3 },
|
|
{ "GPR_T0", GPR_T0 },
|
|
{ "GPR_T1", GPR_T1 },
|
|
{ "GPR_T2", GPR_T2 },
|
|
{ "GPR_T3", GPR_T3 },
|
|
{ "GPR_T4", GPR_T4 },
|
|
{ "GPR_T5", GPR_T5 },
|
|
{ "GPR_T6", GPR_T6 },
|
|
{ "GPR_T7", GPR_T7 },
|
|
{ "GPR_S0", GPR_S0 },
|
|
{ "GPR_S1", GPR_S1 },
|
|
{ "GPR_S2", GPR_S2 },
|
|
{ "GPR_S3", GPR_S3 },
|
|
{ "GPR_S4", GPR_S4 },
|
|
{ "GPR_S5", GPR_S5 },
|
|
{ "GPR_S6", GPR_S6 },
|
|
{ "GPR_S7", GPR_S7 },
|
|
{ "GPR_T8", GPR_T8 },
|
|
{ "GPR_T9", GPR_T9 },
|
|
{ "GPR_K0", GPR_K0 },
|
|
{ "GPR_K1", GPR_K1 },
|
|
{ "GPR_GP", GPR_GP },
|
|
{ "GPR_SP", GPR_SP },
|
|
{ "GPR_FP", GPR_FP },
|
|
{ "GPR_RA", GPR_RA },
|
|
//{ "GPR_S8", GPR_S8 },
|
|
{ "GPR_ANY", 0xFFFFFFFF },
|
|
|
|
{ "RDRAM_CONFIG_REG", 0xA3F00000 },
|
|
{ "RDRAM_DEVICE_TYPE_REG", 0xA3F00000 },
|
|
{ "RDRAM_DEVICE_ID_REG", 0xA3F00004 },
|
|
{ "RDRAM_DELAY_REG", 0xA3F00008 },
|
|
{ "RDRAM_MODE_REG", 0xA3F0000C },
|
|
{ "RDRAM_REF_INTERVAL_REG", 0xA3F00010 },
|
|
{ "RDRAM_REF_ROW_REG", 0xA3F00014 },
|
|
{ "RDRAM_RAS_INTERVAL_REG", 0xA3F00018 },
|
|
{ "RDRAM_MIN_INTERVAL_REG", 0xA3F0001C },
|
|
{ "RDRAM_ADDR_SELECT_REG", 0xA3F00020 },
|
|
{ "RDRAM_DEVICE_MANUF_REG", 0xA3F00024 },
|
|
{ "SP_MEM_ADDR_REG", 0xA4040000 },
|
|
{ "SP_DRAM_ADDR_REG", 0xA4040004 },
|
|
{ "SP_RD_LEN_REG", 0xA4040008 },
|
|
{ "SP_WR_LEN_REG", 0xA404000C },
|
|
{ "SP_STATUS_REG", 0xA4040010 },
|
|
{ "SP_DMA_FULL_REG", 0xA4040014 },
|
|
{ "SP_DMA_BUSY_REG", 0xA4040018 },
|
|
{ "SP_SEMAPHORE_REG", 0xA404001C },
|
|
{ "SP_PC_REG", 0xA4080000 },
|
|
{ "SP_IBIST_REG", 0xA4080004 },
|
|
{ "DPC_START_REG", 0xA4100000 },
|
|
{ "DPC_END_REG", 0xA4100004 },
|
|
{ "DPC_CURRENT_REG", 0xA4100008 },
|
|
{ "DPC_STATUS_REG", 0xA410000C },
|
|
{ "DPC_CLOCK_REG", 0xA4100010 },
|
|
{ "DPC_BUFBUSY_REG", 0xA4100014 },
|
|
{ "DPC_PIPEBUSY_REG", 0xA4100018 },
|
|
{ "DPC_TMEM_REG", 0xA410001C },
|
|
{ "DPS_TBIST_REG", 0xA4200000 },
|
|
{ "DPS_TEST_MODE_REG", 0xA4200004 },
|
|
{ "DPS_BUFTEST_ADDR_REG", 0xA4200008 },
|
|
{ "DPS_BUFTEST_DATA_REG", 0xA420000C },
|
|
{ "MI_INIT_MODE_REG", 0xA4300000 },
|
|
{ "MI_MODE_REG", 0xA4300000 },
|
|
{ "MI_VERSION_REG", 0xA4300004 },
|
|
{ "MI_NOOP_REG", 0xA4300004 },
|
|
{ "MI_INTR_REG", 0xA4300008 },
|
|
{ "MI_INTR_MASK_REG", 0xA430000C },
|
|
{ "VI_STATUS_REG", 0xA4400000 },
|
|
{ "VI_CONTROL_REG", 0xA4400000 },
|
|
{ "VI_ORIGIN_REG", 0xA4400004 },
|
|
{ "VI_DRAM_ADDR_REG", 0xA4400004 },
|
|
{ "VI_WIDTH_REG", 0xA4400008 },
|
|
{ "VI_H_WIDTH_REG", 0xA4400008 },
|
|
{ "VI_INTR_REG", 0xA440000C },
|
|
{ "VI_V_INTR_REG", 0xA440000C },
|
|
{ "VI_CURRENT_REG", 0xA4400010 },
|
|
{ "VI_V_CURRENT_LINE_REG", 0xA4400010 },
|
|
{ "VI_BURST_REG", 0xA4400014 },
|
|
{ "VI_TIMING_REG", 0xA4400014 },
|
|
{ "VI_V_SYNC_REG", 0xA4400018 },
|
|
{ "VI_H_SYNC_REG", 0xA440001C },
|
|
{ "VI_LEAP_REG", 0xA4400020 },
|
|
{ "VI_H_SYNC_LEAP_REG", 0xA4400020 },
|
|
{ "VI_H_START_REG", 0xA4400024 },
|
|
{ "VI_H_VIDEO_REG", 0xA4400024 },
|
|
{ "VI_V_START_REG", 0xA4400028 },
|
|
{ "VI_V_VIDEO_REG", 0xA4400028 },
|
|
{ "VI_V_BURST_REG", 0xA440002C },
|
|
{ "VI_X_SCALE_REG", 0xA4400030 },
|
|
{ "VI_Y_SCALE_REG", 0xA4400034 },
|
|
{ "AI_DRAM_ADDR_REG", 0xA4500000 },
|
|
{ "AI_LEN_REG", 0xA4500004 },
|
|
{ "AI_CONTROL_REG", 0xA4500008 },
|
|
{ "AI_STATUS_REG", 0xA450000C },
|
|
{ "AI_DACRATE_REG", 0xA4500010 },
|
|
{ "AI_BITRATE_REG", 0xA4500014 },
|
|
{ "PI_DRAM_ADDR_REG", 0xA4600000 },
|
|
{ "PI_CART_ADDR_REG", 0xA4600004 },
|
|
{ "PI_RD_LEN_REG", 0xA4600008 },
|
|
{ "PI_WR_LEN_REG", 0xA460000C },
|
|
{ "PI_STATUS_REG", 0xA4600010 },
|
|
{ "PI_BSD_DOM1_LAT_REG", 0xA4600014 },
|
|
{ "PI_BSD_DOM1_PWD_REG", 0xA4600018 },
|
|
{ "PI_BSD_DOM1_PGS_REG", 0xA460001C },
|
|
{ "PI_BSD_DOM1_RLS_REG", 0xA4600020 },
|
|
{ "PI_BSD_DOM2_LAT_REG", 0xA4600024 },
|
|
{ "PI_BSD_DOM2_PWD_REG", 0xA4600028 },
|
|
{ "PI_BSD_DOM2_PGS_REG", 0xA460002C },
|
|
{ "PI_BSD_DOM2_RLS_REG", 0xA4600030 },
|
|
{ "RI_MODE_REG", 0xA4700000 },
|
|
{ "RI_CONFIG_REG", 0xA4700004 },
|
|
{ "RI_CURRENT_LOAD_REG", 0xA4700008 },
|
|
{ "RI_SELECT_REG", 0xA470000C },
|
|
{ "RI_REFRESH_REG", 0xA4700010 },
|
|
{ "RI_COUNT_REG", 0xA4700010 },
|
|
{ "RI_LATENCY_REG", 0xA4700014 },
|
|
{ "RI_RERROR_REG", 0xA4700018 },
|
|
{ "RI_WERROR_REG", 0xA470001C },
|
|
{ "SI_DRAM_ADDR_REG", 0xA4800000 },
|
|
{ "SI_PIF_ADDR_RD64B_REG", 0xA4800004 },
|
|
{ "SI_PIF_ADDR_WR64B_REG", 0xA4800010 },
|
|
{ "SI_STATUS_REG", 0xA4800018 },
|
|
|
|
{ "PIF_ROM_START", 0xBFC00000 },
|
|
{ "PIF_RAM_START", 0xBFC007C0 },
|
|
|
|
{ "SP_DMEM_START", 0xA4000000 },
|
|
{ "SP_IMEM_START", 0xA4001000 },
|
|
|
|
{ "KUBASE", 0x00000000 },
|
|
{ "K0BASE", 0x80000000 },
|
|
{ "K1BASE", 0xA0000000 },
|
|
{ "K2BASE", 0xC0000000 },
|
|
|
|
{ "UT_VEC", 0x80000000 },
|
|
{ "R_VEC", 0xBFC00000 },
|
|
{ "XUT_VEC", 0x80000080 },
|
|
{ "ECC_VEC", 0x80000100 },
|
|
{ "E_VEC", 0x80000180 },
|
|
|
|
{ "M_GFXTASK", 1 },
|
|
{ "M_AUDTASK", 2 },
|
|
{ "OS_READ", 0 },
|
|
{ "OS_WRITE", 1 },
|
|
|
|
{ "COLOR_BLACK", 0x000000FF },
|
|
{ "COLOR_WHITE", 0xFFFFFFFF },
|
|
{ "COLOR_GRAY", 0x808080FF },
|
|
{ "COLOR_RED", 0xFF0000FF },
|
|
{ "COLOR_GREEN", 0x00FF00FF },
|
|
{ "COLOR_BLUE", 0x0000FFFF },
|
|
{ "COLOR_YELLOW", 0xFFFF00FF },
|
|
{ "COLOR_CYAN", 0x00FFFFFF },
|
|
{ "COLOR_MAGENTA", 0xFF00FFFF },
|
|
|
|
{ "EMU_STARTED", JS_EMU_STARTED },
|
|
{ "EMU_STOPPED", JS_EMU_STOPPED },
|
|
{ "EMU_PAUSED", JS_EMU_PAUSED },
|
|
{ "EMU_RESUMED", JS_EMU_RESUMED },
|
|
{ "EMU_RESETTING", JS_EMU_RESETTING },
|
|
{ "EMU_RESET", JS_EMU_RESET },
|
|
{ "EMU_LOADED_ROM", JS_EMU_LOADED_ROM },
|
|
{ "EMU_LOADED_STATE", JS_EMU_LOADED_STATE },
|
|
{ "EMU_DEBUG_PAUSED", JS_EMU_DEBUG_PAUSED },
|
|
{ "EMU_DEBUG_RESUMED", JS_EMU_DEBUG_RESUMED },
|
|
|
|
{ "IMG_I4", IMG_I4 },
|
|
{ "IMG_I8", IMG_I8 },
|
|
{ "IMG_IA4", IMG_IA4 },
|
|
{ "IMG_IA8", IMG_IA8 },
|
|
{ "IMG_IA16", IMG_IA16 },
|
|
{ "IMG_RGBA16", IMG_RGBA16 },
|
|
{ "IMG_RGBA32", IMG_RGBA32 },
|
|
{ "IMG_CI8_RGBA16", IMG_CI8_RGBA16 },
|
|
{ "IMG_CI4_RGBA16", IMG_CI4_RGBA16 },
|
|
{ "IMG_CI8_IA16", IMG_CI8_IA16 },
|
|
{ "IMG_CI4_IA16", IMG_CI4_IA16 },
|
|
|
|
{ "G_IM_FMT_RGBA", G_IM_FMT_RGBA },
|
|
{ "G_IM_FMT_YUV", G_IM_FMT_YUV },
|
|
{ "G_IM_FMT_CI", G_IM_FMT_CI },
|
|
{ "G_IM_FMT_IA", G_IM_FMT_IA },
|
|
{ "G_IM_FMT_I", G_IM_FMT_I },
|
|
|
|
{ "G_IM_SIZ_4b", G_IM_SIZ_4b },
|
|
{ "G_IM_SIZ_8b", G_IM_SIZ_8b },
|
|
{ "G_IM_SIZ_16b", G_IM_SIZ_16b },
|
|
{ "G_IM_SIZ_32b", G_IM_SIZ_32b },
|
|
|
|
{ "G_TT_NONE", G_TT_NONE },
|
|
{ "G_TT_RGBA16", G_TT_RGBA16 },
|
|
{ "G_TT_IA16", G_TT_IA16 },
|
|
|
|
{ nullptr, 0 },
|
|
};
|
|
|
|
duk_push_global_object(ctx);
|
|
duk_put_number_list(ctx, -1, numbers);
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
CScriptInstance* ScriptAPI::GetInstance(duk_context* ctx)
|
|
{
|
|
duk_get_global_string(ctx, HS_gInstancePtr);
|
|
CScriptInstance* instance = (CScriptInstance*)duk_get_pointer(ctx, -1);
|
|
duk_pop(ctx);
|
|
return instance;
|
|
}
|
|
|
|
JSAppCallbackID ScriptAPI::AddAppCallback(duk_context* ctx, duk_idx_t callbackIdx, JSAppHookID hookId,
|
|
JSDukArgSetupFunc argSetupFunc, JSAppCallbackCondFunc conditionFunc, JSAppCallbackCleanupFunc cleanupFunc)
|
|
{
|
|
void* dukFuncHeapPtr = duk_get_heapptr(ctx, callbackIdx);
|
|
JSAppCallback cb(GetInstance(ctx), dukFuncHeapPtr, conditionFunc, argSetupFunc, cleanupFunc);
|
|
return AddAppCallback(ctx, hookId, cb);
|
|
}
|
|
|
|
JSAppCallbackID ScriptAPI::AddAppCallback(duk_context* ctx, JSAppHookID hookId, JSAppCallback& callback)
|
|
{
|
|
CScriptInstance* inst = GetInstance(ctx);
|
|
JSAppCallbackID callbackId = inst->System()->RawAddAppCallback(hookId, callback);
|
|
|
|
if(callbackId == JS_INVALID_CALLBACK)
|
|
{
|
|
inst->System()->ConsoleLog("[SCRIPTSYS]: error: callback was not added");
|
|
return JS_INVALID_CALLBACK;
|
|
}
|
|
|
|
duk_get_global_string(ctx, HS_gAppCallbacks);
|
|
|
|
duk_push_object(ctx);
|
|
duk_push_number(ctx, hookId);
|
|
duk_put_prop_string(ctx, -2, "hookId");
|
|
duk_push_number(ctx, callbackId);
|
|
duk_put_prop_string(ctx, -2, "callbackId");
|
|
duk_push_heapptr(ctx, callback.m_DukFuncHeapPtr);
|
|
duk_put_prop_string(ctx, -2, "func");
|
|
|
|
duk_push_c_function(ctx, js__AppCallbackFinalizer, 1);
|
|
duk_set_finalizer(ctx, -2);
|
|
|
|
duk_put_prop_index(ctx, -2, callbackId);
|
|
|
|
duk_pop(ctx);
|
|
|
|
inst->IncRefCount();
|
|
|
|
return callbackId;
|
|
}
|
|
|
|
bool ScriptAPI::RemoveAppCallback(duk_context* ctx, JSAppCallbackID callbackId)
|
|
{
|
|
duk_get_global_string(ctx, HS_gAppCallbacks);
|
|
duk_bool_t bExists = duk_has_prop_index(ctx, -1, callbackId);
|
|
|
|
if(bExists)
|
|
{
|
|
// will invoke CallbackFinalizer
|
|
duk_del_prop_index(ctx, -1, callbackId);
|
|
}
|
|
|
|
duk_pop(ctx);
|
|
return bExists != 0;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js__AppCallbackFinalizer(duk_context* ctx)
|
|
{
|
|
CScriptInstance* inst = ScriptAPI::GetInstance(ctx);
|
|
|
|
duk_get_prop_string(ctx, 0, "hookId");
|
|
duk_get_prop_string(ctx, 0, "callbackId");
|
|
|
|
JSAppHookID hookId = (JSAppHookID)duk_get_uint(ctx, -2);
|
|
JSAppCallbackID callbackId = (JSAppCallbackID)duk_get_uint(ctx, -1);
|
|
duk_pop_n(ctx, 2);
|
|
|
|
bool bRemoved = inst->System()->RawRemoveAppCallback(hookId, callbackId);
|
|
|
|
if(bRemoved)
|
|
{
|
|
inst->DecRefCount();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void ScriptAPI::RefObject(duk_context* ctx, duk_idx_t idx)
|
|
{
|
|
idx = duk_normalize_index(ctx, idx);
|
|
CScriptInstance* inst = GetInstance(ctx);
|
|
|
|
if (duk_has_prop_string(ctx, idx, HS_objectRefId))
|
|
{
|
|
return;
|
|
}
|
|
|
|
duk_push_global_object(ctx);
|
|
|
|
duk_get_prop_string(ctx, -1, HS_gNextObjectRefId);
|
|
int curObjectId = duk_get_int(ctx, -1);
|
|
duk_pop(ctx);
|
|
duk_push_int(ctx, curObjectId + 1);
|
|
duk_put_prop_string(ctx, -2, HS_gNextObjectRefId);
|
|
|
|
duk_push_int(ctx, curObjectId);
|
|
duk_put_prop_string(ctx, idx, HS_objectRefId);
|
|
|
|
duk_get_prop_string(ctx, -1, HS_gObjectRefs);
|
|
duk_dup(ctx, idx);
|
|
duk_put_prop_index(ctx, -2, curObjectId);
|
|
|
|
duk_pop_n(ctx, 2);
|
|
|
|
inst->IncRefCount();
|
|
}
|
|
|
|
void ScriptAPI::UnrefObject(duk_context* ctx, duk_idx_t idx)
|
|
{
|
|
idx = duk_normalize_index(ctx, idx);
|
|
CScriptInstance* inst = GetInstance(ctx);
|
|
|
|
if (!duk_has_prop_string(ctx, idx, HS_objectRefId))
|
|
{
|
|
return;
|
|
}
|
|
|
|
duk_get_prop_string(ctx, idx, HS_objectRefId);
|
|
int objectId = duk_get_int(ctx, -1);
|
|
duk_del_prop_string(ctx, idx, HS_objectRefId);
|
|
duk_pop(ctx);
|
|
|
|
duk_push_global_object(ctx);
|
|
duk_get_prop_string(ctx, -1, HS_gObjectRefs);
|
|
duk_del_prop_index(ctx, -1, objectId);
|
|
duk_pop_n(ctx, 2);
|
|
|
|
inst->DecRefCount();
|
|
}
|
|
|
|
// PostCMethodCall variant
|
|
duk_ret_t ScriptAPI::js__UnrefObject(duk_context* ctx)
|
|
{
|
|
duk_push_this(ctx);
|
|
UnrefObject(ctx, -1);
|
|
return 0;
|
|
}
|
|
|
|
void ScriptAPI::InitEmitter(duk_context* ctx, duk_idx_t obj_idx, const std::vector<std::string>& eventNames)
|
|
{
|
|
obj_idx = duk_normalize_index(ctx, obj_idx);
|
|
|
|
duk_push_object(ctx);
|
|
|
|
std::vector<std::string>::const_iterator it;
|
|
for (it = eventNames.begin(); it != eventNames.end(); it++)
|
|
{
|
|
duk_push_object(ctx);
|
|
duk_put_prop_string(ctx, -2, (*it).c_str());
|
|
}
|
|
|
|
duk_put_prop_string(ctx, obj_idx, HS_emitterListeners);
|
|
|
|
duk_push_int(ctx, 0);
|
|
duk_put_prop_string(ctx, obj_idx, HS_emitterNextListenerId);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js__Emitter_emit(duk_context* ctx)
|
|
{
|
|
const char* eventName = duk_get_string(ctx, 0);
|
|
duk_idx_t numListenerArgs = duk_get_top(ctx) - 1;
|
|
|
|
duk_push_this(ctx);
|
|
|
|
duk_get_prop_string(ctx, -1, HS_emitterListeners);
|
|
if (!duk_has_prop_string(ctx, -1, eventName))
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "emit: invalid event name '%s'", eventName);
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_get_prop_string(ctx, -1, eventName);
|
|
duk_enum(ctx, -1, 0);
|
|
|
|
while (duk_next(ctx, -1, (duk_bool_t)true))
|
|
{
|
|
duk_push_this(ctx);
|
|
for (duk_idx_t nArg = 0; nArg < numListenerArgs; nArg++)
|
|
{
|
|
duk_dup(ctx, 1 + nArg);
|
|
}
|
|
|
|
// [ listenerFunc this args... ] -> [ retval ]
|
|
if (duk_pcall_method(ctx, numListenerArgs) != 0)
|
|
{
|
|
duk_throw(ctx);
|
|
}
|
|
|
|
duk_pop_n(ctx, 2);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js__Emitter_on(duk_context* ctx)
|
|
{
|
|
CheckArgs(ctx, { Arg_String, Arg_Function });
|
|
|
|
const char* eventName = duk_get_string(ctx, 0);
|
|
|
|
duk_push_this(ctx);
|
|
duk_get_prop_string(ctx, -1, HS_emitterListeners);
|
|
|
|
if (!duk_has_prop_string(ctx, -1, eventName))
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "invalid event name");
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_get_prop_string(ctx, -2, HS_emitterNextListenerId);
|
|
duk_size_t nextIdx = duk_get_int(ctx, -1);
|
|
duk_pop(ctx);
|
|
duk_push_int(ctx, nextIdx + 1);
|
|
duk_put_prop_string(ctx, -3, HS_emitterNextListenerId);
|
|
|
|
duk_get_prop_string(ctx, -1, eventName);
|
|
|
|
duk_pull(ctx, 1);
|
|
duk_put_prop_index(ctx, -2, nextIdx);
|
|
|
|
duk_push_this(ctx);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js__Emitter_off(duk_context* ctx)
|
|
{
|
|
CheckArgs(ctx, { Arg_String, Arg_Function });
|
|
|
|
const char* eventName = duk_get_string(ctx, 0);
|
|
duk_push_this(ctx);
|
|
duk_get_prop_string(ctx, -1, HS_emitterListeners);
|
|
|
|
if (!duk_has_prop_string(ctx, -1, eventName))
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "invalid event name");
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_get_prop_string(ctx, -1, eventName);
|
|
|
|
duk_enum(ctx, -1, 0);
|
|
while (duk_next(ctx, -1, (duk_bool_t)true))
|
|
{
|
|
if (duk_equals(ctx, 1, -1))
|
|
{
|
|
duk_pop(ctx);
|
|
duk_del_prop(ctx, -3);
|
|
}
|
|
else
|
|
{
|
|
duk_pop_n(ctx, 2);
|
|
}
|
|
}
|
|
|
|
duk_push_this(ctx);
|
|
return 1;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_Duktape_modSearch(duk_context* ctx)
|
|
{
|
|
if (!duk_is_string(ctx, 0))
|
|
{
|
|
return ThrowInvalidArgsError(ctx);
|
|
}
|
|
|
|
const char* id = duk_get_string(ctx, 0);
|
|
|
|
stdstr strPath = GetInstance(ctx)->System()->ModulesDirPath() + id;
|
|
CPath path(strPath);
|
|
|
|
if (path.GetExtension() == "dll")
|
|
{
|
|
HMODULE hModule = LoadLibraryA(strPath.c_str());
|
|
|
|
if (hModule == nullptr)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_ERROR,
|
|
"failed to load native module (\"%s\")", strPath.c_str());
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
stdstr strProcName = stdstr_f("dukopen_%s", path.GetName().c_str());
|
|
duk_c_function fnEntryPoint = (duk_c_function)GetProcAddress(hModule, strProcName.c_str());
|
|
|
|
if (fnEntryPoint == nullptr)
|
|
{
|
|
FreeLibrary(hModule);
|
|
duk_push_error_object(ctx, DUK_ERR_ERROR,
|
|
"failed to locate module entry-point (\"%s\")", strProcName.c_str());
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_push_c_function(ctx, fnEntryPoint, 0);
|
|
|
|
if (duk_pcall(ctx, 0) != 0)
|
|
{
|
|
FreeLibrary(hModule);
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
RegisterNativeModule(ctx, hModule);
|
|
|
|
duk_put_prop_string(ctx, 3, "exports");
|
|
return 0;
|
|
}
|
|
|
|
CFile file(strPath.c_str(), CFile::modeRead);
|
|
|
|
if (!file.IsOpen())
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
uint32_t length = file.GetLength();
|
|
|
|
char* sourceCode = new char[length + 1];
|
|
sourceCode[length] = '\0';
|
|
|
|
if (file.Read(sourceCode, length) != length)
|
|
{
|
|
delete[] sourceCode;
|
|
return 0;
|
|
}
|
|
|
|
duk_push_string(ctx, sourceCode);
|
|
delete[] sourceCode;
|
|
return 1;
|
|
}
|
|
|
|
void ScriptAPI::RegisterNativeModule(duk_context* ctx, HMODULE hModule)
|
|
{
|
|
duk_get_global_string(ctx, HS_gNativeModules);
|
|
duk_size_t index = duk_get_length(ctx, -1);
|
|
duk_push_object(ctx);
|
|
duk_push_pointer(ctx, hModule);
|
|
duk_put_prop_string(ctx, -2, "modPtr");
|
|
duk_push_c_function(ctx, js__NativeModuleFinalizer, 1);
|
|
duk_set_finalizer(ctx, -2);
|
|
duk_put_prop_index(ctx, -2, index);
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js__NativeModuleFinalizer(duk_context* ctx)
|
|
{
|
|
duk_get_prop_string(ctx, 0, "modPtr");
|
|
HMODULE hModule = (HMODULE)duk_get_pointer(ctx, -1);
|
|
FreeLibrary(hModule);
|
|
return 0;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::ThrowInvalidArgsError(duk_context* ctx)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "invalid argument(s)");
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::ThrowInvalidArgError(duk_context * ctx, duk_idx_t idx, ArgType wantType)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "argument %d invalid, expected %s", idx, ArgTypeName(wantType));
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::ThrowTooManyArgsError(duk_context * ctx)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "too many arguments");
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::ThrowInvalidAssignmentError(duk_context* ctx, ArgType wantType)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "invalid assignment, expected %s", ArgTypeName(wantType));
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::ThrowNotCallableError(duk_context* ctx)
|
|
{
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "not callable");
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
void ScriptAPI::DebugStack(duk_context* ctx, const char* file, int line)
|
|
{
|
|
duk_push_context_dump(ctx);
|
|
GetInstance(ctx)->System()->ConsoleLog("[SCRIPTSYS] <%s:%d> %s", file, line, duk_to_string(ctx, -1));
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
void ScriptAPI::AllowPrivateCall(duk_context* ctx, bool bAllow)
|
|
{
|
|
duk_push_boolean(ctx, (duk_bool_t)bAllow);
|
|
duk_put_global_string(ctx, HS_gPrivateCallEnabled);
|
|
}
|
|
|
|
bool ScriptAPI::PrivateCallAllowed(duk_context* ctx)
|
|
{
|
|
if (!duk_get_global_string(ctx, HS_gPrivateCallEnabled))
|
|
{
|
|
duk_pop(ctx);
|
|
return false;
|
|
}
|
|
|
|
bool bAllowed = duk_get_boolean(ctx, -1);
|
|
duk_pop(ctx);
|
|
return bAllowed;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::js_DummyConstructor(duk_context* ctx)
|
|
{
|
|
return ThrowNotCallableError(ctx);
|
|
}
|
|
|
|
void ScriptAPI::PushNewDummyConstructor(duk_context* ctx, bool bFrozen)
|
|
{
|
|
duk_push_c_function(ctx, js_DummyConstructor, 0);
|
|
duk_push_object(ctx);
|
|
duk_put_prop_string(ctx, -2, "prototype");
|
|
|
|
if (bFrozen)
|
|
{
|
|
duk_freeze(ctx, -1);
|
|
}
|
|
}
|
|
|
|
void ScriptAPI::DefineGlobalDummyConstructors(duk_context* ctx, const char* constructorNames[], bool bFreeze)
|
|
{
|
|
duk_push_global_object(ctx);
|
|
|
|
for (size_t i = 0;; i++)
|
|
{
|
|
if (constructorNames[i] == nullptr)
|
|
{
|
|
break;
|
|
}
|
|
duk_push_string(ctx, constructorNames[i]);
|
|
PushNewDummyConstructor(ctx, bFreeze);
|
|
duk_def_prop(ctx, -3, DUK_DEFPROP_HAVE_VALUE | DUK_DEFPROP_SET_ENUMERABLE);
|
|
}
|
|
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
void ScriptAPI::SetDummyConstructor(duk_context* ctx, duk_idx_t obj_idx, const char* globalConstructorName)
|
|
{
|
|
obj_idx = duk_normalize_index(ctx, obj_idx);
|
|
duk_get_global_string(ctx, globalConstructorName);
|
|
duk_get_prop_string(ctx, -1, "prototype");
|
|
duk_set_prototype(ctx, obj_idx);
|
|
duk_pop(ctx);
|
|
}
|
|
|
|
void ScriptAPI::DukPutPropList(duk_context* ctx, duk_idx_t obj_idx, const DukPropListEntry* props)
|
|
{
|
|
obj_idx = duk_normalize_index(ctx, obj_idx);
|
|
|
|
for (size_t i = 0;; i++)
|
|
{
|
|
const DukPropListEntry& prop = props[i];
|
|
|
|
if (prop.key == nullptr)
|
|
{
|
|
break;
|
|
}
|
|
|
|
duk_uint_t propFlags = 0;
|
|
bool bHiddenSymbol = (prop.key[0] == '\xFF');
|
|
|
|
if (!bHiddenSymbol)
|
|
{
|
|
propFlags |= prop.writable ? DUK_DEFPROP_SET_WRITABLE : 0;
|
|
propFlags |= prop.enumerable ? DUK_DEFPROP_SET_ENUMERABLE : 0;
|
|
}
|
|
else
|
|
{
|
|
if (prop.typeId == Type_DukGetter ||
|
|
prop.typeId == Type_DukGetterSetter)
|
|
{
|
|
// not compatible
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
}
|
|
|
|
duk_push_string(ctx, prop.key);
|
|
|
|
switch (prop.typeId)
|
|
{
|
|
case Type_DukNumber:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_number(ctx, prop.value.dukNumber.value);
|
|
break;
|
|
case Type_DukInt:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_int(ctx, prop.value.dukInt.value);
|
|
break;
|
|
case Type_DukUInt:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_uint(ctx, prop.value.dukUInt.value);
|
|
break;
|
|
case Type_DukBoolean:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_boolean(ctx, prop.value.dukBoolean.value);
|
|
break;
|
|
case Type_DukString:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_string(ctx, prop.value.dukString.value);
|
|
break;
|
|
case Type_DukPointer:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_pointer(ctx, prop.value.dukPointer.value);
|
|
break;
|
|
case Type_DukCFunction:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_c_function(ctx, prop.value.dukCFunction.func, prop.value.dukCFunction.nargs);
|
|
break;
|
|
case Type_DukDupIndex:
|
|
{
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_idx_t fixedDupIndex = prop.value.dukDupIndex.value;
|
|
if (fixedDupIndex < 0)
|
|
{
|
|
// -1 to account for prop.key push above
|
|
fixedDupIndex = duk_normalize_index(ctx, fixedDupIndex - 1);
|
|
}
|
|
duk_dup(ctx, fixedDupIndex);
|
|
}
|
|
break;
|
|
case Type_DukGetter:
|
|
propFlags |= DUK_DEFPROP_HAVE_GETTER;
|
|
duk_push_c_function(ctx, prop.value.dukGetter.value, 0);
|
|
break;
|
|
case Type_DukGetterSetter:
|
|
propFlags |= DUK_DEFPROP_HAVE_GETTER | DUK_DEFPROP_HAVE_SETTER;
|
|
duk_push_c_function(ctx, prop.value.dukGetterSetter.getter, 0);
|
|
duk_push_c_function(ctx, prop.value.dukGetterSetter.setter, 1);
|
|
break;
|
|
case Type_DukObject:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_object(ctx);
|
|
if (prop.value.dukObject.props != nullptr)
|
|
{
|
|
DukPutPropList(ctx, -1, prop.value.dukObject.props);
|
|
}
|
|
break;
|
|
case Type_DukProxy:
|
|
propFlags |= DUK_DEFPROP_HAVE_VALUE;
|
|
duk_push_object(ctx); // empty target
|
|
duk_push_object(ctx); // handler
|
|
duk_push_c_function(ctx, prop.value.dukProxy.getter, 2);
|
|
duk_put_prop_string(ctx, -2, "get");
|
|
duk_push_c_function(ctx, prop.value.dukProxy.setter, 3);
|
|
duk_put_prop_string(ctx, -2, "set");
|
|
duk_push_proxy(ctx, 0);
|
|
break;
|
|
default:
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
break;
|
|
}
|
|
|
|
if (bHiddenSymbol)
|
|
{
|
|
duk_put_prop(ctx, obj_idx);
|
|
}
|
|
else
|
|
{
|
|
duk_def_prop(ctx, obj_idx, propFlags);
|
|
}
|
|
}
|
|
}
|
|
|
|
duk_bool_t ScriptAPI::ArgTypeMatches(duk_context* ctx, duk_idx_t idx, ArgType wantType)
|
|
{
|
|
ArgType argType = (ArgType)(wantType & (~ArgAttrs));
|
|
|
|
switch (argType)
|
|
{
|
|
case Arg_Any:
|
|
return true;
|
|
case Arg_Number:
|
|
return duk_is_number(ctx, idx);
|
|
case Arg_BufferData:
|
|
return duk_is_buffer_data(ctx, idx);
|
|
case Arg_String:
|
|
return duk_is_string(ctx, idx);
|
|
case Arg_Function:
|
|
return duk_is_function(ctx, idx);
|
|
case Arg_Object:
|
|
return duk_is_object(ctx, idx);
|
|
case Arg_Array:
|
|
return duk_is_array(ctx, idx);
|
|
case Arg_Boolean:
|
|
return duk_is_boolean(ctx, idx);
|
|
default:
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
return false;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
const char* ScriptAPI::ArgTypeName(ArgType argType)
|
|
{
|
|
static const std::map<ArgType, std::string> argTypeNames = {
|
|
{ Arg_Any, "any" },
|
|
{ Arg_Number, "number" },
|
|
{ Arg_BufferData, "bufferdata" },
|
|
{ Arg_String, "string" },
|
|
{ Arg_Function, "function" },
|
|
{ Arg_Object, "object" },
|
|
{ Arg_Array, "array" },
|
|
{ Arg_Boolean, "boolean" }
|
|
};
|
|
|
|
if (argTypeNames.count(argType) == 0)
|
|
{
|
|
g_Notify->BreakPoint(__FILE__, __LINE__);
|
|
}
|
|
|
|
return argTypeNames.at(argType).c_str();
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::CheckSetterAssignment(duk_context* ctx, ArgType wantType)
|
|
{
|
|
if (!ArgTypeMatches(ctx, 0, wantType))
|
|
{
|
|
return ThrowInvalidAssignmentError(ctx, wantType);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
duk_ret_t ScriptAPI::CheckArgs(duk_context* ctx, const std::vector<ArgType>& argTypes)
|
|
{
|
|
duk_idx_t nargs = duk_get_top(ctx);
|
|
|
|
if ((size_t)nargs > argTypes.size())
|
|
{
|
|
return ThrowTooManyArgsError(ctx);
|
|
}
|
|
|
|
duk_idx_t idx = 0;
|
|
std::vector<ArgType>::const_iterator it;
|
|
for (it = argTypes.begin(); it != argTypes.end(); it++)
|
|
{
|
|
bool bOptional = (*it & ArgAttr_Optional) != 0;
|
|
ArgType argType = (ArgType)(*it & (~ArgAttrs));
|
|
|
|
if (idx >= nargs)
|
|
{
|
|
if (bOptional)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
duk_push_error_object(ctx, DUK_ERR_TYPE_ERROR, "argument(s) missing", idx);
|
|
return duk_throw(ctx);
|
|
}
|
|
|
|
if (!ArgTypeMatches(ctx, idx, argType))
|
|
{
|
|
return ThrowInvalidArgError(ctx, idx, argType);
|
|
}
|
|
|
|
idx++;
|
|
}
|
|
|
|
return 0;
|
|
}
|