#include "Font.h" #include #include #include #include #include #include #include #include struct [[gnu::packed]] FontFileHeader { char magic[4]; byte glyph_width; byte glyph_height; byte type; byte is_variable_width; byte unused[6]; char name[64]; }; Font& Font::default_font() { static Font* s_default_font; static const char* default_font_path = "/res/fonts/Katica10.font"; if (!s_default_font) { s_default_font = Font::load_from_file(default_font_path).leak_ref(); ASSERT(s_default_font); } return *s_default_font; } Font& Font::default_fixed_width_font() { static Font* s_default_fixed_width_font; static const char* default_fixed_width_font_path = "/res/fonts/CsillaThin7x10.font"; if (!s_default_fixed_width_font) { s_default_fixed_width_font = Font::load_from_file(default_fixed_width_font_path).leak_ref(); ASSERT(s_default_fixed_width_font); } return *s_default_fixed_width_font; } Font& Font::default_bold_font() { static Font* s_default_bold_font; static const char* default_bold_font_path = "/res/fonts/KaticaBold10.font"; if (!s_default_bold_font) { s_default_bold_font = Font::load_from_file(default_bold_font_path).leak_ref(); ASSERT(s_default_bold_font); } return *s_default_bold_font; } RetainPtr Font::clone() const { size_t bytes_per_glyph = sizeof(dword) * glyph_height(); // FIXME: This is leaked! auto* new_rows = static_cast(kmalloc(bytes_per_glyph * 256)); memcpy(new_rows, m_rows, bytes_per_glyph * 256); auto* new_widths = static_cast(kmalloc(256)); if (m_glyph_widths) memcpy(new_widths, m_glyph_widths, 256); else memset(new_widths, m_glyph_width, 256); return adopt(*new Font(m_name, new_rows, new_widths, m_fixed_width, m_glyph_width, m_glyph_height)); } Font::Font(const String& name, unsigned* rows, byte* widths, bool is_fixed_width, byte glyph_width, byte glyph_height) : m_name(name) , m_rows(rows) , m_glyph_widths(widths) , m_glyph_width(glyph_width) , m_glyph_height(glyph_height) , m_min_glyph_width(glyph_width) , m_max_glyph_width(glyph_width) , m_fixed_width(is_fixed_width) { if (!m_fixed_width) { byte maximum = 0; byte minimum = 255; for (int i = 0; i < 256; ++i) { minimum = min(minimum, m_glyph_widths[i]); maximum = max(maximum, m_glyph_widths[i]); } m_min_glyph_width = minimum; m_max_glyph_width = maximum; } } Font::~Font() { if (m_mmap_ptr) { int rc = munmap(m_mmap_ptr, 4096 * 3); ASSERT(rc == 0); } } RetainPtr Font::load_from_memory(const byte* data) { auto& header = *reinterpret_cast(data); if (memcmp(header.magic, "!Fnt", 4)) { dbgprintf("header.magic != '!Fnt', instead it's '%c%c%c%c'\n", header.magic[0], header.magic[1], header.magic[2], header.magic[3]); return nullptr; } if (header.name[63] != '\0') { dbgprintf("Font name not fully null-terminated\n"); return nullptr; } size_t bytes_per_glyph = sizeof(unsigned) * header.glyph_height; auto* rows = (unsigned*)(data + sizeof(FontFileHeader)); byte* widths = nullptr; if (header.is_variable_width) widths = (byte*)(rows) + 256 * bytes_per_glyph; return adopt(*new Font(String(header.name), rows, widths, !header.is_variable_width, header.glyph_width, header.glyph_height)); } RetainPtr Font::load_from_file(const String& path) { int fd = open(path.characters(), O_RDONLY, 0644); if (fd < 0) { dbgprintf("open(%s) got fd=%d, failed: %s\n", path.characters(), fd, strerror(errno)); perror("open"); return nullptr; } auto* mapped_file = (byte*)mmap(nullptr, 4096 * 3, PROT_READ, MAP_SHARED, fd, 0); if (mapped_file == MAP_FAILED) { int rc = close(fd); ASSERT(rc == 0); return nullptr; } auto font = load_from_memory(mapped_file); font->m_mmap_ptr = mapped_file; int rc = close(fd); ASSERT(rc == 0); return font; } bool Font::write_to_file(const String& path) { int fd = open(path.characters(), O_WRONLY | O_CREAT, 0644); if (fd < 0) { perror("open"); return false; } FontFileHeader header; memset(&header, 0, sizeof(FontFileHeader)); memcpy(header.magic, "!Fnt", 4); header.glyph_width = m_glyph_width; header.glyph_height = m_glyph_height; header.type = 0; header.is_variable_width = !m_fixed_width; memcpy(header.name, m_name.characters(), min(m_name.length(), 63)); size_t bytes_per_glyph = sizeof(unsigned) * m_glyph_height; auto buffer = ByteBuffer::create_uninitialized(sizeof(FontFileHeader) + (256 * bytes_per_glyph) + 256); BufferStream stream(buffer); stream << ByteBuffer::wrap(&header, sizeof(FontFileHeader)); stream << ByteBuffer::wrap(m_rows, (256 * bytes_per_glyph)); stream << ByteBuffer::wrap(m_glyph_widths, 256); ASSERT(stream.at_end()); ssize_t nwritten = write(fd, buffer.pointer(), buffer.size()); ASSERT(nwritten == (ssize_t)buffer.size()); int rc = close(fd); ASSERT(rc == 0); return true; } int Font::width(const String& string) const { if (string.is_empty()) return 0; if (m_fixed_width) return string.length() * m_glyph_width; int width = 0; for (int i = 0; i < string.length(); ++i) width += glyph_width(string[i]) + 1; return width - 1; }