diff options
author | Marvin Borner | 2020-10-24 23:30:42 +0200 |
---|---|---|
committer | Marvin Borner | 2020-10-24 23:30:42 +0200 |
commit | ffb2c74435ad0e313b7c33ae1f00f02824bb6fc0 (patch) | |
tree | 9200b51e77ea66fb439e15a9bd41287a00cfa654 | |
parent | 37b2fde1efd1527ca1462ab10f56ae049ca3525b (diff) |
Started new GFX/GUI abstraction
-rw-r--r-- | Makefile | 8 | ||||
-rw-r--r-- | apps/mandelbrot.c | 42 | ||||
-rw-r--r-- | apps/window.c | 27 | ||||
-rw-r--r-- | apps/wm.c | 148 | ||||
-rw-r--r-- | kernel/Makefile | 4 | ||||
-rw-r--r-- | kernel/drivers/keyboard.c | 2 | ||||
-rw-r--r-- | libgui/Makefile | 1 | ||||
-rw-r--r-- | libgui/gfx.c | 184 | ||||
-rw-r--r-- | libgui/gui.c | 196 | ||||
-rw-r--r-- | libgui/inc/gfx.h | 91 | ||||
-rw-r--r-- | libgui/inc/gui.h | 84 | ||||
-rw-r--r-- | libgui/psf.c | 2 | ||||
-rwxr-xr-x | run | 2 |
13 files changed, 434 insertions, 357 deletions
@@ -1,5 +1,13 @@ # MIT License, Copyright (c) 2020 Marvin Borner +# Kernel optimization +OPTIMIZATION = -Ofast + +# Remove tree optimizations for kernel - TODO: Why? +CFLAGS_EXTRA = -fno-tree-bit-ccp -fno-tree-builtin-call-dce -fno-tree-ccp -fno-tree-ch -fno-tree-coalesce-vars -fno-tree-copy-prop -fno-tree-dce -fno-tree-dominator-opts -fno-tree-dse -fno-tree-fre -fno-tree-pta -fno-tree-sink -fno-tree-slsr -fno-tree-sra -fno-tree-ter + +export + all: compile clean compile: diff --git a/apps/mandelbrot.c b/apps/mandelbrot.c index 7a7aee1..dad9100 100644 --- a/apps/mandelbrot.c +++ b/apps/mandelbrot.c @@ -2,26 +2,26 @@ #include <conv.h> #include <def.h> -#include <gui.h> +#include <gfx.h> #include <input.h> #include <print.h> #include <random.h> #include <str.h> #include <sys.h> -void draw_pixel(struct window *win, int x, int y, u32 c) +void draw_pixel(struct context *ctx, int x, int y, u32 c) { - int pos = x * (win->bpp >> 3) + y * win->pitch; - win->fb[pos + 0] = GET_BLUE(c); - win->fb[pos + 1] = GET_GREEN(c); - win->fb[pos + 2] = GET_RED(c); - win->fb[pos + 3] = GET_ALPHA(c); + int pos = x * (ctx->bpp >> 3) + y * ctx->pitch; + ctx->fb[pos + 0] = GET_BLUE(c); + ctx->fb[pos + 1] = GET_GREEN(c); + ctx->fb[pos + 2] = GET_RED(c); + ctx->fb[pos + 3] = GET_ALPHA(c); } -void draw_mandelbrot(struct window *win, int resolution) +void draw_mandelbrot(struct context *ctx, int resolution) { - int height = win->height; - int width = win->width; + int height = ctx->height; + int width = ctx->width; int max = resolution; for (int row = 0; row < height; row++) { @@ -38,31 +38,31 @@ void draw_mandelbrot(struct window *win, int resolution) } srand(iteration); if (iteration < max) - draw_pixel(win, col, row, + draw_pixel(ctx, col, row, rand() << 16 | rand() << 8 | rand() | 0xff000000); else - draw_pixel(win, col, row, 0xff000000); + draw_pixel(ctx, col, row, 0xff000000); if (row % 50 == 0 && col == 0) - gui_redraw(); + gfx_redraw(); } } - gui_redraw(); + gfx_redraw(); print("Rendered mandelbrot successfully\n"); yield(); } int main() { - print("[mandelbrot window loaded]\n"); + print("[mandelbrot context loaded]\n"); - struct window win = { 0 }; - win.width = 500; - win.height = 300; - gui_new_window(&win); - gui_fill(&win, COLOR_BG); + struct context ctx = { 0 }; + ctx.width = 500; + ctx.height = 300; + gfx_new_ctx(&ctx); + gfx_fill(&ctx, COLOR_BG); - draw_mandelbrot(&win, 50); + draw_mandelbrot(&ctx, 50); while (1) { yield(); diff --git a/apps/window.c b/apps/window.c index a796b52..70131a6 100644 --- a/apps/window.c +++ b/apps/window.c @@ -2,6 +2,7 @@ #include <conv.h> #include <def.h> +#include <gfx.h> #include <gui.h> #include <input.h> #include <print.h> @@ -9,24 +10,16 @@ int main() { - print("[test window loaded]\n"); + print("[test context loaded]\n"); - struct window win = { 0 }; - win.height = 400; - win.width = 600; - win.x = 50; - win.y = 50; - gui_new_window(&win); + struct window *win = gui_init("test", 0, 0); + struct context *ctx = win->ctx; - gui_fill(&win, COLOR_BG); - /* gui_border(&win, COLOR_FG, 2); */ - - gui_init("/font/spleen-12x24.psfu"); - int font_height = gui_font_height(); - int font_width = gui_font_width(); + int font_height = gfx_font_height(); + int font_width = gfx_font_width(); char *hello = "Hello, world!"; - gui_write(&win, win.width / 2 - (strlen(hello) * font_width) / 2, 0, COLOR_GREEN, hello); + gfx_write(ctx, ctx->width / 2 - (strlen(hello) * font_width) / 2, 0, COLOR_GREEN, hello); struct message *msg; int char_x = 0; @@ -45,7 +38,7 @@ int main() if (!event->press) break; - if (char_x * font_width >= (int)win.width) { + if (char_x * font_width >= (int)ctx->width) { char_y++; char_x = 0; } @@ -58,7 +51,7 @@ int main() } else if (ch == '\b') { if (char_x > 0) { char_x--; - gui_draw_rectangle(&win, font_width * char_x, + gfx_draw_rectangle(ctx, font_width * char_x, font_height * char_y, font_width * (char_x + 1), font_height * (char_y + 1), COLOR_BG); @@ -66,7 +59,7 @@ int main() } else if (ch == ' ' && event->scancode == KEY_SPACE) { char_x++; } else if (ch != ' ' && ch != '\0') { - gui_write_char(&win, font_width * char_x++, font_height * char_y, + gfx_write_char(ctx, font_width * char_x++, font_height * char_y, COLOR_CYAN, ch); } break; @@ -3,7 +3,7 @@ #include <assert.h> #include <cpu.h> #include <def.h> -#include <gui.h> +#include <gfx.h> #include <input.h> #include <keymap.h> #include <list.h> @@ -14,52 +14,52 @@ #include <vesa.h> static int MOUSE_SKIP = 0; // => Every move % n != 0 gets skipped -static int window_count; +static int context_count; static struct vbe vbe; -static struct window direct; // Direct video memory window -static struct window root; // Root window (wallpaper etc.) -static struct window exchange; // Exchange buffer -static struct window cursor; // Cursor bitmap window -static struct window *focused; // The focused window -static struct list *windows; // List of all windows +static struct context direct; // Direct video memory context +static struct context root; // Root context (wallpaper etc.) +static struct context exchange; // Exchange buffer +static struct context cursor; // Cursor bitmap context +static struct context *focused; // The focused context +static struct list *contexts; // List of all contexts static struct keymap *keymap; static int mouse_x = 0; static int mouse_y = 0; -static struct window *new_window(struct window *win, u32 pid, int x, int y, u16 width, u16 height, - int flags) +static struct context *new_context(struct context *ctx, u32 pid, int x, int y, u16 width, + u16 height, int flags) { - win->pid = pid; - win->x = x; - win->y = y; - win->width = width; - win->height = height; - win->bpp = vbe.bpp; - win->pitch = win->width * (win->bpp >> 3); - win->fb = malloc(height * win->pitch); - memset(win->fb, 0, height * win->pitch); - win->flags = flags; - window_count++; - if (window_count % 2 == 1) + ctx->pid = pid; + ctx->x = x; + ctx->y = y; + ctx->width = width; + ctx->height = height; + ctx->bpp = vbe.bpp; + ctx->pitch = ctx->width * (ctx->bpp >> 3); + ctx->fb = malloc(height * ctx->pitch); + memset(ctx->fb, 0, height * ctx->pitch); + ctx->flags = flags; + context_count++; + if (context_count % 2 == 1) MOUSE_SKIP++; - return win; + return ctx; } -static struct window *window_at(int x, int y) +static struct context *context_at(int x, int y) { - if (!windows->head || !windows->head->data) + if (!contexts->head || !contexts->head->data) return NULL; - struct window *ret = NULL; - struct node *iterator = windows->head; + struct context *ret = NULL; + struct node *iterator = contexts->head; while (iterator != NULL) { - struct window *win = iterator->data; - if (win != &root && x >= win->x && x <= win->x + (int)win->width && y >= win->y && - y <= win->y + (int)win->height) - ret = win; + struct context *ctx = iterator->data; + if (ctx != &root && x >= ctx->x && x <= ctx->x + (int)ctx->width && y >= ctx->y && + y <= ctx->y + (int)ctx->height) + ret = ctx; iterator = iterator->next; } return ret; @@ -67,24 +67,24 @@ static struct window *window_at(int x, int y) static void redraw_all() { - if (!windows->head || !windows->head->data) + if (!contexts->head || !contexts->head->data) return; - struct node *iterator = windows->head; + struct node *iterator = contexts->head; while (iterator != NULL) { - struct window *win = iterator->data; - if (win != focused) - gui_win_on_win(&exchange, win, win->x, win->y); + struct context *ctx = iterator->data; + if (ctx != focused) + gfx_ctx_on_ctx(&exchange, ctx, ctx->x, ctx->y); iterator = iterator->next; } if (focused) - gui_win_on_win(&exchange, focused, focused->x, focused->y); + gfx_ctx_on_ctx(&exchange, focused, focused->x, focused->y); memcpy(direct.fb, exchange.fb, exchange.pitch * exchange.height); } -// TODO: Send relative mouse position event to focused window +// TODO: Send relative mouse position event to focused context static int mouse_skip = 0; static int mouse_pressed[3] = { 0 }; static void handle_mouse(struct event_mouse *event) @@ -107,13 +107,13 @@ static void handle_mouse(struct event_mouse *event) mouse_y = vbe.height - cursor.height - 1; // Restore cursor buffer backup - gui_copy(&direct, &exchange, cursor.x, cursor.y, cursor.width, cursor.height); + gfx_copy(&direct, &exchange, cursor.x, cursor.y, cursor.width, cursor.height); - // Window focus + // Context focus if (!mouse_pressed[0] && !mouse_pressed[1]) - focused = window_at(mouse_x, mouse_y); + focused = context_at(mouse_x, mouse_y); - // Window position + // Context position if (event->but1 && !mouse_pressed[1]) { mouse_pressed[0] = 1; if (focused && !(focused->flags & WF_NO_DRAG)) { @@ -121,7 +121,7 @@ static void handle_mouse(struct event_mouse *event) focused->y = mouse_y; if (mouse_skip % MOUSE_SKIP == 0) { mouse_skip = 0; - redraw_all(); // TODO: Function to redraw one window + redraw_all(); // TODO: Function to redraw one context } } } else if (mouse_pressed[0]) { @@ -129,7 +129,7 @@ static void handle_mouse(struct event_mouse *event) redraw_all(); } - // Window size + // Context size if (event->but2 && !mouse_pressed[0]) { if (focused && !mouse_pressed[1]) { mouse_x = focused->x + focused->width; @@ -141,7 +141,7 @@ static void handle_mouse(struct event_mouse *event) focused->width = mouse_x - focused->x; if (mouse_y - focused->y > 0) focused->height = mouse_y - focused->y; - redraw_all(); // TODO: Function to redraw one window + redraw_all(); // TODO: Function to redraw one context } mouse_pressed[1] = 1; } else if (mouse_pressed[1]) { @@ -151,7 +151,7 @@ static void handle_mouse(struct event_mouse *event) cursor.x = mouse_x; cursor.y = mouse_y; - gui_win_on_win(&direct, &cursor, cursor.x, cursor.y); + gfx_ctx_on_ctx(&direct, &cursor, cursor.x, cursor.y); mouse_skip++; } @@ -191,26 +191,26 @@ int main(int argc, char **argv) printf("VBE: %dx%d\n", vbe.width, vbe.height); keymap = keymap_parse("/res/keymaps/en.keymap"); - gui_init("/font/spleen-16x32.psfu"); - - windows = list_new(); - new_window(&root, pid, 0, 0, vbe.width, vbe.height, - WF_NO_FOCUS | WF_NO_DRAG | WF_NO_RESIZE); - new_window(&exchange, pid, 0, 0, vbe.width, vbe.height, - WF_NO_FOCUS | WF_NO_DRAG | WF_NO_RESIZE); - new_window(&cursor, pid, 0, 0, 32, 32, WF_NO_FOCUS | WF_NO_RESIZE); + gfx_init("/font/spleen-16x32.psfu"); + + contexts = list_new(); + new_context(&root, pid, 0, 0, vbe.width, vbe.height, + WF_NO_FOCUS | WF_NO_DRAG | WF_NO_RESIZE); + new_context(&exchange, pid, 0, 0, vbe.width, vbe.height, + WF_NO_FOCUS | WF_NO_DRAG | WF_NO_RESIZE); + new_context(&cursor, pid, 0, 0, 32, 32, WF_NO_FOCUS | WF_NO_RESIZE); memcpy(&direct, &root, sizeof(direct)); direct.fb = vbe.fb; - list_add(windows, &root); + list_add(contexts, &root); - gui_fill(&direct, COLOR_BG); - gui_write(&direct, 0, 0, COLOR_FG, "Welcome to Melvix!"); - gui_write(&direct, 0, 32, COLOR_FG, "Loading resources..."); + gfx_fill(&direct, COLOR_BG); + gfx_write(&direct, 0, 0, COLOR_FG, "Welcome to Melvix!"); + gfx_write(&direct, 0, 32, COLOR_FG, "Loading resources..."); - gui_fill(&root, COLOR_BG); - gui_border(&root, COLOR_FG, 2); - gui_load_image(&cursor, "/res/cursor.bmp", 0, 0); - gui_load_wallpaper(&root, "/res/wall.bmp"); + gfx_fill(&root, COLOR_BG); + gfx_border(&root, COLOR_FG, 2); + gfx_load_image(&cursor, "/res/cursor.bmp", 0, 0); + gfx_load_wallpaper(&root, "/res/wall.bmp"); redraw_all(); event_register(EVENT_MOUSE); @@ -224,19 +224,19 @@ int main(int argc, char **argv) } switch (msg->type) { - case WM_NEW_WINDOW: - printf("New window for pid %d\n", msg->src); - struct window *win = msg->data; - int width = win->width ? win->width : 1000; - int height = win->height ? win->height : 800; - int x = win->x ? win->x : vbe.width / 2 - (width / 2); - int y = win->y ? win->y : vbe.height / 2 - (height / 2); - win->pid = msg->src; - new_window(win, msg->src, x, y, width, height, win->flags); - msg_send(msg->src, WM_NEW_WINDOW, win); - list_add(windows, win); - focused = win; + case WM_NEW_CONTEXT: + printf("New context for pid %d\n", msg->src); + struct context *ctx = msg->data; + int width = ctx->width ? ctx->width : 1000; + int height = ctx->height ? ctx->height : 800; + int x = ctx->x ? ctx->x : vbe.width / 2 - (width / 2); + int y = ctx->y ? ctx->y : vbe.height / 2 - (height / 2); + ctx->pid = msg->src; + new_context(ctx, msg->src, x, y, width, height, ctx->flags); + list_add(contexts, ctx); + focused = ctx; redraw_all(); + msg_send(msg->src, WM_NEW_CONTEXT, ctx); break; case WM_REDRAW: redraw_all(); diff --git a/kernel/Makefile b/kernel/Makefile index fcee759..c449cd9 100644 --- a/kernel/Makefile +++ b/kernel/Makefile @@ -23,9 +23,7 @@ LD = ccache ../cross/opt/bin/i686-elf-ld OC = ccache ../cross/opt/bin/i686-elf-objcopy AS = ccache nasm -# TODO: Remove fixflags? -FIXFLAGS = -fno-tree-bit-ccp -fno-tree-builtin-call-dce -fno-tree-ccp -fno-tree-ch -fno-tree-coalesce-vars -fno-tree-copy-prop -fno-tree-dce -fno-tree-dominator-opts -fno-tree-dse -fno-tree-fre -fno-tree-pta -fno-tree-sink -fno-tree-slsr -fno-tree-sra -fno-tree-ter -CFLAGS = -Wall -Wextra -Wno-address-of-packed-member -nostdlib -nostdinc -ffreestanding -fno-builtin -mno-red-zone -mgeneral-regs-only -std=c99 -m32 -pedantic-errors -Wl,-ekernel_main -I../libc/inc/ -Iinc/ -Dkernel $(FIXFLAGS) -Ofast +CFLAGS = -Wall -Wextra -Wno-address-of-packed-member -nostdlib -nostdinc -ffreestanding -fno-builtin -mno-red-zone -mgeneral-regs-only -std=c99 -m32 -pedantic-errors -Wl,-ekernel_main -I../libc/inc/ -Iinc/ -Dkernel $(CFLAGS_EXTRA) $(OPTIMIZATION) ASFLAGS = -f elf32 all: compile diff --git a/kernel/drivers/keyboard.c b/kernel/drivers/keyboard.c index 854ab3b..2bd67ba 100644 --- a/kernel/drivers/keyboard.c +++ b/kernel/drivers/keyboard.c @@ -67,7 +67,7 @@ char keymap[128] = { 0, // Alt key ' ', // Space bar 15, // Caps lock - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // F keys + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // F keys 0, // Num lock 0, // Scroll lock 0, // Home key diff --git a/libgui/Makefile b/libgui/Makefile index 29c8db4..a29a393 100644 --- a/libgui/Makefile +++ b/libgui/Makefile @@ -2,6 +2,7 @@ COBJS = psf.o \ bmp.o \ + gfx.o \ gui.o CC = ccache ../cross/opt/bin/i686-elf-gcc LD = ccache ../cross/opt/bin/i686-elf-ld diff --git a/libgui/gfx.c b/libgui/gfx.c new file mode 100644 index 0000000..30c10e3 --- /dev/null +++ b/libgui/gfx.c @@ -0,0 +1,184 @@ +// MIT License, Copyright (c) 2020 Marvin Borner +// Some GFX functions +// TODO: Better support for bpp < 32 + +#include <assert.h> +#include <bmp.h> +#include <gfx.h> +#include <mem.h> +#include <psf.h> +#include <str.h> +#include <sys.h> +#include <vesa.h> + +struct font *font; + +static void write_char(struct context *ctx, int x, int y, u32 c, char ch) +{ + int bypp = ctx->bpp >> 3; + + int pos = x * bypp + y * ctx->pitch; + char *draw = (char *)&ctx->fb[pos]; + + u32 stride = font->char_size / font->height; + for (int cy = 0; cy < font->height; cy++) { + for (int cx = 0; cx < font->width; cx++) { + u8 bits = font->chars[ch * font->char_size + cy * stride + cx / 8]; + u8 bit = bits >> (7 - cx % 8) & 1; + if (bit) { + draw[bypp * cx] = GET_BLUE(c); + draw[bypp * cx + 1] = GET_GREEN(c); + draw[bypp * cx + 2] = GET_RED(c); + draw[bypp * cx + 3] = GET_ALPHA(c); + } + } + draw += ctx->pitch; + } +} + +static void draw_rectangle(struct context *ctx, int x1, int y1, int x2, int y2, u32 c) +{ + int bypp = ctx->bpp >> 3; + u8 *draw = &ctx->fb[x1 * bypp + y1 * ctx->pitch]; + for (int i = 0; i < y2 - y1; i++) { + for (int j = 0; j < x2 - x1; j++) { + draw[bypp * j] = GET_BLUE(c); + draw[bypp * j + 1] = GET_GREEN(c); + draw[bypp * j + 2] = GET_RED(c); + draw[bypp * j + 3] = GET_ALPHA(c); + } + draw += ctx->pitch; + } +} + +void gfx_write_char(struct context *ctx, int x, int y, u32 c, char ch) +{ + write_char(ctx, x, y, c, ch); + gfx_redraw(); +} + +void gfx_write(struct context *ctx, int x, int y, u32 c, char *text) +{ + for (u32 i = 0; i < strlen(text); i++) { + write_char(ctx, x + i * font->width, y, c, text[i]); + } + gfx_redraw(); +} + +void gfx_load_image(struct context *ctx, char *path, int x, int y) +{ + // TODO: Support x, y + struct bmp *bmp = bmp_load(path); + assert(bmp && bmp->width + x <= ctx->width); + assert(bmp && bmp->height + y <= ctx->height); + + // TODO: Support padding with odd widths + int bypp = bmp->bpp >> 3; + u8 *srcfb = &bmp->data[bypp + (bmp->height - 1) * bmp->pitch]; + u8 *destfb = &ctx->fb[bypp]; + for (u32 cy = 0; cy < bmp->height; cy++) { + memcpy(destfb, srcfb, bmp->pitch); + srcfb -= bmp->pitch; + destfb += ctx->pitch; + } + gfx_redraw(); +} + +void gfx_load_wallpaper(struct context *ctx, char *path) +{ + gfx_load_image(ctx, path, 0, 0); +} + +void gfx_copy(struct context *dest, struct context *src, int x, int y, u32 width, u32 height) +{ + int bypp = dest->bpp >> 3; + u8 *srcfb = &src->fb[x * bypp + y * src->pitch]; + u8 *destfb = &dest->fb[x * bypp + y * dest->pitch]; + for (u32 cy = 0; cy < height; cy++) { + memcpy(destfb, srcfb, width * (dest->bpp >> 3)); + srcfb += src->pitch; + destfb += dest->pitch; + } +} + +// TODO: Support alpha values other than 0x0 and 0xff (blending) +// TODO: Optimize! +void gfx_ctx_on_ctx(struct context *dest, struct context *src, int x, int y) +{ + if (src->width == dest->width && src->height == dest->height && src->x == 0 && + dest->y == 0) { + memcpy(dest->fb, src->fb, dest->pitch * dest->height); + return; + } + + if (src->width > dest->width || src->height > dest->height) + return; + + // TODO: Negative x and y + int bypp = dest->bpp >> 3; + u8 *srcfb = src->fb; + u8 *destfb = &dest->fb[x * bypp + y * dest->pitch]; + for (u32 cy = 0; cy < src->height && cy + y < dest->height; cy++) { + for (u32 cx = 0; cx < src->width && cx + x < dest->width; cx++) { + if (srcfb[bypp * cx + 3]) { + destfb[bypp * cx + 0] = srcfb[bypp * cx + 0]; + destfb[bypp * cx + 1] = srcfb[bypp * cx + 1]; + destfb[bypp * cx + 2] = srcfb[bypp * cx + 2]; + destfb[bypp * cx + 3] = srcfb[bypp * cx + 3]; + } + } + srcfb += src->pitch; + destfb += dest->pitch; + } +} + +void gfx_draw_rectangle(struct context *ctx, int x1, int y1, int x2, int y2, u32 c) +{ + draw_rectangle(ctx, x1, y1, x2, y2, c); + gfx_redraw(); +} + +void gfx_fill(struct context *ctx, u32 c) +{ + draw_rectangle(ctx, 0, 0, ctx->width, ctx->height, c); + gfx_redraw(); +} + +void gfx_border(struct context *ctx, u32 c, u32 width) +{ + if (width <= 0) + return; + + int bypp = ctx->bpp >> 3; + u8 *draw = ctx->fb; + for (u32 i = 0; i < ctx->height; i++) { + for (u32 j = 0; j < ctx->width; j++) { + if (j <= width - 1 || i <= width - 1 || + j - ctx->width + width + 1 <= width || + i - ctx->height + width <= width) { + draw[bypp * j + 0] = GET_BLUE(c); + draw[bypp * j + 1] = GET_GREEN(c); + draw[bypp * j + 2] = GET_RED(c); + draw[bypp * j + 3] = GET_ALPHA(c); + } + } + draw += ctx->pitch; + } + gfx_redraw(); +} + +int gfx_font_height() +{ + return font->height; +} + +int gfx_font_width() +{ + return font->width; +} + +void gfx_init(char *font_path) +{ + font = psf_parse(read(font_path)); + assert(font); +} diff --git a/libgui/gui.c b/libgui/gui.c index 3bbf3d2..8ed79af 100644 --- a/libgui/gui.c +++ b/libgui/gui.c @@ -1,184 +1,50 @@ // MIT License, Copyright (c) 2020 Marvin Borner -// Some GUI functions -// TODO: Better support for bpp < 32 +// Mostly GFX function wrappers -#include <assert.h> -#include <bmp.h> +#include <def.h> +#include <gfx.h> #include <gui.h> #include <mem.h> -#include <psf.h> -#include <str.h> -#include <sys.h> -#include <vesa.h> -struct font *font; +#define MAX_WINDOWS 10 -static void write_char(struct window *win, int x, int y, u32 c, char ch) -{ - int bypp = win->bpp >> 3; - - int pos = x * bypp + y * win->pitch; - char *draw = (char *)&win->fb[pos]; - - u32 stride = font->char_size / font->height; - for (int cy = 0; cy < font->height; cy++) { - for (int cx = 0; cx < font->width; cx++) { - u8 bits = font->chars[ch * font->char_size + cy * stride + cx / 8]; - u8 bit = bits >> (7 - cx % 8) & 1; - if (bit) { - draw[bypp * cx] = GET_BLUE(c); - draw[bypp * cx + 1] = GET_GREEN(c); - draw[bypp * cx + 2] = GET_RED(c); - draw[bypp * cx + 3] = GET_ALPHA(c); - } - } - draw += win->pitch; - } -} - -static void draw_rectangle(struct window *win, int x1, int y1, int x2, int y2, u32 c) -{ - int bypp = win->bpp >> 3; - u8 *draw = &win->fb[x1 * bypp + y1 * win->pitch]; - for (int i = 0; i < y2 - y1; i++) { - for (int j = 0; j < x2 - x1; j++) { - draw[bypp * j] = GET_BLUE(c); - draw[bypp * j + 1] = GET_GREEN(c); - draw[bypp * j + 2] = GET_RED(c); - draw[bypp * j + 3] = GET_ALPHA(c); - } - draw += win->pitch; - } -} - -void gui_write_char(struct window *win, int x, int y, u32 c, char ch) -{ - write_char(win, x, y, c, ch); - gui_redraw(); -} - -void gui_write(struct window *win, int x, int y, u32 c, char *text) -{ - for (u32 i = 0; i < strlen(text); i++) { - write_char(win, x + i * font->width, y, c, text[i]); - } - gui_redraw(); -} - -void gui_load_image(struct window *win, char *path, int x, int y) -{ - // TODO: Support x, y - struct bmp *bmp = bmp_load(path); - assert(bmp && bmp->width + x <= win->width); - assert(bmp && bmp->height + y <= win->height); - - // TODO: Support padding with odd widths - int bypp = bmp->bpp >> 3; - u8 *srcfb = &bmp->data[bypp + (bmp->height - 1) * bmp->pitch]; - u8 *destfb = &win->fb[bypp]; - for (u32 cy = 0; cy < bmp->height; cy++) { - memcpy(destfb, srcfb, bmp->pitch); - srcfb -= bmp->pitch; - destfb += win->pitch; - } - gui_redraw(); -} +u32 window_count = 0; +static struct window windows[MAX_WINDOWS] = { 0 }; -void gui_load_wallpaper(struct window *win, char *path) +struct window *new_window(const char *title, int x, int y, u32 width, u32 height, int flags) { - gui_load_image(win, path, 0, 0); -} + if (window_count + 1 >= MAX_WINDOWS) + return NULL; -void gui_copy(struct window *dest, struct window *src, int x, int y, u32 width, u32 height) -{ - int bypp = dest->bpp >> 3; - u8 *srcfb = &src->fb[x * bypp + y * src->pitch]; - u8 *destfb = &dest->fb[x * bypp + y * dest->pitch]; - for (u32 cy = 0; cy < height; cy++) { - memcpy(destfb, srcfb, width * (dest->bpp >> 3)); - srcfb += src->pitch; - destfb += dest->pitch; - } -} + struct window *win = &windows[window_count + 1]; + win->ctx = malloc(sizeof(*win->ctx)); + win->ctx->x = x > 0 ? x : 50; + win->ctx->y = y > 0 ? y : 50; + win->ctx->width = width > 0 ? width : 600; + win->ctx->height = height > 0 ? height : 400; + win->ctx->flags = flags; + win->title = title; + gfx_new_ctx(windows[window_count + 1].ctx); -// TODO: Support alpha values other than 0x0 and 0xff (blending) -// TODO: Optimize! -void gui_win_on_win(struct window *dest, struct window *src, int x, int y) -{ - if (src->width == dest->width && src->height == dest->height && src->x == 0 && - dest->y == 0) { - memcpy(dest->fb, src->fb, dest->pitch * dest->height); - return; - } + if (!win->ctx->fb) + return NULL; - if (src->width > dest->width || src->height > dest->height) - return; + window_count++; - // TODO: Negative x and y - int bypp = dest->bpp >> 3; - u8 *srcfb = src->fb; - u8 *destfb = &dest->fb[x * bypp + y * dest->pitch]; - for (u32 cy = 0; cy < src->height && cy + y < dest->height; cy++) { - for (u32 cx = 0; cx < src->width && cx + x < dest->width; cx++) { - if (srcfb[bypp * cx + 3]) { - destfb[bypp * cx + 0] = srcfb[bypp * cx + 0]; - destfb[bypp * cx + 1] = srcfb[bypp * cx + 1]; - destfb[bypp * cx + 2] = srcfb[bypp * cx + 2]; - destfb[bypp * cx + 3] = srcfb[bypp * cx + 3]; - } - } - srcfb += src->pitch; - destfb += dest->pitch; - } + return win; } -void gui_draw_rectangle(struct window *win, int x1, int y1, int x2, int y2, u32 c) +struct window *gui_init(const char *title, u32 width, u32 height) { - draw_rectangle(win, x1, y1, x2, y2, c); - gui_redraw(); -} + if (window_count != 0) + return NULL; -void gui_fill(struct window *win, u32 c) -{ - draw_rectangle(win, 0, 0, win->width, win->height, c); - gui_redraw(); -} + struct window *win = new_window(title, 0, 0, width, height, WF_DEFAULT); + if (!win) + return NULL; -void gui_border(struct window *win, u32 c, u32 width) -{ - if (width <= 0) - return; + gfx_fill(win->ctx, COLOR_BG); + gfx_init("/font/spleen-12x24.psfu"); - int bypp = win->bpp >> 3; - u8 *draw = win->fb; - for (u32 i = 0; i < win->height; i++) { - for (u32 j = 0; j < win->width; j++) { - if (j <= width - 1 || i <= width - 1 || - j - win->width + width + 1 <= width || - i - win->height + width <= width) { - draw[bypp * j + 0] = GET_BLUE(c); - draw[bypp * j + 1] = GET_GREEN(c); - draw[bypp * j + 2] = GET_RED(c); - draw[bypp * j + 3] = GET_ALPHA(c); - } - } - draw += win->pitch; - } - gui_redraw(); -} - -int gui_font_height() -{ - return font->height; -} - -int gui_font_width() -{ - return font->width; -} - -void gui_init(char *font_path) -{ - font = psf_parse(read(font_path)); - assert(font); + return win; } diff --git a/libgui/inc/gfx.h b/libgui/inc/gfx.h new file mode 100644 index 0000000..342fe84 --- /dev/null +++ b/libgui/inc/gfx.h @@ -0,0 +1,91 @@ +// MIT License, Copyright (c) 2020 Marvin Borner +// Some GFX functions + +#ifndef GFX_H +#define GFX_H + +#include <def.h> +#include <gui.h> +#include <sys.h> +#include <vesa.h> + +#define GET_ALPHA(color) ((color >> 24) & 0x000000FF) +#define GET_RED(color) ((color >> 16) & 0x000000FF) +#define GET_GREEN(color) ((color >> 8) & 0x000000FF) +#define GET_BLUE(color) ((color >> 0) & 0X000000FF) + +#define COLOR_BLACK 0xff0f0f0f +#define COLOR_RED 0xfff07f7f +#define COLOR_GREEN 0xff7ff088 +#define COLOR_YELLOW 0xffeef07f +#define COLOR_BLUE 0xff7facf0 +#define COLOR_MAGENTA 0xffd67ff0 +#define COLOR_CYAN 0xff7fe7f0 +#define COLOR_WHITE 0xffe9e9e9 +#define COLOR_BRIGHT_BLACK 0xff928374 +#define COLOR_BRIGHT_RED 0xffed9a9a +#define COLOR_BRIGHT_GREEN 0xff9ef0a5 +#define COLOR_BRIGHT_YELLOW 0xffe7e897 +#define COLOR_BRIGHT_BLUE 0xff98b9eb +#define COLOR_BRIGHT_MAGENTA 0xffd196e3 +#define COLOR_BRIGHT_CYAN 0xff94dae0 +#define COLOR_BRIGHT_WHITE 0xffe3e3e3 +#define COLOR_FG COLOR_WHITE +#define COLOR_BG COLOR_BLACK + +#define WF_DEFAULT (0 << 0) +#define WF_NO_FOCUS (1 << 0) +#define WF_NO_DRAG (1 << 1) +#define WF_NO_RESIZE (1 << 2) + +enum message_type { WM_NEW_CONTEXT = EVENT_MAX + 1, WM_REDRAW, WM_KEYBOARD }; + +// Generalized font struct +struct font { + char *chars; + int height; + int width; + int char_size; +}; + +struct context { + u32 pid; + int x; + int y; + u32 width; + u32 height; + u8 *fb; + u32 bpp; + u32 pitch; + int flags; +}; + +struct msg_keyboard { + char ch; + int press; + int scancode; +}; + +void gfx_write_char(struct context *ctx, int x, int y, u32 c, char ch); +void gfx_write(struct context *ctx, int x, int y, u32 c, char *text); +void gfx_load_image(struct context *ctx, char *path, int x, int y); +void gfx_load_wallpaper(struct context *ctx, char *path); +void gfx_copy(struct context *dest, struct context *src, int x, int y, u32 width, u32 height); +void gfx_ctx_on_ctx(struct context *dest, struct context *src, int x, int y); +void gfx_draw_rectangle(struct context *ctx, int x1, int y1, int x2, int y2, u32 c); +void gfx_fill(struct context *ctx, u32 c); +void gfx_border(struct context *ctx, u32 c, u32 width); +void gfx_init(char *font_path); + +int gfx_font_height(); +int gfx_font_width(); + +/** + * Wrappers + */ + +#define gfx_new_ctx(ctx) \ + (msg_send(2, WM_NEW_CONTEXT, (ctx)), (struct context *)msg_receive_loop()->data) +#define gfx_redraw() (msg_send(2, WM_REDRAW, NULL)) // TODO: Partial redraw (optimization) + +#endif diff --git a/libgui/inc/gui.h b/libgui/inc/gui.h index 4186d43..565bcf6 100644 --- a/libgui/inc/gui.h +++ b/libgui/inc/gui.h @@ -1,89 +1,25 @@ // MIT License, Copyright (c) 2020 Marvin Borner -// Some GUI functions +// Mostly GFX function wrappers #ifndef GUI_H #define GUI_H #include <def.h> -#include <sys.h> -#include <vesa.h> +#include <gfx.h> -#define GET_ALPHA(color) ((color >> 24) & 0x000000FF) -#define GET_RED(color) ((color >> 16) & 0x000000FF) -#define GET_GREEN(color) ((color >> 8) & 0x000000FF) -#define GET_BLUE(color) ((color >> 0) & 0X000000FF) +#define MAX_CHILDS 100 -#define COLOR_BLACK 0xff0f0f0f -#define COLOR_RED 0xfff07f7f -#define COLOR_GREEN 0xff7ff088 -#define COLOR_YELLOW 0xffeef07f -#define COLOR_BLUE 0xff7facf0 -#define COLOR_MAGENTA 0xffd67ff0 -#define COLOR_CYAN 0xff7fe7f0 -#define COLOR_WHITE 0xffe9e9e9 -#define COLOR_BRIGHT_BLACK 0xff928374 -#define COLOR_BRIGHT_RED 0xffed9a9a -#define COLOR_BRIGHT_GREEN 0xff9ef0a5 -#define COLOR_BRIGHT_YELLOW 0xffe7e897 -#define COLOR_BRIGHT_BLUE 0xff98b9eb -#define COLOR_BRIGHT_MAGENTA 0xffd196e3 -#define COLOR_BRIGHT_CYAN 0xff94dae0 -#define COLOR_BRIGHT_WHITE 0xffe3e3e3 -#define COLOR_FG COLOR_WHITE -#define COLOR_BG COLOR_BLACK - -#define WF_DEFAULT (0 << 0) -#define WF_NO_FOCUS (1 << 0) -#define WF_NO_DRAG (1 << 1) -#define WF_NO_RESIZE (1 << 2) - -enum message_type { WM_NEW_WINDOW = EVENT_MAX + 1, WM_REDRAW, WM_KEYBOARD }; - -// Generalized font struct -struct font { - char *chars; - int height; - int width; - int char_size; +struct element { + struct context *ctx; }; struct window { - u32 pid; - int x; - int y; - u32 width; - u32 height; - u8 *fb; - u32 bpp; - u32 pitch; - int flags; + const char *title; + struct element *childs[MAX_CHILDS]; + struct context *ctx; }; -struct msg_keyboard { - char ch; - int press; - int scancode; -}; - -void gui_write_char(struct window *win, int x, int y, u32 c, char ch); -void gui_write(struct window *win, int x, int y, u32 c, char *text); -void gui_load_image(struct window *win, char *path, int x, int y); -void gui_load_wallpaper(struct window *win, char *path); -void gui_copy(struct window *dest, struct window *src, int x, int y, u32 width, u32 height); -void gui_win_on_win(struct window *dest, struct window *src, int x, int y); -void gui_draw_rectangle(struct window *win, int x1, int y1, int x2, int y2, u32 c); -void gui_fill(struct window *win, u32 c); -void gui_border(struct window *win, u32 c, u32 width); -void gui_init(char *font_path); - -int gui_font_height(); -int gui_font_width(); - -/** - * Wrappers - */ +// TODO: Remove window return (internal) +struct window *gui_init(const char *title, u32 width, u32 height); -#define gui_new_window(flags) \ - (msg_send(2, WM_NEW_WINDOW, flags), (struct window *)msg_receive_loop()->data) -#define gui_redraw() (msg_send(2, WM_REDRAW, NULL)) // TODO: Partial redraw (optimization) #endif diff --git a/libgui/psf.c b/libgui/psf.c index dbbf032..d6dc9c3 100644 --- a/libgui/psf.c +++ b/libgui/psf.c @@ -2,7 +2,7 @@ // PSF parser #include <def.h> -#include <gui.h> +#include <gfx.h> #include <mem.h> #include <print.h> #include <psf.h> @@ -193,7 +193,7 @@ make_debug() { } make_disasm() { - objdump -drwC -Mintel build/debug.o --visualize-jumps=color | less -R + objdump -drwC -Mintel build/debug.o | less -R #hexdump -C build/kernel.bin | less -R } |