project64/Source/Project64/UserInterface/Debugger/ScriptAPI/ScriptAPI.cpp

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, (int)((UINT_PTR)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, (int)((UINT_PTR)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, (int)((UINT_PTR)callbackId));
if (bExists)
{
// will invoke CallbackFinalizer
duk_del_prop_index(ctx, -1, (int)((UINT_PTR)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, (int)((UINT_PTR)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, (int)((UINT_PTR)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, (int)((UINT_PTR)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) != 0;
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;
}