| Index: third_party/ots/src/gasp.cc
|
| diff --git a/third_party/ots/src/gasp.cc b/third_party/ots/src/gasp.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..1e2327f19bc4c2ca10cc88adeb2390827d870488
|
| --- /dev/null
|
| +++ b/third_party/ots/src/gasp.cc
|
| @@ -0,0 +1,114 @@
|
| +// Copyright (c) 2009 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "gasp.h"
|
| +
|
| +// gasp - Grid-fitting And Scan-conversion Procedure
|
| +// http://www.microsoft.com/typography/otspec/gasp.htm
|
| +
|
| +#define TABLE_NAME "gasp"
|
| +
|
| +#define DROP_THIS_TABLE(...) \
|
| + do { \
|
| + OTS_FAILURE_MSG_(file, TABLE_NAME ": " __VA_ARGS__); \
|
| + OTS_FAILURE_MSG("Table discarded"); \
|
| + delete file->gasp; \
|
| + file->gasp = 0; \
|
| + } while (0)
|
| +
|
| +namespace ots {
|
| +
|
| +bool ots_gasp_parse(OpenTypeFile *file, const uint8_t *data, size_t length) {
|
| + Buffer table(data, length);
|
| +
|
| + OpenTypeGASP *gasp = new OpenTypeGASP;
|
| + file->gasp = gasp;
|
| +
|
| + uint16_t num_ranges = 0;
|
| + if (!table.ReadU16(&gasp->version) ||
|
| + !table.ReadU16(&num_ranges)) {
|
| + return OTS_FAILURE_MSG("Failed to read table header");
|
| + }
|
| +
|
| + if (gasp->version > 1) {
|
| + // Lots of Linux fonts have bad version numbers...
|
| + DROP_THIS_TABLE("bad version: %u", gasp->version);
|
| + return true;
|
| + }
|
| +
|
| + if (num_ranges == 0) {
|
| + DROP_THIS_TABLE("num_ranges is zero");
|
| + return true;
|
| + }
|
| +
|
| + gasp->gasp_ranges.reserve(num_ranges);
|
| + for (unsigned i = 0; i < num_ranges; ++i) {
|
| + uint16_t max_ppem = 0;
|
| + uint16_t behavior = 0;
|
| + if (!table.ReadU16(&max_ppem) ||
|
| + !table.ReadU16(&behavior)) {
|
| + return OTS_FAILURE_MSG("Failed to read subrange %d", i);
|
| + }
|
| + if ((i > 0) && (gasp->gasp_ranges[i - 1].first >= max_ppem)) {
|
| + // The records in the gaspRange[] array must be sorted in order of
|
| + // increasing rangeMaxPPEM value.
|
| + DROP_THIS_TABLE("ranges are not sorted");
|
| + return true;
|
| + }
|
| + if ((i == num_ranges - 1u) && // never underflow.
|
| + (max_ppem != 0xffffu)) {
|
| + DROP_THIS_TABLE("The last record should be 0xFFFF as a sentinel value "
|
| + "for rangeMaxPPEM");
|
| + return true;
|
| + }
|
| +
|
| + if (behavior >> 8) {
|
| + OTS_WARNING("undefined bits are used: %x", behavior);
|
| + // mask undefined bits.
|
| + behavior &= 0x000fu;
|
| + }
|
| +
|
| + if (gasp->version == 0 && (behavior >> 2) != 0) {
|
| + OTS_WARNING("changed the version number to 1");
|
| + gasp->version = 1;
|
| + }
|
| +
|
| + gasp->gasp_ranges.push_back(std::make_pair(max_ppem, behavior));
|
| + }
|
| +
|
| + return true;
|
| +}
|
| +
|
| +bool ots_gasp_should_serialise(OpenTypeFile *file) {
|
| + return file->gasp != NULL;
|
| +}
|
| +
|
| +bool ots_gasp_serialise(OTSStream *out, OpenTypeFile *file) {
|
| + const OpenTypeGASP *gasp = file->gasp;
|
| +
|
| + const uint16_t num_ranges = static_cast<uint16_t>(gasp->gasp_ranges.size());
|
| + if (num_ranges != gasp->gasp_ranges.size() ||
|
| + !out->WriteU16(gasp->version) ||
|
| + !out->WriteU16(num_ranges)) {
|
| + return OTS_FAILURE_MSG("failed to write gasp header");
|
| + }
|
| +
|
| + for (uint16_t i = 0; i < num_ranges; ++i) {
|
| + if (!out->WriteU16(gasp->gasp_ranges[i].first) ||
|
| + !out->WriteU16(gasp->gasp_ranges[i].second)) {
|
| + return OTS_FAILURE_MSG("Failed to write gasp subtable %d", i);
|
| + }
|
| + }
|
| +
|
| + return true;
|
| +}
|
| +
|
| +void ots_gasp_free(OpenTypeFile *file) {
|
| + delete file->gasp;
|
| +}
|
| +
|
| +} // namespace ots
|
| +
|
| +#undef TABLE_NAME
|
| +#undef DROP_THIS_TABLE
|
|
|