2023-02-22 20:19:06 +01:00
|
|
|
/*
|
|
|
|
* Copyright © 2011 Google, Inc.
|
|
|
|
*
|
|
|
|
* This is part of HarfBuzz, a text shaping library.
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, without written agreement and without
|
|
|
|
* license or royalty fees, to use, copy, modify, and distribute this
|
|
|
|
* software and its documentation for any purpose, provided that the
|
|
|
|
* above copyright notice and the following two paragraphs appear in
|
|
|
|
* all copies of this software.
|
|
|
|
*
|
|
|
|
* IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
|
|
|
|
* DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
|
|
|
|
* IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
|
|
|
|
* DAMAGE.
|
|
|
|
*
|
|
|
|
* THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
|
|
|
|
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
|
|
|
|
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
|
|
|
|
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
|
|
|
|
*
|
|
|
|
* Google Author(s): Behdad Esfahbod
|
|
|
|
*/
|
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
#define HB_DEBUG_WASM 1
|
|
|
|
|
2023-02-22 20:19:06 +01:00
|
|
|
#include "hb-shaper-impl.hh"
|
|
|
|
|
|
|
|
#ifdef HAVE_WASM
|
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
#include <wasm_export.h>
|
2023-02-22 20:57:19 +01:00
|
|
|
|
|
|
|
|
2023-02-22 20:19:06 +01:00
|
|
|
/*
|
|
|
|
* shaper face data
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define HB_WASM_TAG_WASM HB_TAG('W','a','s','m')
|
|
|
|
|
|
|
|
struct hb_wasm_face_data_t {
|
2023-02-23 07:30:28 +01:00
|
|
|
hb_blob_t *wasm_blob;
|
|
|
|
wasm_module_t wasm_module;
|
2023-02-22 20:19:06 +01:00
|
|
|
};
|
|
|
|
|
2023-02-23 15:55:49 +01:00
|
|
|
extern "C" {
|
|
|
|
static void debugprint(wasm_exec_env_t exec_env, char *the_string, uint8_t len) {
|
|
|
|
printf("%*s", len, the_string);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
static bool
|
|
|
|
init_wasm ()
|
|
|
|
{
|
|
|
|
static bool initialized;
|
|
|
|
if (initialized)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
RuntimeInitArgs init_args;
|
|
|
|
memset (&init_args, 0, sizeof (RuntimeInitArgs));
|
|
|
|
|
|
|
|
// Define an array of NativeSymbol for the APIs to be exported.
|
|
|
|
// Note: the array must be static defined since runtime
|
|
|
|
// will keep it after registration
|
|
|
|
// For the function signature specifications, goto the link:
|
|
|
|
// https://github.com/bytecodealliance/wasm-micro-runtime/blob/main/doc/export_native_api.md
|
|
|
|
|
|
|
|
static NativeSymbol native_symbols[] = {
|
2023-02-23 15:55:49 +01:00
|
|
|
{
|
|
|
|
"debugprint",
|
|
|
|
(void *)debugprint,
|
|
|
|
"($i)",
|
|
|
|
NULL
|
|
|
|
}
|
2023-02-23 07:30:28 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
init_args.mem_alloc_type = Alloc_With_Allocator;
|
|
|
|
init_args.mem_alloc_option.allocator.malloc_func = (void *) hb_malloc;
|
|
|
|
init_args.mem_alloc_option.allocator.realloc_func = (void *) hb_realloc;
|
|
|
|
init_args.mem_alloc_option.allocator.free_func = (void *) hb_free;
|
|
|
|
|
|
|
|
init_args.mem_alloc_type = Alloc_With_System_Allocator;
|
|
|
|
|
|
|
|
// Native symbols need below registration phase
|
|
|
|
init_args.n_native_symbols = sizeof(native_symbols) / sizeof(NativeSymbol);
|
|
|
|
init_args.native_module_name = "env";
|
|
|
|
init_args.native_symbols = native_symbols;
|
|
|
|
|
|
|
|
if (!wasm_runtime_full_init (&init_args))
|
|
|
|
{
|
|
|
|
DEBUG_MSG (WASM, nullptr, "Init runtime environment failed.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
initialized = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-02-22 20:19:06 +01:00
|
|
|
hb_wasm_face_data_t *
|
|
|
|
_hb_wasm_shaper_face_data_create (hb_face_t *face)
|
|
|
|
{
|
2023-02-22 20:57:19 +01:00
|
|
|
hb_wasm_face_data_t *data = nullptr;
|
2023-02-23 07:30:28 +01:00
|
|
|
hb_blob_t *wasm_blob = nullptr;
|
|
|
|
wasm_module_t wasm_module = nullptr;
|
2023-02-22 20:57:19 +01:00
|
|
|
|
|
|
|
wasm_blob = hb_face_reference_table (face, HB_WASM_TAG_WASM);
|
|
|
|
unsigned length = hb_blob_get_length (wasm_blob);
|
|
|
|
if (!length)
|
|
|
|
goto fail;
|
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
if (!init_wasm ())
|
2023-02-22 20:57:19 +01:00
|
|
|
goto fail;
|
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
wasm_module = wasm_runtime_load ((uint8_t *) hb_blob_get_data_writable (wasm_blob, nullptr),
|
|
|
|
length, nullptr, 0);
|
|
|
|
if (!wasm_module)
|
2023-02-22 21:33:34 +01:00
|
|
|
{
|
2023-02-23 07:30:28 +01:00
|
|
|
DEBUG_MSG (WASM, nullptr, "Load wasm module failed.");
|
2023-02-22 21:33:34 +01:00
|
|
|
goto fail;
|
2023-02-23 07:30:28 +01:00
|
|
|
}
|
|
|
|
|
2023-02-22 21:33:34 +01:00
|
|
|
|
2023-02-22 20:57:19 +01:00
|
|
|
data = (hb_wasm_face_data_t *) hb_calloc (1, sizeof (hb_wasm_face_data_t));
|
2023-02-22 20:19:06 +01:00
|
|
|
if (unlikely (!data))
|
2023-02-22 20:57:19 +01:00
|
|
|
goto fail;
|
2023-02-22 20:19:06 +01:00
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
data->wasm_blob = wasm_blob;
|
|
|
|
data->wasm_module = wasm_module;
|
2023-02-22 20:19:06 +01:00
|
|
|
|
|
|
|
return data;
|
2023-02-22 20:57:19 +01:00
|
|
|
|
|
|
|
fail:
|
|
|
|
if (wasm_module)
|
2023-02-23 07:30:28 +01:00
|
|
|
wasm_runtime_unload (wasm_module);
|
2023-02-22 20:57:19 +01:00
|
|
|
hb_blob_destroy (wasm_blob);
|
|
|
|
hb_free (data);
|
|
|
|
return nullptr;
|
2023-02-22 20:19:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_hb_wasm_shaper_face_data_destroy (hb_wasm_face_data_t *data)
|
|
|
|
{
|
2023-02-23 07:30:28 +01:00
|
|
|
wasm_runtime_unload (data->wasm_module);
|
|
|
|
hb_blob_destroy (data->wasm_blob);
|
2023-02-22 20:19:06 +01:00
|
|
|
hb_free (data);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* shaper font data
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct hb_wasm_font_data_t {};
|
|
|
|
|
|
|
|
hb_wasm_font_data_t *
|
|
|
|
_hb_wasm_shaper_font_data_create (hb_font_t *font HB_UNUSED)
|
|
|
|
{
|
|
|
|
return (hb_wasm_font_data_t *) HB_SHAPER_DATA_SUCCEEDED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_hb_wasm_shaper_font_data_destroy (hb_wasm_font_data_t *data HB_UNUSED)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* shaper
|
|
|
|
*/
|
|
|
|
|
|
|
|
hb_bool_t
|
|
|
|
_hb_wasm_shape (hb_shape_plan_t *shape_plan,
|
|
|
|
hb_font_t *font,
|
|
|
|
hb_buffer_t *buffer,
|
|
|
|
const hb_feature_t *features,
|
|
|
|
unsigned int num_features)
|
|
|
|
{
|
2023-02-22 21:33:34 +01:00
|
|
|
bool ret = true;
|
|
|
|
const hb_wasm_face_data_t *face_data = font->face->data.wasm;
|
2023-02-23 07:30:28 +01:00
|
|
|
constexpr uint32_t stack_size = 8092, heap_size = 8092;
|
|
|
|
|
|
|
|
wasm_module_inst_t module_inst = nullptr;
|
|
|
|
wasm_exec_env_t exec_env = nullptr;
|
|
|
|
wasm_function_inst_t shape_func = nullptr;
|
2023-02-22 21:33:34 +01:00
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
module_inst = wasm_runtime_instantiate(face_data->wasm_module,
|
|
|
|
stack_size, heap_size,
|
|
|
|
nullptr, 0);
|
|
|
|
if (!module_inst)
|
|
|
|
{
|
|
|
|
DEBUG_MSG (WASM, face_data->wasm_module, "Instantiate wasm module failed.");
|
2023-02-22 21:33:34 +01:00
|
|
|
goto fail;
|
2023-02-23 07:30:28 +01:00
|
|
|
}
|
2023-02-22 21:33:34 +01:00
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
exec_env = wasm_runtime_create_exec_env (module_inst, stack_size);
|
|
|
|
if (!exec_env) {
|
|
|
|
DEBUG_MSG (WASM, module_inst, "Create wasm execution environment failed.");
|
2023-02-22 21:33:34 +01:00
|
|
|
goto fail;
|
2023-02-23 07:30:28 +01:00
|
|
|
}
|
|
|
|
|
2023-02-23 16:16:49 +01:00
|
|
|
if (!(shape_func = wasm_runtime_lookup_function (module_inst, "shape", nullptr)))
|
|
|
|
{
|
2023-02-23 07:30:28 +01:00
|
|
|
DEBUG_MSG (WASM, module_inst, "Shape function not found.");
|
2023-02-23 16:16:49 +01:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t font_ref;
|
|
|
|
uint32_t buffer_ref;
|
|
|
|
// cmake -DWAMR_BUILD_REF_TYPES=1 for these to work
|
|
|
|
if (!wasm_externref_obj2ref (module_inst, font, &font_ref) ||
|
|
|
|
!wasm_externref_obj2ref (module_inst, buffer, &buffer_ref))
|
|
|
|
{
|
|
|
|
DEBUG_MSG (WASM, module_inst, "Failed to register objects.");
|
|
|
|
goto fail;
|
2023-02-23 07:30:28 +01:00
|
|
|
}
|
2023-02-22 21:33:34 +01:00
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
wasm_val_t results[1];
|
2023-02-23 16:16:49 +01:00
|
|
|
wasm_val_t arguments[2];
|
2023-02-23 07:30:28 +01:00
|
|
|
|
|
|
|
results[0].kind = WASM_I32;
|
|
|
|
arguments[0].kind = WASM_I32;
|
2023-02-23 16:16:49 +01:00
|
|
|
arguments[0].of.i32 = font_ref;
|
|
|
|
arguments[1].kind = WASM_I32;
|
|
|
|
arguments[1].of.i32 = buffer_ref;
|
2023-02-23 07:30:28 +01:00
|
|
|
|
2023-02-23 16:16:49 +01:00
|
|
|
if (!wasm_runtime_call_wasm_a (exec_env, shape_func,
|
|
|
|
ARRAY_LENGTH (results), results,
|
|
|
|
ARRAY_LENGTH (arguments), arguments))
|
2023-02-22 21:33:34 +01:00
|
|
|
{
|
2023-02-23 07:30:28 +01:00
|
|
|
DEBUG_MSG (WASM, module_inst, "Calling shape function failed: %s",
|
|
|
|
wasm_runtime_get_exception(module_inst));
|
2023-02-22 21:33:34 +01:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
printf ("%i\n", results[0].of.i32);
|
|
|
|
|
|
|
|
|
|
|
|
if (!face_data)
|
|
|
|
goto fail;
|
2023-02-22 21:33:34 +01:00
|
|
|
|
|
|
|
if (0)
|
|
|
|
{
|
|
|
|
fail:
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
2023-02-23 07:30:28 +01:00
|
|
|
if (exec_env)
|
|
|
|
wasm_runtime_destroy_exec_env (exec_env);
|
|
|
|
if (module_inst)
|
|
|
|
wasm_runtime_deinstantiate (module_inst);
|
2023-02-22 21:33:34 +01:00
|
|
|
|
|
|
|
return ret;
|
2023-02-22 20:19:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|