xref: /aosp_15_r20/external/harfbuzz_ng/src/hb-bimap.hh (revision 2d1272b857b1f7575e6e246373e1cb218663db8a)
1*2d1272b8SAndroid Build Coastguard Worker /*
2*2d1272b8SAndroid Build Coastguard Worker  * Copyright © 2019 Adobe Inc.
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  * Adobe Author(s): Michiharu Ariza
25*2d1272b8SAndroid Build Coastguard Worker  */
26*2d1272b8SAndroid Build Coastguard Worker 
27*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_BIMAP_HH
28*2d1272b8SAndroid Build Coastguard Worker #define HB_BIMAP_HH
29*2d1272b8SAndroid Build Coastguard Worker 
30*2d1272b8SAndroid Build Coastguard Worker #include "hb.hh"
31*2d1272b8SAndroid Build Coastguard Worker #include "hb-map.hh"
32*2d1272b8SAndroid Build Coastguard Worker 
33*2d1272b8SAndroid Build Coastguard Worker /* Bi-directional map */
34*2d1272b8SAndroid Build Coastguard Worker struct hb_bimap_t
35*2d1272b8SAndroid Build Coastguard Worker {
resethb_bimap_t36*2d1272b8SAndroid Build Coastguard Worker   void reset ()
37*2d1272b8SAndroid Build Coastguard Worker   {
38*2d1272b8SAndroid Build Coastguard Worker     forw_map.reset ();
39*2d1272b8SAndroid Build Coastguard Worker     back_map.reset ();
40*2d1272b8SAndroid Build Coastguard Worker   }
41*2d1272b8SAndroid Build Coastguard Worker 
allochb_bimap_t42*2d1272b8SAndroid Build Coastguard Worker   void alloc (unsigned pop)
43*2d1272b8SAndroid Build Coastguard Worker   {
44*2d1272b8SAndroid Build Coastguard Worker     forw_map.alloc (pop);
45*2d1272b8SAndroid Build Coastguard Worker     back_map.alloc (pop);
46*2d1272b8SAndroid Build Coastguard Worker   }
47*2d1272b8SAndroid Build Coastguard Worker 
in_errorhb_bimap_t48*2d1272b8SAndroid Build Coastguard Worker   bool in_error () const { return forw_map.in_error () || back_map.in_error (); }
49*2d1272b8SAndroid Build Coastguard Worker 
sethb_bimap_t50*2d1272b8SAndroid Build Coastguard Worker   void set (hb_codepoint_t lhs, hb_codepoint_t rhs)
51*2d1272b8SAndroid Build Coastguard Worker   {
52*2d1272b8SAndroid Build Coastguard Worker     if (in_error ()) return;
53*2d1272b8SAndroid Build Coastguard Worker     if (unlikely (lhs == HB_MAP_VALUE_INVALID)) return;
54*2d1272b8SAndroid Build Coastguard Worker     if (unlikely (rhs == HB_MAP_VALUE_INVALID)) { del (lhs); return; }
55*2d1272b8SAndroid Build Coastguard Worker 
56*2d1272b8SAndroid Build Coastguard Worker     forw_map.set (lhs, rhs);
57*2d1272b8SAndroid Build Coastguard Worker     if (unlikely (in_error ())) return;
58*2d1272b8SAndroid Build Coastguard Worker 
59*2d1272b8SAndroid Build Coastguard Worker     back_map.set (rhs, lhs);
60*2d1272b8SAndroid Build Coastguard Worker     if (unlikely (in_error ())) forw_map.del (lhs);
61*2d1272b8SAndroid Build Coastguard Worker   }
62*2d1272b8SAndroid Build Coastguard Worker 
gethb_bimap_t63*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t get (hb_codepoint_t lhs) const { return forw_map.get (lhs); }
backwardhb_bimap_t64*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t backward (hb_codepoint_t rhs) const { return back_map.get (rhs); }
65*2d1272b8SAndroid Build Coastguard Worker 
operator []hb_bimap_t66*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t operator [] (hb_codepoint_t lhs) const { return get (lhs); }
hashb_bimap_t67*2d1272b8SAndroid Build Coastguard Worker   bool has (hb_codepoint_t lhs) const { return forw_map.has (lhs); }
68*2d1272b8SAndroid Build Coastguard Worker 
69*2d1272b8SAndroid Build Coastguard Worker 
delhb_bimap_t70*2d1272b8SAndroid Build Coastguard Worker   void del (hb_codepoint_t lhs)
71*2d1272b8SAndroid Build Coastguard Worker   {
72*2d1272b8SAndroid Build Coastguard Worker     back_map.del (get (lhs));
73*2d1272b8SAndroid Build Coastguard Worker     forw_map.del (lhs);
74*2d1272b8SAndroid Build Coastguard Worker   }
75*2d1272b8SAndroid Build Coastguard Worker 
clearhb_bimap_t76*2d1272b8SAndroid Build Coastguard Worker   void clear ()
77*2d1272b8SAndroid Build Coastguard Worker   {
78*2d1272b8SAndroid Build Coastguard Worker     forw_map.clear ();
79*2d1272b8SAndroid Build Coastguard Worker     back_map.clear ();
80*2d1272b8SAndroid Build Coastguard Worker   }
81*2d1272b8SAndroid Build Coastguard Worker 
is_emptyhb_bimap_t82*2d1272b8SAndroid Build Coastguard Worker   bool is_empty () const { return forw_map.is_empty (); }
83*2d1272b8SAndroid Build Coastguard Worker 
get_populationhb_bimap_t84*2d1272b8SAndroid Build Coastguard Worker   unsigned int get_population () const { return forw_map.get_population (); }
85*2d1272b8SAndroid Build Coastguard Worker 
86*2d1272b8SAndroid Build Coastguard Worker   protected:
87*2d1272b8SAndroid Build Coastguard Worker   hb_map_t  forw_map;
88*2d1272b8SAndroid Build Coastguard Worker   hb_map_t  back_map;
89*2d1272b8SAndroid Build Coastguard Worker 
90*2d1272b8SAndroid Build Coastguard Worker   public:
91*2d1272b8SAndroid Build Coastguard Worker   auto keys () const HB_AUTO_RETURN (+ forw_map.keys())
92*2d1272b8SAndroid Build Coastguard Worker   auto values () const HB_AUTO_RETURN (+ forw_map.values())
93*2d1272b8SAndroid Build Coastguard Worker   auto iter () const HB_AUTO_RETURN (+ forw_map.iter())
94*2d1272b8SAndroid Build Coastguard Worker };
95*2d1272b8SAndroid Build Coastguard Worker 
96*2d1272b8SAndroid Build Coastguard Worker /* Incremental bimap: only lhs is given, rhs is incrementally assigned */
97*2d1272b8SAndroid Build Coastguard Worker struct hb_inc_bimap_t
98*2d1272b8SAndroid Build Coastguard Worker {
in_errorhb_inc_bimap_t99*2d1272b8SAndroid Build Coastguard Worker   bool in_error () const { return forw_map.in_error () || back_map.in_error (); }
100*2d1272b8SAndroid Build Coastguard Worker 
get_populationhb_inc_bimap_t101*2d1272b8SAndroid Build Coastguard Worker   unsigned int get_population () const { return forw_map.get_population (); }
102*2d1272b8SAndroid Build Coastguard Worker 
resethb_inc_bimap_t103*2d1272b8SAndroid Build Coastguard Worker   void reset ()
104*2d1272b8SAndroid Build Coastguard Worker   {
105*2d1272b8SAndroid Build Coastguard Worker     forw_map.reset ();
106*2d1272b8SAndroid Build Coastguard Worker     back_map.reset ();
107*2d1272b8SAndroid Build Coastguard Worker   }
108*2d1272b8SAndroid Build Coastguard Worker 
allochb_inc_bimap_t109*2d1272b8SAndroid Build Coastguard Worker   void alloc (unsigned pop)
110*2d1272b8SAndroid Build Coastguard Worker   {
111*2d1272b8SAndroid Build Coastguard Worker     forw_map.alloc (pop);
112*2d1272b8SAndroid Build Coastguard Worker     back_map.alloc (pop);
113*2d1272b8SAndroid Build Coastguard Worker   }
114*2d1272b8SAndroid Build Coastguard Worker 
clearhb_inc_bimap_t115*2d1272b8SAndroid Build Coastguard Worker   void clear ()
116*2d1272b8SAndroid Build Coastguard Worker   {
117*2d1272b8SAndroid Build Coastguard Worker     forw_map.clear ();
118*2d1272b8SAndroid Build Coastguard Worker     back_map.resize (0);
119*2d1272b8SAndroid Build Coastguard Worker   }
120*2d1272b8SAndroid Build Coastguard Worker 
121*2d1272b8SAndroid Build Coastguard Worker   /* Add a mapping from lhs to rhs with a unique value if lhs is unknown.
122*2d1272b8SAndroid Build Coastguard Worker    * Return the rhs value as the result.
123*2d1272b8SAndroid Build Coastguard Worker    */
addhb_inc_bimap_t124*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t add (hb_codepoint_t lhs)
125*2d1272b8SAndroid Build Coastguard Worker   {
126*2d1272b8SAndroid Build Coastguard Worker     hb_codepoint_t  rhs = forw_map[lhs];
127*2d1272b8SAndroid Build Coastguard Worker     if (rhs == HB_MAP_VALUE_INVALID)
128*2d1272b8SAndroid Build Coastguard Worker     {
129*2d1272b8SAndroid Build Coastguard Worker       rhs = back_map.length;
130*2d1272b8SAndroid Build Coastguard Worker       forw_map.set (lhs, rhs);
131*2d1272b8SAndroid Build Coastguard Worker       back_map.push (lhs);
132*2d1272b8SAndroid Build Coastguard Worker     }
133*2d1272b8SAndroid Build Coastguard Worker     return rhs;
134*2d1272b8SAndroid Build Coastguard Worker   }
135*2d1272b8SAndroid Build Coastguard Worker 
skiphb_inc_bimap_t136*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t skip ()
137*2d1272b8SAndroid Build Coastguard Worker   {
138*2d1272b8SAndroid Build Coastguard Worker     hb_codepoint_t start = back_map.length;
139*2d1272b8SAndroid Build Coastguard Worker     back_map.push (HB_MAP_VALUE_INVALID);
140*2d1272b8SAndroid Build Coastguard Worker     return start;
141*2d1272b8SAndroid Build Coastguard Worker   }
142*2d1272b8SAndroid Build Coastguard Worker 
skiphb_inc_bimap_t143*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t skip (unsigned count)
144*2d1272b8SAndroid Build Coastguard Worker   {
145*2d1272b8SAndroid Build Coastguard Worker     hb_codepoint_t start = back_map.length;
146*2d1272b8SAndroid Build Coastguard Worker     back_map.alloc (back_map.length + count);
147*2d1272b8SAndroid Build Coastguard Worker     for (unsigned i = 0; i < count; i++)
148*2d1272b8SAndroid Build Coastguard Worker       back_map.push (HB_MAP_VALUE_INVALID);
149*2d1272b8SAndroid Build Coastguard Worker     return start;
150*2d1272b8SAndroid Build Coastguard Worker   }
151*2d1272b8SAndroid Build Coastguard Worker 
get_next_valuehb_inc_bimap_t152*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t get_next_value () const
153*2d1272b8SAndroid Build Coastguard Worker   { return back_map.length; }
154*2d1272b8SAndroid Build Coastguard Worker 
add_sethb_inc_bimap_t155*2d1272b8SAndroid Build Coastguard Worker   void add_set (const hb_set_t *set)
156*2d1272b8SAndroid Build Coastguard Worker   {
157*2d1272b8SAndroid Build Coastguard Worker     for (auto i : *set) add (i);
158*2d1272b8SAndroid Build Coastguard Worker   }
159*2d1272b8SAndroid Build Coastguard Worker 
160*2d1272b8SAndroid Build Coastguard Worker   /* Create an identity map. */
identityhb_inc_bimap_t161*2d1272b8SAndroid Build Coastguard Worker   bool identity (unsigned int size)
162*2d1272b8SAndroid Build Coastguard Worker   {
163*2d1272b8SAndroid Build Coastguard Worker     clear ();
164*2d1272b8SAndroid Build Coastguard Worker     for (hb_codepoint_t i = 0; i < size; i++) add (i);
165*2d1272b8SAndroid Build Coastguard Worker     return !in_error ();
166*2d1272b8SAndroid Build Coastguard Worker   }
167*2d1272b8SAndroid Build Coastguard Worker 
168*2d1272b8SAndroid Build Coastguard Worker   protected:
cmp_idhb_inc_bimap_t169*2d1272b8SAndroid Build Coastguard Worker   static int cmp_id (const void* a, const void* b)
170*2d1272b8SAndroid Build Coastguard Worker   { return (int)*(const hb_codepoint_t *)a - (int)*(const hb_codepoint_t *)b; }
171*2d1272b8SAndroid Build Coastguard Worker 
172*2d1272b8SAndroid Build Coastguard Worker   public:
173*2d1272b8SAndroid Build Coastguard Worker   /* Optional: after finished adding all mappings in a random order,
174*2d1272b8SAndroid Build Coastguard Worker    * reassign rhs to lhs so that they are in the same order. */
sorthb_inc_bimap_t175*2d1272b8SAndroid Build Coastguard Worker   void sort ()
176*2d1272b8SAndroid Build Coastguard Worker   {
177*2d1272b8SAndroid Build Coastguard Worker     hb_codepoint_t  count = get_population ();
178*2d1272b8SAndroid Build Coastguard Worker     hb_vector_t <hb_codepoint_t> work;
179*2d1272b8SAndroid Build Coastguard Worker     if (unlikely (!work.resize (count, false))) return;
180*2d1272b8SAndroid Build Coastguard Worker 
181*2d1272b8SAndroid Build Coastguard Worker     for (hb_codepoint_t rhs = 0; rhs < count; rhs++)
182*2d1272b8SAndroid Build Coastguard Worker       work.arrayZ[rhs] = back_map[rhs];
183*2d1272b8SAndroid Build Coastguard Worker 
184*2d1272b8SAndroid Build Coastguard Worker     work.qsort (cmp_id);
185*2d1272b8SAndroid Build Coastguard Worker 
186*2d1272b8SAndroid Build Coastguard Worker     clear ();
187*2d1272b8SAndroid Build Coastguard Worker     for (hb_codepoint_t rhs = 0; rhs < count; rhs++)
188*2d1272b8SAndroid Build Coastguard Worker       add (work.arrayZ[rhs]);
189*2d1272b8SAndroid Build Coastguard Worker   }
190*2d1272b8SAndroid Build Coastguard Worker 
gethb_inc_bimap_t191*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t get (hb_codepoint_t lhs) const { return forw_map.get (lhs); }
backwardhb_inc_bimap_t192*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t backward (hb_codepoint_t rhs) const { return back_map[rhs]; }
193*2d1272b8SAndroid Build Coastguard Worker 
operator []hb_inc_bimap_t194*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t operator [] (hb_codepoint_t lhs) const { return get (lhs); }
hashb_inc_bimap_t195*2d1272b8SAndroid Build Coastguard Worker   bool has (hb_codepoint_t lhs) const { return forw_map.has (lhs); }
196*2d1272b8SAndroid Build Coastguard Worker 
197*2d1272b8SAndroid Build Coastguard Worker   protected:
198*2d1272b8SAndroid Build Coastguard Worker   hb_map_t forw_map;
199*2d1272b8SAndroid Build Coastguard Worker   hb_vector_t<hb_codepoint_t> back_map;
200*2d1272b8SAndroid Build Coastguard Worker 
201*2d1272b8SAndroid Build Coastguard Worker   public:
202*2d1272b8SAndroid Build Coastguard Worker   auto keys () const HB_AUTO_RETURN (+ back_map.iter())
203*2d1272b8SAndroid Build Coastguard Worker };
204*2d1272b8SAndroid Build Coastguard Worker 
205*2d1272b8SAndroid Build Coastguard Worker #endif /* HB_BIMAP_HH */
206