[wasm-shape] Cache wasm-shape-plan

This commit is contained in:
Behdad Esfahbod 2023-02-24 17:56:14 -07:00
parent 73de7d4d05
commit cb382e489d
2 changed files with 140 additions and 65 deletions

View File

@ -94,6 +94,9 @@ HB_WASM_API_COMPOUND (buffer_contents_t, buffer_copy_contents) (HB_WASM_EXEC_ENV
ret.length = length; ret.length = length;
ret.info = wasm_runtime_module_dup_data (module_inst, (const char *) buffer->info, length * sizeof (buffer->info[0])); ret.info = wasm_runtime_module_dup_data (module_inst, (const char *) buffer->info, length * sizeof (buffer->info[0]));
ret.pos = wasm_runtime_module_dup_data (module_inst, (const char *) buffer->pos, length * sizeof (buffer->pos[0])); ret.pos = wasm_runtime_module_dup_data (module_inst, (const char *) buffer->pos, length * sizeof (buffer->pos[0]));
if (unlikely (buffer->successful && (!ret.info || !ret.pos)))
buffer->successful = false;
} }
HB_WASM_API (bool_t, buffer_set_contents) (HB_WASM_EXEC_ENV HB_WASM_API (bool_t, buffer_set_contents) (HB_WASM_EXEC_ENV

View File

@ -40,9 +40,16 @@
#define HB_WASM_TAG_WASM HB_TAG('W','a','s','m') #define HB_WASM_TAG_WASM HB_TAG('W','a','s','m')
struct hb_wasm_shape_plan_t {
wasm_module_inst_t module_inst;
wasm_exec_env_t exec_env;
ptr_d(void, wasm_shape_plan);
};
struct hb_wasm_face_data_t { struct hb_wasm_face_data_t {
hb_blob_t *wasm_blob; hb_blob_t *wasm_blob;
wasm_module_t wasm_module; wasm_module_t wasm_module;
mutable hb_atomic_ptr_t<hb_wasm_shape_plan_t> plan;
}; };
static bool static bool
@ -122,9 +129,125 @@ fail:
return nullptr; return nullptr;
} }
static hb_wasm_shape_plan_t *
acquire_shape_plan (hb_face_t *face,
const hb_wasm_face_data_t *face_data)
{
constexpr uint32_t stack_size = 32 * 1024, heap_size = 2 * 1024 * 1024;
wasm_module_inst_t module_inst = nullptr;
wasm_exec_env_t exec_env = nullptr;
wasm_function_inst_t func = nullptr;
/* Fetch cached one if available. */
hb_wasm_shape_plan_t *plan = face_data->plan.get_acquire ();
if (likely (plan && face_data->plan.cmpexch (plan, nullptr)))
return plan;
plan = (hb_wasm_shape_plan_t *) hb_calloc (1, sizeof (hb_wasm_shape_plan_t));
module_inst = plan->module_inst = wasm_runtime_instantiate(face_data->wasm_module,
stack_size, heap_size,
nullptr, 0);
if (unlikely (!module_inst))
{
DEBUG_MSG (WASM, face_data, "Create wasm module instance failed.");
goto fail;
}
exec_env = plan->exec_env = wasm_runtime_create_exec_env (module_inst,
stack_size);
if (unlikely (!exec_env)) {
DEBUG_MSG (WASM, face_data, "Create wasm execution environment failed.");
goto fail;
}
func = wasm_runtime_lookup_function (module_inst, "shape_plan_create", nullptr);
if (func)
{
wasm_val_t results[1];
wasm_val_t arguments[1];
HB_OBJ2REF (face);
if (unlikely (!faceref))
{
DEBUG_MSG (WASM, face_data, "Failed to register face object.");
goto fail;
}
results[0].kind = WASM_I32;
arguments[0].kind = WASM_I32;
arguments[0].of.i32 = faceref;
bool ret = wasm_runtime_call_wasm_a (exec_env, func,
ARRAY_LENGTH (results), results,
ARRAY_LENGTH (arguments), arguments);
if (unlikely (!ret))
{
DEBUG_MSG (WASM, module_inst, "Calling shape_plan_create() failed: %s",
wasm_runtime_get_exception(module_inst));
goto fail;
}
plan->wasm_shape_planptr = results[0].of.i32;
}
return plan;
fail:
if (exec_env)
wasm_runtime_destroy_exec_env (exec_env);
if (module_inst)
wasm_runtime_deinstantiate (module_inst);
hb_free (plan);
return nullptr;
}
static void
release_shape_plan (const hb_wasm_face_data_t *face_data,
hb_wasm_shape_plan_t *plan,
bool cache = false)
{
if (cache && face_data->plan.cmpexch (nullptr, plan))
return;
auto *module_inst = plan->module_inst;
auto *exec_env = plan->exec_env;
/* Is there even any point to having a shape_plan_destroy function
* and calling it? */
if (plan->wasm_shape_planptr)
{
auto *func = wasm_runtime_lookup_function (module_inst, "shape_plan_destroy", nullptr);
if (func)
{
wasm_val_t arguments[1];
arguments[0].kind = WASM_I32;
arguments[0].of.i32 = plan->wasm_shape_planptr;
bool ret = wasm_runtime_call_wasm_a (exec_env, func,
0, nullptr,
ARRAY_LENGTH (arguments), arguments);
if (unlikely (!ret))
{
DEBUG_MSG (WASM, module_inst, "Calling shape_plan_destroy() failed: %s",
wasm_runtime_get_exception(module_inst));
}
}
}
wasm_runtime_destroy_exec_env (exec_env);
wasm_runtime_deinstantiate (module_inst);
hb_free (plan);
}
void void
_hb_wasm_shaper_face_data_destroy (hb_wasm_face_data_t *data) _hb_wasm_shaper_face_data_destroy (hb_wasm_face_data_t *data)
{ {
if (data->plan.get_relaxed ())
release_shape_plan (data, data->plan);
wasm_runtime_unload (data->wasm_module); wasm_runtime_unload (data->wasm_module);
hb_blob_destroy (data->wasm_blob); hb_blob_destroy (data->wasm_blob);
hb_free (data); hb_free (data);
@ -163,61 +286,28 @@ _hb_wasm_shape (hb_shape_plan_t *shape_plan,
bool ret = true; bool ret = true;
hb_face_t *face = font->face; hb_face_t *face = font->face;
const hb_wasm_face_data_t *face_data = face->data.wasm; const hb_wasm_face_data_t *face_data = face->data.wasm;
constexpr uint32_t stack_size = 32 * 1024, heap_size = 2 * 1024 * 1024;
wasm_module_inst_t module_inst = nullptr;
wasm_exec_env_t exec_env = nullptr;
wasm_function_inst_t func = nullptr; wasm_function_inst_t func = nullptr;
uint32_t shape_planref = nullref;
module_inst = wasm_runtime_instantiate(face_data->wasm_module, hb_wasm_shape_plan_t *plan = acquire_shape_plan (face, face_data);
stack_size, heap_size, if (unlikely (!plan))
nullptr, 0);
if (unlikely (!module_inst))
{ {
DEBUG_MSG (WASM, face_data->wasm_module, "Instantiate wasm module failed."); DEBUG_MSG (WASM, face_data, "Acquiring shape-plan failed.");
return false; return false;
} }
auto *module_inst = plan->module_inst;
auto *exec_env = plan->exec_env;
// cmake -DWAMR_BUILD_REF_TYPES=1 for these to work // cmake -DWAMR_BUILD_REF_TYPES=1 for these to work
HB_OBJ2REF (face);
HB_OBJ2REF (font); HB_OBJ2REF (font);
HB_OBJ2REF (buffer); HB_OBJ2REF (buffer);
if (unlikely (!faceref || !fontref || !bufferref)) if (unlikely (!fontref || !bufferref))
{ {
DEBUG_MSG (WASM, module_inst, "Failed to register objects."); DEBUG_MSG (WASM, module_inst, "Failed to register objects.");
goto fail; goto fail;
} }
exec_env = wasm_runtime_create_exec_env (module_inst, stack_size);
if (unlikely (!exec_env)) {
DEBUG_MSG (WASM, module_inst, "Create wasm execution environment failed.");
goto fail;
}
func = wasm_runtime_lookup_function (module_inst, "shape_plan_create", nullptr);
if (func)
{
wasm_val_t results[1];
wasm_val_t arguments[1];
results[0].kind = WASM_I32;
arguments[0].kind = WASM_I32;
arguments[0].of.i32 = faceref;
ret = wasm_runtime_call_wasm_a (exec_env, func,
ARRAY_LENGTH (results), results,
ARRAY_LENGTH (arguments), arguments);
if (unlikely (!ret))
{
DEBUG_MSG (WASM, module_inst, "Calling shape_plan_create() failed: %s",
wasm_runtime_get_exception(module_inst));
goto fail;
}
shape_planref = results[0].of.i32;
}
func = wasm_runtime_lookup_function (module_inst, "shape", nullptr); func = wasm_runtime_lookup_function (module_inst, "shape", nullptr);
if (unlikely (!func)) if (unlikely (!func))
{ {
@ -230,7 +320,7 @@ _hb_wasm_shape (hb_shape_plan_t *shape_plan,
results[0].kind = WASM_I32; results[0].kind = WASM_I32;
arguments[0].kind = WASM_I32; arguments[0].kind = WASM_I32;
arguments[0].of.i32 = shape_planref; arguments[0].of.i32 = plan->wasm_shape_planptr;
arguments[1].kind = WASM_I32; arguments[1].kind = WASM_I32;
arguments[1].of.i32 = fontref; arguments[1].of.i32 = fontref;
arguments[2].kind = WASM_I32; arguments[2].kind = WASM_I32;
@ -248,6 +338,12 @@ _hb_wasm_shape (hb_shape_plan_t *shape_plan,
wasm_runtime_module_free (module_inst, arguments[2].of.i32); wasm_runtime_module_free (module_inst, arguments[2].of.i32);
if (unlikely (buffer->in_error ()))
{
DEBUG_MSG (WASM, module_inst, "Buffer in error. Memory allocation fail?");
goto fail;
}
if (unlikely (!ret)) if (unlikely (!ret))
{ {
DEBUG_MSG (WASM, module_inst, "Calling shape() failed: %s", DEBUG_MSG (WASM, module_inst, "Calling shape() failed: %s",
@ -264,31 +360,7 @@ fail:
ret = false; ret = false;
} }
if (ret && shape_planref) release_shape_plan (face_data, plan, ret);
{
func = wasm_runtime_lookup_function (module_inst, "shape_plan_destroy", nullptr);
if (func)
{
wasm_val_t arguments[1];
arguments[0].kind = WASM_I32;
arguments[0].of.i32 = shape_planref;
ret = wasm_runtime_call_wasm_a (exec_env, func,
0, nullptr,
ARRAY_LENGTH (arguments), arguments);
if (unlikely (!ret))
{
DEBUG_MSG (WASM, module_inst, "Calling shape_plan_destroy() failed: %s",
wasm_runtime_get_exception(module_inst));
}
}
}
if (exec_env)
wasm_runtime_destroy_exec_env (exec_env);
if (module_inst)
wasm_runtime_deinstantiate (module_inst);
return ret; return ret;
} }