[FL-3579, FL-3601, FL-3714] JavaScript runner (#3286)
* FBT: cdefines to env, libs order
* API: strtod, modf, itoa, calloc
* Apps: elk js
* Apps: mjs
* JS: scripts as assets
* mjs: composite resolver
* mjs: stack trace
* ELK JS example removed
* MJS thread, MJS lib modified to support script interruption
* JS console UI
* Module system, BadUSB bindings rework
* JS notifications, simple dialog, BadUSB demo
* Custom dialogs, dialog demo
* MJS as system library, some dirty hacks to make it compile
* Plugin-based js modules
* js_uart(BadUART) module
* js_uart: support for byte array arguments
* Script icon and various fixes
* File browser: multiple extensions filter, running js scripts from app loader
* Running js scripts from archive browser
* JS Runner as system app
* Example scripts moved to /ext/apps/Scripts
* JS bytecode listing generation
* MJS builtin printf cleanup
* JS examples cleanup
* mbedtls version fix
* Unused lib cleanup
* Making PVS happy & TODOs cleanup
* TODOs cleanup #2
* MJS: initial typed arrays support
* JS: fix mem leak in uart destructor
Co-authored-by: SG <who.just.the.doctor@gmail.com>
Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
2024-02-12 11:54:32 +03:00
|
|
|
#include <core/common_defines.h>
|
|
|
|
|
#include "js_modules.h"
|
|
|
|
|
#include <m-dict.h>
|
|
|
|
|
#include "modules/js_flipper.h"
|
|
|
|
|
|
|
|
|
|
#define TAG "JS modules"
|
|
|
|
|
|
2024-03-25 19:35:38 +03:00
|
|
|
// Absolute path is used to make possible plugin load from CLI
|
|
|
|
|
#define MODULES_PATH "/ext/apps_data/js_app/plugins"
|
|
|
|
|
|
[FL-3579, FL-3601, FL-3714] JavaScript runner (#3286)
* FBT: cdefines to env, libs order
* API: strtod, modf, itoa, calloc
* Apps: elk js
* Apps: mjs
* JS: scripts as assets
* mjs: composite resolver
* mjs: stack trace
* ELK JS example removed
* MJS thread, MJS lib modified to support script interruption
* JS console UI
* Module system, BadUSB bindings rework
* JS notifications, simple dialog, BadUSB demo
* Custom dialogs, dialog demo
* MJS as system library, some dirty hacks to make it compile
* Plugin-based js modules
* js_uart(BadUART) module
* js_uart: support for byte array arguments
* Script icon and various fixes
* File browser: multiple extensions filter, running js scripts from app loader
* Running js scripts from archive browser
* JS Runner as system app
* Example scripts moved to /ext/apps/Scripts
* JS bytecode listing generation
* MJS builtin printf cleanup
* JS examples cleanup
* mbedtls version fix
* Unused lib cleanup
* Making PVS happy & TODOs cleanup
* TODOs cleanup #2
* MJS: initial typed arrays support
* JS: fix mem leak in uart destructor
Co-authored-by: SG <who.just.the.doctor@gmail.com>
Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
2024-02-12 11:54:32 +03:00
|
|
|
typedef struct {
|
|
|
|
|
JsModeConstructor create;
|
|
|
|
|
JsModeDestructor destroy;
|
|
|
|
|
void* context;
|
|
|
|
|
} JsModuleData;
|
|
|
|
|
|
|
|
|
|
DICT_DEF2(JsModuleDict, FuriString*, FURI_STRING_OPLIST, JsModuleData, M_POD_OPLIST);
|
|
|
|
|
|
|
|
|
|
static const JsModuleDescriptor modules_builtin[] = {
|
|
|
|
|
{"flipper", js_flipper_create, NULL},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct JsModules {
|
|
|
|
|
struct mjs* mjs;
|
|
|
|
|
JsModuleDict_t module_dict;
|
|
|
|
|
PluginManager* plugin_manager;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
JsModules* js_modules_create(struct mjs* mjs, CompositeApiResolver* resolver) {
|
|
|
|
|
JsModules* modules = malloc(sizeof(JsModules));
|
|
|
|
|
modules->mjs = mjs;
|
|
|
|
|
JsModuleDict_init(modules->module_dict);
|
|
|
|
|
|
|
|
|
|
modules->plugin_manager = plugin_manager_alloc(
|
|
|
|
|
PLUGIN_APP_ID, PLUGIN_API_VERSION, composite_api_resolver_get(resolver));
|
|
|
|
|
|
|
|
|
|
return modules;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_modules_destroy(JsModules* modules) {
|
|
|
|
|
JsModuleDict_it_t it;
|
|
|
|
|
for(JsModuleDict_it(it, modules->module_dict); !JsModuleDict_end_p(it);
|
|
|
|
|
JsModuleDict_next(it)) {
|
|
|
|
|
const JsModuleDict_itref_t* module_itref = JsModuleDict_cref(it);
|
|
|
|
|
if(module_itref->value.destroy) {
|
|
|
|
|
module_itref->value.destroy(module_itref->value.context);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
plugin_manager_free(modules->plugin_manager);
|
|
|
|
|
JsModuleDict_clear(modules->module_dict);
|
|
|
|
|
free(modules);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mjs_val_t js_module_require(JsModules* modules, const char* name, size_t name_len) {
|
|
|
|
|
FuriString* module_name = furi_string_alloc_set_str(name);
|
|
|
|
|
// Check if module is already installed
|
|
|
|
|
JsModuleData* module_inst = JsModuleDict_get(modules->module_dict, module_name);
|
|
|
|
|
if(module_inst) { //-V547
|
|
|
|
|
furi_string_free(module_name);
|
|
|
|
|
mjs_prepend_errorf(
|
|
|
|
|
modules->mjs, MJS_BAD_ARGS_ERROR, "\"%s\" module is already installed", name);
|
|
|
|
|
return MJS_UNDEFINED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool module_found = false;
|
|
|
|
|
// Check built-in modules
|
|
|
|
|
for(size_t i = 0; i < COUNT_OF(modules_builtin); i++) { //-V1008
|
|
|
|
|
size_t name_compare_len = strlen(modules_builtin[i].name);
|
|
|
|
|
|
|
|
|
|
if(name_compare_len != name_len) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(strncmp(name, modules_builtin[i].name, name_compare_len) == 0) {
|
|
|
|
|
JsModuleData module = {
|
|
|
|
|
.create = modules_builtin[i].create, .destroy = modules_builtin[i].destroy};
|
|
|
|
|
JsModuleDict_set_at(modules->module_dict, module_name, module);
|
|
|
|
|
module_found = true;
|
|
|
|
|
FURI_LOG_I(TAG, "Using built-in module %s", name);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// External module load
|
|
|
|
|
if(!module_found) {
|
|
|
|
|
FuriString* module_path = furi_string_alloc();
|
2024-03-25 19:35:38 +03:00
|
|
|
furi_string_printf(module_path, "%s/js_%s.fal", MODULES_PATH, name);
|
[FL-3579, FL-3601, FL-3714] JavaScript runner (#3286)
* FBT: cdefines to env, libs order
* API: strtod, modf, itoa, calloc
* Apps: elk js
* Apps: mjs
* JS: scripts as assets
* mjs: composite resolver
* mjs: stack trace
* ELK JS example removed
* MJS thread, MJS lib modified to support script interruption
* JS console UI
* Module system, BadUSB bindings rework
* JS notifications, simple dialog, BadUSB demo
* Custom dialogs, dialog demo
* MJS as system library, some dirty hacks to make it compile
* Plugin-based js modules
* js_uart(BadUART) module
* js_uart: support for byte array arguments
* Script icon and various fixes
* File browser: multiple extensions filter, running js scripts from app loader
* Running js scripts from archive browser
* JS Runner as system app
* Example scripts moved to /ext/apps/Scripts
* JS bytecode listing generation
* MJS builtin printf cleanup
* JS examples cleanup
* mbedtls version fix
* Unused lib cleanup
* Making PVS happy & TODOs cleanup
* TODOs cleanup #2
* MJS: initial typed arrays support
* JS: fix mem leak in uart destructor
Co-authored-by: SG <who.just.the.doctor@gmail.com>
Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
2024-02-12 11:54:32 +03:00
|
|
|
FURI_LOG_I(TAG, "Loading external module %s", furi_string_get_cstr(module_path));
|
|
|
|
|
do {
|
|
|
|
|
uint32_t plugin_cnt_last = plugin_manager_get_count(modules->plugin_manager);
|
|
|
|
|
PluginManagerError load_error = plugin_manager_load_single(
|
|
|
|
|
modules->plugin_manager, furi_string_get_cstr(module_path));
|
|
|
|
|
if(load_error != PluginManagerErrorNone) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
const JsModuleDescriptor* plugin =
|
|
|
|
|
plugin_manager_get_ep(modules->plugin_manager, plugin_cnt_last);
|
|
|
|
|
furi_assert(plugin);
|
|
|
|
|
|
|
|
|
|
if(strncmp(name, plugin->name, name_len) != 0) {
|
|
|
|
|
FURI_LOG_E(TAG, "Module name missmatch %s", plugin->name);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
JsModuleData module = {.create = plugin->create, .destroy = plugin->destroy};
|
|
|
|
|
JsModuleDict_set_at(modules->module_dict, module_name, module);
|
|
|
|
|
|
|
|
|
|
module_found = true;
|
|
|
|
|
} while(0);
|
|
|
|
|
furi_string_free(module_path);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Run module constructor
|
|
|
|
|
mjs_val_t module_object = MJS_UNDEFINED;
|
|
|
|
|
if(module_found) {
|
|
|
|
|
module_inst = JsModuleDict_get(modules->module_dict, module_name);
|
|
|
|
|
furi_assert(module_inst);
|
|
|
|
|
if(module_inst->create) { //-V779
|
|
|
|
|
module_inst->context = module_inst->create(modules->mjs, &module_object);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(module_object == MJS_UNDEFINED) { //-V547
|
|
|
|
|
mjs_prepend_errorf(modules->mjs, MJS_BAD_ARGS_ERROR, "\"%s\" module load fail", name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
furi_string_free(module_name);
|
|
|
|
|
|
|
|
|
|
return module_object;
|
|
|
|
|
}
|