From 519a1de7d9fe809efc4077933fabbe93a8da9439 Mon Sep 17 00:00:00 2001 From: Marvin Borner Date: Thu, 20 May 2021 20:19:05 +0200 Subject: More generic makefile layout --- apps/chess/Makefile | 17 +- apps/chess/chess.c | 252 --------------------- apps/chess/main.c | 252 +++++++++++++++++++++ apps/generic.mk | 17 ++ apps/idle/Makefile | 17 +- apps/idle/idle.c | 16 -- apps/idle/main.c | 16 ++ apps/init/Makefile | 17 +- apps/init/init.c | 15 -- apps/init/main.c | 15 ++ apps/test/Makefile | 17 +- apps/test/main.c | 108 +++++++++ apps/test/test.c | 108 --------- apps/wm/Makefile | 17 +- apps/wm/main.c | 626 ++++++++++++++++++++++++++++++++++++++++++++++++++++ apps/wm/wm.c | 626 ---------------------------------------------------- 16 files changed, 1054 insertions(+), 1082 deletions(-) delete mode 100644 apps/chess/chess.c create mode 100644 apps/chess/main.c create mode 100644 apps/generic.mk delete mode 100644 apps/idle/idle.c create mode 100644 apps/idle/main.c delete mode 100644 apps/init/init.c create mode 100644 apps/init/main.c create mode 100644 apps/test/main.c delete mode 100644 apps/test/test.c create mode 100644 apps/wm/main.c delete mode 100644 apps/wm/wm.c (limited to 'apps') diff --git a/apps/chess/Makefile b/apps/chess/Makefile index e84f65f..bc65e93 100644 --- a/apps/chess/Makefile +++ b/apps/chess/Makefile @@ -1,16 +1,7 @@ # MIT License, Copyright (c) 2021 Marvin Borner -OBJS = chess.o +NAME = chess +OBJS = main.o +LIBS = -lgui -lc -all: $(OBJS) - @mkdir -p $(BUILD)/apps/chess/ - @$(LD) -o $(BUILD)/apps/chess/exec $(LDFLAGS) $^ -lgui -ltxt -lc -ifeq ($(CONFIG_STRIP), true) - @$(ST) --strip-all $(BUILD)/apps/chess/exec -endif - -clean: - @$(RM) -f $(OBJS) - -%.o: %.c - @$(CC) -c $(CFLAGS) $< -o $@ +include ../generic.mk diff --git a/apps/chess/chess.c b/apps/chess/chess.c deleted file mode 100644 index 94739ef..0000000 --- a/apps/chess/chess.c +++ /dev/null @@ -1,252 +0,0 @@ -// MIT License, Copyright (c) 2021 Marvin Borner -// Ugly chess implementation to find the limits of libgui -// Asserts are generally not needed but don't hurt either - -#include -#include -#include - -// Config -#define SIZE 8 -#define TILE 48 -#define WHITE_STARTS 1 -#define DARK_COLOR 0xff946f51 -#define LIGHT_COLOR 0xfff0d9b5 -#define START_FEN "rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1" - -// Pieces -#define NONE 0 -#define KING 1 -#define PAWN 2 -#define KNIGHT 3 -#define BISHOP 5 -#define ROOK 6 -#define QUEEN 7 -#define WHITE 8 -#define BLACK 16 - -// Masks -#define TYPE_MASK 7 -#define BLACK_MASK BLACK -#define WHITE_MASK WHITE -#define COLOR_MASK (WHITE_MASK | BLACK_MASK) - -// Macros -#define COLOR(piece) (piece & COLOR_MASK) -#define IS_COLOR(piece, color) (COLOR(piece) == color) -#define TYPE(piece) (piece & TYPE_MASK) -#define IS_ROOK_OR_QUEEN(piece) ((piece & 6) == 6) -#define IS_BISHOP_OR_QUEEN(piece) ((piece & 5) == 5) -#define IS_SLIDING_PIECE(piece) ((piece & 4) != 0) - -struct piece { - u32 piece; - u32 widget; - char name[8]; - struct { - u8 moved : 1; - // idk - } bits; -}; - -typedef struct piece board[SIZE][SIZE]; - -static u32 win = 0; // Window -static board tiles = { 0 }; // Matrix -static vec2 selected = { -1, -1 }; // Selected tile - -static void load_image(struct piece *tile) -{ - assert(gui_clear(win, tile->widget, GUI_LAYER_FG) == EOK); - - char icon[48] = { 0 }; - snprintf(icon, sizeof(icon), "/icons/chess-%s-%d.png", tile->name, TILE); - enum gfx_filter filter = IS_COLOR(tile->piece, BLACK) ? GFX_FILTER_NONE : GFX_FILTER_INVERT; - - /* assert(gui_fill(win, tile->widget, GUI_LAYER_FG, 0) == EOK); */ - assert(gui_load_image_filter(win, tile->widget, GUI_LAYER_FG, vec2(0, 0), vec2(TILE, TILE), - filter, icon) == EOK); -} - -static void mouseclick(u32 widget_id, vec2 pos) -{ - UNUSED(pos); - - vec2 clicked = vec2(0, 0); - for (u32 x = 0; x < SIZE; x++) - for (u32 y = 0; y < SIZE; y++) - if (tiles[x][y].widget == widget_id) - clicked = vec2(x, y); - - struct piece *clicked_piece = &tiles[clicked.x][clicked.y]; - - if (vec2_eq(clicked, selected)) - return; - - if (selected.x != (u32)-1) { - struct piece *selected_piece = &tiles[selected.x][selected.y]; - - clicked_piece->piece = selected_piece->piece; - selected_piece->piece = 0; - - strlcpy(clicked_piece->name, selected_piece->name, sizeof(clicked_piece->name)); - selected_piece->name[0] = '\0'; - - assert(gui_clear(win, selected_piece->widget, GUI_LAYER_FG) == EOK); - load_image(clicked_piece); - - assert(gui_redraw_window(win) == EOK); - - selected = vec2(-1, -1); - } else if (clicked_piece->piece) { - assert(gui_redraw_widget(win, clicked_piece->widget) == EOK); - selected = clicked; - } -} - -static const char *resolve_name(u32 piece, char buf[8]) -{ - const char *name = NULL; - switch (piece & TYPE_MASK) { - case KING: - name = "king"; - break; - case PAWN: - name = "pawn"; - break; - case KNIGHT: - name = "knight"; - break; - case BISHOP: - name = "bishop"; - break; - case ROOK: - name = "rook"; - break; - case QUEEN: - name = "queen"; - break; - default: - err(1, "Unknown piece %d\n", piece); - } - - strlcpy(buf, name, 8); - - return buf; -} - -static u32 fen_resolve_letter(char ch) -{ - u32 piece = 0; - - switch (ch) { - case 'k': - piece = KING | BLACK; - break; - case 'K': - piece = KING | WHITE; - break; - case 'p': - piece = PAWN | BLACK; - break; - case 'P': - piece = PAWN | WHITE; - break; - case 'n': - piece = KNIGHT | BLACK; - break; - case 'N': - piece = KNIGHT | WHITE; - break; - case 'b': - piece = BISHOP | BLACK; - break; - case 'B': - piece = BISHOP | WHITE; - break; - case 'r': - piece = ROOK | BLACK; - break; - case 'R': - piece = ROOK | WHITE; - break; - case 'q': - piece = QUEEN | BLACK; - break; - case 'Q': - piece = QUEEN | WHITE; - break; - default: - err(1, "Invalid letter (%c)!\n", ch); - } - - return piece; -} - -// TODO: Add more than basic fen support -static void fen_parse(const char *fen) -{ - if (!fen || !*fen) - return; - - u8 x = 0, y = 0; - for (const char *p = fen; p && *p; p++) { - if (*p == ' ') - break; - - if (*p == '/') { - x = 0; - y++; - continue; - } - - if (*p >= '0' && *p <= '9') - continue; - - u32 piece = fen_resolve_letter(*p); - - tiles[x][y].piece = piece; - resolve_name(piece, tiles[x][y].name); - - x++; - } -} - -static void draw_board(void) -{ - for (u8 x = 0; x < 8; x++) { - for (u8 y = 0; y < 8; y++) { - u32 widget = - gui_new_widget(win, vec2(TILE, TILE), vec2(TILE * x, TILE * y)); - assert((signed)widget > 0); - - u8 colored = (x + y + 1) % 2 == 0; -#if !WHITE_STARTS - colored = !colored; -#endif - assert(gui_fill(win, widget, GUI_LAYER_BG, - colored ? DARK_COLOR : LIGHT_COLOR) == EOK); - - struct piece *tile = &tiles[x][y]; - assert(gui_listen_widget(win, widget, GUI_LISTEN_MOUSECLICK, - (u32)mouseclick) == EOK); - - tile->widget = widget; - - if (tile->piece) - load_image(tile); - } - } - - assert(gui_redraw_window(win) == EOK); -} - -int main(void) -{ - assert(gui_new_window(&win) == EOK); - fen_parse(START_FEN); - draw_board(); - - gui_loop(); - return 0; -} diff --git a/apps/chess/main.c b/apps/chess/main.c new file mode 100644 index 0000000..94739ef --- /dev/null +++ b/apps/chess/main.c @@ -0,0 +1,252 @@ +// MIT License, Copyright (c) 2021 Marvin Borner +// Ugly chess implementation to find the limits of libgui +// Asserts are generally not needed but don't hurt either + +#include +#include +#include + +// Config +#define SIZE 8 +#define TILE 48 +#define WHITE_STARTS 1 +#define DARK_COLOR 0xff946f51 +#define LIGHT_COLOR 0xfff0d9b5 +#define START_FEN "rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1" + +// Pieces +#define NONE 0 +#define KING 1 +#define PAWN 2 +#define KNIGHT 3 +#define BISHOP 5 +#define ROOK 6 +#define QUEEN 7 +#define WHITE 8 +#define BLACK 16 + +// Masks +#define TYPE_MASK 7 +#define BLACK_MASK BLACK +#define WHITE_MASK WHITE +#define COLOR_MASK (WHITE_MASK | BLACK_MASK) + +// Macros +#define COLOR(piece) (piece & COLOR_MASK) +#define IS_COLOR(piece, color) (COLOR(piece) == color) +#define TYPE(piece) (piece & TYPE_MASK) +#define IS_ROOK_OR_QUEEN(piece) ((piece & 6) == 6) +#define IS_BISHOP_OR_QUEEN(piece) ((piece & 5) == 5) +#define IS_SLIDING_PIECE(piece) ((piece & 4) != 0) + +struct piece { + u32 piece; + u32 widget; + char name[8]; + struct { + u8 moved : 1; + // idk + } bits; +}; + +typedef struct piece board[SIZE][SIZE]; + +static u32 win = 0; // Window +static board tiles = { 0 }; // Matrix +static vec2 selected = { -1, -1 }; // Selected tile + +static void load_image(struct piece *tile) +{ + assert(gui_clear(win, tile->widget, GUI_LAYER_FG) == EOK); + + char icon[48] = { 0 }; + snprintf(icon, sizeof(icon), "/icons/chess-%s-%d.png", tile->name, TILE); + enum gfx_filter filter = IS_COLOR(tile->piece, BLACK) ? GFX_FILTER_NONE : GFX_FILTER_INVERT; + + /* assert(gui_fill(win, tile->widget, GUI_LAYER_FG, 0) == EOK); */ + assert(gui_load_image_filter(win, tile->widget, GUI_LAYER_FG, vec2(0, 0), vec2(TILE, TILE), + filter, icon) == EOK); +} + +static void mouseclick(u32 widget_id, vec2 pos) +{ + UNUSED(pos); + + vec2 clicked = vec2(0, 0); + for (u32 x = 0; x < SIZE; x++) + for (u32 y = 0; y < SIZE; y++) + if (tiles[x][y].widget == widget_id) + clicked = vec2(x, y); + + struct piece *clicked_piece = &tiles[clicked.x][clicked.y]; + + if (vec2_eq(clicked, selected)) + return; + + if (selected.x != (u32)-1) { + struct piece *selected_piece = &tiles[selected.x][selected.y]; + + clicked_piece->piece = selected_piece->piece; + selected_piece->piece = 0; + + strlcpy(clicked_piece->name, selected_piece->name, sizeof(clicked_piece->name)); + selected_piece->name[0] = '\0'; + + assert(gui_clear(win, selected_piece->widget, GUI_LAYER_FG) == EOK); + load_image(clicked_piece); + + assert(gui_redraw_window(win) == EOK); + + selected = vec2(-1, -1); + } else if (clicked_piece->piece) { + assert(gui_redraw_widget(win, clicked_piece->widget) == EOK); + selected = clicked; + } +} + +static const char *resolve_name(u32 piece, char buf[8]) +{ + const char *name = NULL; + switch (piece & TYPE_MASK) { + case KING: + name = "king"; + break; + case PAWN: + name = "pawn"; + break; + case KNIGHT: + name = "knight"; + break; + case BISHOP: + name = "bishop"; + break; + case ROOK: + name = "rook"; + break; + case QUEEN: + name = "queen"; + break; + default: + err(1, "Unknown piece %d\n", piece); + } + + strlcpy(buf, name, 8); + + return buf; +} + +static u32 fen_resolve_letter(char ch) +{ + u32 piece = 0; + + switch (ch) { + case 'k': + piece = KING | BLACK; + break; + case 'K': + piece = KING | WHITE; + break; + case 'p': + piece = PAWN | BLACK; + break; + case 'P': + piece = PAWN | WHITE; + break; + case 'n': + piece = KNIGHT | BLACK; + break; + case 'N': + piece = KNIGHT | WHITE; + break; + case 'b': + piece = BISHOP | BLACK; + break; + case 'B': + piece = BISHOP | WHITE; + break; + case 'r': + piece = ROOK | BLACK; + break; + case 'R': + piece = ROOK | WHITE; + break; + case 'q': + piece = QUEEN | BLACK; + break; + case 'Q': + piece = QUEEN | WHITE; + break; + default: + err(1, "Invalid letter (%c)!\n", ch); + } + + return piece; +} + +// TODO: Add more than basic fen support +static void fen_parse(const char *fen) +{ + if (!fen || !*fen) + return; + + u8 x = 0, y = 0; + for (const char *p = fen; p && *p; p++) { + if (*p == ' ') + break; + + if (*p == '/') { + x = 0; + y++; + continue; + } + + if (*p >= '0' && *p <= '9') + continue; + + u32 piece = fen_resolve_letter(*p); + + tiles[x][y].piece = piece; + resolve_name(piece, tiles[x][y].name); + + x++; + } +} + +static void draw_board(void) +{ + for (u8 x = 0; x < 8; x++) { + for (u8 y = 0; y < 8; y++) { + u32 widget = + gui_new_widget(win, vec2(TILE, TILE), vec2(TILE * x, TILE * y)); + assert((signed)widget > 0); + + u8 colored = (x + y + 1) % 2 == 0; +#if !WHITE_STARTS + colored = !colored; +#endif + assert(gui_fill(win, widget, GUI_LAYER_BG, + colored ? DARK_COLOR : LIGHT_COLOR) == EOK); + + struct piece *tile = &tiles[x][y]; + assert(gui_listen_widget(win, widget, GUI_LISTEN_MOUSECLICK, + (u32)mouseclick) == EOK); + + tile->widget = widget; + + if (tile->piece) + load_image(tile); + } + } + + assert(gui_redraw_window(win) == EOK); +} + +int main(void) +{ + assert(gui_new_window(&win) == EOK); + fen_parse(START_FEN); + draw_board(); + + gui_loop(); + return 0; +} diff --git a/apps/generic.mk b/apps/generic.mk new file mode 100644 index 0000000..f2bc936 --- /dev/null +++ b/apps/generic.mk @@ -0,0 +1,17 @@ +# MIT License, Copyright (c) 2021 Marvin Borner + +NAME ?= unknown +LIBS ?= -lc + +all: $(OBJS) + @mkdir -p $(BUILD)/apps/$(NAME)/ + @$(LD) -o $(BUILD)/apps/$(NAME)/exec $(LDFLAGS) $^ $(LIBS) +ifeq ($(CONFIG_STRIP), true) + @$(ST) --strip-all $(BUILD)/apps/wm/exec +endif + +clean: + @$(RM) -f $(OBJS) + +%.o: %.c + @$(CC) -c $(CFLAGS) $< -o $@ diff --git a/apps/idle/Makefile b/apps/idle/Makefile index f4f9730..0e65edd 100644 --- a/apps/idle/Makefile +++ b/apps/idle/Makefile @@ -1,16 +1,7 @@ # MIT License, Copyright (c) 2021 Marvin Borner -OBJS = idle.o +NAME = idle +OBJS = main.o +LIBS = -lc -all: $(OBJS) - @mkdir -p $(BUILD)/apps/idle/ - @$(LD) -o $(BUILD)/apps/idle/exec $(LDFLAGS) $^ -lc -ifeq ($(CONFIG_STRIP), true) - @$(ST) --strip-all $(BUILD)/apps/idle/exec -endif - -clean: - @$(RM) -f $(OBJS) - -%.o: %.c - @$(CC) -c $(CFLAGS) $< -o $@ +include ../generic.mk diff --git a/apps/idle/idle.c b/apps/idle/idle.c deleted file mode 100644 index 43186ff..0000000 --- a/apps/idle/idle.c +++ /dev/null @@ -1,16 +0,0 @@ -// MIT License, Copyright (c) 2021 Marvin Borner - -#include - -int main(int argc, char **argv) -{ - UNUSED(argc); - UNUSED(argv); - - // Just chillin' - while (1) - ; - /* __asm__ volatile("hlt"); */ - - return 0; -} diff --git a/apps/idle/main.c b/apps/idle/main.c new file mode 100644 index 0000000..43186ff --- /dev/null +++ b/apps/idle/main.c @@ -0,0 +1,16 @@ +// MIT License, Copyright (c) 2021 Marvin Borner + +#include + +int main(int argc, char **argv) +{ + UNUSED(argc); + UNUSED(argv); + + // Just chillin' + while (1) + ; + /* __asm__ volatile("hlt"); */ + + return 0; +} diff --git a/apps/init/Makefile b/apps/init/Makefile index 68038d0..aa0ac8b 100644 --- a/apps/init/Makefile +++ b/apps/init/Makefile @@ -1,16 +1,7 @@ # MIT License, Copyright (c) 2021 Marvin Borner -OBJS = init.o +NAME = init +OBJS = main.o +LIBS = -lc -all: $(OBJS) - @mkdir -p $(BUILD)/apps/init/ - @$(LD) -o $(BUILD)/apps/init/exec $(LDFLAGS) $^ -lc -ifeq ($(CONFIG_STRIP), true) - @$(ST) --strip-all $(BUILD)/apps/init/exec -endif - -clean: - @$(RM) -f $(OBJS) - -%.o: %.c - @$(CC) -c $(CFLAGS) $< -o $@ +include ../generic.mk diff --git a/apps/init/init.c b/apps/init/init.c deleted file mode 100644 index 44cf461..0000000 --- a/apps/init/init.c +++ /dev/null @@ -1,15 +0,0 @@ -// MIT License, Copyright (c) 2020 Marvin Borner - -#include -#include -#include - -int main(int argc, char **argv) -{ - UNUSED(argc); - UNUSED(argv); - - assert(exec("wm", NULL) == EOK); - - return 0; -} diff --git a/apps/init/main.c b/apps/init/main.c new file mode 100644 index 0000000..44cf461 --- /dev/null +++ b/apps/init/main.c @@ -0,0 +1,15 @@ +// MIT License, Copyright (c) 2020 Marvin Borner + +#include +#include +#include + +int main(int argc, char **argv) +{ + UNUSED(argc); + UNUSED(argv); + + assert(exec("wm", NULL) == EOK); + + return 0; +} diff --git a/apps/test/Makefile b/apps/test/Makefile index e80b54b..b0e05f3 100644 --- a/apps/test/Makefile +++ b/apps/test/Makefile @@ -1,16 +1,7 @@ # MIT License, Copyright (c) 2021 Marvin Borner -OBJS = test.o fuzz.o +NAME = test +OBJS = main.o fuzz.o +LIBS = -lc -all: $(OBJS) - @mkdir -p $(BUILD)/apps/test/ - @$(LD) -o $(BUILD)/apps/test/exec $(LDFLAGS) $^ -lc -ifeq ($(CONFIG_STRIP), true) - @$(ST) --strip-all $(BUILD)/apps/test/exec -endif - -clean: - @$(RM) -f $(OBJS) - -%.o: %.c - @$(CC) -c $(CFLAGS) $< -o $@ +include ../generic.mk diff --git a/apps/test/main.c b/apps/test/main.c new file mode 100644 index 0000000..2101412 --- /dev/null +++ b/apps/test/main.c @@ -0,0 +1,108 @@ +// MIT License, Copyright (c) 2020 Marvin Borner + +#include "test.h" + +#include +#include +#include +#include +#include +#include +#include + +#define a_mag 0x55 +#define b_mag 0x42 + +#define TEST(name) static void test_##name(void) +#define CHECK(exp) pass_or_fail(__FILE__, __LINE__, __func__, #exp, "1", exp); +#define EQUALS(first, second) \ + pass_or_fail(__FILE__, __LINE__, __func__, #first, #second, (first) == (second)); +#define EQUALS_STR(first, second) \ + pass_or_fail(__FILE__, __LINE__, __func__, #first, #second, strcmp((first), (second)) == 0); + +static u32 failed; + +static void pass_or_fail(const char *file_name, int line_num, const char *func, const char *first, + const char *second, int success) +{ + failed += success ? 0 : 1; + log("\x1B[%s\x1B[0m %s:%d: %s: %s == %s\n", success ? "32m[PASS]" : "31m[FAIL]", file_name, + line_num, func, first, second); +} + +TEST(math) +{ + EQUALS(pow(2, 3), 8); + EQUALS(pow(0, 3), 0); + EQUALS(pow(0, 0), 1); +} + +TEST(crypto) +{ + const char *text = "Melvix"; + u32 length = 6; + + EQUALS(crc32(0, text, length), 0x98bb3595); + + const u8 md5_text[16] = { + 0x01, 0xdc, 0xaf, 0x55, 0x2a, 0xe5, 0x7a, 0xf2, + 0xe5, 0xb4, 0x75, 0xac, 0x0f, 0x38, 0x97, 0x9c, + }; + u8 md5_res[16] = { 0 }; + md5(text, length, md5_res); + EQUALS(memcmp(md5_res, md5_text, 16), 0); +} + +TEST(conv) +{ + char buf1[1] = { 0 }; + char buf2[7] = { 0 }; + char buf3[5] = { 0 }; + char buf4[3] = { 0 }; + EQUALS(atoi("42"), 42); + EQUALS_STR(htoa(0x42), "42"); + EQUALS(htoi("42"), 0x42); + EQUALS_STR(itoa(42), "42"); + EQUALS_STR(conv_base(42, buf1, 0, 0), ""); + EQUALS_STR(conv_base(42, buf2, 2, 0), "101010"); + EQUALS_STR(conv_base(424242, buf3, 36, 0), "93ci"); + EQUALS_STR(conv_base(0xffffffff, buf4, 10, 1), "-1"); +} + +TEST(mem) +{ + const char *str0 = ""; + const char *str1 = ""; + const char *str2 = "12345"; + const char *str3 = "12345"; + const char *str4 = "12354"; + EQUALS(memcmp(str4, str2, strlen(str2)), 1); + EQUALS(memcmp(str2, str4, strlen(str2)), -1); + EQUALS(memcmp(str2, str3, strlen(str2)), 0); + EQUALS(memcmp(str0, str1, strlen(str0)), 0); + + char buf[6] = { 0 }; + EQUALS_STR(memcpy(buf, "hallo", 6), "hallo"); + + char buf2[6] = { 0 }; + EQUALS_STR(memset(buf2, 'x', 5), "xxxxx"); +} + +int main(void) +{ + test_math(); + test_crypto(); + test_conv(); + test_mem(); + + /* fuzz(); */ + + if (failed) + log("%d tests failed\n", failed); + else + log("All tests passed\n"); + + boot(SYS_BOOT_SHUTDOWN); + + return 0; +} diff --git a/apps/test/test.c b/apps/test/test.c deleted file mode 100644 index 2101412..0000000 --- a/apps/test/test.c +++ /dev/null @@ -1,108 +0,0 @@ -// MIT License, Copyright (c) 2020 Marvin Borner - -#include "test.h" - -#include -#include -#include -#include -#include -#include -#include - -#define a_mag 0x55 -#define b_mag 0x42 - -#define TEST(name) static void test_##name(void) -#define CHECK(exp) pass_or_fail(__FILE__, __LINE__, __func__, #exp, "1", exp); -#define EQUALS(first, second) \ - pass_or_fail(__FILE__, __LINE__, __func__, #first, #second, (first) == (second)); -#define EQUALS_STR(first, second) \ - pass_or_fail(__FILE__, __LINE__, __func__, #first, #second, strcmp((first), (second)) == 0); - -static u32 failed; - -static void pass_or_fail(const char *file_name, int line_num, const char *func, const char *first, - const char *second, int success) -{ - failed += success ? 0 : 1; - log("\x1B[%s\x1B[0m %s:%d: %s: %s == %s\n", success ? "32m[PASS]" : "31m[FAIL]", file_name, - line_num, func, first, second); -} - -TEST(math) -{ - EQUALS(pow(2, 3), 8); - EQUALS(pow(0, 3), 0); - EQUALS(pow(0, 0), 1); -} - -TEST(crypto) -{ - const char *text = "Melvix"; - u32 length = 6; - - EQUALS(crc32(0, text, length), 0x98bb3595); - - const u8 md5_text[16] = { - 0x01, 0xdc, 0xaf, 0x55, 0x2a, 0xe5, 0x7a, 0xf2, - 0xe5, 0xb4, 0x75, 0xac, 0x0f, 0x38, 0x97, 0x9c, - }; - u8 md5_res[16] = { 0 }; - md5(text, length, md5_res); - EQUALS(memcmp(md5_res, md5_text, 16), 0); -} - -TEST(conv) -{ - char buf1[1] = { 0 }; - char buf2[7] = { 0 }; - char buf3[5] = { 0 }; - char buf4[3] = { 0 }; - EQUALS(atoi("42"), 42); - EQUALS_STR(htoa(0x42), "42"); - EQUALS(htoi("42"), 0x42); - EQUALS_STR(itoa(42), "42"); - EQUALS_STR(conv_base(42, buf1, 0, 0), ""); - EQUALS_STR(conv_base(42, buf2, 2, 0), "101010"); - EQUALS_STR(conv_base(424242, buf3, 36, 0), "93ci"); - EQUALS_STR(conv_base(0xffffffff, buf4, 10, 1), "-1"); -} - -TEST(mem) -{ - const char *str0 = ""; - const char *str1 = ""; - const char *str2 = "12345"; - const char *str3 = "12345"; - const char *str4 = "12354"; - EQUALS(memcmp(str4, str2, strlen(str2)), 1); - EQUALS(memcmp(str2, str4, strlen(str2)), -1); - EQUALS(memcmp(str2, str3, strlen(str2)), 0); - EQUALS(memcmp(str0, str1, strlen(str0)), 0); - - char buf[6] = { 0 }; - EQUALS_STR(memcpy(buf, "hallo", 6), "hallo"); - - char buf2[6] = { 0 }; - EQUALS_STR(memset(buf2, 'x', 5), "xxxxx"); -} - -int main(void) -{ - test_math(); - test_crypto(); - test_conv(); - test_mem(); - - /* fuzz(); */ - - if (failed) - log("%d tests failed\n", failed); - else - log("All tests passed\n"); - - boot(SYS_BOOT_SHUTDOWN); - - return 0; -} diff --git a/apps/wm/Makefile b/apps/wm/Makefile index 8fb0096..556d526 100644 --- a/apps/wm/Makefile +++ b/apps/wm/Makefile @@ -1,16 +1,7 @@ # MIT License, Copyright (c) 2021 Marvin Borner -OBJS = wm.o +NAME = wm +OBJS = main.o +LIBS = -lgui -ltxt -lc -all: $(OBJS) - @mkdir -p $(BUILD)/apps/wm/ - @$(LD) -o $(BUILD)/apps/wm/exec $(LDFLAGS) $^ -lgui -ltxt -lc -ifeq ($(CONFIG_STRIP), true) - @$(ST) --strip-all $(BUILD)/apps/wm/exec -endif - -clean: - @$(RM) -f $(OBJS) - -%.o: %.c - @$(CC) -c $(CFLAGS) $< -o $@ +include ../generic.mk diff --git a/apps/wm/main.c b/apps/wm/main.c new file mode 100644 index 0000000..83dc20b --- /dev/null +++ b/apps/wm/main.c @@ -0,0 +1,626 @@ +// MIT License, Copyright (c) 2020 Marvin Borner +// Window manager / compositor (does everything basically) + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define WINDOW_MOVE_TIMEOUT 20 + +struct client { + u32 conn; // Bus conn +}; + +struct window { + u32 id; + u32 shid; + struct context ctx; + struct client client; + u32 flags; + vec2 pos; + vec2 pos_prev; +}; + +struct rectangle { + vec2 pos1; // Upper left + vec2 pos2; // Lower right + void *data; +}; + +// Global vars ftw! +static u8 bypp = 4; +static struct vbe screen = { 0 }; +static struct list *windows = NULL; // THIS LIST SHALL BE SORTED BY Z-INDEX! +static struct window *direct = NULL; +static struct window *wallpaper = NULL; +static struct window *cursor = NULL; +static struct window *focused = NULL; +static struct keymap *keymap = NULL; +static struct client wm_client = { 0 }; +static struct { + u8 shift : 1; + u8 alt : 1; + u8 ctrl : 1; +} special_keys = { 0 }; +static struct { + vec2 pos; + u8 left : 1; + u8 mid : 1; + u8 right : 1; +} mouse = { 0 }; + +/** + * 5head algorithms + * Thanks to @LarsVomMars for the help + */ + +static void windows_at_rec(vec2 pos1, vec2 pos2, struct list *list) +{ + u32 width = pos2.x - pos1.x; + u32 height = pos2.y - pos1.y; + vec2 rec_corners[] = { + pos1, + vec2_add(pos1, vec2(width, 0)), + pos2, + vec2_add(pos1, vec2(0, height)), + }; + + u8 cursor_found = 0; + + struct node *iterator = windows->head; + while (iterator) { + struct window *win = iterator->data; + + if ((win->flags & WF_NO_WINDOW) != 0) + goto next; + + vec2 corners[] = { + win->pos, + vec2_add(win->pos, vec2(win->ctx.size.x, 0)), + vec2_add(win->pos, vec2(win->ctx.size.x, win->ctx.size.y)), + vec2_add(win->pos, vec2(0, win->ctx.size.y)), + }; + + for (u8 i = 0; i < 4; i++) { + vec2 corner = corners[i]; + if ((pos1.x <= corner.x && pos1.y <= corner.y) && + (pos2.x >= corner.x && pos2.y >= corner.y)) { + if (win == cursor) + cursor_found = 1; + else + list_add(list, win); + goto next; + } + } + + vec2 win_pos1 = win->pos; + vec2 win_pos2 = vec2_add(win->pos, win->ctx.size); + for (u8 i = 0; i < 4; i++) { + vec2 corner = rec_corners[i]; + if ((win_pos1.x <= corner.x && win_pos1.y <= corner.y) && + (win_pos2.x >= corner.x && win_pos2.y >= corner.y)) { + if (win == cursor) + cursor_found = 1; + else + list_add(list, win); + goto next; + } + } + + next: + iterator = iterator->next; + } + + if (cursor_found) + list_add(list, cursor); +} + +static struct rectangle rectangle_at(vec2 pos1, vec2 pos2) +{ + u32 width = ABS(pos2.x - pos1.x); + u32 height = ABS(pos2.y - pos1.y); + u32 pitch = width * bypp; + u8 *data = zalloc(width * height * bypp); + + struct list *windows_at = list_new(); + windows_at_rec(pos1, pos2, windows_at); + struct node *iterator = windows_at->head; + while (iterator) { + struct window *win = iterator->data; + + s32 start_x = win->pos.x - pos1.x; + u32 end_x = width; + if (start_x <= 0) { // Either right side or background + u32 right = start_x + win->ctx.size.x; + if (right <= width) { // Right side + start_x = 0; + end_x = right; + } else { // Background + start_x = 0; + } + } + + s32 start_y = win->pos.y - pos1.y; + u32 end_y = height; + if (start_y <= 0) { // Either bottom side or background + u32 bot = start_y + win->ctx.size.y; + if (bot <= height) { // Bottom side + start_y = 0; + end_y = bot; + } else { // Background + start_y = 0; + } + } + + vec2 pos = vec2_sub(pos1, win->pos); + + u8 *srcfb = + &win->ctx.fb[(pos.x + start_x) * bypp + (pos.y + start_y) * win->ctx.pitch]; + u8 *destfb = &data[start_x * bypp + start_y * pitch]; + + // Copy window data to rectangle buffer + for (u32 cy = start_y; cy < end_y; cy++) { + u32 diff = 0; + for (u32 cx = start_x; cx < end_x; cx++) { + if (srcfb[bypp - 1]) + memcpy(destfb, srcfb, bypp); + + srcfb += bypp; + destfb += bypp; + diff += bypp; + } + srcfb += win->ctx.pitch - diff; + destfb += pitch - diff; + } + + iterator = iterator->next; + } + list_destroy(windows_at); + + return (struct rectangle){ .pos1 = pos1, .pos2 = pos2, .data = data }; +} + +static void rectangle_redraw(vec2 pos1, vec2 pos2) +{ + assert(pos1.x <= pos2.x && pos1.y <= pos2.y); + struct rectangle rec = rectangle_at(pos1, pos2); + + u32 width = ABS(pos2.x - pos1.x); + u32 height = ABS(pos2.y - pos1.y); + + if (!width || !height) + return; + + u8 *srcfb = rec.data; + u8 *destfb = &direct->ctx.fb[rec.pos1.x * bypp + rec.pos1.y * direct->ctx.pitch]; + for (u32 cy = 0; cy < height; cy++) { + memcpy(destfb, srcfb, width * bypp); + srcfb += width * bypp; + destfb += direct->ctx.pitch; + } + + free(rec.data); +} + +/** + * Window operations + */ + +static struct window *window_new(struct client client, struct vec2 pos, struct vec2 size, u32 flags) +{ + assert(windows); + + struct window *win = malloc(sizeof(*win)); + static u32 id = 0; + win->id = id++; + win->ctx.size = size; + win->ctx.bpp = screen.bpp; + win->ctx.pitch = size.x * bypp; + win->ctx.bytes = win->ctx.pitch * win->ctx.size.y; + if ((flags & WF_NO_FB) != 0) { + win->ctx.fb = NULL; + } else { + assert(shalloc(win->ctx.bytes, (u32 *)&win->ctx.fb, &win->shid) == EOK); + memset(win->ctx.fb, COLOR_BLACK, win->ctx.bytes); + } + win->client = client; + win->flags = flags; + win->pos = pos; + win->pos_prev = pos; + list_add(windows, win); + return win; +} + +static struct window *window_find(u32 id) +{ + struct node *iterator = windows->head; + while (iterator) { + struct window *win = iterator->data; + if (win->id == id) + return win; + iterator = iterator->next; + } + return NULL; +} + +static struct window *window_at(vec2 pos) +{ + struct window *ret = NULL; + + struct node *iterator = windows->head; + while (iterator) { + struct window *win = iterator->data; + if (!(win->flags & (WF_NO_WINDOW | WF_NO_FOCUS)) && pos.x >= win->pos.x && + pos.x <= win->pos.x + win->ctx.size.x && pos.y >= win->pos.y && + pos.y <= win->pos.y + win->ctx.size.y) + ret = win; + iterator = iterator->next; + } + return ret; +} + +// Transparent windows can't use efficient rectangle-based redrawing (I think!) +static void window_redraw_alpha(struct window *win) +{ + vec2 pos1 = win->pos_prev; + vec2 pos2 = vec2(pos1.x + win->ctx.size.x, pos1.y + win->ctx.size.y); + + rectangle_redraw(pos1, pos2); + gfx_ctx_on_ctx(&direct->ctx, &win->ctx, win->pos, GFX_ALPHA); +} + +// TODO: Make more efficient... +static void window_redraw_non_alpha(struct window *win) +{ + s32 diff_x = win->pos_prev.x - win->pos.x; + s32 diff_y = win->pos_prev.y - win->pos.y; + + if (!diff_x && !diff_y) { + gfx_ctx_on_ctx(&direct->ctx, &win->ctx, win->pos, GFX_NON_ALPHA); + return; + } + + vec2 pos1 = { 0 }; + vec2 pos2 = { 0 }; + + /** + * Redraw left/right diff rectangle (only one!) + */ + + if (diff_x <= 0) { // Right + pos1.x = win->pos_prev.x; + pos2.x = pos1.x - diff_x + 2; + } else if (diff_x > 0) { // Left + pos1.x = win->pos.x + win->ctx.size.x; + pos2.x = pos1.x + diff_x + 2; // TODO: Why +2? + } + + if (diff_y <= 0) { // Down + pos1.y = win->pos_prev.y; + pos2.y = win->pos.y + win->ctx.size.y; + } else if (diff_y > 0) { // Up + pos1.y = win->pos.y; + pos2.y = win->pos_prev.y + win->ctx.size.y; + } + + rectangle_redraw(pos1, pos2); + + /** + * Redraw bottom/top diff rectangle (only one!) + */ + + if (diff_y <= 0) { // Down + pos1.y = win->pos_prev.y; + pos2.y = pos1.y - diff_y; + } else if (diff_y > 0) { // Up + pos1.y = win->pos.y + win->ctx.size.y; + pos2.y = pos1.y + diff_y; + } + + if (diff_x <= 0) { // Right + pos1.x = win->pos_prev.x; + pos2.x = win->pos.x + win->ctx.size.x; + } else if (diff_x > 0) { // Left + pos1.x = win->pos.x; + pos2.x = win->pos_prev.x + win->ctx.size.x; + } + + rectangle_redraw(pos1, pos2); + + // Redraw window on top of everything + gfx_ctx_on_ctx(&direct->ctx, &win->ctx, win->pos, GFX_NON_ALPHA); +} + +static void window_redraw(struct window *win) +{ + if (win->flags & WF_ALPHA) + window_redraw_alpha(win); + else + window_redraw_non_alpha(win); +} + +// TODO: Fix strange artifacts after destroying +static void window_destroy(struct window *win) +{ + memset(win->ctx.fb, 0, win->ctx.bytes); + rectangle_redraw(win->pos, vec2_add(win->pos, win->ctx.size)); + list_remove(windows, list_first_data(windows, win)); + sys_free(win->ctx.fb); + free(win); +} + +/** + * Event handlers + */ + +static void handle_event_keyboard(struct event_keyboard *event) +{ + if (event->magic != KEYBOARD_MAGIC) { + log("Keyboard magic doesn't match!\n"); + return; + } + + if (event->scancode == KEY_LEFTSHIFT || event->scancode == KEY_RIGHTSHIFT) + special_keys.shift ^= 1; + else if (event->scancode == KEY_LEFTALT || event->scancode == KEY_RIGHTALT) + special_keys.alt ^= 1; + else if (event->scancode == KEY_LEFTCTRL || event->scancode == KEY_RIGHTCTRL) + special_keys.ctrl ^= 1; + + if (event->scancode > KEYMAP_LENGTH) + return; + + char ch; + if (special_keys.shift) + ch = keymap->shift_map[event->scancode]; + else if (special_keys.alt) + ch = keymap->alt_map[event->scancode]; + else + ch = keymap->map[event->scancode]; + + UNUSED(ch); +} + +static struct timer mouse_timer = { 0 }; +static void handle_event_mouse(struct event_mouse *event) +{ + if (event->magic != MOUSE_MAGIC) { + log("Mouse magic doesn't match!\n"); + return; + } + + cursor->pos_prev = mouse.pos; + + if (event->rel) { + mouse.pos.x += (s32)event->pos.x; + mouse.pos.y -= (s32)event->pos.y; + } else { + // TODO: Support other absolute scaling than 0xffff (VMWare default) + mouse.pos.x = event->pos.x * screen.width / 0xffff; + mouse.pos.y = event->pos.y * screen.height / 0xffff; + } + + // Fix x overflow + if ((signed)mouse.pos.x < 0) + mouse.pos.x = 0; + else if (mouse.pos.x + cursor->ctx.size.x > (unsigned)screen.width - 1) + mouse.pos.x = screen.width - cursor->ctx.size.x - 1; + + // Fix y overflow + if ((signed)mouse.pos.y < 0) + mouse.pos.y = 0; + else if (mouse.pos.y + cursor->ctx.size.y > (unsigned)screen.height - 1) + mouse.pos.y = screen.height - cursor->ctx.size.y - 1; + + cursor->pos = mouse.pos; + + struct window *win = window_at(mouse.pos); + if (win && !(win->flags & WF_NO_FOCUS) && !event->but.left && !event->but.right && + !event->but.middle) + focused = win; + + if (focused && !(focused->flags & WF_NO_DRAG) && event->but.left && special_keys.alt) { + struct timer timer = { 0 }; + io_read(IO_TIMER, &timer, 0, sizeof(timer)); + if (timer.time - mouse_timer.time > WINDOW_MOVE_TIMEOUT) { + focused->pos_prev = focused->pos; + focused->pos = mouse.pos; + window_redraw(focused); + mouse_timer = timer; + } + return; + } else if (!vec2_eq(cursor->pos, cursor->pos_prev)) { + window_redraw(cursor); + } + + if (!win) + return; + + struct message_mouse msg = { 0 }; + msg.header.state = MSG_GO_ON; + msg.id = win->id; + msg.pos = vec2_sub(mouse.pos, win->pos); + msg.bits.click = event->but.left; + + if (msg_connect_conn(win->client.conn) == EOK) + msg_send(GUI_MOUSE, &msg, sizeof(msg)); + else + log("Failed to connect to window\n"); +} + +/** + * Message handlers + */ + +static void handle_message_new_window(struct message_new_window *msg) +{ + struct window *win = window_new((struct client){ .conn = msg->header.bus.conn }, + vec2(500, 600), vec2(600, 400), 0); + msg->ctx = win->ctx; + msg->shid = win->shid; + msg->id = win->id; + + if (msg->header.state == MSG_NEED_ANSWER) { + msg_connect_conn(msg->header.bus.conn); + msg_send(GUI_NEW_WINDOW | MSG_SUCCESS, msg, sizeof(*msg)); + } +} + +static void handle_message_redraw_window(struct message_redraw_window *msg) +{ + u32 id = msg->id; + struct window *win = window_find(id); + if (!win || win->client.conn != msg->header.bus.conn) { + if (msg->header.state == MSG_NEED_ANSWER) { + msg_connect_conn(msg->header.bus.conn); + msg_send(GUI_REDRAW_WINDOW | MSG_FAILURE, msg, sizeof(msg->header)); + } + return; + } + + window_redraw(win); + + if (msg->header.state == MSG_NEED_ANSWER) { + msg_connect_conn(msg->header.bus.conn); + msg_send(GUI_REDRAW_WINDOW | MSG_SUCCESS, msg, sizeof(msg->header)); + } +} + +static void handle_message_destroy_window(struct message_destroy_window *msg) +{ + u32 id = msg->id; + struct window *win = window_find(id); + if (!win || win->client.conn != msg->header.bus.conn) { + if (msg->header.state == MSG_NEED_ANSWER) { + msg_connect_conn(msg->header.bus.conn); + msg_send(GUI_DESTROY_WINDOW | MSG_FAILURE, msg, sizeof(msg->header)); + } + return; + } + + window_destroy(win); + + if (msg->header.state == MSG_NEED_ANSWER) { + msg_connect_conn(msg->header.bus.conn); + msg_send(GUI_DESTROY_WINDOW | MSG_SUCCESS, msg, sizeof(msg->header)); + } +} + +static void handle_message(void *msg) +{ + struct message_header *header = msg; + + switch (header->type) { + case GUI_NEW_WINDOW: + handle_message_new_window(msg); + break; + case GUI_REDRAW_WINDOW: + handle_message_redraw_window(msg); + break; + case GUI_DESTROY_WINDOW: + handle_message_destroy_window(msg); + break; + default: + log("Message type %d not implemented!\n", header->type); + msg_connect_conn(header->bus.conn); + msg_send(GUI_DESTROY_WINDOW | MSG_SUCCESS, msg, sizeof(header)); + } +} + +static void handle_exit(void) +{ + if (keymap) + free(keymap); + + if (screen.fb) + memset(screen.fb, COLOR_RED, screen.height * screen.pitch); + + if (windows) { + struct node *iterator = windows->head; + while (iterator) { + struct window *win = iterator->data; + if (win->ctx.fb) + sys_free(win->ctx.fb); + free(win); + iterator = iterator->next; + } + list_destroy(windows); + } +} + +/** + * Main loop + */ + +int main(int argc, char **argv) +{ + UNUSED(argc); + UNUSED(argv); + + atexit(handle_exit); + + assert(io_control(IO_FRAMEBUFFER, IOCTL_FB_GET, &screen, sizeof(screen)) == EOK); + log("WM loaded: %dx%d\n", screen.width, screen.height); + wm_client = (struct client){ .conn = 0 }; + bypp = (screen.bpp >> 3); + + windows = list_new(); + keymap = keymap_parse("/res/keymaps/en.keymap"); + + direct = window_new(wm_client, vec2(0, 0), vec2(screen.width, screen.height), + WF_NO_WINDOW | WF_NO_FB | WF_NO_DRAG | WF_NO_FOCUS | WF_NO_RESIZE); + direct->ctx.fb = screen.fb; + direct->flags ^= WF_NO_FB; + wallpaper = window_new(wm_client, vec2(0, 0), vec2(screen.width, screen.height), + WF_NO_DRAG | WF_NO_FOCUS | WF_NO_RESIZE); + cursor = window_new(wm_client, vec2(0, 0), vec2(32, 32), + WF_NO_DRAG | WF_NO_FOCUS | WF_NO_RESIZE | WF_ALPHA); + + gfx_load_wallpaper(&wallpaper->ctx, "/res/wall.png"); + memset(cursor->ctx.fb, 0, cursor->ctx.bytes); + gfx_load_wallpaper(&cursor->ctx, "/res/cursor.png"); + window_redraw(wallpaper); + + assert(io_control(IO_BUS, IOCTL_BUS_REGISTER, "wm") == EOK); + + assert(exec("chess", NULL) == EOK); + + u8 msg[1024] = { 0 }; + struct event_keyboard event_keyboard = { 0 }; + struct event_mouse event_mouse = { 0 }; + enum io_type listeners[] = { IO_KEYBOARD, IO_MOUSE, IO_BUS, 0 }; + + while (1) { + res poll_ret = 0; + if ((poll_ret = io_poll(listeners)) >= 0) { + if (poll_ret == IO_KEYBOARD) { + if (io_read(IO_KEYBOARD, &event_keyboard, 0, + sizeof(event_keyboard)) > 0) { + handle_event_keyboard(&event_keyboard); + continue; + } + } else if (poll_ret == IO_MOUSE) { + if (io_read(IO_MOUSE, &event_mouse, 0, sizeof(event_mouse)) > 0) { + handle_event_mouse(&event_mouse); + continue; + } + } else if (poll_ret == IO_BUS) { + if (msg_receive(msg, sizeof(msg)) > 0) { + handle_message(msg); + continue; + } + } + } + panic("Poll/read error: %s\n", strerror(errno)); + } + + return 1; +} diff --git a/apps/wm/wm.c b/apps/wm/wm.c deleted file mode 100644 index 6aed58e..0000000 --- a/apps/wm/wm.c +++ /dev/null @@ -1,626 +0,0 @@ -// MIT License, Copyright (c) 2020 Marvin Borner -// Window manager / compositor (does everything basically) - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#define WINDOW_MOVE_TIMEOUT 20 - -struct client { - u32 conn; // Bus conn -}; - -struct window { - u32 id; - u32 shid; - struct context ctx; - struct client client; - u32 flags; - vec2 pos; - vec2 pos_prev; -}; - -struct rectangle { - vec2 pos1; // Upper left - vec2 pos2; // Lower right - void *data; -}; - -// Global vars ftw! -static u8 bypp = 4; -static struct vbe screen = { 0 }; -static struct list *windows = NULL; // THIS LIST SHALL BE SORTED BY Z-INDEX! -static struct window *direct = NULL; -static struct window *wallpaper = NULL; -static struct window *cursor = NULL; -static struct window *focused = NULL; -static struct keymap *keymap = NULL; -static struct client wm_client = { 0 }; -static struct { - u8 shift : 1; - u8 alt : 1; - u8 ctrl : 1; -} special_keys = { 0 }; -static struct { - vec2 pos; - u8 left : 1; - u8 mid : 1; - u8 right : 1; -} mouse = { 0 }; - -/** - * 5head algorithms - * Thanks to @LarsVomMars for the help - */ - -static void windows_at_rec(vec2 pos1, vec2 pos2, struct list *list) -{ - u32 width = pos2.x - pos1.x; - u32 height = pos2.y - pos1.y; - vec2 rec_corners[] = { - pos1, - vec2_add(pos1, vec2(width, 0)), - pos2, - vec2_add(pos1, vec2(0, height)), - }; - - u8 cursor_found = 0; - - struct node *iterator = windows->head; - while (iterator) { - struct window *win = iterator->data; - - if ((win->flags & WF_NO_WINDOW) != 0) - goto next; - - vec2 corners[] = { - win->pos, - vec2_add(win->pos, vec2(win->ctx.size.x, 0)), - vec2_add(win->pos, vec2(win->ctx.size.x, win->ctx.size.y)), - vec2_add(win->pos, vec2(0, win->ctx.size.y)), - }; - - for (u8 i = 0; i < 4; i++) { - vec2 corner = corners[i]; - if ((pos1.x <= corner.x && pos1.y <= corner.y) && - (pos2.x >= corner.x && pos2.y >= corner.y)) { - if (win == cursor) - cursor_found = 1; - else - list_add(list, win); - goto next; - } - } - - vec2 win_pos1 = win->pos; - vec2 win_pos2 = vec2_add(win->pos, win->ctx.size); - for (u8 i = 0; i < 4; i++) { - vec2 corner = rec_corners[i]; - if ((win_pos1.x <= corner.x && win_pos1.y <= corner.y) && - (win_pos2.x >= corner.x && win_pos2.y >= corner.y)) { - if (win == cursor) - cursor_found = 1; - else - list_add(list, win); - goto next; - } - } - - next: - iterator = iterator->next; - } - - if (cursor_found) - list_add(list, cursor); -} - -static struct rectangle rectangle_at(vec2 pos1, vec2 pos2) -{ - u32 width = ABS(pos2.x - pos1.x); - u32 height = ABS(pos2.y - pos1.y); - u32 pitch = width * bypp; - u8 *data = zalloc(width * height * bypp); - - struct list *windows_at = list_new(); - windows_at_rec(pos1, pos2, windows_at); - struct node *iterator = windows_at->head; - while (iterator) { - struct window *win = iterator->data; - - s32 start_x = win->pos.x - pos1.x; - u32 end_x = width; - if (start_x <= 0) { // Either right side or background - u32 right = start_x + win->ctx.size.x; - if (right <= width) { // Right side - start_x = 0; - end_x = right; - } else { // Background - start_x = 0; - } - } - - s32 start_y = win->pos.y - pos1.y; - u32 end_y = height; - if (start_y <= 0) { // Either bottom side or background - u32 bot = start_y + win->ctx.size.y; - if (bot <= height) { // Bottom side - start_y = 0; - end_y = bot; - } else { // Background - start_y = 0; - } - } - - vec2 pos = vec2_sub(pos1, win->pos); - - u8 *srcfb = - &win->ctx.fb[(pos.x + start_x) * bypp + (pos.y + start_y) * win->ctx.pitch]; - u8 *destfb = &data[start_x * bypp + start_y * pitch]; - - // Copy window data to rectangle buffer - for (u32 cy = start_y; cy < end_y; cy++) { - u32 diff = 0; - for (u32 cx = start_x; cx < end_x; cx++) { - if (srcfb[bypp - 1]) - memcpy(destfb, srcfb, bypp); - - srcfb += bypp; - destfb += bypp; - diff += bypp; - } - srcfb += win->ctx.pitch - diff; - destfb += pitch - diff; - } - - iterator = iterator->next; - } - list_destroy(windows_at); - - return (struct rectangle){ .pos1 = pos1, .pos2 = pos2, .data = data }; -} - -static void rectangle_redraw(vec2 pos1, vec2 pos2) -{ - assert(pos1.x <= pos2.x && pos1.y <= pos2.y); - struct rectangle rec = rectangle_at(pos1, pos2); - - u32 width = ABS(pos2.x - pos1.x); - u32 height = ABS(pos2.y - pos1.y); - - if (!width || !height) - return; - - u8 *srcfb = rec.data; - u8 *destfb = &direct->ctx.fb[rec.pos1.x * bypp + rec.pos1.y * direct->ctx.pitch]; - for (u32 cy = 0; cy < height; cy++) { - memcpy(destfb, srcfb, width * bypp); - srcfb += width * bypp; - destfb += direct->ctx.pitch; - } - - free(rec.data); -} - -/** - * Window operations - */ - -static struct window *window_new(struct client client, struct vec2 pos, struct vec2 size, u32 flags) -{ - assert(windows); - - struct window *win = malloc(sizeof(*win)); - static u32 id = 0; - win->id = id++; - win->ctx.size = size; - win->ctx.bpp = screen.bpp; - win->ctx.pitch = size.x * bypp; - win->ctx.bytes = win->ctx.pitch * win->ctx.size.y; - if ((flags & WF_NO_FB) != 0) { - win->ctx.fb = NULL; - } else { - assert(shalloc(win->ctx.bytes, (u32 *)&win->ctx.fb, &win->shid) == EOK); - memset(win->ctx.fb, COLOR_BLACK, win->ctx.bytes); - } - win->client = client; - win->flags = flags; - win->pos = pos; - win->pos_prev = pos; - list_add(windows, win); - return win; -} - -static struct window *window_find(u32 id) -{ - struct node *iterator = windows->head; - while (iterator) { - struct window *win = iterator->data; - if (win->id == id) - return win; - iterator = iterator->next; - } - return NULL; -} - -static struct window *window_at(vec2 pos) -{ - struct window *ret = NULL; - - struct node *iterator = windows->head; - while (iterator) { - struct window *win = iterator->data; - if (!(win->flags & (WF_NO_WINDOW | WF_NO_FOCUS)) && pos.x >= win->pos.x && - pos.x <= win->pos.x + win->ctx.size.x && pos.y >= win->pos.y && - pos.y <= win->pos.y + win->ctx.size.y) - ret = win; - iterator = iterator->next; - } - return ret; -} - -// Transparent windows can't use efficient rectangle-based redrawing (I think!) -static void window_redraw_alpha(struct window *win) -{ - vec2 pos1 = win->pos_prev; - vec2 pos2 = vec2(pos1.x + win->ctx.size.x, pos1.y + win->ctx.size.y); - - rectangle_redraw(pos1, pos2); - gfx_ctx_on_ctx(&direct->ctx, &win->ctx, win->pos, GFX_ALPHA); -} - -// TODO: Make more efficient... -static void window_redraw_non_alpha(struct window *win) -{ - s32 diff_x = win->pos_prev.x - win->pos.x; - s32 diff_y = win->pos_prev.y - win->pos.y; - - if (!diff_x && !diff_y) { - gfx_ctx_on_ctx(&direct->ctx, &win->ctx, win->pos, GFX_NON_ALPHA); - return; - } - - vec2 pos1 = { 0 }; - vec2 pos2 = { 0 }; - - /** - * Redraw left/right diff rectangle (only one!) - */ - - if (diff_x <= 0) { // Right - pos1.x = win->pos_prev.x; - pos2.x = pos1.x - diff_x + 2; - } else if (diff_x > 0) { // Left - pos1.x = win->pos.x + win->ctx.size.x; - pos2.x = pos1.x + diff_x + 2; // TODO: Why +2? - } - - if (diff_y <= 0) { // Down - pos1.y = win->pos_prev.y; - pos2.y = win->pos.y + win->ctx.size.y; - } else if (diff_y > 0) { // Up - pos1.y = win->pos.y; - pos2.y = win->pos_prev.y + win->ctx.size.y; - } - - rectangle_redraw(pos1, pos2); - - /** - * Redraw bottom/top diff rectangle (only one!) - */ - - if (diff_y <= 0) { // Down - pos1.y = win->pos_prev.y; - pos2.y = pos1.y - diff_y; - } else if (diff_y > 0) { // Up - pos1.y = win->pos.y + win->ctx.size.y; - pos2.y = pos1.y + diff_y; - } - - if (diff_x <= 0) { // Right - pos1.x = win->pos_prev.x; - pos2.x = win->pos.x + win->ctx.size.x; - } else if (diff_x > 0) { // Left - pos1.x = win->pos.x; - pos2.x = win->pos_prev.x + win->ctx.size.x; - } - - rectangle_redraw(pos1, pos2); - - // Redraw window on top of everything - gfx_ctx_on_ctx(&direct->ctx, &win->ctx, win->pos, GFX_NON_ALPHA); -} - -static void window_redraw(struct window *win) -{ - if (win->flags & WF_ALPHA) - window_redraw_alpha(win); - else - window_redraw_non_alpha(win); -} - -// TODO: Fix strange artifacts after destroying -static void window_destroy(struct window *win) -{ - memset(win->ctx.fb, 0, win->ctx.bytes); - rectangle_redraw(win->pos, vec2_add(win->pos, win->ctx.size)); - list_remove(windows, list_first_data(windows, win)); - sys_free(win->ctx.fb); - free(win); -} - -/** - * Event handlers - */ - -static void handle_event_keyboard(struct event_keyboard *event) -{ - if (event->magic != KEYBOARD_MAGIC) { - log("Keyboard magic doesn't match!\n"); - return; - } - - if (event->scancode == KEY_LEFTSHIFT || event->scancode == KEY_RIGHTSHIFT) - special_keys.shift ^= 1; - else if (event->scancode == KEY_LEFTALT || event->scancode == KEY_RIGHTALT) - special_keys.alt ^= 1; - else if (event->scancode == KEY_LEFTCTRL || event->scancode == KEY_RIGHTCTRL) - special_keys.ctrl ^= 1; - - if (event->scancode > KEYMAP_LENGTH) - return; - - char ch; - if (special_keys.shift) - ch = keymap->shift_map[event->scancode]; - else if (special_keys.alt) - ch = keymap->alt_map[event->scancode]; - else - ch = keymap->map[event->scancode]; - - UNUSED(ch); -} - -static struct timer mouse_timer = { 0 }; -static void handle_event_mouse(struct event_mouse *event) -{ - if (event->magic != MOUSE_MAGIC) { - log("Mouse magic doesn't match!\n"); - return; - } - - cursor->pos_prev = mouse.pos; - - if (event->rel) { - mouse.pos.x += (s32)event->pos.x; - mouse.pos.y -= (s32)event->pos.y; - } else { - // TODO: Support other absolute scaling than 0xffff (VMWare default) - mouse.pos.x = event->pos.x * screen.width / 0xffff; - mouse.pos.y = event->pos.y * screen.height / 0xffff; - } - - // Fix x overflow - if ((signed)mouse.pos.x < 0) - mouse.pos.x = 0; - else if (mouse.pos.x + cursor->ctx.size.x > (unsigned)screen.width - 1) - mouse.pos.x = screen.width - cursor->ctx.size.x - 1; - - // Fix y overflow - if ((signed)mouse.pos.y < 0) - mouse.pos.y = 0; - else if (mouse.pos.y + cursor->ctx.size.y > (unsigned)screen.height - 1) - mouse.pos.y = screen.height - cursor->ctx.size.y - 1; - - cursor->pos = mouse.pos; - - struct window *win = window_at(mouse.pos); - if (win && !(win->flags & WF_NO_FOCUS) && !event->but.left && !event->but.right && - !event->but.middle) - focused = win; - - if (focused && !(focused->flags & WF_NO_DRAG) && event->but.left && special_keys.alt) { - struct timer timer = { 0 }; - io_read(IO_TIMER, &timer, 0, sizeof(timer)); - if (timer.time - mouse_timer.time > WINDOW_MOVE_TIMEOUT) { - focused->pos_prev = focused->pos; - focused->pos = mouse.pos; - window_redraw(focused); - mouse_timer = timer; - } - return; - } else if (!vec2_eq(cursor->pos, cursor->pos_prev)) { - window_redraw(cursor); - } - - if (!win) - return; - - struct message_mouse msg = { 0 }; - msg.header.state = MSG_GO_ON; - msg.id = win->id; - msg.pos = vec2_sub(mouse.pos, win->pos); - msg.bits.click = event->but.left; - - if (msg_connect_conn(win->client.conn) == EOK) - msg_send(GUI_MOUSE, &msg, sizeof(msg)); - else - log("Failed to connect to window\n"); -} - -/** - * Message handlers - */ - -static void handle_message_new_window(struct message_new_window *msg) -{ - struct window *win = window_new((struct client){ .conn = msg->header.bus.conn }, - vec2(500, 600), vec2(600, 400), 0); - msg->ctx = win->ctx; - msg->shid = win->shid; - msg->id = win->id; - - if (msg->header.state == MSG_NEED_ANSWER) { - msg_connect_conn(msg->header.bus.conn); - msg_send(GUI_NEW_WINDOW | MSG_SUCCESS, msg, sizeof(*msg)); - } -} - -static void handle_message_redraw_window(struct message_redraw_window *msg) -{ - u32 id = msg->id; - struct window *win = window_find(id); - if (!win || win->client.conn != msg->header.bus.conn) { - if (msg->header.state == MSG_NEED_ANSWER) { - msg_connect_conn(msg->header.bus.conn); - msg_send(GUI_REDRAW_WINDOW | MSG_FAILURE, msg, sizeof(msg->header)); - } - return; - } - - window_redraw(win); - - if (msg->header.state == MSG_NEED_ANSWER) { - msg_connect_conn(msg->header.bus.conn); - msg_send(GUI_REDRAW_WINDOW | MSG_SUCCESS, msg, sizeof(msg->header)); - } -} - -static void handle_message_destroy_window(struct message_destroy_window *msg) -{ - u32 id = msg->id; - struct window *win = window_find(id); - if (!win || win->client.conn != msg->header.bus.conn) { - if (msg->header.state == MSG_NEED_ANSWER) { - msg_connect_conn(msg->header.bus.conn); - msg_send(GUI_DESTROY_WINDOW | MSG_FAILURE, msg, sizeof(msg->header)); - } - return; - } - - window_destroy(win); - - if (msg->header.state == MSG_NEED_ANSWER) { - msg_connect_conn(msg->header.bus.conn); - msg_send(GUI_DESTROY_WINDOW | MSG_SUCCESS, msg, sizeof(msg->header)); - } -} - -static void handle_message(void *msg) -{ - struct message_header *header = msg; - - switch (header->type) { - case GUI_NEW_WINDOW: - handle_message_new_window(msg); - break; - case GUI_REDRAW_WINDOW: - handle_message_redraw_window(msg); - break; - case GUI_DESTROY_WINDOW: - handle_message_destroy_window(msg); - break; - default: - log("Message type %d not implemented!\n", header->type); - msg_connect_conn(header->bus.conn); - msg_send(GUI_DESTROY_WINDOW | MSG_SUCCESS, msg, sizeof(header)); - } -} - -static void handle_exit(void) -{ - if (keymap) - free(keymap); - - if (screen.fb) - memset(screen.fb, COLOR_RED, screen.height * screen.pitch); - - if (windows) { - struct node *iterator = windows->head; - while (iterator) { - struct window *win = iterator->data; - if (win->ctx.fb) - sys_free(win->ctx.fb); - free(win); - iterator = iterator->next; - } - list_destroy(windows); - } -} - -/** - * Main loop - */ - -int main(int argc, char **argv) -{ - UNUSED(argc); - UNUSED(argv); - - atexit(handle_exit); - - assert(io_control(IO_FRAMEBUFFER, IOCTL_FB_GET, &screen, sizeof(screen)) == EOK); - log("WM loaded: %dx%d\n", screen.width, screen.height); - wm_client = (struct client){ .conn = 0 }; - bypp = (screen.bpp >> 3); - - windows = list_new(); - keymap = keymap_parse("/res/keymaps/en.keymap"); - - direct = window_new(wm_client, vec2(0, 0), vec2(screen.width, screen.height), - WF_NO_WINDOW | WF_NO_FB | WF_NO_DRAG | WF_NO_FOCUS | WF_NO_RESIZE); - direct->ctx.fb = screen.fb; - direct->flags ^= WF_NO_FB; - wallpaper = window_new(wm_client, vec2(0, 0), vec2(screen.width, screen.height), - WF_NO_DRAG | WF_NO_FOCUS | WF_NO_RESIZE); - cursor = window_new(wm_client, vec2(0, 0), vec2(32, 32), - WF_NO_DRAG | WF_NO_FOCUS | WF_NO_RESIZE | WF_ALPHA); - - gfx_load_wallpaper(&wallpaper->ctx, "/res/wall.png"); - memset(cursor->ctx.fb, 0, cursor->ctx.bytes); - gfx_load_wallpaper(&cursor->ctx, "/res/cursor.png"); - window_redraw(wallpaper); - - assert(io_control(IO_BUS, IOCTL_BUS_REGISTER, "wm") == EOK); - - /* assert(exec("chess", NULL) == EOK); */ - - u8 msg[1024] = { 0 }; - struct event_keyboard event_keyboard = { 0 }; - struct event_mouse event_mouse = { 0 }; - enum io_type listeners[] = { IO_KEYBOARD, IO_MOUSE, IO_BUS, 0 }; - - while (1) { - res poll_ret = 0; - if ((poll_ret = io_poll(listeners)) >= 0) { - if (poll_ret == IO_KEYBOARD) { - if (io_read(IO_KEYBOARD, &event_keyboard, 0, - sizeof(event_keyboard)) > 0) { - handle_event_keyboard(&event_keyboard); - continue; - } - } else if (poll_ret == IO_MOUSE) { - if (io_read(IO_MOUSE, &event_mouse, 0, sizeof(event_mouse)) > 0) { - handle_event_mouse(&event_mouse); - continue; - } - } else if (poll_ret == IO_BUS) { - if (msg_receive(msg, sizeof(msg)) > 0) { - handle_message(msg); - continue; - } - } - } - panic("Poll/read error: %s\n", strerror(errno)); - } - - return 1; -} -- cgit v1.2.3