[wasm] More boilerplate

This commit is contained in:
Behdad Esfahbod 2023-02-22 13:33:34 -07:00
parent fcc8be409b
commit db8e51e01b
1 changed files with 67 additions and 2 deletions

View File

@ -56,38 +56,59 @@ get_wasm_store ()
struct hb_wasm_face_data_t { struct hb_wasm_face_data_t {
hb_blob_t *blob; hb_blob_t *blob;
wasm_module_t *mod; wasm_module_t *mod;
unsigned shape_func_idx;
}; };
hb_wasm_face_data_t * hb_wasm_face_data_t *
_hb_wasm_shaper_face_data_create (hb_face_t *face) _hb_wasm_shaper_face_data_create (hb_face_t *face)
{ {
wasm_store_t *wasm_store = nullptr;
hb_blob_t *wasm_blob = nullptr; hb_blob_t *wasm_blob = nullptr;
own wasm_module_t *wasm_module = nullptr; own wasm_module_t *wasm_module = nullptr;
hb_wasm_face_data_t *data = nullptr; hb_wasm_face_data_t *data = nullptr;
wasm_exporttype_vec_t exports = {};
unsigned shape_func_idx = (unsigned) -1;
wasm_blob = hb_face_reference_table (face, HB_WASM_TAG_WASM); wasm_blob = hb_face_reference_table (face, HB_WASM_TAG_WASM);
unsigned length = hb_blob_get_length (wasm_blob); unsigned length = hb_blob_get_length (wasm_blob);
if (!length) if (!length)
goto fail; goto fail;
wasm_store = get_wasm_store (); // Do before others to initialize
wasm_byte_vec_t binary; wasm_byte_vec_t binary;
wasm_byte_vec_new_uninitialized (&binary, length); wasm_byte_vec_new_uninitialized (&binary, length);
memcpy (binary.data, hb_blob_get_data (wasm_blob, nullptr), length); memcpy (binary.data, hb_blob_get_data (wasm_blob, nullptr), length);
wasm_module = wasm_module_new (get_wasm_store (), &binary); wasm_module = wasm_module_new (wasm_store, &binary);
if (!wasm_module) if (!wasm_module)
goto fail; goto fail;
wasm_byte_vec_delete(&binary); wasm_byte_vec_delete(&binary);
wasm_module_exports (wasm_module, &exports);
for (unsigned i = 0; i < exports.size; i++)
{
auto *name = wasm_exporttype_name(exports.data[i]);
if (0 == memcmp ("shape", name->data, name->size))
{
shape_func_idx = i;
break;
}
}
if (shape_func_idx == (unsigned) -1)
goto fail;
data = (hb_wasm_face_data_t *) hb_calloc (1, sizeof (hb_wasm_face_data_t)); data = (hb_wasm_face_data_t *) hb_calloc (1, sizeof (hb_wasm_face_data_t));
if (unlikely (!data)) if (unlikely (!data))
goto fail; goto fail;
data->blob = wasm_blob; data->blob = wasm_blob;
data->mod = wasm_module; data->mod = wasm_module;
data->shape_func_idx = shape_func_idx;
return data; return data;
fail: fail:
wasm_exporttype_vec_delete (&exports);
if (wasm_module) if (wasm_module)
wasm_module_delete (wasm_module); wasm_module_delete (wasm_module);
hb_blob_destroy (wasm_blob); hb_blob_destroy (wasm_blob);
@ -133,7 +154,51 @@ _hb_wasm_shape (hb_shape_plan_t *shape_plan,
const hb_feature_t *features, const hb_feature_t *features,
unsigned int num_features) unsigned int num_features)
{ {
return false; bool ret = true;
const hb_wasm_face_data_t *face_data = font->face->data.wasm;
own wasm_instance_t *instance = nullptr;
const wasm_func_t *run_func = nullptr;
wasm_trap_t *trap = nullptr;
own wasm_extern_vec_t exports = {};
wasm_val_t as[1] = { WASM_I32_VAL(6) };
wasm_val_t rs[1] = { WASM_INIT_VAL };
wasm_val_vec_t args = WASM_ARRAY_VEC (as);
wasm_val_vec_t results = WASM_ARRAY_VEC (rs);
instance = wasm_instance_new (get_wasm_store (), face_data->mod, nullptr/*imports*/, nullptr);
if (!instance)
goto fail;
wasm_instance_exports (instance, &exports);
if (exports.size == 0)
goto fail;
run_func = wasm_extern_as_func (exports.data[face_data->shape_func_idx]);
if (!run_func)
goto fail;
trap = wasm_func_call (run_func, &args, &results);
if (trap)
{
goto fail;
}
ret = bool (rs[0].of.i32);
if (0)
{
fail:
ret = false;
}
wasm_extern_vec_delete (&exports);
if (instance)
wasm_instance_delete(instance);
if (trap)
wasm_trap_delete (trap);
return ret;
} }
#endif #endif