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 <fcntl.h> |
| 6 #include <ft2build.h> |
| 7 #include FT_FREETYPE_H |
| 8 #include FT_OUTLINE_H |
| 9 #include <sys/stat.h> |
| 10 #include <sys/types.h> |
| 11 #include <unistd.h> |
| 12 |
| 13 #include <cstdio> |
| 14 #include <cstdlib> |
| 15 #include <cstring> |
| 16 |
| 17 #include "opentype-sanitiser.h" |
| 18 #include "ots-memory-stream.h" |
| 19 |
| 20 namespace { |
| 21 |
| 22 void DumpBitmap(const FT_Bitmap *bitmap) { |
| 23 for (int i = 0; i < bitmap->rows * bitmap->width; ++i) { |
| 24 if (bitmap->buffer[i] > 192) { |
| 25 std::fprintf(stderr, "#"); |
| 26 } else if (bitmap->buffer[i] > 128) { |
| 27 std::fprintf(stderr, "*"); |
| 28 } else if (bitmap->buffer[i] > 64) { |
| 29 std::fprintf(stderr, "+"); |
| 30 } else if (bitmap->buffer[i] > 32) { |
| 31 std::fprintf(stderr, "."); |
| 32 } else { |
| 33 std::fprintf(stderr, " "); |
| 34 } |
| 35 |
| 36 if ((i + 1) % bitmap->width == 0) { |
| 37 std::fprintf(stderr, "\n"); |
| 38 } |
| 39 } |
| 40 } |
| 41 |
| 42 int CompareBitmaps(const FT_Bitmap *orig, const FT_Bitmap *trans) { |
| 43 int ret = 0; |
| 44 |
| 45 if (orig->width == trans->width && |
| 46 orig->rows == trans->rows) { |
| 47 for (int i = 0; i < orig->rows * orig->width; ++i) { |
| 48 if (orig->buffer[i] != trans->buffer[i]) { |
| 49 std::fprintf(stderr, "bitmap data doesn't match!\n"); |
| 50 ret = 1; |
| 51 break; |
| 52 } |
| 53 } |
| 54 } else { |
| 55 std::fprintf(stderr, "bitmap metrics doesn't match! (%d, %d), (%d, %d)\n", |
| 56 orig->width, orig->rows, trans->width, trans->rows); |
| 57 ret = 1; |
| 58 } |
| 59 |
| 60 if (ret) { |
| 61 std::fprintf(stderr, "EXPECTED:\n"); |
| 62 DumpBitmap(orig); |
| 63 std::fprintf(stderr, "\nACTUAL:\n"); |
| 64 DumpBitmap(trans); |
| 65 std::fprintf(stderr, "\n\n"); |
| 66 } |
| 67 |
| 68 delete[] orig->buffer; |
| 69 delete[] trans->buffer; |
| 70 return ret; |
| 71 } |
| 72 |
| 73 int GetBitmap(FT_Library library, FT_Outline *outline, FT_Bitmap *bitmap) { |
| 74 FT_BBox bbox; |
| 75 FT_Outline_Get_CBox(outline, &bbox); |
| 76 |
| 77 bbox.xMin &= ~63; |
| 78 bbox.yMin &= ~63; |
| 79 bbox.xMax = (bbox.xMax + 63) & ~63; |
| 80 bbox.yMax = (bbox.yMax + 63) & ~63; |
| 81 FT_Outline_Translate(outline, -bbox.xMin, -bbox.yMin); |
| 82 |
| 83 const int w = (bbox.xMax - bbox.xMin) >> 6; |
| 84 const int h = (bbox.yMax - bbox.yMin) >> 6; |
| 85 |
| 86 if (w == 0 || h == 0) { |
| 87 return -1; // white space |
| 88 } |
| 89 if (w < 0 || h < 0) { |
| 90 std::fprintf(stderr, "bad width/height\n"); |
| 91 return 1; // error |
| 92 } |
| 93 |
| 94 uint8_t *buf = new uint8_t[w * h]; |
| 95 std::memset(buf, 0x0, w * h); |
| 96 |
| 97 bitmap->width = w; |
| 98 bitmap->rows = h; |
| 99 bitmap->pitch = w; |
| 100 bitmap->buffer = buf; |
| 101 bitmap->pixel_mode = FT_PIXEL_MODE_GRAY; |
| 102 bitmap->num_grays = 256; |
| 103 if (FT_Outline_Get_Bitmap(library, outline, bitmap)) { |
| 104 std::fprintf(stderr, "can't get outline\n"); |
| 105 delete[] buf; |
| 106 return 1; // error. |
| 107 } |
| 108 |
| 109 return 0; |
| 110 } |
| 111 |
| 112 int LoadChar(FT_Face face, bool use_bitmap, int pt, FT_ULong c) { |
| 113 static const int kDpi = 72; |
| 114 |
| 115 FT_Matrix matrix; |
| 116 matrix.xx = matrix.yy = 1 << 16; |
| 117 matrix.xy = matrix.yx = 0 << 16; |
| 118 |
| 119 FT_Int32 flags = FT_LOAD_DEFAULT | FT_LOAD_TARGET_NORMAL; |
| 120 if (!use_bitmap) { |
| 121 // Since the transcoder drops embedded bitmaps from the transcoded one, |
| 122 // we have to use FT_LOAD_NO_BITMAP flag for the original face. |
| 123 flags |= FT_LOAD_NO_BITMAP; |
| 124 } |
| 125 |
| 126 FT_Error error = FT_Set_Char_Size(face, pt * (1 << 6), 0, kDpi, 0); |
| 127 if (error) { |
| 128 std::fprintf(stderr, "Failed to set the char size!\n"); |
| 129 return 1; |
| 130 } |
| 131 |
| 132 FT_Set_Transform(face, &matrix, 0); |
| 133 |
| 134 error = FT_Load_Char(face, c, flags); |
| 135 if (error) return -1; // no such glyf in the font. |
| 136 |
| 137 if (face->glyph->format != FT_GLYPH_FORMAT_OUTLINE) { |
| 138 std::fprintf(stderr, "bad format\n"); |
| 139 return 1; |
| 140 } |
| 141 |
| 142 return 0; |
| 143 } |
| 144 |
| 145 int LoadCharThenCompare(FT_Library library, |
| 146 FT_Face orig_face, FT_Face trans_face, |
| 147 int pt, FT_ULong c) { |
| 148 FT_Bitmap orig_bitmap, trans_bitmap; |
| 149 |
| 150 // Load original bitmap. |
| 151 int ret = LoadChar(orig_face, false, pt, c); |
| 152 if (ret) return ret; // 1: error, -1: no such glyph |
| 153 |
| 154 FT_Outline *outline = &orig_face->glyph->outline; |
| 155 ret = GetBitmap(library, outline, &orig_bitmap); |
| 156 if (ret) return ret; // white space? |
| 157 |
| 158 // Load transformed bitmap. |
| 159 ret = LoadChar(trans_face, true, pt, c); |
| 160 if (ret == -1) { |
| 161 std::fprintf(stderr, "the glyph is not found on the transcoded font\n"); |
| 162 } |
| 163 if (ret) return 1; // -1 should be treated as error. |
| 164 outline = &trans_face->glyph->outline; |
| 165 ret = GetBitmap(library, outline, &trans_bitmap); |
| 166 if (ret) return ret; // white space? |
| 167 |
| 168 return CompareBitmaps(&orig_bitmap, &trans_bitmap); |
| 169 } |
| 170 |
| 171 int SideBySide(FT_Library library, const char *file_name, |
| 172 uint8_t *orig_font, size_t orig_len, |
| 173 uint8_t *trans_font, size_t trans_len) { |
| 174 FT_Face orig_face; |
| 175 FT_Error error |
| 176 = FT_New_Memory_Face(library, orig_font, orig_len, 0, &orig_face); |
| 177 if (error) { |
| 178 std::fprintf(stderr, "Failed to open the original font: %s!\n", file_name); |
| 179 return 1; |
| 180 } |
| 181 |
| 182 FT_Face trans_face; |
| 183 error = FT_New_Memory_Face(library, trans_font, trans_len, 0, &trans_face); |
| 184 if (error) { |
| 185 std::fprintf(stderr, "Failed to open the transcoded font: %s!\n", |
| 186 file_name); |
| 187 return 1; |
| 188 } |
| 189 |
| 190 static const int kPts[] = {100, 20, 18, 16, 12, 10, 8}; // pt |
| 191 static const size_t kPtsLen = sizeof(kPts) / sizeof(kPts[0]); |
| 192 |
| 193 static const int kUnicodeRanges[] = { |
| 194 0x0020, 0x007E, // Basic Latin (ASCII) |
| 195 0x00A1, 0x017F, // Latin-1 |
| 196 0x1100, 0x11FF, // Hangul |
| 197 0x3040, 0x309F, // Japanese HIRAGANA letters |
| 198 0x3130, 0x318F, // Hangul |
| 199 0x4E00, 0x4F00, // CJK Kanji/Hanja |
| 200 0xAC00, 0xAD00, // Hangul |
| 201 }; |
| 202 static const size_t kUnicodeRangesLen |
| 203 = sizeof(kUnicodeRanges) / sizeof(kUnicodeRanges[0]); |
| 204 |
| 205 for (size_t i = 0; i < kPtsLen; ++i) { |
| 206 for (size_t j = 0; j < kUnicodeRangesLen; j += 2) { |
| 207 for (int k = 0; k <= kUnicodeRanges[j + 1] - kUnicodeRanges[j]; ++k) { |
| 208 int ret = LoadCharThenCompare(library, orig_face, trans_face, |
| 209 kPts[i], |
| 210 kUnicodeRanges[j] + k); |
| 211 if (ret > 0) { |
| 212 std::fprintf(stderr, "Glyph mismatch! (file: %s, U+%04x, %dpt)!\n", |
| 213 file_name, kUnicodeRanges[j] + k, kPts[i]); |
| 214 return 1; |
| 215 } |
| 216 } |
| 217 } |
| 218 } |
| 219 |
| 220 return 0; |
| 221 } |
| 222 |
| 223 } // namespace |
| 224 |
| 225 int main(int argc, char **argv) { |
| 226 if (argc != 2) { |
| 227 std::fprintf(stderr, "Usage: %s ttf_or_otf_filename\n", argv[0]); |
| 228 return 1; |
| 229 } |
| 230 |
| 231 // load the font to memory. |
| 232 const int fd = ::open(argv[1], O_RDONLY); |
| 233 if (fd < 0) { |
| 234 ::perror("open"); |
| 235 return 1; |
| 236 } |
| 237 |
| 238 struct stat st; |
| 239 ::fstat(fd, &st); |
| 240 const off_t orig_len = st.st_size; |
| 241 |
| 242 uint8_t *orig_font = new uint8_t[orig_len]; |
| 243 if (::read(fd, orig_font, orig_len) != orig_len) { |
| 244 std::fprintf(stderr, "Failed to read file!\n"); |
| 245 return 1; |
| 246 } |
| 247 ::close(fd); |
| 248 |
| 249 // check if FreeType2 can open the original font. |
| 250 FT_Library library; |
| 251 FT_Error error = FT_Init_FreeType(&library); |
| 252 if (error) { |
| 253 std::fprintf(stderr, "Failed to initialize FreeType2!\n"); |
| 254 return 1; |
| 255 } |
| 256 FT_Face dummy; |
| 257 error = FT_New_Memory_Face(library, orig_font, orig_len, 0, &dummy); |
| 258 if (error) { |
| 259 std::fprintf(stderr, "Failed to open the original font with FT2! %s\n", |
| 260 argv[1]); |
| 261 return 1; |
| 262 } |
| 263 |
| 264 // transcode the original font. |
| 265 static const size_t kPadLen = 20 * 1024; |
| 266 uint8_t *trans_font = new uint8_t[orig_len + kPadLen]; |
| 267 ots::MemoryStream output(trans_font, orig_len + kPadLen); |
| 268 ots::OTSContext context; |
| 269 |
| 270 bool result = context.Process(&output, orig_font, orig_len); |
| 271 if (!result) { |
| 272 std::fprintf(stderr, "Failed to sanitise file! %s\n", argv[1]); |
| 273 return 1; |
| 274 } |
| 275 const size_t trans_len = output.Tell(); |
| 276 |
| 277 // perform side-by-side tests. |
| 278 return SideBySide(library, argv[1], |
| 279 orig_font, orig_len, |
| 280 trans_font, trans_len); |
| 281 } |
OLD | NEW |