Skip to content

Commit

Permalink
subsys: fb: add support for generating CFB font headers at build time
Browse files Browse the repository at this point in the history
Add script and cmake functions for automatically generating Character
Frame Buffer (CFB) font header files from image files, TrueType, or
OpenType font files.

Signed-off-by: Henrik Brix Andersen <[email protected]>
  • Loading branch information
henrikbrixandersen authored and carlescufi committed Nov 28, 2018
1 parent 7a5640f commit 9e8c9ca
Show file tree
Hide file tree
Showing 3 changed files with 230 additions and 0 deletions.
61 changes: 61 additions & 0 deletions cmake/cfb.cmake
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
# These functions can be used to generate a CFB font include file from
# a TrueType/OpenType font file or an image file.
function(generate_cfb_font
input_file # The TrueType/OpenType font file or the image file
output_file # The generated header file
width # Width of the CFB font elements in pixels
height # Height of the CFB font elements in pixels
)
add_custom_command(
OUTPUT ${output_file}
COMMAND
${PYTHON_EXECUTABLE}
${ZEPHYR_BASE}/scripts/gen_cfb_font_header.py
--input ${input_file}
--output ${output_file}
--width ${width}
--height ${height}
${ARGN} # Extra arguments are passed to gen_cfb_font_header.py
DEPENDS ${source_file}
WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
)
endfunction()

function(generate_cfb_font_for_gen_target
target # The cmake target that depends on the generated file
input_file # The TrueType/OpenType font file or the image file
output_file # The generated header file
width # Width of the CFB font elements in pixels
height # Height of the CFB font elements in pixels
gen_target # The generated file target we depend on
# Any additional arguments are passed on to
# gen_cfb_font_header.py
)
generate_cfb_font(${input_file} ${output_file} ${width} ${height} ${ARGN})

# Ensure 'output_file' is generated before 'target' by creating a
# dependency between the two targets

add_dependencies(${target} ${gen_target})
endfunction()

function(generate_cfb_font_for_target
target # The cmake target that depends on the generated file
input_file # The TrueType/OpenType font file or image file
output_file # The generated header file
width # Width of the CFB font elements in pixels
height # Height of the CFB font elements in pixels
# Any additional arguments are passed on to
# gen_cfb_font_header.py
)
# Ensure 'output_file' is generated before 'target' by creating a
# 'custom_target' for it and setting up a dependency between the two
# targets

# But first create a unique name for the custom target
generate_unique_target_name_from_filename(${output_file} generated_target_name)

add_custom_target(${generated_target_name} DEPENDS ${output_file})
generate_cfb_font_for_gen_target(${target} ${input_file} ${output_file}
${width} ${height} ${generated_target_name} ${ARGN})
endfunction()
168 changes: 168 additions & 0 deletions scripts/gen_cfb_font_header.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,168 @@
#!/usr/bin/env python3
#
# Copyright (c) 2018 Henrik Brix Andersen <[email protected]>
#
# SPDX-License-Identifier: Apache-2.0

import argparse
import sys

from PIL import ImageFont
from PIL import Image
from PIL import ImageDraw

PRINTABLE_MIN = 32
PRINTABLE_MAX = 127

def generate_element(image, charcode):
"""Generate CFB font element for a given character code from an image"""
blackwhite = image.convert("1", dither=Image.NONE)
pixels = blackwhite.load()

if args.dump:
blackwhite.save("{}_{}.png".format(args.name, charcode))

if PRINTABLE_MIN <= charcode <= PRINTABLE_MAX:
char = " ({:c})".format(charcode)
else:
char = ""

args.output.write("""\t/* {:d}{} */\n\t{{\n""".format(charcode, char))

for col in range(0, args.width):
args.output.write("\t\t")
for octet in range(0, int(args.height / 8)):
value = ""
for bit in range(0, 8):
row = octet * 8 + bit
if pixels[col, row]:
value = "0" + value
else:
value = "1" + value
args.output.write("0x{:02x},".format(int(value, 2)))
args.output.write("\n")
args.output.write("\t},\n")

