blob: 33bc8cb2d9d1d06b0f1005cf634829f795bc8e63 [file] [log] [blame]
/*
* 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 <stdio.h>
#include <stdlib.h>
#include <string.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
#define GRUB_FONT_DEFAULT_SIZE 16
#define ARRAY_SIZE(array) (sizeof (array) / sizeof (array[0]))
#define MAX_CODE 65536
static unsigned char result[MAX_CODE / 8];
static void
add_glyph (FT_Face face,
unsigned int char_code)
{
int flag = FT_LOAD_RENDER | FT_LOAD_MONOCHROME;
FT_Error err;
FT_UInt glyph_idx;
glyph_idx = FT_Get_Char_Index (face, char_code);
if (!glyph_idx)
return;
err = FT_Load_Glyph (face, glyph_idx, flag);
if (err)
{
printf ("Freetype Error %d loading glyph 0x%x for U+0x%x",
err, glyph_idx, char_code);
if (err > 0 && err < (signed) ARRAY_SIZE (ft_errmsgs))
printf (": %s\n", ft_errmsgs[err]);
else
printf ("\n");
return;
}
if (face->glyph->bitmap.width > 12 && char_code < MAX_CODE)
result[char_code >> 3] |= (1 << (char_code & 7));
}
int
main (int argc, char *argv[])
{
FT_Library ft_lib;
FILE *file;
int i;
if (argc != 3)
{
fprintf (stderr, "grub-gen-widthspec: usage: INPUT OUTPUT");
return 1;
}
if (FT_Init_FreeType (&ft_lib))
{
fprintf (stderr, "grub-gen-widthspec: error: FT_Init_FreeType fails");
return 1;
}
{
FT_Face ft_face;
int size;
FT_Error err;
unsigned int char_code, glyph_index;
err = FT_New_Face (ft_lib, argv[1], 0, &ft_face);
if (err)
{
fprintf (stderr, "can't open file %s, index %d: error %d",
argv[1], 0, err);
if (err > 0 && err < (signed) ARRAY_SIZE (ft_errmsgs))
fprintf (stderr, ": %s\n", ft_errmsgs[err]);
else
fprintf (stderr, "\n");
return 1;
}
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;
if (FT_Set_Pixel_Sizes (ft_face, size, size))
{
fprintf (stderr, "grub-gen-widthspec: error: can't set %dx%d font size", size, size);
return 1;
}
for (char_code = FT_Get_First_Char (ft_face, &glyph_index);
glyph_index;
char_code = FT_Get_Next_Char (ft_face, char_code, &glyph_index))
add_glyph (ft_face, char_code);
FT_Done_Face (ft_face);
}
FT_Done_FreeType (ft_lib);
file = fopen (argv[2], "w");
if (! file)
{
fprintf (stderr, "grub-gen-asciih: error: cannot write to `%s': %s", argv[2],
strerror (errno));
return 1;
}
fprintf (file, "/* THIS CHUNK OF BYTES IS AUTOMATICALLY GENERATED */\n");
fprintf (file, "unsigned char widthspec[] =\n");
fprintf (file, "{\n");
for (i = 0; i < MAX_CODE / 8; i++)
fprintf (file, "0x%02x,%c", result[i], ((i & 0xf) == 0xf) ? '\n' : ' ');
fprintf (file, "};\n");
fclose (file);
return 0;
}