blob: 64217a041f4ec10ec805c770f5d6229ed3fe8860 [file] [log] [blame]
/*
* 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 */