def extract_font_glyphs():
"""Extract font glyphs from a TrueType/OpenType font file"""
font = ImageFont.truetype(args.input, args.size)
for i in range(args.first, args.last + 1):
image = Image.new("RGB", (args.width, args.height), (255, 255, 255))
draw = ImageDraw.Draw(image)
draw.text((0, 0), chr(i), (0, 0, 0), font=font)
generate_element(image, i)

def extract_image_glyphs():
"""Extract font glyphs from an image file"""
image = Image.open(args.input)

x_offset = 0
for i in range(args.first, args.last + 1):
glyph = image.crop((x_offset, 0, x_offset + args.width, args.height))
generate_element(glyph, i)
x_offset += args.width

def generate_header():
"""Generate CFB font header file"""
guard = "__CFB_FONT_{:s}_{:d}{:d}_H__".format(args.name.upper(), args.width,
args.height)

args.output.write("""/*
* This file was automatically generated using the following command:
* {cmd}
*
*/
#ifndef {guard}
#define {guard}
#include <zephyr.h>
#include <display/cfb.h>
const u8_t cfb_font_{name:s}_{width:d}{height:d}[{elem:d}][{b:.0f}] = {{\n"""
.format(cmd=" ".join(sys.argv),
guard=guard,
name=args.name,
width=args.width,
height=args.height,
elem=args.last - args.first + 1,
b=args.width / 8 * args.height))

if args.type == "font":
extract_font_glyphs()
elif args.type == "image":
extract_image_glyphs()
elif args.input.name.lower().endswith((".otf", ".otc", ".ttf", ".ttc")):
extract_font_glyphs()
else:
extract_image_glyphs()

args.output.write("""
}};
FONT_ENTRY_DEFINE({name}_{width}{height},
{width},
{height},
CFB_FONT_MONO_VPACKED,
cfb_font_{name}_{width}{height},
{first},
{last}
);
#endif /* {guard} */""" .format(name=args.name, width=args.width,
height=args.height, first=args.first,
last=args.last, guard=guard))

def parse_args():
"""Parse arguments"""
global args
parser = argparse.ArgumentParser(
description="Character Frame Buffer (CFB) font header file generator",
formatter_class=argparse.RawDescriptionHelpFormatter)

parser.add_argument(
"-d", "--dump", action="store_true",
help="dump generated CFB font elements as images for preview")

group = parser.add_argument_group("input arguments")
group.add_argument(
"-i", "--input", required=True, type=argparse.FileType('rb'), metavar="FILE",
help="TrueType/OpenType file or image input file")
group.add_argument(
"-t", "--type", default="auto", choices=["auto", "font", "image"],
help="Input file type (default: %(default)s)")

group = parser.add_argument_group("font arguments")
group.add_argument(
"-s", "--size", type=int, default=10, metavar="POINTS",
help="TrueType/OpenType font size in points (default: %(default)s)")

group = parser.add_argument_group("output arguments")
group.add_argument(
"-o", "--output", type=argparse.FileType('w'), default="-", metavar="FILE",
help="CFB font header file (default: stdout)")
group.add_argument(
"-x", "--width", required=True, type=int,
help="width of the CFB font elements in pixels")
group.add_argument(
"-y", "--height", required=True, type=int, choices=range(8, 128, 8),
help="height of the CFB font elements in pixels")
group.add_argument(
"-n", "--name", default="custom",
help="name of the CFB font entry (default: %(default)s)")
group.add_argument(
"--first", type=int, default=PRINTABLE_MIN, metavar="CHARCODE",
help="character code mapped to the first CFB font element (default: %(default)s)")
group.add_argument(
"--last", type=int, default=PRINTABLE_MAX, metavar="CHARCODE",
help="character code mapped to the last CFB font element (default: %(default)s)")

args = parser.parse_args()

def main():
"""Parse arguments and generate CFB font header file"""
parse_args()
generate_header()

if __name__ == "__main__":
main()
1 change: 1 addition & 0 deletions scripts/requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -16,3 +16,4 @@ pykwalify
# "win32" is used for 64-bit Windows as well
windows-curses; sys_platform == "win32"
colorama
Pillow

0 comments on commit 9e8c9ca

Please sign in to comment.