xref: /aosp_15_r20/external/harfbuzz_ng/src/hb-wasm-api-buffer.hh (revision 2d1272b857b1f7575e6e246373e1cb218663db8a)
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