1*2d1272b8SAndroid Build Coastguard Worker /*
2*2d1272b8SAndroid Build Coastguard Worker * Copyright © 2023 Behdad Esfahbod
3*2d1272b8SAndroid Build Coastguard Worker *
4*2d1272b8SAndroid Build Coastguard Worker * This is part of HarfBuzz, a text shaping library.
5*2d1272b8SAndroid Build Coastguard Worker *
6*2d1272b8SAndroid Build Coastguard Worker * Permission is hereby granted, without written agreement and without
7*2d1272b8SAndroid Build Coastguard Worker * license or royalty fees, to use, copy, modify, and distribute this
8*2d1272b8SAndroid Build Coastguard Worker * software and its documentation for any purpose, provided that the
9*2d1272b8SAndroid Build Coastguard Worker * above copyright notice and the following two paragraphs appear in
10*2d1272b8SAndroid Build Coastguard Worker * all copies of this software.
11*2d1272b8SAndroid Build Coastguard Worker *
12*2d1272b8SAndroid Build Coastguard Worker * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
13*2d1272b8SAndroid Build Coastguard Worker * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
14*2d1272b8SAndroid Build Coastguard Worker * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
15*2d1272b8SAndroid Build Coastguard Worker * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
16*2d1272b8SAndroid Build Coastguard Worker * DAMAGE.
17*2d1272b8SAndroid Build Coastguard Worker *
18*2d1272b8SAndroid Build Coastguard Worker * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
19*2d1272b8SAndroid Build Coastguard Worker * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
20*2d1272b8SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
21*2d1272b8SAndroid Build Coastguard Worker * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
22*2d1272b8SAndroid Build Coastguard Worker * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
23*2d1272b8SAndroid Build Coastguard Worker */
24*2d1272b8SAndroid Build Coastguard Worker
25*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_WASM_API_BUFFER_HH
26*2d1272b8SAndroid Build Coastguard Worker #define HB_WASM_API_BUFFER_HH
27*2d1272b8SAndroid Build Coastguard Worker
28*2d1272b8SAndroid Build Coastguard Worker #include "hb-wasm-api.hh"
29*2d1272b8SAndroid Build Coastguard Worker
30*2d1272b8SAndroid Build Coastguard Worker #include "hb-buffer.hh"
31*2d1272b8SAndroid Build Coastguard Worker
32*2d1272b8SAndroid Build Coastguard Worker namespace hb {
33*2d1272b8SAndroid Build Coastguard Worker namespace wasm {
34*2d1272b8SAndroid Build Coastguard Worker
35*2d1272b8SAndroid Build Coastguard Worker static_assert (sizeof (glyph_info_t) == sizeof (hb_glyph_info_t), "");
36*2d1272b8SAndroid Build Coastguard Worker static_assert (sizeof (glyph_position_t) == sizeof (hb_glyph_position_t), "");
37*2d1272b8SAndroid Build Coastguard Worker
HB_WASM_API(bool_t,buffer_contents_realloc)38*2d1272b8SAndroid Build Coastguard Worker HB_WASM_API (bool_t, buffer_contents_realloc) (HB_WASM_EXEC_ENV
39*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_contents_t, contents),
40*2d1272b8SAndroid Build Coastguard Worker uint32_t size)
41*2d1272b8SAndroid Build Coastguard Worker {
42*2d1272b8SAndroid Build Coastguard Worker HB_PTR_PARAM (buffer_contents_t, contents);
43*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!contents))
44*2d1272b8SAndroid Build Coastguard Worker return false;
45*2d1272b8SAndroid Build Coastguard Worker
46*2d1272b8SAndroid Build Coastguard Worker if (size <= contents->length)
47*2d1272b8SAndroid Build Coastguard Worker return true;
48*2d1272b8SAndroid Build Coastguard Worker
49*2d1272b8SAndroid Build Coastguard Worker unsigned bytes;
50*2d1272b8SAndroid Build Coastguard Worker if (hb_unsigned_mul_overflows (size, sizeof (glyph_info_t), &bytes))
51*2d1272b8SAndroid Build Coastguard Worker return false;
52*2d1272b8SAndroid Build Coastguard Worker
53*2d1272b8SAndroid Build Coastguard Worker glyph_info_t *info = HB_ARRAY_APP2NATIVE (glyph_info_t, contents->info, contents->length);
54*2d1272b8SAndroid Build Coastguard Worker glyph_position_t *pos = HB_ARRAY_APP2NATIVE (glyph_position_t, contents->pos, contents->length);
55*2d1272b8SAndroid Build Coastguard Worker
56*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!info || !pos))
57*2d1272b8SAndroid Build Coastguard Worker return false;
58*2d1272b8SAndroid Build Coastguard Worker
59*2d1272b8SAndroid Build Coastguard Worker glyph_info_t *new_info = nullptr;
60*2d1272b8SAndroid Build Coastguard Worker uint32_t new_inforef = module_malloc (bytes, (void **) &new_info);
61*2d1272b8SAndroid Build Coastguard Worker glyph_position_t *new_pos = nullptr;
62*2d1272b8SAndroid Build Coastguard Worker uint32_t new_posref = module_malloc (bytes, (void **) &new_pos);
63*2d1272b8SAndroid Build Coastguard Worker
64*2d1272b8SAndroid Build Coastguard Worker unsigned old_bytes = contents->length * sizeof (glyph_info_t);
65*2d1272b8SAndroid Build Coastguard Worker if (likely (new_inforef))
66*2d1272b8SAndroid Build Coastguard Worker {
67*2d1272b8SAndroid Build Coastguard Worker hb_memcpy (new_info, info, old_bytes);
68*2d1272b8SAndroid Build Coastguard Worker module_free (contents->info);
69*2d1272b8SAndroid Build Coastguard Worker contents->info = new_inforef;
70*2d1272b8SAndroid Build Coastguard Worker }
71*2d1272b8SAndroid Build Coastguard Worker if (likely (new_posref))
72*2d1272b8SAndroid Build Coastguard Worker {
73*2d1272b8SAndroid Build Coastguard Worker hb_memcpy (new_pos, pos, old_bytes);
74*2d1272b8SAndroid Build Coastguard Worker module_free (contents->pos);
75*2d1272b8SAndroid Build Coastguard Worker contents->pos = new_posref;
76*2d1272b8SAndroid Build Coastguard Worker }
77*2d1272b8SAndroid Build Coastguard Worker
78*2d1272b8SAndroid Build Coastguard Worker if (likely (new_info && new_pos))
79*2d1272b8SAndroid Build Coastguard Worker {
80*2d1272b8SAndroid Build Coastguard Worker contents->length = size;
81*2d1272b8SAndroid Build Coastguard Worker return true;
82*2d1272b8SAndroid Build Coastguard Worker }
83*2d1272b8SAndroid Build Coastguard Worker
84*2d1272b8SAndroid Build Coastguard Worker return false;
85*2d1272b8SAndroid Build Coastguard Worker }
86*2d1272b8SAndroid Build Coastguard Worker
HB_WASM_API(void,buffer_contents_free)87*2d1272b8SAndroid Build Coastguard Worker HB_WASM_API (void, buffer_contents_free) (HB_WASM_EXEC_ENV
88*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_contents_t, contents))
89*2d1272b8SAndroid Build Coastguard Worker {
90*2d1272b8SAndroid Build Coastguard Worker HB_PTR_PARAM (buffer_contents_t, contents);
91*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!contents))
92*2d1272b8SAndroid Build Coastguard Worker return;
93*2d1272b8SAndroid Build Coastguard Worker
94*2d1272b8SAndroid Build Coastguard Worker module_free (contents->info);
95*2d1272b8SAndroid Build Coastguard Worker module_free (contents->pos);
96*2d1272b8SAndroid Build Coastguard Worker
97*2d1272b8SAndroid Build Coastguard Worker contents->info = nullref;
98*2d1272b8SAndroid Build Coastguard Worker contents->pos = nullref;
99*2d1272b8SAndroid Build Coastguard Worker contents->length = 0;
100*2d1272b8SAndroid Build Coastguard Worker }
101*2d1272b8SAndroid Build Coastguard Worker
HB_WASM_API(bool_t,buffer_copy_contents)102*2d1272b8SAndroid Build Coastguard Worker HB_WASM_API (bool_t, buffer_copy_contents) (HB_WASM_EXEC_ENV
103*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_t, buffer),
104*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_contents_t, contents))
105*2d1272b8SAndroid Build Coastguard Worker {
106*2d1272b8SAndroid Build Coastguard Worker HB_REF2OBJ (buffer);
107*2d1272b8SAndroid Build Coastguard Worker HB_PTR_PARAM (buffer_contents_t, contents);
108*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!contents))
109*2d1272b8SAndroid Build Coastguard Worker return false;
110*2d1272b8SAndroid Build Coastguard Worker
111*2d1272b8SAndroid Build Coastguard Worker if (buffer->have_output)
112*2d1272b8SAndroid Build Coastguard Worker buffer->sync ();
113*2d1272b8SAndroid Build Coastguard Worker if (!buffer->have_positions)
114*2d1272b8SAndroid Build Coastguard Worker buffer->clear_positions ();
115*2d1272b8SAndroid Build Coastguard Worker
116*2d1272b8SAndroid Build Coastguard Worker unsigned length = buffer->len;
117*2d1272b8SAndroid Build Coastguard Worker
118*2d1272b8SAndroid Build Coastguard Worker if (length <= contents->length)
119*2d1272b8SAndroid Build Coastguard Worker {
120*2d1272b8SAndroid Build Coastguard Worker glyph_info_t *info = HB_ARRAY_APP2NATIVE (glyph_info_t, contents->info, length);
121*2d1272b8SAndroid Build Coastguard Worker glyph_position_t *pos = HB_ARRAY_APP2NATIVE (glyph_position_t, contents->pos, length);
122*2d1272b8SAndroid Build Coastguard Worker
123*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!info || !pos))
124*2d1272b8SAndroid Build Coastguard Worker {
125*2d1272b8SAndroid Build Coastguard Worker contents->length = 0;
126*2d1272b8SAndroid Build Coastguard Worker return false;
127*2d1272b8SAndroid Build Coastguard Worker }
128*2d1272b8SAndroid Build Coastguard Worker
129*2d1272b8SAndroid Build Coastguard Worker unsigned bytes = length * sizeof (hb_glyph_info_t);
130*2d1272b8SAndroid Build Coastguard Worker hb_memcpy (info, buffer->info, bytes);
131*2d1272b8SAndroid Build Coastguard Worker hb_memcpy (pos, buffer->pos, bytes);
132*2d1272b8SAndroid Build Coastguard Worker
133*2d1272b8SAndroid Build Coastguard Worker return true;
134*2d1272b8SAndroid Build Coastguard Worker }
135*2d1272b8SAndroid Build Coastguard Worker
136*2d1272b8SAndroid Build Coastguard Worker module_free (contents->info);
137*2d1272b8SAndroid Build Coastguard Worker module_free (contents->pos);
138*2d1272b8SAndroid Build Coastguard Worker
139*2d1272b8SAndroid Build Coastguard Worker contents->length = length;
140*2d1272b8SAndroid Build Coastguard Worker unsigned bytes = length * sizeof (hb_glyph_info_t);
141*2d1272b8SAndroid Build Coastguard Worker contents->info = wasm_runtime_module_dup_data (module_inst, (const char *) buffer->info, bytes);
142*2d1272b8SAndroid Build Coastguard Worker contents->pos = wasm_runtime_module_dup_data (module_inst, (const char *) buffer->pos, bytes);
143*2d1272b8SAndroid Build Coastguard Worker
144*2d1272b8SAndroid Build Coastguard Worker if (length && (!contents->info || !contents->pos))
145*2d1272b8SAndroid Build Coastguard Worker {
146*2d1272b8SAndroid Build Coastguard Worker contents->length = 0;
147*2d1272b8SAndroid Build Coastguard Worker return false;
148*2d1272b8SAndroid Build Coastguard Worker }
149*2d1272b8SAndroid Build Coastguard Worker
150*2d1272b8SAndroid Build Coastguard Worker return true;
151*2d1272b8SAndroid Build Coastguard Worker }
152*2d1272b8SAndroid Build Coastguard Worker
HB_WASM_API(bool_t,buffer_set_contents)153*2d1272b8SAndroid Build Coastguard Worker HB_WASM_API (bool_t, buffer_set_contents) (HB_WASM_EXEC_ENV
154*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_t, buffer),
155*2d1272b8SAndroid Build Coastguard Worker ptr_d(const buffer_contents_t, contents))
156*2d1272b8SAndroid Build Coastguard Worker {
157*2d1272b8SAndroid Build Coastguard Worker HB_REF2OBJ (buffer);
158*2d1272b8SAndroid Build Coastguard Worker HB_PTR_PARAM (buffer_contents_t, contents);
159*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!contents))
160*2d1272b8SAndroid Build Coastguard Worker return false;
161*2d1272b8SAndroid Build Coastguard Worker
162*2d1272b8SAndroid Build Coastguard Worker unsigned length = contents->length;
163*2d1272b8SAndroid Build Coastguard Worker unsigned bytes;
164*2d1272b8SAndroid Build Coastguard Worker if (unlikely (hb_unsigned_mul_overflows (length, sizeof (buffer->info[0]), &bytes)))
165*2d1272b8SAndroid Build Coastguard Worker return false;
166*2d1272b8SAndroid Build Coastguard Worker
167*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!buffer->resize (length)))
168*2d1272b8SAndroid Build Coastguard Worker return false;
169*2d1272b8SAndroid Build Coastguard Worker
170*2d1272b8SAndroid Build Coastguard Worker glyph_info_t *info = (glyph_info_t *) (validate_app_addr (contents->info, bytes) ? addr_app_to_native (contents->info) : nullptr);
171*2d1272b8SAndroid Build Coastguard Worker glyph_position_t *pos = (glyph_position_t *) (validate_app_addr (contents->pos, bytes) ? addr_app_to_native (contents->pos) : nullptr);
172*2d1272b8SAndroid Build Coastguard Worker
173*2d1272b8SAndroid Build Coastguard Worker if (!buffer->have_positions)
174*2d1272b8SAndroid Build Coastguard Worker buffer->clear_positions (); /* This is wasteful. */
175*2d1272b8SAndroid Build Coastguard Worker
176*2d1272b8SAndroid Build Coastguard Worker hb_memcpy (buffer->info, info, bytes);
177*2d1272b8SAndroid Build Coastguard Worker hb_memcpy (buffer->pos, pos, bytes);
178*2d1272b8SAndroid Build Coastguard Worker buffer->len = length;
179*2d1272b8SAndroid Build Coastguard Worker
180*2d1272b8SAndroid Build Coastguard Worker return true;
181*2d1272b8SAndroid Build Coastguard Worker }
182*2d1272b8SAndroid Build Coastguard Worker
HB_WASM_API(direction_t,buffer_get_direction)183*2d1272b8SAndroid Build Coastguard Worker HB_WASM_API (direction_t, buffer_get_direction) (HB_WASM_EXEC_ENV
184*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_t, buffer))
185*2d1272b8SAndroid Build Coastguard Worker {
186*2d1272b8SAndroid Build Coastguard Worker HB_REF2OBJ (buffer);
187*2d1272b8SAndroid Build Coastguard Worker
188*2d1272b8SAndroid Build Coastguard Worker return (direction_t) hb_buffer_get_direction (buffer);
189*2d1272b8SAndroid Build Coastguard Worker }
190*2d1272b8SAndroid Build Coastguard Worker
HB_WASM_API(script_t,buffer_get_script)191*2d1272b8SAndroid Build Coastguard Worker HB_WASM_API (script_t, buffer_get_script) (HB_WASM_EXEC_ENV
192*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_t, buffer))
193*2d1272b8SAndroid Build Coastguard Worker {
194*2d1272b8SAndroid Build Coastguard Worker HB_REF2OBJ (buffer);
195*2d1272b8SAndroid Build Coastguard Worker
196*2d1272b8SAndroid Build Coastguard Worker return hb_script_to_iso15924_tag (hb_buffer_get_script (buffer));
197*2d1272b8SAndroid Build Coastguard Worker }
198*2d1272b8SAndroid Build Coastguard Worker
HB_WASM_API(void,buffer_reverse)199*2d1272b8SAndroid Build Coastguard Worker HB_WASM_API (void, buffer_reverse) (HB_WASM_EXEC_ENV
200*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_t, buffer))
201*2d1272b8SAndroid Build Coastguard Worker {
202*2d1272b8SAndroid Build Coastguard Worker HB_REF2OBJ (buffer);
203*2d1272b8SAndroid Build Coastguard Worker
204*2d1272b8SAndroid Build Coastguard Worker hb_buffer_reverse (buffer);
205*2d1272b8SAndroid Build Coastguard Worker }
206*2d1272b8SAndroid Build Coastguard Worker
HB_WASM_API(void,buffer_reverse_clusters)207*2d1272b8SAndroid Build Coastguard Worker HB_WASM_API (void, buffer_reverse_clusters) (HB_WASM_EXEC_ENV
208*2d1272b8SAndroid Build Coastguard Worker ptr_d(buffer_t, buffer))
209*2d1272b8SAndroid Build Coastguard Worker {
210*2d1272b8SAndroid Build Coastguard Worker HB_REF2OBJ (buffer);
211*2d1272b8SAndroid Build Coastguard Worker
212*2d1272b8SAndroid Build Coastguard Worker hb_buffer_reverse_clusters (buffer);
213*2d1272b8SAndroid Build Coastguard Worker }
214*2d1272b8SAndroid Build Coastguard Worker
215*2d1272b8SAndroid Build Coastguard Worker }}
216*2d1272b8SAndroid Build Coastguard Worker
217*2d1272b8SAndroid Build Coastguard Worker #endif /* HB_WASM_API_BUFFER_HH */
218