aboutsummaryrefslogtreecommitdiff
path: root/libs/libgui/psf.c
diff options
context:
space:
mode:
Diffstat (limited to 'libs/libgui/psf.c')
-rw-r--r--libs/libgui/psf.c108
1 files changed, 83 insertions, 25 deletions
diff --git a/libs/libgui/psf.c b/libs/libgui/psf.c
index 5a6b1bb..5af04a7 100644
--- a/libs/libgui/psf.c
+++ b/libs/libgui/psf.c
@@ -1,58 +1,116 @@
// MIT License, Copyright (c) 2020 Marvin Borner
// PSF parser
+#include <assert.h>
#include <def.h>
#include <libgui/gfx.h>
#include <libgui/psf.h>
#include <mem.h>
#include <print.h>
-// Verifies the PSF magics
-// Returns the PSF version or 0
-static int psf_verify(char *data)
+static u8 psf_verify(void *data)
{
struct psf1_header *header1 = (struct psf1_header *)data;
struct psf2_header *header2 = (struct psf2_header *)data;
- if (header1->magic[0] == PSF1_MAGIC_0 && header1->magic[1] == PSF1_MAGIC_1)
+ if (header1->magic == PSF1_MAGIC)
return 1;
- else if (header2->magic[0] == PSF2_MAGIC_0 && header2->magic[1] == PSF2_MAGIC_1 &&
- header2->magic[2] == PSF2_MAGIC_2 && header2->magic[3] == PSF2_MAGIC_3)
+ else if (header2->magic == PSF2_MAGIC)
return 2;
else
return 0;
}
-struct gfx_font *psf_parse(char *data)
+void psf_free(struct gfx_font *font)
{
- int version = psf_verify(data);
-
- char *chars;
- int height;
- int width;
- int char_size;
-
- if (version == 1) {
- chars = data + sizeof(struct psf1_header);
- height = ((struct psf1_header *)data)->char_size;
- width = 8;
- char_size = ((struct psf1_header *)data)->char_size;
- } else if (version == 2) {
- chars = data + ((struct psf2_header *)data)->size;
- height = ((struct psf2_header *)data)->height;
- width = ((struct psf2_header *)data)->width;
- char_size = ((struct psf2_header *)data)->char_size;
- } else {
+ free(font->raw);
+ free(font);
+}
+
+// TODO: Improve bruteforce unitab search
+u32 psf_unicode(struct gfx_font *font, u32 needle)
+{
+ if (!font->flags.unicode)
+ return 0;
+
+ u8 *data = font->raw;
+
+ u8 version = psf_verify(data);
+ assert(version == 2);
+
+ struct psf2_header *header = (struct psf2_header *)data;
+
+ u8 *table = (u8 *)(data + header->size + header->char_count * header->char_size);
+
+ u32 glyph = 0;
+ while (table < data + font->total_size) {
+ u32 ch = table[0] & 0xff;
+ if (ch == 0xff) {
+ glyph++;
+ table++;
+ continue;
+ } else if (ch & 128) {
+ if ((ch & 32) == 0) {
+ ch = ((table[0] & 0x1f) << 6) + (table[1] & 0x3f);
+ table++;
+ } else if ((ch & 16) == 0) {
+ ch = ((((table[0] & 0xf) << 6) + (table[1] & 0x3f)) << 6) +
+ (table[2] & 0x3f);
+ table += 2;
+ } else if ((ch & 8) == 0) {
+ ch = ((((((table[0] & 0x7) << 6) + (table[1] & 0x3f)) << 6) +
+ (table[2] & 0x3f))
+ << 6) +
+ (table[3] & 0x3f);
+ table += 3;
+ } else {
+ ch = 0;
+ }
+ }
+
+ if (ch == needle)
+ return glyph;
+
+ table++;
+ }
+
+ return 0;
+}
+
+struct gfx_font *psf_parse(const char *path)
+{
+ struct stat s = { 0 };
+ if (stat(path, &s) != 0 || !s.size)
+ return NULL;
+ u8 *data = malloc(s.size);
+ read(path, data, 0, s.size);
+
+ u8 version = psf_verify(data);
+ if (version == 0) {
print("Unknown font!\n");
return NULL;
+ } else if (version == 1) {
+ log("PSF version 1 is no longer supported\n");
+ return NULL;
}
+ struct psf2_header *header = (struct psf2_header *)data;
+ u8 *chars = data + header->size;
+ u32 height = header->height;
+ u32 width = header->width;
+ u32 char_size = header->char_size;
+
+ if (!(header->flags & PSF2_UNICODE))
+ log("No unicode table found, fonts may not render correctly!\n");
+
struct gfx_font *font = malloc(sizeof(*font));
font->raw = data;
font->chars = chars;
font->size.x = width;
font->size.y = height;
font->char_size = char_size;
+ font->total_size = s.size;
+ font->flags.unicode = (header->flags & PSF2_UNICODE) == PSF2_UNICODE;
return font;
}