xref: /aosp_15_r20/external/harfbuzz_ng/src/hb-ot-hdmx-table.hh (revision 2d1272b857b1f7575e6e246373e1cb218663db8a)
1*2d1272b8SAndroid Build Coastguard Worker /*
2*2d1272b8SAndroid Build Coastguard Worker  * Copyright © 2018  Google, 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  * Google Author(s): Garret Rieger
25*2d1272b8SAndroid Build Coastguard Worker  */
26*2d1272b8SAndroid Build Coastguard Worker 
27*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_OT_HDMX_TABLE_HH
28*2d1272b8SAndroid Build Coastguard Worker #define HB_OT_HDMX_TABLE_HH
29*2d1272b8SAndroid Build Coastguard Worker 
30*2d1272b8SAndroid Build Coastguard Worker #include "hb-open-type.hh"
31*2d1272b8SAndroid Build Coastguard Worker 
32*2d1272b8SAndroid Build Coastguard Worker /*
33*2d1272b8SAndroid Build Coastguard Worker  * hdmx -- Horizontal Device Metrics
34*2d1272b8SAndroid Build Coastguard Worker  * https://docs.microsoft.com/en-us/typography/opentype/spec/hdmx
35*2d1272b8SAndroid Build Coastguard Worker  */
36*2d1272b8SAndroid Build Coastguard Worker #define HB_OT_TAG_hdmx HB_TAG('h','d','m','x')
37*2d1272b8SAndroid Build Coastguard Worker 
38*2d1272b8SAndroid Build Coastguard Worker 
39*2d1272b8SAndroid Build Coastguard Worker namespace OT {
40*2d1272b8SAndroid Build Coastguard Worker 
41*2d1272b8SAndroid Build Coastguard Worker 
42*2d1272b8SAndroid Build Coastguard Worker struct DeviceRecord
43*2d1272b8SAndroid Build Coastguard Worker {
get_sizeOT::DeviceRecord44*2d1272b8SAndroid Build Coastguard Worker   static unsigned int get_size (unsigned count)
45*2d1272b8SAndroid Build Coastguard Worker   { return hb_ceil_to_4 (min_size + count * HBUINT8::static_size); }
46*2d1272b8SAndroid Build Coastguard Worker 
47*2d1272b8SAndroid Build Coastguard Worker   template<typename Iterator,
48*2d1272b8SAndroid Build Coastguard Worker 	   hb_requires (hb_is_iterator (Iterator))>
serializeOT::DeviceRecord49*2d1272b8SAndroid Build Coastguard Worker   bool serialize (hb_serialize_context_t *c,
50*2d1272b8SAndroid Build Coastguard Worker 		  unsigned pixelSize,
51*2d1272b8SAndroid Build Coastguard Worker 		  Iterator it,
52*2d1272b8SAndroid Build Coastguard Worker 		  const hb_vector_t<hb_codepoint_pair_t> new_to_old_gid_list,
53*2d1272b8SAndroid Build Coastguard Worker 		  unsigned num_glyphs)
54*2d1272b8SAndroid Build Coastguard Worker   {
55*2d1272b8SAndroid Build Coastguard Worker     TRACE_SERIALIZE (this);
56*2d1272b8SAndroid Build Coastguard Worker 
57*2d1272b8SAndroid Build Coastguard Worker     if (unlikely (!c->extend (this, num_glyphs)))  return_trace (false);
58*2d1272b8SAndroid Build Coastguard Worker 
59*2d1272b8SAndroid Build Coastguard Worker     this->pixelSize = pixelSize;
60*2d1272b8SAndroid Build Coastguard Worker     this->maxWidth =
61*2d1272b8SAndroid Build Coastguard Worker     + it
62*2d1272b8SAndroid Build Coastguard Worker     | hb_reduce (hb_max, 0u);
63*2d1272b8SAndroid Build Coastguard Worker 
64*2d1272b8SAndroid Build Coastguard Worker     for (auto &_ : new_to_old_gid_list)
65*2d1272b8SAndroid Build Coastguard Worker       widthsZ[_.first] = *it++;
66*2d1272b8SAndroid Build Coastguard Worker 
67*2d1272b8SAndroid Build Coastguard Worker     return_trace (true);
68*2d1272b8SAndroid Build Coastguard Worker   }
69*2d1272b8SAndroid Build Coastguard Worker 
sanitizeOT::DeviceRecord70*2d1272b8SAndroid Build Coastguard Worker   bool sanitize (hb_sanitize_context_t *c, unsigned sizeDeviceRecord) const
71*2d1272b8SAndroid Build Coastguard Worker   {
72*2d1272b8SAndroid Build Coastguard Worker     TRACE_SANITIZE (this);
73*2d1272b8SAndroid Build Coastguard Worker     return_trace (likely (c->check_struct (this) &&
74*2d1272b8SAndroid Build Coastguard Worker 			  hb_barrier () &&
75*2d1272b8SAndroid Build Coastguard Worker 			  c->check_range (this, sizeDeviceRecord)));
76*2d1272b8SAndroid Build Coastguard Worker   }
77*2d1272b8SAndroid Build Coastguard Worker 
78*2d1272b8SAndroid Build Coastguard Worker   HBUINT8			pixelSize;	/* Pixel size for following widths (as ppem). */
79*2d1272b8SAndroid Build Coastguard Worker   HBUINT8			maxWidth;	/* Maximum width. */
80*2d1272b8SAndroid Build Coastguard Worker   UnsizedArrayOf<HBUINT8>	widthsZ;	/* Array of widths (numGlyphs is from the 'maxp' table). */
81*2d1272b8SAndroid Build Coastguard Worker   public:
82*2d1272b8SAndroid Build Coastguard Worker   DEFINE_SIZE_UNBOUNDED (2);
83*2d1272b8SAndroid Build Coastguard Worker };
84*2d1272b8SAndroid Build Coastguard Worker 
85*2d1272b8SAndroid Build Coastguard Worker 
86*2d1272b8SAndroid Build Coastguard Worker struct hdmx
87*2d1272b8SAndroid Build Coastguard Worker {
88*2d1272b8SAndroid Build Coastguard Worker   static constexpr hb_tag_t tableTag = HB_OT_TAG_hdmx;
89*2d1272b8SAndroid Build Coastguard Worker 
get_sizeOT::hdmx90*2d1272b8SAndroid Build Coastguard Worker   unsigned int get_size () const
91*2d1272b8SAndroid Build Coastguard Worker   { return min_size + numRecords * sizeDeviceRecord; }
92*2d1272b8SAndroid Build Coastguard Worker 
93*2d1272b8SAndroid Build Coastguard Worker   template<typename Iterator,
94*2d1272b8SAndroid Build Coastguard Worker 	   hb_requires (hb_is_iterator (Iterator))>
serializeOT::hdmx95*2d1272b8SAndroid Build Coastguard Worker   bool serialize (hb_serialize_context_t *c,
96*2d1272b8SAndroid Build Coastguard Worker 		  unsigned version,
97*2d1272b8SAndroid Build Coastguard Worker 		  Iterator it,
98*2d1272b8SAndroid Build Coastguard Worker 		  const hb_vector_t<hb_codepoint_pair_t> &new_to_old_gid_list,
99*2d1272b8SAndroid Build Coastguard Worker 		  unsigned num_glyphs)
100*2d1272b8SAndroid Build Coastguard Worker   {
101*2d1272b8SAndroid Build Coastguard Worker     TRACE_SERIALIZE (this);
102*2d1272b8SAndroid Build Coastguard Worker 
103*2d1272b8SAndroid Build Coastguard Worker     if (unlikely (!c->extend_min ((*this))))  return_trace (false);
104*2d1272b8SAndroid Build Coastguard Worker 
105*2d1272b8SAndroid Build Coastguard Worker     this->version = version;
106*2d1272b8SAndroid Build Coastguard Worker     this->numRecords = it.len ();
107*2d1272b8SAndroid Build Coastguard Worker     this->sizeDeviceRecord = DeviceRecord::get_size (num_glyphs);
108*2d1272b8SAndroid Build Coastguard Worker 
109*2d1272b8SAndroid Build Coastguard Worker     for (const hb_item_type<Iterator>& _ : +it)
110*2d1272b8SAndroid Build Coastguard Worker       c->start_embed<DeviceRecord> ()->serialize (c, _.first, _.second, new_to_old_gid_list, num_glyphs);
111*2d1272b8SAndroid Build Coastguard Worker 
112*2d1272b8SAndroid Build Coastguard Worker     return_trace (c->successful ());
113*2d1272b8SAndroid Build Coastguard Worker   }
114*2d1272b8SAndroid Build Coastguard Worker 
115*2d1272b8SAndroid Build Coastguard Worker 
subsetOT::hdmx116*2d1272b8SAndroid Build Coastguard Worker   bool subset (hb_subset_context_t *c) const
117*2d1272b8SAndroid Build Coastguard Worker   {
118*2d1272b8SAndroid Build Coastguard Worker     TRACE_SUBSET (this);
119*2d1272b8SAndroid Build Coastguard Worker 
120*2d1272b8SAndroid Build Coastguard Worker     auto *hdmx_prime = c->serializer->start_embed <hdmx> ();
121*2d1272b8SAndroid Build Coastguard Worker 
122*2d1272b8SAndroid Build Coastguard Worker     unsigned num_input_glyphs = get_num_glyphs ();
123*2d1272b8SAndroid Build Coastguard Worker     auto it =
124*2d1272b8SAndroid Build Coastguard Worker     + hb_range ((unsigned) numRecords)
125*2d1272b8SAndroid Build Coastguard Worker     | hb_map ([c, num_input_glyphs, this] (unsigned _)
126*2d1272b8SAndroid Build Coastguard Worker 	{
127*2d1272b8SAndroid Build Coastguard Worker 	  const DeviceRecord *device_record =
128*2d1272b8SAndroid Build Coastguard Worker 	    &StructAtOffset<DeviceRecord> (&firstDeviceRecord,
129*2d1272b8SAndroid Build Coastguard Worker 					   _ * sizeDeviceRecord);
130*2d1272b8SAndroid Build Coastguard Worker 	  auto row =
131*2d1272b8SAndroid Build Coastguard Worker 	    + hb_iter (c->plan->new_to_old_gid_list)
132*2d1272b8SAndroid Build Coastguard Worker 	    | hb_map ([num_input_glyphs, device_record] (hb_codepoint_pair_t _)
133*2d1272b8SAndroid Build Coastguard Worker 		      {
134*2d1272b8SAndroid Build Coastguard Worker 			return device_record->widthsZ.as_array (num_input_glyphs) [_.second];
135*2d1272b8SAndroid Build Coastguard Worker 		      })
136*2d1272b8SAndroid Build Coastguard Worker 	    ;
137*2d1272b8SAndroid Build Coastguard Worker 	  return hb_pair ((unsigned) device_record->pixelSize, +row);
138*2d1272b8SAndroid Build Coastguard Worker 	})
139*2d1272b8SAndroid Build Coastguard Worker     ;
140*2d1272b8SAndroid Build Coastguard Worker 
141*2d1272b8SAndroid Build Coastguard Worker     hdmx_prime->serialize (c->serializer, version, it,
142*2d1272b8SAndroid Build Coastguard Worker 			   c->plan->new_to_old_gid_list,
143*2d1272b8SAndroid Build Coastguard Worker 			   c->plan->num_output_glyphs ());
144*2d1272b8SAndroid Build Coastguard Worker     return_trace (true);
145*2d1272b8SAndroid Build Coastguard Worker   }
146*2d1272b8SAndroid Build Coastguard Worker 
get_num_glyphsOT::hdmx147*2d1272b8SAndroid Build Coastguard Worker   unsigned get_num_glyphs () const
148*2d1272b8SAndroid Build Coastguard Worker   {
149*2d1272b8SAndroid Build Coastguard Worker     return sizeDeviceRecord - DeviceRecord::min_size;
150*2d1272b8SAndroid Build Coastguard Worker   }
151*2d1272b8SAndroid Build Coastguard Worker 
sanitizeOT::hdmx152*2d1272b8SAndroid Build Coastguard Worker   bool sanitize (hb_sanitize_context_t *c) const
153*2d1272b8SAndroid Build Coastguard Worker   {
154*2d1272b8SAndroid Build Coastguard Worker     TRACE_SANITIZE (this);
155*2d1272b8SAndroid Build Coastguard Worker     return_trace (c->check_struct (this) &&
156*2d1272b8SAndroid Build Coastguard Worker 		  hb_barrier () &&
157*2d1272b8SAndroid Build Coastguard Worker 		  !hb_unsigned_mul_overflows (numRecords, sizeDeviceRecord) &&
158*2d1272b8SAndroid Build Coastguard Worker                   min_size + numRecords * sizeDeviceRecord > numRecords * sizeDeviceRecord &&
159*2d1272b8SAndroid Build Coastguard Worker 		  sizeDeviceRecord >= DeviceRecord::min_size &&
160*2d1272b8SAndroid Build Coastguard Worker 		  c->check_range (this, get_size ()));
161*2d1272b8SAndroid Build Coastguard Worker   }
162*2d1272b8SAndroid Build Coastguard Worker 
163*2d1272b8SAndroid Build Coastguard Worker   protected:
164*2d1272b8SAndroid Build Coastguard Worker   HBUINT16	version;	/* Table version number (0) */
165*2d1272b8SAndroid Build Coastguard Worker   HBUINT16	numRecords;	/* Number of device records. */
166*2d1272b8SAndroid Build Coastguard Worker   HBUINT32	sizeDeviceRecord;
167*2d1272b8SAndroid Build Coastguard Worker 				/* Size of a device record, 32-bit aligned. */
168*2d1272b8SAndroid Build Coastguard Worker   DeviceRecord	firstDeviceRecord;
169*2d1272b8SAndroid Build Coastguard Worker 				/* Array of device records. */
170*2d1272b8SAndroid Build Coastguard Worker   public:
171*2d1272b8SAndroid Build Coastguard Worker   DEFINE_SIZE_MIN (8);
172*2d1272b8SAndroid Build Coastguard Worker };
173*2d1272b8SAndroid Build Coastguard Worker 
174*2d1272b8SAndroid Build Coastguard Worker } /* namespace OT */
175*2d1272b8SAndroid Build Coastguard Worker 
176*2d1272b8SAndroid Build Coastguard Worker 
177*2d1272b8SAndroid Build Coastguard Worker #endif /* HB_OT_HDMX_TABLE_HH */
178