genode/repos/gems/include/gems/vfs_font.h

172 lines
4.4 KiB
C
Raw Normal View History

/*
* \brief Implementation of 'Text_painter::Font' for VFS-mounted fonts
* \author Norman Feske
* \date 2018-03-26
*/
/*
* Copyright (C) 2018 Genode Labs GmbH
*
* This file is part of the Genode OS framework, which is distributed
* under the terms of the GNU Affero General Public License version 3.
*/
#ifndef _INCLUDE__GEMS__VFS_FONT_T_
#define _INCLUDE__GEMS__VFS_FONT_T_
#include <os/vfs.h>
#include <nitpicker_gfx/text_painter.h>
namespace Genode { class Vfs_font; }
class Genode::Vfs_font : public Text_painter::Font
{
public:
typedef Glyph_painter::Glyph Glyph;
static constexpr Vfs::file_size GLYPH_SLOT_BYTES = 64*1024;
class Glyph_header
{
private:
uint8_t _width = 0;
uint8_t _height = 0;
uint8_t _vpos = 0;
int8_t _advance_decimal = 0;
uint8_t _advance_fractional = 0;
uint8_t _reserved[3] { };
Glyph::Opacity _values[];
float _advance() const
{
float value = 256.0f*_advance_decimal + _advance_fractional;
return value/256;
}
public:
Glyph_header(Glyph const &glyph)
:
_width ((uint8_t)min(255U, glyph.width)),
_height((uint8_t)min(255U, glyph.height)),
_vpos ((uint8_t)min(255U, glyph.vpos)),
_advance_decimal((int8_t)max(-127, min(127, glyph.advance.decimal()))),
_advance_fractional((uint8_t)glyph.advance.value & 0xff)
{ }
Glyph_header() { }
Glyph glyph() const { return Glyph { .width = _width,
.height = _height,
.vpos = _vpos,
.advance = _advance(),
.values = _values }; }
} __attribute__((packed));
private:
typedef Text_painter::Codepoint Codepoint;
typedef Text_painter::Area Area;
typedef Directory::Path Path;
Directory const _font_dir;
unsigned const _baseline;
Area const _bounding_box;
unsigned const _height;
struct Glyph_buffer : Byte_range_ptr
{
Allocator &_alloc;
Glyph_header &header { *(Glyph_header *)start };
static size_t _bytes(Area size) { return sizeof(Glyph_buffer) + size.count()*4; }
Glyph_buffer(Allocator &alloc, Area size)
:
Byte_range_ptr((char *)alloc.alloc(_bytes(size)), _bytes(size)),
_alloc(alloc)
{ }
~Glyph_buffer() { _alloc.free(start, num_bytes); }
};
Glyph_buffer mutable _buffer;
Readonly_file _glyphs_file;
template <typename T, unsigned MAX_LEN = 128>
static T _value_from_file(Directory const &dir, Path const &path,
T const &default_value)
{
T result = default_value;
try {
Readonly_file const file(dir, path);
char buf[MAX_LEN + 1] { };
if (file.read(Byte_range_ptr { buf, sizeof(buf) }) <= MAX_LEN)
if (ascii_to(buf, result))
return result;
}
catch (Readonly_file::Open_failed) { }
return default_value;
}
static Readonly_file::At _file_pos(Codepoint c)
{
return Readonly_file::At{(Vfs::file_size)c.value*GLYPH_SLOT_BYTES};
}
public:
struct Unavailable : Exception { };
/**
* Constructor
*
* \param alloc allocator for glyph buffer
* \param dir directory
* \param path path to font
*
* \throw Unavailable unable to obtain font data
*/
Vfs_font(Allocator &alloc, Directory const &dir, Path const &path)
:
_font_dir(dir, path),
_baseline(_value_from_file(_font_dir, "baseline", 0U)),
_bounding_box(_value_from_file(_font_dir, "max_width", 0U),
_value_from_file(_font_dir, "max_height", 0U)),
_height(_value_from_file(_font_dir, "height", 0U)),
_buffer(alloc, _bounding_box),
_glyphs_file(_font_dir, "glyphs")
{ }
void _apply_glyph(Codepoint c, Apply_fn const &fn) const override
{
_glyphs_file.read(_file_pos(c), _buffer);
fn.apply(_buffer.header.glyph());
}
Advance_info advance_info(Codepoint c) const override
{
Byte_range_ptr header_buffer { _buffer.start, sizeof(Glyph_header) };
_glyphs_file.read(_file_pos(c), header_buffer);
Glyph const glyph = _buffer.header.glyph();
return Advance_info { .width = glyph.width, .advance = glyph.advance };
}
unsigned baseline() const override { return _baseline; }
unsigned height() const override { return _height; }
Area bounding_box() const override { return _bounding_box; }
};
#endif /* _INCLUDE__GEMS__TTF_FONT_T_ */