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 |