/* * Copyright © 2023 Behdad Esfahbod * * 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. */ #ifndef HB_WASM_API_BUFFER_HH #define HB_WASM_API_BUFFER_HH #include "hb-wasm-api.hh" #include "hb-buffer.hh" namespace hb { namespace wasm { static_assert (sizeof (glyph_info_t) == sizeof (hb_glyph_info_t), ""); static_assert (sizeof (glyph_position_t) == sizeof (hb_glyph_position_t), ""); HB_WASM_API (bool_t, buffer_contents_realloc) (HB_WASM_EXEC_ENV ptr_d(buffer_contents_t, contents), uint32_t size) { HB_PTR_PARAM (buffer_contents_t, contents); if (unlikely (!contents)) return false; if (size <= contents->length) return true; unsigned bytes; if (hb_unsigned_mul_overflows (size, sizeof (glyph_info_t), &bytes)) return false; glyph_info_t *info = HB_ARRAY_APP2NATIVE (glyph_info_t, contents->info, contents->length); glyph_position_t *pos = HB_ARRAY_APP2NATIVE (glyph_position_t, contents->pos, contents->length); if (unlikely (!info || !pos)) return false; glyph_info_t *new_info = nullptr; uint32_t new_inforef = module_malloc (bytes, (void **) &new_info); glyph_position_t *new_pos = nullptr; uint32_t new_posref = module_malloc (bytes, (void **) &new_pos); unsigned old_bytes = contents->length * sizeof (glyph_info_t); if (likely (new_inforef)) { hb_memcpy (new_info, info, old_bytes); module_free (contents->info); contents->info = new_inforef; } if (likely (new_posref)) { hb_memcpy (new_pos, pos, old_bytes); module_free (contents->pos); contents->pos = new_posref; } if (likely (new_info && new_pos)) { contents->length = size; return true; } return false; } HB_WASM_API (void, buffer_contents_free) (HB_WASM_EXEC_ENV ptr_d(buffer_contents_t, contents)) { HB_PTR_PARAM (buffer_contents_t, contents); if (unlikely (!contents)) return; module_free (contents->info); module_free (contents->pos); contents->info = nullref; contents->pos = nullref; contents->length = 0; } HB_WASM_API (bool_t, buffer_copy_contents) (HB_WASM_EXEC_ENV ptr_d(buffer_t, buffer), ptr_d(buffer_contents_t, contents)) { HB_REF2OBJ (buffer); HB_PTR_PARAM (buffer_contents_t, contents); if (unlikely (!contents)) return false; if (buffer->have_output) buffer->sync (); if (!buffer->have_positions) buffer->clear_positions (); unsigned length = buffer->len; if (length <= contents->length) { glyph_info_t *info = HB_ARRAY_APP2NATIVE (glyph_info_t, contents->info, length); glyph_position_t *pos = HB_ARRAY_APP2NATIVE (glyph_position_t, contents->pos, length); if (unlikely (!info || !pos)) { contents->length = 0; return false; } unsigned bytes = length * sizeof (hb_glyph_info_t); hb_memcpy (info, buffer->info, bytes); hb_memcpy (pos, buffer->pos, bytes); return true; } module_free (contents->info); module_free (contents->pos); contents->length = length; unsigned bytes = length * sizeof (hb_glyph_info_t); contents->info = wasm_runtime_module_dup_data (module_inst, (const char *) buffer->info, bytes); contents->pos = wasm_runtime_module_dup_data (module_inst, (const char *) buffer->pos, bytes); if (length && (!contents->info || !contents->pos)) { contents->length = 0; return false; } return true; } HB_WASM_API (bool_t, buffer_set_contents) (HB_WASM_EXEC_ENV ptr_d(buffer_t, buffer), ptr_d(const buffer_contents_t, contents)) { HB_REF2OBJ (buffer); HB_PTR_PARAM (buffer_contents_t, contents); if (unlikely (!contents)) return false; unsigned length = contents->length; unsigned bytes; if (unlikely (hb_unsigned_mul_overflows (length, sizeof (buffer->info[0]), &bytes))) return false; if (unlikely (!buffer->resize (length))) return false; glyph_info_t *info = (glyph_info_t *) (validate_app_addr (contents->info, bytes) ? addr_app_to_native (contents->info) : nullptr); glyph_position_t *pos = (glyph_position_t *) (validate_app_addr (contents->pos, bytes) ? addr_app_to_native (contents->pos) : nullptr); if (!buffer->have_positions) buffer->clear_positions (); /* This is wasteful. */ hb_memcpy (buffer->info, info, bytes); hb_memcpy (buffer->pos, pos, bytes); buffer->len = length; return true; } HB_WASM_API (direction_t, buffer_get_direction) (HB_WASM_EXEC_ENV ptr_d(buffer_t, buffer)) { HB_REF2OBJ (buffer); return (direction_t) hb_buffer_get_direction (buffer); } HB_WASM_API (script_t, buffer_get_script) (HB_WASM_EXEC_ENV ptr_d(buffer_t, buffer)) { HB_REF2OBJ (buffer); return hb_script_to_iso15924_tag (hb_buffer_get_script (buffer)); } HB_WASM_API (void, buffer_reverse) (HB_WASM_EXEC_ENV ptr_d(buffer_t, buffer)) { HB_REF2OBJ (buffer); hb_buffer_reverse (buffer); } HB_WASM_API (void, buffer_reverse_clusters) (HB_WASM_EXEC_ENV ptr_d(buffer_t, buffer)) { HB_REF2OBJ (buffer); hb_buffer_reverse_clusters (buffer); } }} #endif /* HB_WASM_API_BUFFER_HH */