123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338 |
- /*
- * GRUB -- GRand Unified Bootloader
- * Copyright (C) 2009,2010 Free Software Foundation, Inc.
- *
- * GRUB is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * GRUB is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
- */
- #include <config.h>
- #include <grub/types.h>
- #include <grub/misc.h>
- #include <grub/emu/misc.h>
- #include <grub/util/misc.h>
- #include <grub/misc.h>
- #include <grub/i18n.h>
- #include <grub/fontformat.h>
- #include <grub/font.h>
- #include <grub/unicode.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #ifndef GRUB_BUILD
- #define _GNU_SOURCE 1
- #pragma GCC diagnostic ignored "-Wmissing-prototypes"
- #pragma GCC diagnostic ignored "-Wmissing-declarations"
- #include <argp.h>
- #pragma GCC diagnostic error "-Wmissing-prototypes"
- #pragma GCC diagnostic error "-Wmissing-declarations"
- #endif
- #include <assert.h>
- #include <errno.h>
- #include <ft2build.h>
- #include FT_FREETYPE_H
- #include FT_TRUETYPE_TAGS_H
- #include FT_TRUETYPE_TABLES_H
- #include FT_SYNTHESIS_H
- #undef __FTERRORS_H__
- #define FT_ERROR_START_LIST const char *ft_errmsgs[] = {
- #define FT_ERRORDEF(e, v, s) [e] = s,
- #define FT_ERROR_END_LIST };
- #include FT_ERRORS_H
- #ifndef GRUB_BUILD
- #include "progname.h"
- #endif
- #ifdef GRUB_BUILD
- #define grub_util_fopen fopen
- #endif
- #define GRUB_FONT_DEFAULT_SIZE 16
- #define GRUB_FONT_RANGE_BLOCK 1024
- #define GSUB_PTR_VALID(x, end) assert ((grub_uint8_t *) (x) <= (end))
- #define GSUB_ARRAY_SIZE_VALID(a, sz, end) \
- assert ((sz) >= 0 && ((sz) <= ((end) - (grub_uint8_t *) (a)) / sizeof (*(a))))
- struct grub_glyph_info
- {
- struct grub_glyph_info *next;
- grub_uint32_t char_code;
- int width;
- int height;
- int x_ofs;
- int y_ofs;
- int device_width;
- int bitmap_size;
- grub_uint8_t *bitmap;
- };
- enum file_formats
- {
- PF2
- };
- enum
- {
- GRUB_FONT_FLAG_BOLD = 1,
- GRUB_FONT_FLAG_NOBITMAP = 2,
- GRUB_FONT_FLAG_NOHINTING = 4,
- GRUB_FONT_FLAG_FORCEHINT = 8
- };
- struct grub_font_info
- {
- const char *name;
- int style;
- int desc;
- int asce;
- int size;
- int max_width;
- int max_height;
- int min_y;
- int max_y;
- int flags;
- int num_range;
- grub_uint32_t *ranges;
- struct grub_glyph_info *glyphs_unsorted;
- struct grub_glyph_info *glyphs_sorted;
- int num_glyphs;
- };
- static int font_verbosity;
- static void
- add_pixel (grub_uint8_t **data, int *mask, int not_blank)
- {
- if (*mask == 0)
- {
- (*data)++;
- **data = 0;
- *mask = 128;
- }
- if (not_blank)
- **data |= *mask;
- *mask >>= 1;
- }
- static void
- add_glyph (struct grub_font_info *font_info, FT_UInt glyph_idx, FT_Face face,
- grub_uint32_t char_code, int nocut)
- {
- struct grub_glyph_info *glyph_info;
- int width, height;
- int cuttop, cutbottom, cutleft, cutright;
- grub_uint8_t *data;
- int mask, i, j, bitmap_size;
- FT_GlyphSlot glyph;
- int flag = FT_LOAD_RENDER | FT_LOAD_MONOCHROME;
- FT_Error err;
- if (font_info->flags & GRUB_FONT_FLAG_NOBITMAP)
- flag |= FT_LOAD_NO_BITMAP;
- if (font_info->flags & GRUB_FONT_FLAG_NOHINTING)
- flag |= FT_LOAD_NO_HINTING;
- else if (font_info->flags & GRUB_FONT_FLAG_FORCEHINT)
- flag |= FT_LOAD_FORCE_AUTOHINT;
- err = FT_Load_Glyph (face, glyph_idx, flag);
- if (err)
- {
- printf (_("Freetype Error %d loading glyph 0x%x for U+0x%x%s"),
- err, glyph_idx, char_code & GRUB_FONT_CODE_CHAR_MASK,
- char_code & GRUB_FONT_CODE_RIGHT_JOINED
- /* TRANSLATORS: These qualifiers are used for cursive typography,
- mainly Arabic. Note that the terms refer to the visual position
- and not logical order and if used in left-to-right script then
- leftmost is initial but with right-to-left script like Arabic
- rightmost is the initial. */
- ? ((char_code & GRUB_FONT_CODE_LEFT_JOINED) ? _(" (medial)"):
- _(" (leftmost)"))
- : ((char_code & GRUB_FONT_CODE_LEFT_JOINED) ? _(" (rightmost)"):
- ""));
- if (err > 0 && err < (signed) ARRAY_SIZE (ft_errmsgs))
- printf (": %s\n", ft_errmsgs[err]);
- else
- printf ("\n");
- return;
- }
- glyph = face->glyph;
- if (font_info->flags & GRUB_FONT_FLAG_BOLD)
- FT_GlyphSlot_Embolden (glyph);
- if (nocut)
- cuttop = cutbottom = cutleft = cutright = 0;
- else
- {
- for (cuttop = 0; cuttop < glyph->bitmap.rows; cuttop++)
- {
- for (j = 0; j < glyph->bitmap.width; j++)
- if (glyph->bitmap.buffer[j / 8 + cuttop * glyph->bitmap.pitch]
- & (1 << (7 - (j & 7))))
- break;
- if (j != glyph->bitmap.width)
- break;
- }
- for (cutbottom = glyph->bitmap.rows - 1; cutbottom >= 0; cutbottom--)
- {
- for (j = 0; j < glyph->bitmap.width; j++)
- if (glyph->bitmap.buffer[j / 8 + cutbottom * glyph->bitmap.pitch]
- & (1 << (7 - (j & 7))))
- break;
- if (j != glyph->bitmap.width)
- break;
- }
- cutbottom = glyph->bitmap.rows - 1 - cutbottom;
- if (cutbottom + cuttop >= glyph->bitmap.rows)
- cutbottom = 0;
- for (cutleft = 0; cutleft < glyph->bitmap.width; cutleft++)
- {
- for (j = 0; j < glyph->bitmap.rows; j++)
- if (glyph->bitmap.buffer[cutleft / 8 + j * glyph->bitmap.pitch]
- & (1 << (7 - (cutleft & 7))))
- break;
- if (j != glyph->bitmap.rows)
- break;
- }
- for (cutright = glyph->bitmap.width - 1; cutright >= 0; cutright--)
- {
- for (j = 0; j < glyph->bitmap.rows; j++)
- if (glyph->bitmap.buffer[cutright / 8 + j * glyph->bitmap.pitch]
- & (1 << (7 - (cutright & 7))))
- break;
- if (j != glyph->bitmap.rows)
- break;
- }
- cutright = glyph->bitmap.width - 1 - cutright;
- if (cutright + cutleft >= glyph->bitmap.width)
- cutright = 0;
- }
- width = glyph->bitmap.width - cutleft - cutright;
- height = glyph->bitmap.rows - cutbottom - cuttop;
- bitmap_size = ((width * height + 7) / 8);
- glyph_info = xmalloc (sizeof (struct grub_glyph_info));
- glyph_info->bitmap = xmalloc (bitmap_size);
- glyph_info->bitmap_size = bitmap_size;
- glyph_info->next = font_info->glyphs_unsorted;
- font_info->glyphs_unsorted = glyph_info;
- font_info->num_glyphs++;
- glyph_info->char_code = char_code;
- glyph_info->width = width;
- glyph_info->height = height;
- glyph_info->x_ofs = glyph->bitmap_left + cutleft;
- glyph_info->y_ofs = glyph->bitmap_top - height - cuttop;
- glyph_info->device_width = glyph->metrics.horiAdvance / 64;
- if (width > font_info->max_width)
- font_info->max_width = width;
- if (height > font_info->max_height)
- font_info->max_height = height;
- if (glyph_info->y_ofs < font_info->min_y && glyph_info->y_ofs > -font_info->size)
- font_info->min_y = glyph_info->y_ofs;
- if (glyph_info->y_ofs + height > font_info->max_y)
- font_info->max_y = glyph_info->y_ofs + height;
- mask = 0;
- data = &glyph_info->bitmap[0] - 1;
- for (j = cuttop; j < height + cuttop; j++)
- for (i = cutleft; i < width + cutleft; i++)
- add_pixel (&data, &mask,
- glyph->bitmap.buffer[i / 8 + j * glyph->bitmap.pitch] &
- (1 << (7 - (i & 7))));
- }
- struct glyph_replace *subst_rightjoin, *subst_leftjoin, *subst_medijoin;
- struct glyph_replace
- {
- struct glyph_replace *next;
- grub_uint32_t from, to;
- };
- /* TODO: sort glyph_replace and use binary search if necessary. */
- static void
- add_char (struct grub_font_info *font_info, FT_Face face,
- grub_uint32_t char_code, int nocut)
- {
- FT_UInt glyph_idx;
- struct glyph_replace *cur;
- glyph_idx = FT_Get_Char_Index (face, char_code);
- if (!glyph_idx)
- return;
- add_glyph (font_info, glyph_idx, face, char_code, nocut);
- for (cur = subst_rightjoin; cur; cur = cur->next)
- if (cur->from == glyph_idx)
- {
- add_glyph (font_info, cur->to, face,
- char_code | GRUB_FONT_CODE_RIGHT_JOINED, nocut);
- break;
- }
- if (!cur && char_code >= GRUB_UNICODE_ARABIC_START
- && char_code < GRUB_UNICODE_ARABIC_END)
- {
- int i;
- for (i = 0; grub_unicode_arabic_shapes[i].code; i++)
- if (grub_unicode_arabic_shapes[i].code == char_code
- && grub_unicode_arabic_shapes[i].right_linked)
- {
- FT_UInt idx2;
- idx2 = FT_Get_Char_Index (face, grub_unicode_arabic_shapes[i]
- .right_linked);
- if (idx2)
- add_glyph (font_info, idx2, face,
- char_code | GRUB_FONT_CODE_RIGHT_JOINED, nocut);
- break;
- }
- }
- for (cur = subst_leftjoin; cur; cur = cur->next)
- if (cur->from == glyph_idx)
- {
- add_glyph (font_info, cur->to, face,
- char_code | GRUB_FONT_CODE_LEFT_JOINED, nocut);
- break;
- }
- if (!cur && char_code >= GRUB_UNICODE_ARABIC_START
- && char_code < GRUB_UNICODE_ARABIC_END)
- {
- int i;
- for (i = 0; grub_unicode_arabic_shapes[i].code; i++)
- if (grub_unicode_arabic_shapes[i].code == char_code
- && grub_unicode_arabic_shapes[i].left_linked)
- {
- FT_UInt idx2;
- idx2 = FT_Get_Char_Index (face, grub_unicode_arabic_shapes[i]
- .left_linked);
- if (idx2)
- add_glyph (font_info, idx2, face,
- char_code | GRUB_FONT_CODE_LEFT_JOINED, nocut);
- break;
- }
- }
- for (cur = subst_medijoin; cur; cur = cur->next)
- if (cur->from == glyph_idx)
- {
- add_glyph (font_info, cur->to, face,
- char_code | GRUB_FONT_CODE_LEFT_JOINED
- | GRUB_FONT_CODE_RIGHT_JOINED, nocut);
- break;
- }
- if (!cur && char_code >= GRUB_UNICODE_ARABIC_START
- && char_code < GRUB_UNICODE_ARABIC_END)
- {
- int i;
- for (i = 0; grub_unicode_arabic_shapes[i].code; i++)
- if (grub_unicode_arabic_shapes[i].code == char_code
- && grub_unicode_arabic_shapes[i].both_linked)
- {
- FT_UInt idx2;
- idx2 = FT_Get_Char_Index (face, grub_unicode_arabic_shapes[i]
- .both_linked);
- if (idx2)
- add_glyph (font_info, idx2, face,
- char_code | GRUB_FONT_CODE_LEFT_JOINED
- | GRUB_FONT_CODE_RIGHT_JOINED, nocut);
- break;
- }
- }
- }
- struct gsub_header
- {
- grub_uint32_t version;
- grub_uint16_t scripts_off;
- grub_uint16_t features_off;
- grub_uint16_t lookups_off;
- } GRUB_PACKED;
- struct gsub_features
- {
- grub_uint16_t count;
- struct
- {
- #define FEATURE_FINA 0x66696e61
- #define FEATURE_INIT 0x696e6974
- #define FEATURE_MEDI 0x6d656469
- #define FEATURE_AALT 0x61616c74
- #define FEATURE_LIGA 0x6c696761
- #define FEATURE_RLIG 0x726c6967
- grub_uint32_t feature_tag;
- grub_uint16_t offset;
- } GRUB_PACKED features[0];
- } GRUB_PACKED;
- struct gsub_feature
- {
- grub_uint16_t params;
- grub_uint16_t lookupcount;
- grub_uint16_t lookupindices[0];
- } GRUB_PACKED;
- struct gsub_lookup_list
- {
- grub_uint16_t count;
- grub_uint16_t offsets[0];
- } GRUB_PACKED;
- struct gsub_lookup
- {
- grub_uint16_t type;
- grub_uint16_t flag;
- grub_uint16_t subtablecount;
- grub_uint16_t subtables[0];
- } GRUB_PACKED;
- struct gsub_substitution
- {
- grub_uint16_t type;
- grub_uint16_t coverage_off;
- union
- {
- grub_int16_t delta;
- struct
- {
- grub_int16_t count;
- grub_uint16_t repl[0];
- };
- };
- } GRUB_PACKED;
- struct gsub_coverage_list
- {
- grub_uint16_t type;
- grub_uint16_t count;
- grub_uint16_t glyphs[0];
- } GRUB_PACKED;
- struct gsub_coverage_ranges
- {
- grub_uint16_t type;
- grub_uint16_t count;
- struct
- {
- grub_uint16_t start;
- grub_uint16_t end;
- grub_uint16_t start_index;
- } GRUB_PACKED ranges[0];
- } GRUB_PACKED;
- #define GSUB_SINGLE_SUBSTITUTION 1
- #define GSUB_SUBSTITUTION_DELTA 1
- #define GSUB_SUBSTITUTION_MAP 2
- #define GSUB_COVERAGE_LIST 1
- #define GSUB_COVERAGE_RANGE 2
- #define GSUB_RTL_CHAR 1
- static void
- add_subst (grub_uint32_t from, grub_uint32_t to, struct glyph_replace **target)
- {
- struct glyph_replace *new = xmalloc (sizeof (*new));
- new->next = *target;
- new->from = from;
- new->to = to;
- *target = new;
- }
- static void
- subst (const struct gsub_substitution *sub, grub_uint32_t glyph,
- struct glyph_replace **target, int *i)
- {
- grub_uint16_t substtype;
- substtype = grub_be_to_cpu16 (sub->type);
- if (substtype == GSUB_SUBSTITUTION_DELTA)
- add_subst (glyph, glyph + grub_be_to_cpu16 (sub->delta), target);
- else if (*i >= grub_be_to_cpu16 (sub->count))
- printf (_("Out of range substitution (%d, %u)\n"), *i,
- grub_be_to_cpu16 (sub->count));
- else
- add_subst (glyph, grub_be_to_cpu16 (sub->repl[(*i)++]), target);
- }
- static void
- process_cursive (struct gsub_feature *feature,
- struct gsub_lookup_list *lookups,
- grub_uint32_t feattag,
- grub_uint32_t num_glyphs,
- grub_uint8_t *gsub_end)
- {
- int j, k;
- int i;
- int lookup_count = grub_be_to_cpu16 (feature->lookupcount);
- struct glyph_replace **target = NULL;
- struct gsub_substitution *sub;
- GSUB_ARRAY_SIZE_VALID (feature->lookupindices, lookup_count, gsub_end);
- for (j = 0; j < lookup_count; j++)
- {
- int lookup_index = grub_be_to_cpu16 (feature->lookupindices[j]);
- int sub_count;
- struct gsub_lookup *lookup;
- if (lookup_index >= grub_be_to_cpu16 (lookups->count))
- {
- /* TRANSLATORS: "lookup" is taken directly from font specifications
- which are formulated as "Under condition X replace LOOKUP with
- SUBSTITUITION". "*/
- printf (_("Out of range lookup: %d\n"), lookup_index);
- continue;
- }
- lookup = (struct gsub_lookup *)
- ((grub_uint8_t *) lookups
- + grub_be_to_cpu16 (lookups->offsets[lookup_index]));
- if (grub_be_to_cpu16 (lookup->type) != GSUB_SINGLE_SUBSTITUTION)
- {
- printf (_("Unsupported substitution type: %d\n"),
- grub_be_to_cpu16 (lookup->type));
- continue;
- }
- if (grub_be_to_cpu16 (lookup->flag) & ~GSUB_RTL_CHAR)
- {
- grub_util_info ("unsupported substitution flag: 0x%x",
- grub_be_to_cpu16 (lookup->flag));
- }
- switch (feattag)
- {
- case FEATURE_INIT:
- if (grub_be_to_cpu16 (lookup->flag) & GSUB_RTL_CHAR)
- target = &subst_leftjoin;
- else
- target = &subst_rightjoin;
- break;
- case FEATURE_FINA:
- if (grub_be_to_cpu16 (lookup->flag) & GSUB_RTL_CHAR)
- target = &subst_rightjoin;
- else
- target = &subst_leftjoin;
- break;
- case FEATURE_MEDI:
- target = &subst_medijoin;
- break;
- }
- sub_count = grub_be_to_cpu16 (lookup->subtablecount);
- GSUB_ARRAY_SIZE_VALID (lookup->subtables, sub_count, gsub_end);
- for (k = 0; k < sub_count; k++)
- {
- sub = (struct gsub_substitution *)
- ((grub_uint8_t *) lookup + grub_be_to_cpu16 (lookup->subtables[k]));
- grub_uint16_t substtype;
- substtype = grub_be_to_cpu16 (sub->type);
- if (substtype != GSUB_SUBSTITUTION_MAP
- && substtype != GSUB_SUBSTITUTION_DELTA)
- {
- printf (_("Unsupported substitution specification: %u\n"),
- substtype);
- continue;
- }
- void *coverage = (grub_uint8_t *) sub
- + grub_be_to_cpu16 (sub->coverage_off);
- grub_uint32_t covertype;
- covertype = grub_be_to_cpu16 (grub_get_unaligned16 (coverage));
- i = 0;
- if (covertype == GSUB_COVERAGE_LIST)
- {
- struct gsub_coverage_list *cover = coverage;
- int count = grub_be_to_cpu16 (cover->count);
- int l;
- GSUB_ARRAY_SIZE_VALID (cover->glyphs, count, gsub_end);
- for (l = 0; l < count; l++)
- subst (sub, grub_be_to_cpu16 (cover->glyphs[l]), target, &i);
- }
- else if (covertype == GSUB_COVERAGE_RANGE)
- {
- struct gsub_coverage_ranges *cover = coverage;
- int count = grub_be_to_cpu16 (cover->count);
- int l, m;
- GSUB_ARRAY_SIZE_VALID (cover->ranges, count, gsub_end);
- for (l = 0; l < count; l++)
- {
- grub_uint16_t start = grub_be_to_cpu16 (cover->ranges[l].start);
- grub_uint16_t end = grub_be_to_cpu16 (cover->ranges[l].end);
- if (start > end || end > num_glyphs)
- grub_util_error ("%s", _("invalid font range"));
- for (m = start; m <= end; m++)
- subst (sub, m, target, &i);
- }
- }
- else
- /* TRANSLATORS: most font transformations apply only to
- some glyphs. Those glyphs are described as "coverage".
- There are 2 coverage specifications: list and range.
- This warning is thrown when another coverage specification
- is detected. */
- fprintf (stderr,
- _("Unsupported coverage specification: %u\n"), covertype);
- }
- }
- }
- static void
- add_font (struct grub_font_info *font_info, FT_Face face, int nocut)
- {
- struct gsub_header *gsub = NULL;
- FT_ULong gsub_len = 0;
- grub_uint8_t *gsub_end = NULL;
- if (!FT_Load_Sfnt_Table (face, TTAG_GSUB, 0, NULL, &gsub_len))
- {
- gsub = xmalloc (gsub_len);
- if (FT_Load_Sfnt_Table (face, TTAG_GSUB, 0, (void *) gsub, &gsub_len))
- {
- free (gsub);
- gsub = NULL;
- gsub_len = 0;
- }
- }
- gsub_end = (grub_uint8_t *) gsub + gsub_len;
- if (gsub)
- {
- struct gsub_features *features
- = (struct gsub_features *) (((grub_uint8_t *) gsub)
- + grub_be_to_cpu16 (gsub->features_off));
- struct gsub_lookup_list *lookups
- = (struct gsub_lookup_list *) (((grub_uint8_t *) gsub)
- + grub_be_to_cpu16 (gsub->lookups_off));
- int i;
- int nfeatures = grub_be_to_cpu16 (features->count);
- GSUB_PTR_VALID (features, gsub_end);
- GSUB_PTR_VALID (lookups, gsub_end);
- GSUB_ARRAY_SIZE_VALID (features->features, nfeatures, gsub_end);
- for (i = 0; i < nfeatures; i++)
- {
- struct gsub_feature *feature = (struct gsub_feature *)
- ((grub_uint8_t *) features
- + grub_be_to_cpu16 (features->features[i].offset));
- grub_uint32_t feattag
- = grub_be_to_cpu32 (features->features[i].feature_tag);
- GSUB_PTR_VALID (feature, gsub_end);
- if (feature->params)
- fprintf (stderr,
- _("WARNING: unsupported font feature parameters: %x\n"),
- grub_be_to_cpu16 (feature->params));
- switch (feattag)
- {
- /* Used for retrieving all possible variants. Useless in grub. */
- case FEATURE_AALT:
- break;
- /* FIXME: Add ligature support. */
- case FEATURE_LIGA:
- case FEATURE_RLIG:
- break;
- /* Cursive form variants. */
- case FEATURE_FINA:
- case FEATURE_INIT:
- case FEATURE_MEDI:
- process_cursive (feature, lookups, feattag,
- face->num_glyphs, gsub_end);
- break;
- default:
- {
- char str[5];
- int j;
- memcpy (str, &features->features[i].feature_tag,
- sizeof (features->features[i].feature_tag));
- str[4] = 0;
- for (j = 0; j < 4; j++)
- if (!grub_isgraph (str[j]))
- str[j] = '?';
- /* TRANSLATORS: It's gsub feature, not gsub font. */
- grub_util_info ("Unknown gsub font feature 0x%x (%s)",
- feattag, str);
- }
- }
- }
- free (gsub);
- }
- if (font_info->num_range)
- {
- int i;
- grub_uint32_t j;
- for (i = 0; i < font_info->num_range; i++)
- for (j = font_info->ranges[i * 2]; j <= font_info->ranges[i * 2 + 1];
- j++)
- add_char (font_info, face, j, nocut);
- }
- else
- {
- grub_uint32_t char_code, glyph_index;
- for (char_code = FT_Get_First_Char (face, &glyph_index);
- glyph_index;
- char_code = FT_Get_Next_Char (face, char_code, &glyph_index))
- add_char (font_info, face, char_code, nocut);
- }
- }
- static void
- write_string_section (const char *name, const char *str,
- int *offset, FILE *file,
- const char *filename)
- {
- grub_uint32_t leng, leng_be32;
- leng = strlen (str) + 1;
- leng_be32 = grub_cpu_to_be32 (leng);
- grub_util_write_image (name, 4, file, filename);
- grub_util_write_image ((char *) &leng_be32, 4, file, filename);
- grub_util_write_image (str, leng, file, filename);
- *offset += 8 + leng;
- }
- static void
- write_be16_section (const char *name, grub_uint16_t data, int* offset,
- FILE *file, const char *filename)
- {
- grub_uint32_t leng;
- leng = grub_cpu_to_be32_compile_time (2);
- data = grub_cpu_to_be16 (data);
- grub_util_write_image (name, 4, file, filename);
- grub_util_write_image ((char *) &leng, 4, file, filename);
- grub_util_write_image ((char *) &data, 2, file, filename);
- *offset += 10;
- }
- static void
- print_glyphs (struct grub_font_info *font_info)
- {
- int num;
- struct grub_glyph_info *glyph;
- char line[512];
- for (glyph = font_info->glyphs_sorted, num = 0; num < font_info->num_glyphs;
- glyph++, num++)
- {
- int x, y, xmax, xmin, ymax, ymin;
- grub_uint8_t *bitmap, mask;
- printf ("\nGlyph #%d, U+%04x\n", num, glyph->char_code);
- printf ("Width %d, Height %d, X offset %d, Y offset %d, Device width %d\n",
- glyph->width, glyph->height, glyph->x_ofs, glyph->y_ofs,
- glyph->device_width);
- xmax = glyph->x_ofs + glyph->width;
- if (xmax < glyph->device_width)
- xmax = glyph->device_width;
- xmin = glyph->x_ofs;
- if (xmin > 0)
- xmin = 0;
- ymax = glyph->y_ofs + glyph->height;
- if (ymax < font_info->asce)
- ymax = font_info->asce;
- ymin = glyph->y_ofs;
- if (ymin > - font_info->desc)
- ymin = - font_info->desc;
- bitmap = glyph->bitmap;
- mask = 0x80;
- for (y = ymax - 1; y > ymin - 1; y--)
- {
- int line_pos;
- line_pos = 0;
- for (x = xmin; x < xmax; x++)
- {
- if ((x >= glyph->x_ofs) &&
- (x < glyph->x_ofs + glyph->width) &&
- (y >= glyph->y_ofs) &&
- (y < glyph->y_ofs + glyph->height))
- {
- line[line_pos++] = (*bitmap & mask) ? '#' : '_';
- mask >>= 1;
- if (mask == 0)
- {
- mask = 0x80;
- bitmap++;
- }
- }
- else if ((x >= 0) &&
- (x < glyph->device_width) &&
- (y >= - font_info->desc) &&
- (y < font_info->asce))
- {
- line[line_pos++] = ((x == 0) || (y == 0)) ? '+' : '.';
- }
- else
- line[line_pos++] = '*';
- }
- line[line_pos] = 0;
- printf ("%s\n", line);
- }
- }
- }
- static void
- write_font_pf2 (struct grub_font_info *font_info, char *output_file)
- {
- FILE *file;
- grub_uint32_t leng;
- char style_name[20], *font_name, *ptr;
- int offset;
- struct grub_glyph_info *cur;
- file = grub_util_fopen (output_file, "wb");
- if (! file)
- grub_util_error (_("cannot write to `%s': %s"), output_file,
- strerror (errno));
- offset = 0;
- leng = grub_cpu_to_be32_compile_time (4);
- grub_util_write_image (FONT_FORMAT_SECTION_NAMES_FILE,
- sizeof(FONT_FORMAT_SECTION_NAMES_FILE) - 1, file,
- output_file);
- grub_util_write_image ((char *) &leng, 4, file, output_file);
- grub_util_write_image (FONT_FORMAT_PFF2_MAGIC, 4, file, output_file);
- offset += 12;
- if (! font_info->name)
- font_info->name = "Unknown";
- if (font_info->flags & GRUB_FONT_FLAG_BOLD)
- font_info->style |= FT_STYLE_FLAG_BOLD;
- style_name[0] = 0;
- if (font_info->style & FT_STYLE_FLAG_BOLD)
- strcpy (style_name, " Bold");
- if (font_info->style & FT_STYLE_FLAG_ITALIC)
- strcat (style_name, " Italic");
- if (! style_name[0])
- strcpy (style_name, " Regular");
- font_name = xmalloc (strlen (font_info->name) + strlen (&style_name[1])
- + 3 + 20);
- ptr = grub_stpcpy (font_name, font_info->name);
- *ptr++ = ' ';
- ptr = grub_stpcpy (ptr, &style_name[1]);
- *ptr++ = ' ';
- snprintf (ptr, 20, "%d", font_info->size);
- write_string_section (FONT_FORMAT_SECTION_NAMES_FONT_NAME,
- font_name, &offset, file, output_file);
- write_string_section (FONT_FORMAT_SECTION_NAMES_FAMILY,
- font_info->name, &offset, file, output_file);
- write_string_section (FONT_FORMAT_SECTION_NAMES_WEIGHT,
- (font_info->style & FT_STYLE_FLAG_BOLD) ?
- "bold" : "normal",
- &offset, file, output_file);
- write_string_section (FONT_FORMAT_SECTION_NAMES_SLAN,
- (font_info->style & FT_STYLE_FLAG_ITALIC) ?
- "italic" : "normal",
- &offset, file, output_file);
- write_be16_section (FONT_FORMAT_SECTION_NAMES_POINT_SIZE,
- font_info->size, &offset, file, output_file);
- write_be16_section (FONT_FORMAT_SECTION_NAMES_MAX_CHAR_WIDTH,
- font_info->max_width, &offset, file, output_file);
- write_be16_section (FONT_FORMAT_SECTION_NAMES_MAX_CHAR_HEIGHT,
- font_info->max_height, &offset, file, output_file);
- if (! font_info->desc)
- {
- if (font_info->min_y >= 0)
- font_info->desc = 1;
- else
- font_info->desc = - font_info->min_y;
- }
- if (! font_info->asce)
- {
- if (font_info->max_y <= 0)
- font_info->asce = 1;
- else
- font_info->asce = font_info->max_y;
- }
- write_be16_section (FONT_FORMAT_SECTION_NAMES_ASCENT,
- font_info->asce, &offset, file, output_file);
- write_be16_section (FONT_FORMAT_SECTION_NAMES_DESCENT,
- font_info->desc, &offset, file, output_file);
- if (font_verbosity > 0)
- {
- printf ("Font name: %s\n", font_name);
- printf ("Max width: %d\n", font_info->max_width);
- printf ("Max height: %d\n", font_info->max_height);
- printf ("Font ascent: %d\n", font_info->asce);
- printf ("Font descent: %d\n", font_info->desc);
- }
- if (font_verbosity > 0)
- printf ("Number of glyph: %d\n", font_info->num_glyphs);
- leng = grub_cpu_to_be32 (font_info->num_glyphs * 9);
- grub_util_write_image (FONT_FORMAT_SECTION_NAMES_CHAR_INDEX,
- sizeof(FONT_FORMAT_SECTION_NAMES_CHAR_INDEX) - 1,
- file, output_file);
- grub_util_write_image ((char *) &leng, 4, file, output_file);
- offset += 8 + font_info->num_glyphs * 9 + 8;
- for (cur = font_info->glyphs_sorted;
- cur < font_info->glyphs_sorted + font_info->num_glyphs; cur++)
- {
- grub_uint32_t data32;
- grub_uint8_t data8;
- data32 = grub_cpu_to_be32 (cur->char_code);
- grub_util_write_image ((char *) &data32, 4, file, output_file);
- data8 = 0;
- grub_util_write_image ((char *) &data8, 1, file, output_file);
- data32 = grub_cpu_to_be32 (offset);
- grub_util_write_image ((char *) &data32, 4, file, output_file);
- offset += 10 + cur->bitmap_size;
- }
- leng = 0xffffffff;
- grub_util_write_image (FONT_FORMAT_SECTION_NAMES_DATA,
- sizeof(FONT_FORMAT_SECTION_NAMES_DATA) - 1,
- file, output_file);
- grub_util_write_image ((char *) &leng, 4, file, output_file);
- for (cur = font_info->glyphs_sorted;
- cur < font_info->glyphs_sorted + font_info->num_glyphs; cur++)
- {
- grub_uint16_t data;
- data = grub_cpu_to_be16 (cur->width);
- grub_util_write_image ((char *) &data, 2, file, output_file);
- data = grub_cpu_to_be16 (cur->height);
- grub_util_write_image ((char *) &data, 2, file, output_file);
- data = grub_cpu_to_be16 (cur->x_ofs);
- grub_util_write_image ((char *) &data, 2, file, output_file);
- data = grub_cpu_to_be16 (cur->y_ofs);
- grub_util_write_image ((char *) &data, 2, file, output_file);
- data = grub_cpu_to_be16 (cur->device_width);
- grub_util_write_image ((char *) &data, 2, file, output_file);
- grub_util_write_image ((char *) &cur->bitmap[0], cur->bitmap_size,
- file, output_file);
- }
- free (font_name);
- fclose (file);
- }
- #ifndef GRUB_BUILD
- static struct argp_option options[] = {
- {"output", 'o', N_("FILE"), 0, N_("save output in FILE [required]"), 0},
- /* TRANSLATORS: bitmaps are images like e.g. in JPEG. */
- {"index", 'i', N_("NUM"), 0,
- /* TRANSLATORS: some font files may have multiple faces (fonts).
- This option is used to chose among them, the first face being '0'.
- Rarely used. */
- N_("select face index"), 0},
- {"range", 'r', N_("FROM-TO[,FROM-TO]"), 0,
- /* TRANSLATORS: It refers to the range of characters in font. */
- N_("set font range"), 0},
- {"name", 'n', N_("NAME"), 0,
- /* TRANSLATORS: "family name" for font is just a generic name without suffix
- like "Bold". */
- N_("set font family name"), 0},
- {"size", 's', N_("SIZE"), 0, N_("set font size"), 0},
- {"desc", 'd', N_("NUM"), 0, N_("set font descent"), 0},
- {"asce", 'c', N_("NUM"), 0, N_("set font ascent"), 0},
- {"bold", 'b', 0, 0, N_("convert to bold font"), 0},
- {"force-autohint", 'a', 0, 0, N_("force autohint"), 0},
- {"no-hinting", 0x101, 0, 0, N_("disable hinting"), 0},
- {"no-bitmap", 0x100, 0, 0,
- /* TRANSLATORS: some fonts contain bitmap rendering for
- some sizes. This option forces rerendering even if
- pre-rendered bitmap is available.
- */
- N_("ignore bitmap strikes when loading"), 0},
- {"verbose", 'v', 0, 0, N_("print verbose messages."), 0},
- { 0, 0, 0, 0, 0, 0 }
- };
- #define my_argp_parse argp_parse
- #define MY_ARGP_KEY_ARG ARGP_KEY_ARG
- #define my_error_t error_t
- #define MY_ARGP_ERR_UNKNOWN ARGP_ERR_UNKNOWN
- #define my_argp_state argp_state
- #else
- #define my_error_t int
- #define MY_ARGP_ERR_UNKNOWN -1
- #define MY_ARGP_KEY_ARG -1
- #define my_argp_parse(a, argc, argv, b, c, st) my_argp_parse_real(argc, argv, st)
- struct my_argp_state
- {
- void *input;
- };
- #endif
- struct arguments
- {
- struct grub_font_info font_info;
- size_t nfiles;
- size_t files_max;
- char **files;
- char *output_file;
- int font_index;
- int font_size;
- enum file_formats file_format;
- };
- #ifdef GRUB_BUILD
- static int
- has_argument (int v)
- {
- return v =='o' || v == 'i' || v == 'r' || v == 'n' || v == 's'
- || v == 'd' || v == 'c';
- }
- #endif
- static my_error_t
- argp_parser (int key, char *arg, struct my_argp_state *state)
- {
- /* Get the input argument from argp_parse, which we
- know is a pointer to our arguments structure. */
- struct arguments *arguments = state->input;
- switch (key)
- {
- case 'b':
- arguments->font_info.flags |= GRUB_FONT_FLAG_BOLD;
- break;
- case 0x100:
- arguments->font_info.flags |= GRUB_FONT_FLAG_NOBITMAP;
- break;
- case 0x101:
- arguments->font_info.flags |= GRUB_FONT_FLAG_NOHINTING;
- break;
- case 'a':
- arguments->font_info.flags |= GRUB_FONT_FLAG_FORCEHINT;
- break;
- case 'o':
- arguments->output_file = xstrdup (arg);
- break;
- case 'n':
- arguments->font_info.name = xstrdup (arg);
- break;
- case 'i':
- arguments->font_index = strtoul (arg, NULL, 0);
- break;
- case 's':
- arguments->font_size = strtoul (arg, NULL, 0);
- break;
- case 'r':
- {
- char *p = arg;
- while (1)
- {
- grub_uint32_t a, b;
- a = strtoul (p, &p, 0);
- if (*p != '-')
- /* TRANSLATORS: It refers to the range of characters in font. */
- grub_util_error ("%s", _("invalid font range"));
- b = strtoul (p + 1, &p, 0);
- if ((arguments->font_info.num_range
- & (GRUB_FONT_RANGE_BLOCK - 1)) == 0)
- arguments->font_info.ranges = xrealloc (arguments->font_info.ranges,
- (arguments->font_info.num_range +
- GRUB_FONT_RANGE_BLOCK) *
- sizeof (grub_uint32_t) * 2);
- arguments->font_info.ranges[arguments->font_info.num_range * 2] = a;
- arguments->font_info.ranges[arguments->font_info.num_range * 2 + 1] = b;
- arguments->font_info.num_range++;
- if (*p)
- {
- if (*p != ',')
- grub_util_error ("%s", _("invalid font range"));
- p++;
- }
- else
- break;
- }
- break;
- }
- case 'd':
- arguments->font_info.desc = strtoul (arg, NULL, 0);
- break;
- case 'c':
- arguments->font_info.asce = strtoul (arg, NULL, 0);
- break;
- case 'v':
- font_verbosity++;
- break;
- case MY_ARGP_KEY_ARG:
- assert (arguments->nfiles < arguments->files_max);
- arguments->files[arguments->nfiles++] = xstrdup(arg);
- break;
- default:
- return MY_ARGP_ERR_UNKNOWN;
- }
- return 0;
- }
- #ifdef GRUB_BUILD
- /* We don't require host platform to have argp. In the same time configuring
- gnulib for build would result in even worse mess. So we have our
- minimalistic argp replacement just enough for build system. Most
- argp features are omitted. */
- static int
- my_argp_parse_real (int argc, char **argv, void *st)
- {
- int curar;
- struct my_argp_state p;
- p.input = st;
- for (curar = 1; curar < argc; )
- {
- if (argv[curar][0] == '-')
- {
- if (has_argument (argv[curar][1])
- && curar + 1 >= argc)
- return 1;
- if (has_argument (argv[curar][1]))
- {
- if (argp_parser (argv[curar][1], argv[curar + 1], &p))
- return 1;
- curar += 2;
- continue;
- }
- if (argp_parser (argv[curar][1], NULL, &p))
- return 1;
- curar++;
- continue;
- }
- if (argp_parser (MY_ARGP_KEY_ARG, argv[curar], &p))
- return 1;
- curar++;
- }
- return 0;
- }
- #endif
- #ifndef GRUB_BUILD
- static struct argp argp = {
- options, argp_parser, N_("[OPTIONS] FONT_FILES"),
- N_("Convert common font file formats into PF2"),
- NULL, NULL, NULL
- };
- #endif
- int
- main (int argc, char *argv[])
- {
- FT_Library ft_lib;
- struct arguments arguments;
- #ifndef GRUB_BUILD
- grub_util_host_init (&argc, &argv);
- #endif
- memset (&arguments, 0, sizeof (struct arguments));
- arguments.file_format = PF2;
- arguments.files_max = argc + 1;
- arguments.files = xmalloc ((arguments.files_max + 1)
- * sizeof (arguments.files[0]));
- memset (arguments.files, 0, (arguments.files_max + 1)
- * sizeof (arguments.files[0]));
- if (my_argp_parse (&argp, argc, argv, 0, 0, &arguments) != 0)
- {
- fprintf (stderr, "%s", _("Error in parsing command line arguments\n"));
- exit(1);
- }
- if (! arguments.output_file)
- grub_util_error ("%s", _("output file must be specified"));
- if (FT_Init_FreeType (&ft_lib))
- grub_util_error ("%s", _("FT_Init_FreeType fails"));
- {
- size_t i;
- for (i = 0; i < arguments.nfiles; i++)
- {
- FT_Face ft_face;
- int size;
- FT_Error err;
- err = FT_New_Face (ft_lib, arguments.files[i],
- arguments.font_index, &ft_face);
- if (err)
- {
- printf (_("can't open file %s, index %d: error %d"),
- arguments.files[i],
- arguments.font_index, err);
- if (err > 0 && err < (signed) ARRAY_SIZE (ft_errmsgs))
- printf (": %s\n", ft_errmsgs[err]);
- else
- printf ("\n");
- continue;
- }
- if ((! arguments.font_info.name) && (ft_face->family_name))
- arguments.font_info.name = xstrdup (ft_face->family_name);
- size = arguments.font_size;
- if (! size)
- {
- if ((ft_face->face_flags & FT_FACE_FLAG_SCALABLE) ||
- (! ft_face->num_fixed_sizes))
- size = GRUB_FONT_DEFAULT_SIZE;
- else
- size = ft_face->available_sizes[0].height;
- }
- arguments.font_info.style = ft_face->style_flags;
- arguments.font_info.size = size;
- err = FT_Set_Pixel_Sizes (ft_face, size, size);
- if (err)
- grub_util_error (_("can't set %dx%d font size: Freetype error %d: %s"),
- size, size, err,
- (err > 0 && err < (signed) ARRAY_SIZE (ft_errmsgs))
- ? ft_errmsgs[err] : "");
- add_font (&arguments.font_info, ft_face, arguments.file_format != PF2);
- FT_Done_Face (ft_face);
- }
- }
- FT_Done_FreeType (ft_lib);
- {
- int counter[65537];
- struct grub_glyph_info *tmp, *cur;
- int i;
- memset (counter, 0, sizeof (counter));
- for (cur = arguments.font_info.glyphs_unsorted; cur; cur = cur->next)
- counter[(cur->char_code & 0xffff) + 1]++;
- for (i = 0; i < 0x10000; i++)
- counter[i+1] += counter[i];
- tmp = xmalloc (arguments.font_info.num_glyphs
- * sizeof (tmp[0]));
- for (cur = arguments.font_info.glyphs_unsorted; cur; cur = cur->next)
- tmp[counter[(cur->char_code & 0xffff)]++] = *cur;
- memset (counter, 0, sizeof (counter));
- for (cur = tmp; cur < tmp + arguments.font_info.num_glyphs; cur++)
- counter[((cur->char_code & 0xffff0000) >> 16) + 1]++;
- for (i = 0; i < 0x10000; i++)
- counter[i+1] += counter[i];
- arguments.font_info.glyphs_sorted = xmalloc (arguments.font_info.num_glyphs
- * sizeof (arguments.font_info.glyphs_sorted[0]));
- for (cur = tmp; cur < tmp + arguments.font_info.num_glyphs; cur++)
- arguments.font_info.glyphs_sorted[counter[(cur->char_code & 0xffff0000)
- >> 16]++] = *cur;
- free (tmp);
- }
- switch (arguments.file_format)
- {
- case PF2:
- write_font_pf2 (&arguments.font_info, arguments.output_file);
- break;
- }
- if (font_verbosity > 1)
- print_glyphs (&arguments.font_info);
- free (arguments.font_info.glyphs_sorted);
- {
- size_t i;
- for (i = 0; i < arguments.nfiles; i++)
- free (arguments.files[i]);
- }
- free (arguments.files);
- return 0;
- }
|