2024-05-17 17:45:40 +01:00
|
|
|
#include "../js_modules.h"
|
|
|
|
|
#include "furi_hal_random.h"
|
|
|
|
|
#include <float.h>
|
|
|
|
|
|
2024-07-15 07:38:49 +03:00
|
|
|
#define JS_MATH_PI ((double)M_PI)
|
|
|
|
|
#define JS_MATH_E ((double)M_E)
|
2024-05-17 17:45:40 +01:00
|
|
|
#define JS_MATH_EPSILON ((double)DBL_EPSILON)
|
|
|
|
|
|
|
|
|
|
#define TAG "JsMath"
|
|
|
|
|
|
|
|
|
|
static void ret_bad_args(struct mjs* mjs, const char* error) {
|
|
|
|
|
mjs_prepend_errorf(mjs, MJS_BAD_ARGS_ERROR, "%s", error);
|
|
|
|
|
mjs_return(mjs, MJS_UNDEFINED);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool check_args(struct mjs* mjs, size_t count) {
|
|
|
|
|
size_t num_args = mjs_nargs(mjs);
|
|
|
|
|
if(num_args != count) {
|
|
|
|
|
ret_bad_args(mjs, "Wrong argument count");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
for(size_t i = 0; i < count; i++) {
|
|
|
|
|
if(!mjs_is_number(mjs_arg(mjs, i))) {
|
|
|
|
|
ret_bad_args(mjs, "Wrong argument type");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_is_equal(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 3)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double a = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
double b = mjs_get_double(mjs, mjs_arg(mjs, 1));
|
|
|
|
|
double e = mjs_get_double(mjs, mjs_arg(mjs, 2));
|
|
|
|
|
double f = fabs(a - b);
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_boolean(mjs, (f <= e)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_abs(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, fabs(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_acos(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
if(x < (double)-1. || x > (double)1.) {
|
|
|
|
|
ret_bad_args(mjs, "Invalid input value for math.acos");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, acos(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_acosh(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
if(x < (double)1.) {
|
|
|
|
|
ret_bad_args(mjs, "Invalid input value for math.acosh");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, log(x + sqrt(x * x - (double)1.))));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_asin(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, asin(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_asinh(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, log(x + sqrt(x * x + (double)1.))));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_atan(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, atan(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_atan2(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 2)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double y = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 1));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, atan2(y, x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_atanh(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
if(x < (double)-1. || x > (double)1.) {
|
|
|
|
|
ret_bad_args(mjs, "Invalid input value for math.atanh");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, (double)0.5 * log(((double)1. + x) / ((double)1. - x))));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_cbrt(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, cbrt(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_ceil(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, ceil(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_clz32(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned int x = (unsigned int)mjs_get_int(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
int count = 0;
|
|
|
|
|
while(x) {
|
|
|
|
|
x >>= 1;
|
|
|
|
|
count++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, 32 - count));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_cos(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, cos(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_exp(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, exp(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_floor(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, floor(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_log(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
if(x <= 0) {
|
|
|
|
|
ret_bad_args(mjs, "Invalid input value for math.log");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, log(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_max(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 2)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
double y = mjs_get_double(mjs, mjs_arg(mjs, 1));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, x > y ? x : y));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_min(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 2)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
double y = mjs_get_double(mjs, mjs_arg(mjs, 1));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, x < y ? x : y));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_pow(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 2)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double base = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
double exponent = mjs_get_double(mjs, mjs_arg(mjs, 1));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, pow(base, exponent)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_random(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 0)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// double clearly provides more bits for entropy then we pack
|
|
|
|
|
// 32bit should be enough for now, but fix it maybe
|
|
|
|
|
const uint32_t random_val = furi_hal_random_get();
|
|
|
|
|
double rnd = (double)random_val / (double)FURI_HAL_RANDOM_MAX;
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, rnd));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_sign(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(
|
|
|
|
|
mjs,
|
2024-05-17 18:43:52 +01:00
|
|
|
mjs_mk_number(
|
|
|
|
|
mjs, fabs(x) <= JS_MATH_EPSILON ? 0 : (x < (double)0. ? (double)-1.0 : (double)1.0)));
|
2024-05-17 17:45:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_sin(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, sin(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_sqrt(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
if(x < (double)0.) {
|
|
|
|
|
ret_bad_args(mjs, "Invalid input value for math.sqrt");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, sqrt(x)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void js_math_trunc(struct mjs* mjs) {
|
|
|
|
|
if(!check_args(mjs, 1)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double x = mjs_get_double(mjs, mjs_arg(mjs, 0));
|
|
|
|
|
|
|
|
|
|
mjs_return(mjs, mjs_mk_number(mjs, x < (double)0. ? ceil(x) : floor(x)));
|
|
|
|
|
}
|
|
|
|
|
|
[FL-3893] JS modules (#3841)
* feat: backport js_gpio from unleashed
* feat: backport js_keyboard, TextInputModel::minimum_length from unleashed
* fix: api version inconsistency
* style: js_gpio
* build: fix submodule ._ .
* refactor: js_gpio
* docs: type declarations for gpio
* feat: gpio interrupts
* fix: js_gpio freeing, resetting and minor stylistic changes
* style: js_gpio
* style: mlib array, fixme's
* feat: js_gpio adc
* feat: js_event_loop
* docs: js_event_loop
* feat: js_event_loop subscription cancellation
* feat: js_event_loop + js_gpio integration
* fix: js_event_loop memory leak
* feat: stop event loop on back button
* test: js: basic, math, event_loop
* feat: js_event_loop queue
* feat: js linkage to previously loaded plugins
* build: fix ci errors
* feat: js module ordered teardown
* feat: js_gui_defer_free
* feat: basic hourglass view
* style: JS ASS (Argument Schema for Scripts)
* fix: js_event_loop mem leaks and lifetime problems
* fix: crashing test and pvs false positives
* feat: mjs custom obj destructors, gui submenu view
* refactor: yank js_gui_defer_free (yuck)
* refactor: maybe_unsubscribe
* empty_screen, docs, typing fix-ups
* docs: navigation event & demo
* feat: submenu setHeader
* feat: text_input
* feat: text_box
* docs: text_box availability
* ci: silence irrelevant pvs low priority warning
* style: use furistring
* style: _get_at -> _safe_get
* fix: built-in module name assignment
* feat: js_dialog; refactor, optimize: js_gui
* docs: js_gui
* ci: silence pvs warning: Memory allocation is infallible
* style: fix storage spelling
* feat: foreign pointer signature checks
* feat: js_storage
* docs: js_storage
* fix: my unit test was breaking other tests ;_;
* ci: fix ci?
* Make doxygen happy
* docs: flipper, math, notification, global
* style: review suggestions
* style: review fixups
* fix: badusb demo script
* docs: badusb
* ci: add nofl
* ci: make linter happy
* Bump api version
Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
2024-10-14 21:42:11 +03:00
|
|
|
static void* js_math_create(struct mjs* mjs, mjs_val_t* object, JsModules* modules) {
|
|
|
|
|
UNUSED(modules);
|
2024-05-17 17:45:40 +01:00
|
|
|
mjs_val_t math_obj = mjs_mk_object(mjs);
|
2024-10-31 05:22:05 +00:00
|
|
|
mjs_set(mjs, math_obj, "isEqual", ~0, MJS_MK_FN(js_math_is_equal));
|
2024-05-17 17:45:40 +01:00
|
|
|
mjs_set(mjs, math_obj, "abs", ~0, MJS_MK_FN(js_math_abs));
|
|
|
|
|
mjs_set(mjs, math_obj, "acos", ~0, MJS_MK_FN(js_math_acos));
|
|
|
|
|
mjs_set(mjs, math_obj, "acosh", ~0, MJS_MK_FN(js_math_acosh));
|
|
|
|
|
mjs_set(mjs, math_obj, "asin", ~0, MJS_MK_FN(js_math_asin));
|
|
|
|
|
mjs_set(mjs, math_obj, "asinh", ~0, MJS_MK_FN(js_math_asinh));
|
|
|
|
|
mjs_set(mjs, math_obj, "atan", ~0, MJS_MK_FN(js_math_atan));
|
|
|
|
|
mjs_set(mjs, math_obj, "atan2", ~0, MJS_MK_FN(js_math_atan2));
|
|
|
|
|
mjs_set(mjs, math_obj, "atanh", ~0, MJS_MK_FN(js_math_atanh));
|
|
|
|
|
mjs_set(mjs, math_obj, "cbrt", ~0, MJS_MK_FN(js_math_cbrt));
|
|
|
|
|
mjs_set(mjs, math_obj, "ceil", ~0, MJS_MK_FN(js_math_ceil));
|
|
|
|
|
mjs_set(mjs, math_obj, "clz32", ~0, MJS_MK_FN(js_math_clz32));
|
|
|
|
|
mjs_set(mjs, math_obj, "cos", ~0, MJS_MK_FN(js_math_cos));
|
|
|
|
|
mjs_set(mjs, math_obj, "exp", ~0, MJS_MK_FN(js_math_exp));
|
|
|
|
|
mjs_set(mjs, math_obj, "floor", ~0, MJS_MK_FN(js_math_floor));
|
|
|
|
|
mjs_set(mjs, math_obj, "log", ~0, MJS_MK_FN(js_math_log));
|
|
|
|
|
mjs_set(mjs, math_obj, "max", ~0, MJS_MK_FN(js_math_max));
|
|
|
|
|
mjs_set(mjs, math_obj, "min", ~0, MJS_MK_FN(js_math_min));
|
|
|
|
|
mjs_set(mjs, math_obj, "pow", ~0, MJS_MK_FN(js_math_pow));
|
|
|
|
|
mjs_set(mjs, math_obj, "random", ~0, MJS_MK_FN(js_math_random));
|
|
|
|
|
mjs_set(mjs, math_obj, "sign", ~0, MJS_MK_FN(js_math_sign));
|
|
|
|
|
mjs_set(mjs, math_obj, "sin", ~0, MJS_MK_FN(js_math_sin));
|
|
|
|
|
mjs_set(mjs, math_obj, "sqrt", ~0, MJS_MK_FN(js_math_sqrt));
|
|
|
|
|
mjs_set(mjs, math_obj, "trunc", ~0, MJS_MK_FN(js_math_trunc));
|
|
|
|
|
mjs_set(mjs, math_obj, "PI", ~0, mjs_mk_number(mjs, JS_MATH_PI));
|
|
|
|
|
mjs_set(mjs, math_obj, "E", ~0, mjs_mk_number(mjs, JS_MATH_E));
|
|
|
|
|
mjs_set(mjs, math_obj, "EPSILON", ~0, mjs_mk_number(mjs, JS_MATH_EPSILON));
|
|
|
|
|
*object = math_obj;
|
|
|
|
|
return (void*)1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const JsModuleDescriptor js_math_desc = {
|
|
|
|
|
"math",
|
|
|
|
|
js_math_create,
|
|
|
|
|
NULL,
|
[FL-3893] JS modules (#3841)
* feat: backport js_gpio from unleashed
* feat: backport js_keyboard, TextInputModel::minimum_length from unleashed
* fix: api version inconsistency
* style: js_gpio
* build: fix submodule ._ .
* refactor: js_gpio
* docs: type declarations for gpio
* feat: gpio interrupts
* fix: js_gpio freeing, resetting and minor stylistic changes
* style: js_gpio
* style: mlib array, fixme's
* feat: js_gpio adc
* feat: js_event_loop
* docs: js_event_loop
* feat: js_event_loop subscription cancellation
* feat: js_event_loop + js_gpio integration
* fix: js_event_loop memory leak
* feat: stop event loop on back button
* test: js: basic, math, event_loop
* feat: js_event_loop queue
* feat: js linkage to previously loaded plugins
* build: fix ci errors
* feat: js module ordered teardown
* feat: js_gui_defer_free
* feat: basic hourglass view
* style: JS ASS (Argument Schema for Scripts)
* fix: js_event_loop mem leaks and lifetime problems
* fix: crashing test and pvs false positives
* feat: mjs custom obj destructors, gui submenu view
* refactor: yank js_gui_defer_free (yuck)
* refactor: maybe_unsubscribe
* empty_screen, docs, typing fix-ups
* docs: navigation event & demo
* feat: submenu setHeader
* feat: text_input
* feat: text_box
* docs: text_box availability
* ci: silence irrelevant pvs low priority warning
* style: use furistring
* style: _get_at -> _safe_get
* fix: built-in module name assignment
* feat: js_dialog; refactor, optimize: js_gui
* docs: js_gui
* ci: silence pvs warning: Memory allocation is infallible
* style: fix storage spelling
* feat: foreign pointer signature checks
* feat: js_storage
* docs: js_storage
* fix: my unit test was breaking other tests ;_;
* ci: fix ci?
* Make doxygen happy
* docs: flipper, math, notification, global
* style: review suggestions
* style: review fixups
* fix: badusb demo script
* docs: badusb
* ci: add nofl
* ci: make linter happy
* Bump api version
Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
2024-10-14 21:42:11 +03:00
|
|
|
NULL,
|
2024-05-17 17:45:40 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const FlipperAppPluginDescriptor plugin_descriptor = {
|
|
|
|
|
.appid = PLUGIN_APP_ID,
|
|
|
|
|
.ep_api_version = PLUGIN_API_VERSION,
|
|
|
|
|
.entry_point = &js_math_desc,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const FlipperAppPluginDescriptor* js_math_ep(void) {
|
|
|
|
|
return &plugin_descriptor;
|
|
|
|
|
}
|