OLD | NEW |
| (Empty) |
1 // Copyright (c) 2009 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "hdmx.h" | |
6 #include "head.h" | |
7 #include "maxp.h" | |
8 | |
9 // hdmx - Horizontal Device Metrics | |
10 // http://www.microsoft.com/typography/otspec/hdmx.htm | |
11 | |
12 #define TABLE_NAME "hdmx" | |
13 | |
14 #define DROP_THIS_TABLE(...) \ | |
15 do { \ | |
16 OTS_FAILURE_MSG_(file, TABLE_NAME ": " __VA_ARGS__); \ | |
17 OTS_FAILURE_MSG("Table discarded"); \ | |
18 delete file->hdmx; \ | |
19 file->hdmx = 0; \ | |
20 } while (0) | |
21 | |
22 namespace ots { | |
23 | |
24 bool ots_hdmx_parse(OpenTypeFile *file, const uint8_t *data, size_t length) { | |
25 Buffer table(data, length); | |
26 file->hdmx = new OpenTypeHDMX; | |
27 OpenTypeHDMX * const hdmx = file->hdmx; | |
28 | |
29 if (!file->head || !file->maxp) { | |
30 return OTS_FAILURE_MSG("Missing maxp or head tables in font, needed by hdmx"
); | |
31 } | |
32 | |
33 if ((file->head->flags & 0x14) == 0) { | |
34 // http://www.microsoft.com/typography/otspec/recom.htm | |
35 DROP_THIS_TABLE("the table should not be present when bit 2 and 4 of the " | |
36 "head->flags are not set"); | |
37 return true; | |
38 } | |
39 | |
40 int16_t num_recs; | |
41 if (!table.ReadU16(&hdmx->version) || | |
42 !table.ReadS16(&num_recs) || | |
43 !table.ReadS32(&hdmx->size_device_record)) { | |
44 return OTS_FAILURE_MSG("Failed to read hdmx header"); | |
45 } | |
46 if (hdmx->version != 0) { | |
47 DROP_THIS_TABLE("bad version: %u", hdmx->version); | |
48 return true; | |
49 } | |
50 if (num_recs <= 0) { | |
51 DROP_THIS_TABLE("bad num_recs: %d", num_recs); | |
52 return true; | |
53 } | |
54 const int32_t actual_size_device_record = file->maxp->num_glyphs + 2; | |
55 if (hdmx->size_device_record < actual_size_device_record) { | |
56 DROP_THIS_TABLE("bad hdmx->size_device_record: %d", hdmx->size_device_record
); | |
57 return true; | |
58 } | |
59 | |
60 hdmx->pad_len = hdmx->size_device_record - actual_size_device_record; | |
61 if (hdmx->pad_len > 3) { | |
62 return OTS_FAILURE_MSG("Bad padding %d", hdmx->pad_len); | |
63 } | |
64 | |
65 uint8_t last_pixel_size = 0; | |
66 hdmx->records.reserve(num_recs); | |
67 for (int i = 0; i < num_recs; ++i) { | |
68 OpenTypeHDMXDeviceRecord rec; | |
69 | |
70 if (!table.ReadU8(&rec.pixel_size) || | |
71 !table.ReadU8(&rec.max_width)) { | |
72 return OTS_FAILURE_MSG("Failed to read hdmx record %d", i); | |
73 } | |
74 if ((i != 0) && | |
75 (rec.pixel_size <= last_pixel_size)) { | |
76 DROP_THIS_TABLE("records are not sorted"); | |
77 return true; | |
78 } | |
79 last_pixel_size = rec.pixel_size; | |
80 | |
81 rec.widths.reserve(file->maxp->num_glyphs); | |
82 for (unsigned j = 0; j < file->maxp->num_glyphs; ++j) { | |
83 uint8_t width; | |
84 if (!table.ReadU8(&width)) { | |
85 return OTS_FAILURE_MSG("Failed to read glyph width %d in record %d", j,
i); | |
86 } | |
87 rec.widths.push_back(width); | |
88 } | |
89 | |
90 if ((hdmx->pad_len > 0) && | |
91 !table.Skip(hdmx->pad_len)) { | |
92 return OTS_FAILURE_MSG("Failed to skip padding %d", hdmx->pad_len); | |
93 } | |
94 | |
95 hdmx->records.push_back(rec); | |
96 } | |
97 | |
98 return true; | |
99 } | |
100 | |
101 bool ots_hdmx_should_serialise(OpenTypeFile *file) { | |
102 if (!file->hdmx) return false; | |
103 if (!file->glyf) return false; // this table is not for CFF fonts. | |
104 return true; | |
105 } | |
106 | |
107 bool ots_hdmx_serialise(OTSStream *out, OpenTypeFile *file) { | |
108 OpenTypeHDMX * const hdmx = file->hdmx; | |
109 | |
110 const int16_t num_recs = static_cast<int16_t>(hdmx->records.size()); | |
111 if (hdmx->records.size() > | |
112 static_cast<size_t>(std::numeric_limits<int16_t>::max()) || | |
113 !out->WriteU16(hdmx->version) || | |
114 !out->WriteS16(num_recs) || | |
115 !out->WriteS32(hdmx->size_device_record)) { | |
116 return OTS_FAILURE_MSG("Failed to write hdmx header"); | |
117 } | |
118 | |
119 for (int16_t i = 0; i < num_recs; ++i) { | |
120 const OpenTypeHDMXDeviceRecord& rec = hdmx->records[i]; | |
121 if (!out->Write(&rec.pixel_size, 1) || | |
122 !out->Write(&rec.max_width, 1) || | |
123 !out->Write(&rec.widths[0], rec.widths.size())) { | |
124 return OTS_FAILURE_MSG("Failed to write hdmx record %d", i); | |
125 } | |
126 if ((hdmx->pad_len > 0) && | |
127 !out->Write((const uint8_t *)"\x00\x00\x00", hdmx->pad_len)) { | |
128 return OTS_FAILURE_MSG("Failed to write hdmx padding of length %d", hdmx->
pad_len); | |
129 } | |
130 } | |
131 | |
132 return true; | |
133 } | |
134 | |
135 void ots_hdmx_free(OpenTypeFile *file) { | |
136 delete file->hdmx; | |
137 } | |
138 | |
139 } // namespace ots | |
140 | |
141 #undef TABLE_NAME | |
142 #undef DROP_THIS_TABLE | |
OLD | NEW |