blob: 08056ba77ff46e25cfb4331b5010e82733297ca3 [file] [log] [blame]
yusukes@chromium.orgd257d182009-11-04 04:56:32 +00001// 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 <freetype/ftoutln.h>
7#include <ft2build.h>
8#include FT_FREETYPE_H
9#include <sys/stat.h>
10#include <sys/types.h>
yusukes@chromium.org8ad0a172009-11-04 06:07:58 +000011#include <unistd.h>
yusukes@chromium.orgd257d182009-11-04 04:56:32 +000012
13#include <cstdio>
14#include <cstdlib>
15#include <cstring>
16
17#include "opentype-sanitiser.h"
18#include "ots-memory-stream.h"
19
20namespace {
21
22void 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
42int 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
73int 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
112int 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
145int 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
171int 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
225int main(int argc, char **argv) {
226 ots::DisableDebugOutput(); // turn off ERROR and WARNING outputs.
227
228 if (argc != 2) {
229 std::fprintf(stderr, "Usage: %s ttf_or_otf_filename\n", argv[0]);
230 return 1;
231 }
232
233 // load the font to memory.
234 const int fd = ::open(argv[1], O_RDONLY);
235 if (fd < 0) {
236 ::perror("open");
237 return 1;
238 }
239
240 struct stat st;
241 ::fstat(fd, &st);
242 const off_t orig_len = st.st_size;
243
244 uint8_t *orig_font = new uint8_t[orig_len];
245 if (::read(fd, orig_font, orig_len) != orig_len) {
246 std::fprintf(stderr, "Failed to read file!\n");
247 return 1;
248 }
249 ::close(fd);
250
251 // check if FreeType2 can open the original font.
252 FT_Library library;
253 FT_Error error = FT_Init_FreeType(&library);
254 if (error) {
255 std::fprintf(stderr, "Failed to initialize FreeType2!\n");
256 return 1;
257 }
258 FT_Face dummy;
259 error = FT_New_Memory_Face(library, orig_font, orig_len, 0, &dummy);
260 if (error) {
261 std::fprintf(stderr, "Failed to open the original font with FT2! %s\n",
262 argv[1]);
263 return 1;
264 }
265
266 // transcode the original font.
267 static const size_t kPadLen = 20 * 1024;
268 uint8_t *trans_font = new uint8_t[orig_len + kPadLen];
269 ots::MemoryStream output(trans_font, orig_len + kPadLen);
270
271 bool result = ots::Process(&output, orig_font, orig_len);
272 if (!result) {
273 std::fprintf(stderr, "Failed to sanitise file! %s\n", argv[1]);
274 return 1;
275 }
276 const size_t trans_len = output.Tell();
277
278 // perform side-by-side tests.
279 return SideBySide(library, argv[1],
280 orig_font, orig_len,
281 trans_font, trans_len);
282}