1034 lines
30 KiB
C++
1034 lines
30 KiB
C++
#include "stdafx.h"
|
|
|
|
#include "N64Image.h"
|
|
#include "ScriptAPI.h"
|
|
#include <dwrite.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_push_global_object(ctx);
|
|
duk_push_string(ctx, "PJ64_JSAPI_VERSION");
|
|
duk_push_string(ctx, PJ64_JSAPI_VERSION);
|
|
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);
|
|
|
|
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);
|
|
|
|
inst->System()->RawRemoveAppCallback(hookId, callbackId);
|
|
|
|
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);
|
|
|
|
int count = 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);
|
|
count++;
|
|
}
|
|
|
|
// throw if there are no listeners for error event
|
|
if (count == 0 && strcmp("error", eventName) == 0)
|
|
{
|
|
duk_dup(ctx, 1);
|
|
duk_throw(ctx);
|
|
}
|
|
|
|
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;
|
|
}
|