diff options
author | Mitya Selivanov <automainint@guattari.tech> | 2024-08-12 20:10:40 +0200 |
---|---|---|
committer | Mitya Selivanov <automainint@guattari.tech> | 2024-08-12 20:10:40 +0200 |
commit | ec7ce54de5583ab11de5b19e6efc146cb52c3418 (patch) | |
tree | 8596c1f11d73c0d78273f4455225eb9a1c56c299 | |
parent | 4f20ddeb9e0e08fd32f66ea049eb5d97c4b1c1a8 (diff) | |
download | reduced_system_layer-ec7ce54de5583ab11de5b19e6efc146cb52c3418.zip |
Refactor examples
-rwxr-xr-x | examples/echo.c | 1062 | ||||
-rwxr-xr-x | examples/gravity.c | 803 | ||||
-rwxr-xr-x | examples/landscape.c | 803 | ||||
-rwxr-xr-x | examples/ui.c | 801 | ||||
-rwxr-xr-x | reduced_system_layer.c | 333 |
5 files changed, 303 insertions, 3499 deletions
diff --git a/examples/echo.c b/examples/echo.c index 3a31b18..671d609 100755 --- a/examples/echo.c +++ b/examples/echo.c @@ -3,31 +3,6 @@ #/ #/ echo.c #/ -#/ This is a reduced system layer. -#/ It allows you to create a window, draw graphics in it, handle -#/ input events. -#/ -#/ ---------------------------------------------------------------- -#/ -#/ DESIGN PRINCIPLES -#/ -#/ - Minimalistic feature set. For graphics, you have access to the -#/ pixel buffer, and that's it. -#/ -#/ - No implicit control flow. No callbacks. You write your own -#/ main and call everything explicitly. But the number of things -#/ you have to call to do something is as little as possible. -#/ -#/ - Optimized to use in a single source file. -#/ Installation process? Ctrl+C, Ctrl+V, done. -#/ -#/ If you have an idea how to reduce the feature set further, -#/ let me know! -#/ -#/ ---------------------------------------------------------------- -#/ -#/ (C) 2024 Mitya Selivanov <guattari.tech>, MIT License -#/ #/ ================================================================ #/ #/ Self-compilation shell script @@ -49,163 +24,8 @@ gcc \ ./$BIN $@ && rm $BIN exit $? # */ #endif -// ================================================================ -// -// Basic declarations -// -// ================================================================ - -#define _GNU_SOURCE - -typedef signed char i8; -typedef signed short i16; -typedef signed i32; -typedef signed long long i64; -typedef unsigned char u8; -typedef unsigned short u16; -typedef unsigned u32; -typedef unsigned long long u64; -typedef char c8; -typedef int c32; -typedef signed char b8; -typedef float f32; -typedef double f64; - -// ================================================================ -// -// PLATFORM API -// -// ================================================================ - -enum { - MAX_NUM_PIXELS = 10 * 1024 * 1024, - MAX_INPUT_SIZE = 256, - MAX_CLIPBOARD_SIZE = 10 * 1024 * 1024, - MAX_NUM_AUDIO_SAMPLES = 0, - MAX_NUM_SOCKETS = 64, - - AUDIO_NUM_CHANNELS = 2, - AUDIO_SAMPLE_RATE = 44100, - - IPv4_UDP = 1, - IPv6_UDP = 2, - - KEY_LEFT = 128, - KEY_RIGHT, - KEY_UP, - KEY_DOWN, - KEY_LCTRL, - KEY_RCTRL, - KEY_LSHIFT, - KEY_RSHIFT, - KEY_LALT, - KEY_RALT, - KEY_ESCAPE, - KEY_PRINTSCREEN, - KEY_DELETE, - KEY_PAUSE, - KEY_INSERT, - KEY_HOME, - KEY_END, - KEY_PAGEUP, - KEY_PAGEDOWN, - BUTTON_LEFT, - BUTTON_MIDDLE, - BUTTON_RIGHT, - MOD_CTRL, - MOD_SHIFT, - MOD_ALT, - MOD_CAPS, - MOD_NUM, - MOD_SCROLL, - KEY_F_, - KEY_KP_ = KEY_F_ + 64, -}; - -typedef struct { - b8 ctrl : 1; - b8 shift : 1; - b8 alt : 1; - b8 caps : 1; - b8 num : 1; - b8 scroll : 1; - u16 key; - c32 c; -} Input_Key; - -typedef struct { - c8 * title; - i32 frame_width; - i32 frame_height; - u32 * pixels; - i64 input_size; - Input_Key *input; - i64 clipboard_size; - c8 * clipboard; - b8 done; - b8 graceful_exit; - b8 has_focus; - b8 has_cursor; - i32 cursor_x; - i32 cursor_y; - i32 cursor_dx; - i32 cursor_dy; - i64 wheel_dy; - b8 key_down[512]; - b8 key_pressed[512]; -} Platform; - -typedef struct { - u16 protocol; - u16 port; - union { - u32 v4_address_as_u32; - u8 v4_address[4]; - u8 v6_address[16]; - }; -} IP_Address; - -typedef i64 (*Thread_Proc)(void *user_data); - -// UTF-8 -i32 utf8_size(c32 c); -c32 utf8_read(i64 len, c8 *s); -i32 utf8_write(c32 c, c8 *buffer); - -// Time and sleep -i64 p_time(void); -void p_yield(void); -void p_sleep_for(i64 duration); - -// Window -void p_init(void); -void p_cleanup(void); -i32 p_handle_events(void); -i32 p_wait_events(void); -void p_render_frame(void); - -// Clipboard -void p_clipboard_write(i64 size, c8 *data); -// Sound -void p_handle_audio(i64 time_elapsed); -void p_queue_sound(i64 delay, i64 num_samples, f32 *samples); - -// UDP sockets -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, u16 *local_port, IP_Address *remote_address); -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data, u16 *local_port); - -Platform platform = {0}; - -// ================================================================ -// -// WRITE YOUR CODE HERE -// -// ================================================================ - -#include <stdio.h> -#include <string.h> -#include <errno.h> +#include "../reduced_system_layer.c" enum { MODE_RECV = 1, @@ -296,883 +116,3 @@ i32 main(i32 argc, c8 **argv) { p_cleanup(); return 0; } - -// ================================================================ -// -// PLATFORM IMPLEMENTATION -// -// ---------------------------------------------------------------- -// -// TODO -// - Sound -// - Clipboard daemon -// -// ALSA -// https://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_min_8c-example.html -// -// ================================================================ - -#include <assert.h> -#include <string.h> - -c32 utf8_read(i64 len, c8 *s) { - if (len >= 1 && - (s[0] & 0x80) == 0) - return s[0]; - if (len >= 2 && - (s[0] & 0xe0) == 0xc0 && - (s[1] & 0xc0) == 0x80) - return (s[1] & 0x3f) - | ((s[0] & 0x1f) << 6); - if (len >= 3 && - (s[0] & 0xf0) == 0xe0 && - (s[1] & 0xc0) == 0x80 && - (s[2] & 0xc0) == 0x80) - return (s[2] & 0x3f) - | ((s[1] & 0x3f) << 6) - | ((s[0] & 0x0f) << 12); - if (len >= 4 && - (s[0] & 0xf8) == 0xf0 && - (s[1] & 0xc0) == 0x80 && - (s[2] & 0xc0) == 0x80 && - (s[3] & 0xc0) == 0x80) - return (s[3] & 0x3f) - | ((s[2] & 0x3f) << 6) - | ((s[1] & 0x3f) << 12) - | ((s[0] & 0x07) << 18); - return 0; -} - -i32 utf8_size(c32 c) { - if ((c & 0x00007f) == c) return 1; - if ((c & 0x0007ff) == c) return 2; - if ((c & 0x00ffff) == c) return 3; - if ((c & 0x1fffff) == c) return 4; - return 0; -} - -i32 utf8_write(c32 c, c8 *buffer) { - if ((c & 0x7f) == c) { - buffer[0] = (c8) c; - return 1; - } - - if ((c & 0x7ff) == c) { - buffer[0] = 0xc0 | ((c >> 6) & 0x1f); - buffer[1] = 0x80 | ( c & 0x3f); - return 2; - } - - if ((c & 0xffff) == c) { - buffer[0] = 0xc0 | ((c >> 12) & 0x0f); - buffer[1] = 0x80 | ((c >> 6) & 0x3f); - buffer[2] = 0x80 | ( c & 0x3f); - return 3; - } - - if ((c & 0x1fffff) == c) { - buffer[0] = 0xc0 | ((c >> 18) & 0x07); - buffer[1] = 0x80 | ((c >> 12) & 0x3f); - buffer[2] = 0x80 | ((c >> 6) & 0x3f); - buffer[3] = 0x80 | ( c & 0x3f); - return 4; - } - - return 0; -} - -// ================================================================ -// -// Unix -// -// ================================================================ - -#ifdef __unix__ - -#include <time.h> -#include <sched.h> -#include <unistd.h> - -i64 p_time(void) { - struct timespec t; - timespec_get(&t, TIME_UTC); - return 1000 * t.tv_sec + t.tv_nsec / 1000000; -} - -void p_yield(void) { - sched_yield(); -} - -void p_sleep_for(i64 duration) { - if (duration <= 0) { - usleep(0); - return; - } - - if (duration >= 1000) - // seconds - sleep(duration / 1000); - - if ((duration % 1000) > 0) - // microseconds - usleep((duration % 1000) * 1000); -} - -#endif - -// ================================================================ -// -// UDP sockets -// -// ================================================================ - -#ifdef __unix__ - -#include <arpa/inet.h> -#include <errno.h> -#include <fcntl.h> -#include <netinet/in.h> -#include <signal.h> -#include <sys/ioctl.h> -#include <sys/select.h> -#include <sys/socket.h> -#include <sys/types.h> -#include <unistd.h> -#include <netdb.h> - -typedef struct { - b8 ready; - i32 socket; - u16 local_port; - IP_Address address; -} Socket_Slot; - -b8 _sockets_init = 0; -Socket_Slot _sockets[MAX_NUM_SOCKETS] = {0}; - -void sockets_initialize(void) { - if (_sockets_init) - return; - - signal(SIGPIPE, SIG_IGN); - _sockets_init = 1; -} - -void sockets_cleanup(void) { - for (i64 i = 0; i < MAX_NUM_SOCKETS; ++i) - if (_sockets[i].ready) { - close(_sockets[i].socket); - _sockets[i].ready = 0; - } -} - -b8 sockets_open(u16 slot, IP_Address address, u16 *local_port) { - sockets_initialize(); - - b8 change_address = - !_sockets[slot].ready - || _sockets[slot].address.protocol != address.protocol - || (address.port != 0 && _sockets[slot].local_port != address.port) - || (memcmp(_sockets[slot].address.v6_address, &(u8[sizeof address.v6_address]) {0}, sizeof address.v6_address) != 0 && - memcmp(_sockets[slot].address.v6_address, address.v6_address, sizeof address.v6_address) != 0); - - if (change_address && _sockets[slot].ready) { - close(_sockets[slot].socket); - _sockets[slot].ready = 0; - } - - struct sockaddr *p; - i32 p_len; - - struct sockaddr_in a4 = {0}; - struct sockaddr_in6 a6 = {0}; - - if (address.protocol == IPv4_UDP) { - p = (struct sockaddr *) &a4; - p_len = sizeof a4; - } else { - p = (struct sockaddr *) &a6; - p_len = sizeof a6; - } - - if (!_sockets[slot].ready) { - _sockets[slot].socket = socket(address.protocol == IPv4_UDP ? AF_INET : AF_INET6, SOCK_DGRAM, IPPROTO_UDP); - - if (_sockets[slot].socket == -1) { - fprintf(stderr, "ERROR: socket failed (errno %d)\n", errno); - return 0; - } - - if (address.protocol == IPv4_UDP) { - a4.sin_family = AF_INET; - a4.sin_port = htons(address.port); - a4.sin_addr.s_addr = address.v4_address_as_u32; - } else { - a6.sin6_family = AF_INET6; - a6.sin6_port = htons(address.port); - memcpy(a6.sin6_addr.s6_addr, address.v6_address, sizeof a6.sin6_addr.s6_addr); - } - - if (bind(_sockets[slot].socket, p, p_len) == -1) { - close(_sockets[slot].socket); - - fprintf(stderr, "ERROR: bind failed (errno %d)\n", errno); - return 0; - } - - if (getsockname(_sockets[slot].socket, p, &(socklen_t) {p_len}) == -1) { - close(_sockets[slot].socket); - - fprintf(stderr, "ERROR: getsockname failed (errno %d)\n", errno); - return 0; - } - - if (p->sa_family == AF_INET) - _sockets[slot].local_port = ntohs(a4.sin_port); - else - _sockets[slot].local_port = ntohs(a6.sin6_port); - - _sockets[slot].ready = 1; - _sockets[slot].address = address; - } - - if (local_port != NULL) - *local_port = _sockets[slot].local_port; - - return 1; -} - -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, u16 *local_port, IP_Address *remote_address) { - assert(slot < MAX_NUM_SOCKETS); - assert(address.protocol == IPv4_UDP || address.protocol == IPv6_UDP); - - if (!sockets_open(slot, address, local_port)) - return 0; - if (size <= 0) - return 0; - - struct sockaddr *p; - i32 p_len; - - struct sockaddr_in a4 = {0}; - struct sockaddr_in6 a6 = {0}; - - if (address.protocol == IPv4_UDP) { - p = (struct sockaddr *) &a4; - p_len = sizeof a4; - } else { - p = (struct sockaddr *) &a6; - p_len = sizeof a6; - } - - i64 recieved = recvfrom( - _sockets[slot].socket, - data, - size, - MSG_DONTWAIT, - p, - &(socklen_t) {p_len} - ); - - if (recieved < 0) { - if (errno == EAGAIN || errno == EWOULDBLOCK) - return 0; - - fprintf(stderr, "ERROR: recvfrom failed (errno %d)\n", errno); - return 0; - } - - if (remote_address != NULL) { - memset(remote_address, 0, sizeof *remote_address); - remote_address->protocol = address.protocol; - - if (address.protocol == IPv4_UDP) { - remote_address->port = ntohs(a4.sin_port); - remote_address->v4_address_as_u32 = a4.sin_addr.s_addr; - } else { - remote_address->port = ntohs(a6.sin6_port); - memcpy(remote_address->v6_address, a6.sin6_addr.s6_addr, sizeof remote_address->v6_address); - } - } - - return recieved; -} - -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data, u16 *local_port) { - assert(slot < MAX_NUM_SOCKETS); - assert(address.protocol == IPv4_UDP || address.protocol == IPv6_UDP); - - IP_Address local_address = address; - local_address.port = 0; - - if (!sockets_open(slot, local_address, local_port)) - return 0; - if (size <= 0) - return 0; - - struct sockaddr *p; - i32 p_len; - - struct sockaddr_in a4 = {0}; - struct sockaddr_in6 a6 = {0}; - - if (address.protocol == IPv4_UDP) { - p = (struct sockaddr *) &a4; - p_len = sizeof a4; - - a4.sin_family = AF_INET; - a4.sin_port = htons(address.port); - a4.sin_addr.s_addr = address.v4_address_as_u32; - } else { - p = (struct sockaddr *) &a6; - p_len = sizeof a6; - - a6.sin6_family = AF_INET6; - a6.sin6_port = htons(address.port); - memcpy(a6.sin6_addr.s6_addr, address.v6_address, sizeof a6.sin6_addr.s6_addr); - } - - i64 sent = sendto( - _sockets[0].socket, - data, - size, - MSG_DONTWAIT, - p, - p_len - ); - - if (sent < 0) { - if (errno == EAGAIN || errno == EWOULDBLOCK) - return 0; - - fprintf(stderr, "ERROR: sendto failed (errno %d)\n", errno); - return 0; - } - - return sent; -} - -#endif - -// ================================================================ -// -// X11 -// -// ================================================================ - -#ifdef __linux__ - -#include <X11/Xlib.h> -#include <X11/Xutil.h> -#include <X11/Xatom.h> - -static i16 _key_table[512] = {0}; -static u32 _buffer[MAX_NUM_PIXELS] = {0}; -static Input_Key _input[MAX_INPUT_SIZE] = {0}; -static c8 _clipboard_buffer[MAX_CLIPBOARD_SIZE] = {0}; -static XImage _image = {0}; -static Display * _display = NULL; -static GC _gc = NULL; -static XIM _im = NULL; -static XIC _ic = NULL; -static Window _window = 0; -static Atom _wm_delete_window = 0; -static Atom _clipboard = 0; -static Atom _targets = 0; -static Atom _utf8_string = 0; -static Atom _target = None; - -void p_init(void) { - _display = XOpenDisplay(NULL); - assert(_display != NULL); - - _key_table[XKeysymToKeycode(_display, XK_Left)] = KEY_LEFT; - _key_table[XKeysymToKeycode(_display, XK_Right)] = KEY_RIGHT; - _key_table[XKeysymToKeycode(_display, XK_Up)] = KEY_UP; - _key_table[XKeysymToKeycode(_display, XK_Down)] = KEY_DOWN; - _key_table[XKeysymToKeycode(_display, XK_1)] = '1'; - _key_table[XKeysymToKeycode(_display, XK_2)] = '2'; - _key_table[XKeysymToKeycode(_display, XK_3)] = '3'; - _key_table[XKeysymToKeycode(_display, XK_4)] = '4'; - _key_table[XKeysymToKeycode(_display, XK_5)] = '5'; - _key_table[XKeysymToKeycode(_display, XK_6)] = '6'; - _key_table[XKeysymToKeycode(_display, XK_7)] = '7'; - _key_table[XKeysymToKeycode(_display, XK_8)] = '8'; - _key_table[XKeysymToKeycode(_display, XK_9)] = '9'; - _key_table[XKeysymToKeycode(_display, XK_0)] = '0'; - _key_table[XKeysymToKeycode(_display, XK_A)] = 'a'; - _key_table[XKeysymToKeycode(_display, XK_B)] = 'b'; - _key_table[XKeysymToKeycode(_display, XK_C)] = 'c'; - _key_table[XKeysymToKeycode(_display, XK_D)] = 'd'; - _key_table[XKeysymToKeycode(_display, XK_E)] = 'e'; - _key_table[XKeysymToKeycode(_display, XK_F)] = 'f'; - _key_table[XKeysymToKeycode(_display, XK_G)] = 'g'; - _key_table[XKeysymToKeycode(_display, XK_H)] = 'h'; - _key_table[XKeysymToKeycode(_display, XK_I)] = 'i'; - _key_table[XKeysymToKeycode(_display, XK_J)] = 'j'; - _key_table[XKeysymToKeycode(_display, XK_K)] = 'k'; - _key_table[XKeysymToKeycode(_display, XK_L)] = 'l'; - _key_table[XKeysymToKeycode(_display, XK_M)] = 'm'; - _key_table[XKeysymToKeycode(_display, XK_N)] = 'n'; - _key_table[XKeysymToKeycode(_display, XK_O)] = 'o'; - _key_table[XKeysymToKeycode(_display, XK_P)] = 'p'; - _key_table[XKeysymToKeycode(_display, XK_Q)] = 'q'; - _key_table[XKeysymToKeycode(_display, XK_R)] = 'r'; - _key_table[XKeysymToKeycode(_display, XK_S)] = 's'; - _key_table[XKeysymToKeycode(_display, XK_T)] = 't'; - _key_table[XKeysymToKeycode(_display, XK_U)] = 'u'; - _key_table[XKeysymToKeycode(_display, XK_V)] = 'v'; - _key_table[XKeysymToKeycode(_display, XK_W)] = 'w'; - _key_table[XKeysymToKeycode(_display, XK_X)] = 'x'; - _key_table[XKeysymToKeycode(_display, XK_Y)] = 'y'; - _key_table[XKeysymToKeycode(_display, XK_Z)] = 'z'; - _key_table[XKeysymToKeycode(_display, XK_space)] = ' '; - _key_table[XKeysymToKeycode(_display, XK_braceleft)] = '['; - _key_table[XKeysymToKeycode(_display, XK_braceright)] = ']'; - _key_table[XKeysymToKeycode(_display, XK_colon)] = ';'; - _key_table[XKeysymToKeycode(_display, XK_quotedbl)] = '\''; - _key_table[XKeysymToKeycode(_display, XK_asciitilde)] = '`'; - _key_table[XKeysymToKeycode(_display, XK_backslash)] = '\\'; - _key_table[XKeysymToKeycode(_display, XK_comma)] = ','; - _key_table[XKeysymToKeycode(_display, XK_greater)] = '.'; - _key_table[XKeysymToKeycode(_display, XK_question)] = '/'; - _key_table[XKeysymToKeycode(_display, XK_minus)] = '-'; - _key_table[XKeysymToKeycode(_display, XK_equal)] = '='; - _key_table[XKeysymToKeycode(_display, XK_F1)] = KEY_F_ + 1; - _key_table[XKeysymToKeycode(_display, XK_F2)] = KEY_F_ + 2; - _key_table[XKeysymToKeycode(_display, XK_F3)] = KEY_F_ + 3; - _key_table[XKeysymToKeycode(_display, XK_F4)] = KEY_F_ + 4; - _key_table[XKeysymToKeycode(_display, XK_F5)] = KEY_F_ + 5; - _key_table[XKeysymToKeycode(_display, XK_F6)] = KEY_F_ + 6; - _key_table[XKeysymToKeycode(_display, XK_F7)] = KEY_F_ + 7; - _key_table[XKeysymToKeycode(_display, XK_F8)] = KEY_F_ + 8; - _key_table[XKeysymToKeycode(_display, XK_F9)] = KEY_F_ + 9; - _key_table[XKeysymToKeycode(_display, XK_F10)] = KEY_F_ + 10; - _key_table[XKeysymToKeycode(_display, XK_F11)] = KEY_F_ + 11; - _key_table[XKeysymToKeycode(_display, XK_F12)] = KEY_F_ + 12; - _key_table[XKeysymToKeycode(_display, XK_Control_L)] = KEY_LCTRL; - _key_table[XKeysymToKeycode(_display, XK_Control_R)] = KEY_RCTRL; - _key_table[XKeysymToKeycode(_display, XK_Shift_L)] = KEY_LSHIFT; - _key_table[XKeysymToKeycode(_display, XK_Shift_R)] = KEY_RSHIFT; - _key_table[XKeysymToKeycode(_display, XK_Alt_L)] = KEY_LALT; - _key_table[XKeysymToKeycode(_display, XK_Alt_R)] = KEY_RALT; - _key_table[XKeysymToKeycode(_display, XK_Escape)] = KEY_ESCAPE; - _key_table[XKeysymToKeycode(_display, XK_BackSpace)] = '\b'; - _key_table[XKeysymToKeycode(_display, XK_Tab)] = '\t'; - _key_table[XKeysymToKeycode(_display, XK_Return)] = '\n'; - _key_table[XKeysymToKeycode(_display, XK_Print)] = KEY_PRINTSCREEN; - _key_table[XKeysymToKeycode(_display, XK_Delete)] = KEY_DELETE; - _key_table[XKeysymToKeycode(_display, XK_Pause)] = KEY_PAUSE; - _key_table[XKeysymToKeycode(_display, XK_Insert)] = KEY_INSERT; - _key_table[XKeysymToKeycode(_display, XK_Home)] = KEY_HOME; - _key_table[XKeysymToKeycode(_display, XK_End)] = KEY_END; - _key_table[XKeysymToKeycode(_display, XK_Page_Up)] = KEY_PAGEUP; - _key_table[XKeysymToKeycode(_display, XK_Page_Down)] = KEY_PAGEDOWN; - _key_table[XKeysymToKeycode(_display, XK_KP_0)] = KEY_KP_ + '0'; - _key_table[XKeysymToKeycode(_display, XK_KP_1)] = KEY_KP_ + '1'; - _key_table[XKeysymToKeycode(_display, XK_KP_2)] = KEY_KP_ + '2'; - _key_table[XKeysymToKeycode(_display, XK_KP_3)] = KEY_KP_ + '3'; - _key_table[XKeysymToKeycode(_display, XK_KP_4)] = KEY_KP_ + '4'; - _key_table[XKeysymToKeycode(_display, XK_KP_5)] = KEY_KP_ + '5'; - _key_table[XKeysymToKeycode(_display, XK_KP_6)] = KEY_KP_ + '6'; - _key_table[XKeysymToKeycode(_display, XK_KP_7)] = KEY_KP_ + '7'; - _key_table[XKeysymToKeycode(_display, XK_KP_8)] = KEY_KP_ + '8'; - _key_table[XKeysymToKeycode(_display, XK_KP_9)] = KEY_KP_ + '9'; - _key_table[XKeysymToKeycode(_display, XK_KP_Enter)] = KEY_KP_ + '\n'; - _key_table[XKeysymToKeycode(_display, XK_KP_Divide)] = KEY_KP_ + '/'; - _key_table[XKeysymToKeycode(_display, XK_KP_Multiply)] = KEY_KP_ + '*'; - _key_table[XKeysymToKeycode(_display, XK_KP_Add)] = KEY_KP_ + '+'; - _key_table[XKeysymToKeycode(_display, XK_KP_Subtract)] = KEY_KP_ + '-'; - _key_table[XKeysymToKeycode(_display, XK_KP_Decimal)] = KEY_KP_ + '.'; - _key_table[XKeysymToKeycode(_display, XK_KP_Separator)] = KEY_KP_ + ','; - - i32 screen = DefaultScreen(_display); - i32 depth = DefaultDepth (_display, screen); - Visual *visual = DefaultVisual(_display, screen); - - _gc = DefaultGC(_display, screen); - assert(_gc != NULL); - - XSetGraphicsExposures(_display, _gc, False); - - i32 _display_width = DisplayWidth (_display, screen); - i32 _display_height = DisplayHeight(_display, screen); - - i32 x = (_display_width - platform.frame_width) / 2; - i32 y = (_display_height - platform.frame_height) / 2; - - _window = XCreateWindow(_display, XDefaultRootWindow(_display), x, y, platform.frame_width, platform.frame_height, 0, depth, InputOutput, visual, CWEventMask, &(XSetWindowAttributes) { .event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | SubstructureNotifyMask, }); - - _im = XOpenIM(_display, NULL, NULL, NULL); - assert(_im != NULL); - - _ic = XCreateIC(_im, XNInputStyle, XIMPreeditNothing | XIMStatusNothing, XNClientWindow, _window, NULL); - assert(_ic != NULL); - - platform.pixels = _buffer; - platform.input = _input; - platform.clipboard = _clipboard_buffer; - - _image = (XImage) { - .width = platform.frame_width, - .height = platform.frame_height, - .depth = depth, - .xoffset = 0, - .format = ZPixmap, - .data = (c8 *) _buffer, - .byte_order = LSBFirst, - .bitmap_unit = 32, - .bitmap_bit_order = LSBFirst, - .bitmap_pad = 32, - .bits_per_pixel = 32, - .bytes_per_line = 4 * platform.frame_width, - .red_mask = 0xff0000, - .green_mask = 0x00ff00, - .blue_mask = 0x0000ff, - }; - - XInitImage(&_image); - - _wm_delete_window = XInternAtom(_display, "WM_DELETE_WINDOW", False); - _clipboard = XInternAtom(_display, "CLIPBOARD", False); - _targets = XInternAtom(_display, "TARGETS", False); - _utf8_string = XInternAtom(_display, "UTF8_STRING", False); - - XSetICFocus(_ic); - XSetWMProtocols(_display, _window, &_wm_delete_window, 1); - - XStoreName(_display, _window, platform.title); - XMapWindow(_display, _window); -} - -void p_cleanup(void) { - if (!platform.graceful_exit) - return; - - if (_window != 0) - XDestroyWindow(_display, _window); - if (_display != NULL) - XCloseDisplay (_display); - - _display = NULL; - _window = 0; - - sockets_cleanup(); -} - -i32 p_handle_events(void) { - i32 num_events = 0; - - memset(platform.key_pressed, 0, sizeof platform.key_pressed); - - platform.input_size = 0; - platform.cursor_dx = 0; - platform.cursor_dy = 0; - platform.wheel_dy = 0; - - XEvent ev; - - b8 requested_clipboard = 0; - - while (XEventsQueued(_display, QueuedAlready) > 0) { - ++num_events; - - XNextEvent(_display, &ev); - XFilterEvent(&ev, _window); - - switch (ev.type) { - case DestroyNotify: - platform.done = 1; - break; - - case MotionNotify: - platform.cursor_dx += ev.xmotion.x - platform.cursor_x; - platform.cursor_dy += ev.xmotion.y - platform.cursor_y; - platform.cursor_x = ev.xmotion.x; - platform.cursor_y = ev.xmotion.y; - break; - - case ButtonPress: - platform.cursor_x = ev.xbutton.x; - platform.cursor_y = ev.xbutton.y; - switch (ev.xbutton.button) { - case Button1: - platform.key_down[BUTTON_LEFT] = 1; - ++platform.key_pressed[BUTTON_LEFT]; - break; - case Button2: - platform.key_down[BUTTON_MIDDLE] = 1; - ++platform.key_pressed[BUTTON_MIDDLE]; - break; - case Button3: - platform.key_down[BUTTON_RIGHT] = 1; - ++platform.key_pressed[BUTTON_RIGHT]; - break; - case Button4: ++platform.wheel_dy; break; - case Button5: --platform.wheel_dy; break; - default:; - } - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - break; - - case ButtonRelease: - platform.cursor_x = ev.xbutton.x; - platform.cursor_y = ev.xbutton.y; - switch (ev.xbutton.button) { - case Button1: platform.key_down[BUTTON_LEFT] = 0; break; - case Button2: platform.key_down[BUTTON_MIDDLE] = 0; break; - case Button3: platform.key_down[BUTTON_RIGHT] = 0; break; - default:; - } - break; - - case KeyPress: { - i16 k = _key_table[ev.xkey.keycode]; - platform.cursor_x = ev.xkey.x; - platform.cursor_y = ev.xkey.y; - platform.key_down [k] = 1; - platform.key_pressed[k]++; - platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); - platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); - platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); - platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); - platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); - platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); - if (platform.input_size < MAX_INPUT_SIZE) { - if (k < 32 || k >= 128) - platform.input[platform.input_size++] = (Input_Key) { - .ctrl = !!(ev.xkey.state & ControlMask), - .shift = !!(ev.xkey.state & ShiftMask), - .alt = !!(ev.xkey.state & Mod1Mask), - .caps = !!(ev.xkey.state & LockMask), - .num = !!(ev.xkey.state & Mod2Mask), - .scroll = !!(ev.xkey.state & Mod3Mask), - .key = k, - .c = 0, - }; - else { - c8 buf[16]; - i32 len = Xutf8LookupString(_ic, &ev.xkey, buf, sizeof buf - 1, NULL, NULL); - if (len > 0) - platform.input[platform.input_size++] = (Input_Key) { - .ctrl = !!(ev.xkey.state & ControlMask), - .shift = !!(ev.xkey.state & ShiftMask), - .alt = !!(ev.xkey.state & Mod1Mask), - .caps = !!(ev.xkey.state & LockMask), - .num = !!(ev.xkey.state & Mod2Mask), - .scroll = !!(ev.xkey.state & Mod3Mask), - .key = k, - .c = utf8_read(len, buf), - }; - } - } - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - } break; - - case KeyRelease: { - i16 k = _key_table[ev.xkey.keycode]; - platform.cursor_x = ev.xkey.x; - platform.cursor_y = ev.xkey.y; - platform.key_down [k] = 0; - platform.key_pressed[k]--; - platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); - platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); - platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); - platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); - platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); - platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); - } break; - - case SelectionRequest: - if (ev.xselectionrequest.requestor != _window && - ev.xselectionrequest.selection == _clipboard && - XGetSelectionOwner(_display, _clipboard) == _window) { - if (ev.xselectionrequest.property != None) { - if (ev.xselectionrequest.target == _targets) - XChangeProperty( - ev.xselectionrequest.display, - ev.xselectionrequest.requestor, - ev.xselectionrequest.property, - XA_ATOM, - 32, - PropModeReplace, - (u8 *) &_utf8_string, - 1 - ); - else if (ev.xselectionrequest.target == _utf8_string) - XChangeProperty( - ev.xselectionrequest.display, - ev.xselectionrequest.requestor, - ev.xselectionrequest.property, - ev.xselectionrequest.target, - 8, - PropModeReplace, - (u8 *) platform.clipboard, - platform.clipboard_size - ); - } - - XSendEvent(_display, ev.xselectionrequest.requestor, 0, 0, (XEvent *) &(XSelectionEvent) { - .type = SelectionNotify, - .serial = ev.xselectionrequest.serial, - .send_event = ev.xselectionrequest.send_event, - .display = ev.xselectionrequest.display, - .requestor = ev.xselectionrequest.requestor, - .selection = ev.xselectionrequest.selection, - .target = ev.xselectionrequest.target, - .property = ev.xselectionrequest.property, - .time = ev.xselectionrequest.time, - }); - } - break; - - case SelectionNotify: - if (ev.xselection.property != None) { - i64 len; - u8 *data; - - XGetWindowProperty( - _display, - _window, - _clipboard, - 0, - MAX_CLIPBOARD_SIZE, - False, - AnyPropertyType, - &(Atom) {0}, - &(int) {0}, - (unsigned long *) &len, - &(unsigned long) {0}, - &data - ); - - if (ev.xselection.target == _targets) { - _target = None; - Atom *list = (Atom *) data; - - for (i64 i = 0; i < len; i++) - if (list[i] == XA_STRING) - _target = XA_STRING; - else if (list[i] == _utf8_string) { - _target = _utf8_string; - break; - } - - if (_target != None) - XConvertSelection(_display, _clipboard, _target, _clipboard, _window, CurrentTime); - } else if (ev.xselection.target == _target) { - if (len > MAX_CLIPBOARD_SIZE) - len = MAX_CLIPBOARD_SIZE; - platform.clipboard_size = len; - if (len > 0) - memcpy(platform.clipboard, data, len); - } - - if (data) - XFree(data); - } - break; - - case EnterNotify: platform.has_cursor = 1; break; - case LeaveNotify: platform.has_cursor = 0; break; - case FocusOut: platform.has_focus = 0; break; - - case FocusIn: - platform.has_focus = 1; - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - break; - - case MappingNotify: - XRefreshKeyboardMapping(&ev.xmapping); - break; - - case ClientMessage: - if ((Atom) ev.xclient.data.l[0] == _wm_delete_window) - platform.done = 1; - break; - - default:; - } - } - - for (i64 k = 0; k < (i64) (sizeof platform.key_pressed / sizeof *platform.key_pressed); ++k) - platform.key_pressed[k] = platform.key_pressed[k] > 0 ? 1 : 0; - - XWindowAttributes attrs; - XGetWindowAttributes(_display, _window, &attrs); - - if ((platform.frame_width != attrs.width || platform.frame_height != attrs.height) && attrs.width * attrs.height * 4 <= (i32) sizeof _buffer) { - if (attrs.width > 0 && attrs.height > 0) { - _image.width = attrs.width; - _image.height = attrs.height; - _image.bytes_per_line = 4 * attrs.width; - } - - platform.frame_width = attrs.width; - platform.frame_height = attrs.height; - } - - return num_events; -} - -i32 p_wait_events(void) { - i32 num_events = 0; - - do { - num_events = p_handle_events(); - sched_yield(); - } while (num_events == 0); - - return num_events; -} - -void p_render_frame(void) { - if (platform.done) - return; - - XPutImage(_display, _window, _gc, &_image, 0, 0, 0, 0, platform.frame_width, platform.frame_height); - XFlush(_display); -} - -void p_clipboard_write(i64 size, c8 *data) { - assert(size <= MAX_CLIPBOARD_SIZE); - - XSetSelectionOwner(_display, _clipboard, _window, CurrentTime); - - platform.clipboard_size = size < MAX_CLIPBOARD_SIZE ? size : MAX_CLIPBOARD_SIZE; - if (platform.clipboard_size > 0) - memcpy(platform.clipboard, data, platform.clipboard_size); -} - -#endif - -// ================================================================ -// -// ALSA -// -// ================================================================ - -#ifdef __linux__ - -void p_handle_audio(i64 time_elapsed) { - (void) time_elapsed; - assert(0); -} - -void p_queue_sound(i64 delay, i64 num_samples, f32 *samples) { - (void) delay; - (void) num_samples; - (void) samples; - assert(0); -} - -#endif diff --git a/examples/gravity.c b/examples/gravity.c index 7c6faf0..70f279f 100755 --- a/examples/gravity.c +++ b/examples/gravity.c @@ -3,31 +3,6 @@ #/ #/ gravity.c #/ -#/ This is a reduced system layer. -#/ It allows you to create a window, draw graphics in it, handle -#/ input events. -#/ -#/ ---------------------------------------------------------------- -#/ -#/ DESIGN PRINCIPLES -#/ -#/ - Minimalistic feature set. For graphics, you have access to the -#/ pixel buffer, and that's it. -#/ -#/ - No implicit control flow. No callbacks. You write your own -#/ main and call everything explicitly. But the number of things -#/ you have to call to do something is as little as possible. -#/ -#/ - Optimized to use in a single source file. -#/ Installation process? Ctrl+C, Ctrl+V, done. -#/ -#/ If you have an idea how to reduce the feature set further, -#/ let me know! -#/ -#/ ---------------------------------------------------------------- -#/ -#/ (C) 2024 Mitya Selivanov <guattari.tech>, MIT License -#/ #/ ================================================================ #/ #/ Self-compilation shell script @@ -49,159 +24,8 @@ gcc \ ./$BIN $@ && rm $BIN exit $? # */ #endif -// ================================================================ -// -// Basic declarations -// -// ================================================================ - -#define _GNU_SOURCE - -typedef signed char i8; -typedef signed short i16; -typedef signed i32; -typedef signed long long i64; -typedef unsigned char u8; -typedef unsigned short u16; -typedef unsigned u32; -typedef unsigned long long u64; -typedef char c8; -typedef int c32; -typedef signed char b8; -typedef float f32; -typedef double f64; - -// ================================================================ -// -// PLATFORM API -// -// ================================================================ - -enum { - MAX_NUM_PIXELS = 10 * 1024 * 1024, - MAX_INPUT_SIZE = 256, - MAX_CLIPBOARD_SIZE = 10 * 1024 * 1024, - MAX_NUM_AUDIO_SAMPLES = 0, - MAX_NUM_SOCKETS = 0, - - AUDIO_NUM_CHANNELS = 2, - AUDIO_SAMPLE_RATE = 44100, - - IPv4 = 1, - IPv6 = 2, - - KEY_LEFT = 128, - KEY_RIGHT, - KEY_UP, - KEY_DOWN, - KEY_LCTRL, - KEY_RCTRL, - KEY_LSHIFT, - KEY_RSHIFT, - KEY_LALT, - KEY_RALT, - KEY_ESCAPE, - KEY_PRINTSCREEN, - KEY_DELETE, - KEY_PAUSE, - KEY_INSERT, - KEY_HOME, - KEY_END, - KEY_PAGEUP, - KEY_PAGEDOWN, - BUTTON_LEFT, - BUTTON_MIDDLE, - BUTTON_RIGHT, - MOD_CTRL, - MOD_SHIFT, - MOD_ALT, - MOD_CAPS, - MOD_NUM, - MOD_SCROLL, - KEY_F_, - KEY_KP_ = KEY_F_ + 64, -}; - -typedef struct { - b8 ctrl : 1; - b8 shift : 1; - b8 alt : 1; - b8 caps : 1; - b8 num : 1; - b8 scroll : 1; - u16 key; - c32 c; -} Input_Key; - -typedef struct { - c8 * title; - i32 frame_width; - i32 frame_height; - u32 * pixels; - i64 input_size; - Input_Key *input; - i64 clipboard_size; - c8 * clipboard; - b8 done; - b8 has_focus; - b8 has_cursor; - i32 cursor_x; - i32 cursor_y; - i32 cursor_dx; - i32 cursor_dy; - i64 wheel_dy; - b8 key_down[512]; - b8 key_pressed[512]; -} Platform; -typedef struct { - u16 type; - u16 port; - union { - u8 v4_address[4]; - u8 v6_address[16]; - }; -} IP_Address; - -typedef i64 (*Thread_Proc)(void *user_data); - -// UTF-8 -i32 utf8_size(c32 c); -c32 utf8_read(i64 len, c8 *s); -i32 utf8_write(c32 c, c8 *buffer); - -// Window -void p_init(void); -void p_cleanup(void); -i32 p_handle_events(void); -i32 p_wait_events(void); -void p_render_frame(void); - -// Clipboard -void p_clipboard_write(i64 size, c8 *data); - -// Sound -void p_handle_audio(i64 time_elapsed); -void p_queue_sound(i64 delay, i64 num_samples, f32 *samples); - -// UDP sockets -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address); -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data); - -Platform platform = {0}; - -// ================================================================ -// -// GAME -// -// ================================================================ - -#include <stdio.h> -#include <stdlib.h> -#include <assert.h> -#include <string.h> -#include <time.h> -#include <math.h> +#include "../reduced_system_layer.c" typedef struct { f64 x; @@ -351,628 +175,3 @@ i32 main(i32 argc, c8 **argv) { p_cleanup(); return 0; } - -// ================================================================ -// -// PLATFORM IMPLEMENTATION -// -// ---------------------------------------------------------------- -// -// TODO -// - Sound -// - Sockets -// - Clipboard daemon -// -// ALSA -// https://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_min_8c-example.html -// -// ================================================================ - -#include <assert.h> -#include <string.h> - -c32 utf8_read(i64 len, c8 *s) { - if (len >= 1 && - (s[0] & 0x80) == 0) - return s[0]; - if (len >= 2 && - (s[0] & 0xe0) == 0xc0 && - (s[1] & 0xc0) == 0x80) - return (s[1] & 0x3f) - | ((s[0] & 0x1f) << 6); - if (len >= 3 && - (s[0] & 0xf0) == 0xe0 && - (s[1] & 0xc0) == 0x80 && - (s[2] & 0xc0) == 0x80) - return (s[2] & 0x3f) - | ((s[1] & 0x3f) << 6) - | ((s[0] & 0x0f) << 12); - if (len >= 4 && - (s[0] & 0xf8) == 0xf0 && - (s[1] & 0xc0) == 0x80 && - (s[2] & 0xc0) == 0x80 && - (s[3] & 0xc0) == 0x80) - return (s[3] & 0x3f) - | ((s[2] & 0x3f) << 6) - | ((s[1] & 0x3f) << 12) - | ((s[0] & 0x07) << 18); - return 0; -} - -i32 utf8_size(c32 c) { - if ((c & 0x00007f) == c) return 1; - if ((c & 0x0007ff) == c) return 2; - if ((c & 0x00ffff) == c) return 3; - if ((c & 0x1fffff) == c) return 4; - return 0; -} - -i32 utf8_write(c32 c, c8 *buffer) { - if ((c & 0x7f) == c) { - buffer[0] = (c8) c; - return 1; - } - - if ((c & 0x7ff) == c) { - buffer[0] = 0xc0 | ((c >> 6) & 0x1f); - buffer[1] = 0x80 | ( c & 0x3f); - return 2; - } - - if ((c & 0xffff) == c) { - buffer[0] = 0xc0 | ((c >> 12) & 0x0f); - buffer[1] = 0x80 | ((c >> 6) & 0x3f); - buffer[2] = 0x80 | ( c & 0x3f); - return 3; - } - - if ((c & 0x1fffff) == c) { - buffer[0] = 0xc0 | ((c >> 18) & 0x07); - buffer[1] = 0x80 | ((c >> 12) & 0x3f); - buffer[2] = 0x80 | ((c >> 6) & 0x3f); - buffer[3] = 0x80 | ( c & 0x3f); - return 4; - } - - return 0; -} - -// ================================================================ -// -// UDP sockets -// -// ================================================================ - -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address) { - (void) slot; - (void) address; - (void) size; - (void) data; - (void) remote_address; - assert(0); - return 0; -} - -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data) { - (void) slot; - (void) address; - (void) size; - (void) data; - assert(0); - return 0; -} - -// ================================================================ -// -// X11 -// -// ================================================================ - -#ifdef __linux__ - -#include <X11/Xlib.h> -#include <X11/Xutil.h> -#include <X11/Xatom.h> -#include <sched.h> -#include <time.h> - -static i16 _key_table[512] = {0}; -static u32 _buffer[MAX_NUM_PIXELS] = {0}; -static Input_Key _input[MAX_INPUT_SIZE] = {0}; -static c8 _clipboard_buffer[MAX_CLIPBOARD_SIZE] = {0}; -static XImage _image = {0}; -static Display * _display = NULL; -static GC _gc = NULL; -static XIM _im = NULL; -static XIC _ic = NULL; -static Window _window = 0; -static Atom _wm_delete_window = 0; -static Atom _clipboard = 0; -static Atom _targets = 0; -static Atom _utf8_string = 0; -static Atom _target = None; - -void p_init(void) { - _display = XOpenDisplay(NULL); - assert(_display != NULL); - - _key_table[XKeysymToKeycode(_display, XK_Left)] = KEY_LEFT; - _key_table[XKeysymToKeycode(_display, XK_Right)] = KEY_RIGHT; - _key_table[XKeysymToKeycode(_display, XK_Up)] = KEY_UP; - _key_table[XKeysymToKeycode(_display, XK_Down)] = KEY_DOWN; - _key_table[XKeysymToKeycode(_display, XK_1)] = '1'; - _key_table[XKeysymToKeycode(_display, XK_2)] = '2'; - _key_table[XKeysymToKeycode(_display, XK_3)] = '3'; - _key_table[XKeysymToKeycode(_display, XK_4)] = '4'; - _key_table[XKeysymToKeycode(_display, XK_5)] = '5'; - _key_table[XKeysymToKeycode(_display, XK_6)] = '6'; - _key_table[XKeysymToKeycode(_display, XK_7)] = '7'; - _key_table[XKeysymToKeycode(_display, XK_8)] = '8'; - _key_table[XKeysymToKeycode(_display, XK_9)] = '9'; - _key_table[XKeysymToKeycode(_display, XK_0)] = '0'; - _key_table[XKeysymToKeycode(_display, XK_A)] = 'a'; - _key_table[XKeysymToKeycode(_display, XK_B)] = 'b'; - _key_table[XKeysymToKeycode(_display, XK_C)] = 'c'; - _key_table[XKeysymToKeycode(_display, XK_D)] = 'd'; - _key_table[XKeysymToKeycode(_display, XK_E)] = 'e'; - _key_table[XKeysymToKeycode(_display, XK_F)] = 'f'; - _key_table[XKeysymToKeycode(_display, XK_G)] = 'g'; - _key_table[XKeysymToKeycode(_display, XK_H)] = 'h'; - _key_table[XKeysymToKeycode(_display, XK_I)] = 'i'; - _key_table[XKeysymToKeycode(_display, XK_J)] = 'j'; - _key_table[XKeysymToKeycode(_display, XK_K)] = 'k'; - _key_table[XKeysymToKeycode(_display, XK_L)] = 'l'; - _key_table[XKeysymToKeycode(_display, XK_M)] = 'm'; - _key_table[XKeysymToKeycode(_display, XK_N)] = 'n'; - _key_table[XKeysymToKeycode(_display, XK_O)] = 'o'; - _key_table[XKeysymToKeycode(_display, XK_P)] = 'p'; - _key_table[XKeysymToKeycode(_display, XK_Q)] = 'q'; - _key_table[XKeysymToKeycode(_display, XK_R)] = 'r'; - _key_table[XKeysymToKeycode(_display, XK_S)] = 's'; - _key_table[XKeysymToKeycode(_display, XK_T)] = 't'; - _key_table[XKeysymToKeycode(_display, XK_U)] = 'u'; - _key_table[XKeysymToKeycode(_display, XK_V)] = 'v'; - _key_table[XKeysymToKeycode(_display, XK_W)] = 'w'; - _key_table[XKeysymToKeycode(_display, XK_X)] = 'x'; - _key_table[XKeysymToKeycode(_display, XK_Y)] = 'y'; - _key_table[XKeysymToKeycode(_display, XK_Z)] = 'z'; - _key_table[XKeysymToKeycode(_display, XK_space)] = ' '; - _key_table[XKeysymToKeycode(_display, XK_braceleft)] = '['; - _key_table[XKeysymToKeycode(_display, XK_braceright)] = ']'; - _key_table[XKeysymToKeycode(_display, XK_colon)] = ';'; - _key_table[XKeysymToKeycode(_display, XK_quotedbl)] = '\''; - _key_table[XKeysymToKeycode(_display, XK_asciitilde)] = '`'; - _key_table[XKeysymToKeycode(_display, XK_backslash)] = '\\'; - _key_table[XKeysymToKeycode(_display, XK_comma)] = ','; - _key_table[XKeysymToKeycode(_display, XK_greater)] = '.'; - _key_table[XKeysymToKeycode(_display, XK_question)] = '/'; - _key_table[XKeysymToKeycode(_display, XK_minus)] = '-'; - _key_table[XKeysymToKeycode(_display, XK_equal)] = '='; - _key_table[XKeysymToKeycode(_display, XK_F1)] = KEY_F_ + 1; - _key_table[XKeysymToKeycode(_display, XK_F2)] = KEY_F_ + 2; - _key_table[XKeysymToKeycode(_display, XK_F3)] = KEY_F_ + 3; - _key_table[XKeysymToKeycode(_display, XK_F4)] = KEY_F_ + 4; - _key_table[XKeysymToKeycode(_display, XK_F5)] = KEY_F_ + 5; - _key_table[XKeysymToKeycode(_display, XK_F6)] = KEY_F_ + 6; - _key_table[XKeysymToKeycode(_display, XK_F7)] = KEY_F_ + 7; - _key_table[XKeysymToKeycode(_display, XK_F8)] = KEY_F_ + 8; - _key_table[XKeysymToKeycode(_display, XK_F9)] = KEY_F_ + 9; - _key_table[XKeysymToKeycode(_display, XK_F10)] = KEY_F_ + 10; - _key_table[XKeysymToKeycode(_display, XK_F11)] = KEY_F_ + 11; - _key_table[XKeysymToKeycode(_display, XK_F12)] = KEY_F_ + 12; - _key_table[XKeysymToKeycode(_display, XK_Control_L)] = KEY_LCTRL; - _key_table[XKeysymToKeycode(_display, XK_Control_R)] = KEY_RCTRL; - _key_table[XKeysymToKeycode(_display, XK_Shift_L)] = KEY_LSHIFT; - _key_table[XKeysymToKeycode(_display, XK_Shift_R)] = KEY_RSHIFT; - _key_table[XKeysymToKeycode(_display, XK_Alt_L)] = KEY_LALT; - _key_table[XKeysymToKeycode(_display, XK_Alt_R)] = KEY_RALT; - _key_table[XKeysymToKeycode(_display, XK_Escape)] = KEY_ESCAPE; - _key_table[XKeysymToKeycode(_display, XK_BackSpace)] = '\b'; - _key_table[XKeysymToKeycode(_display, XK_Tab)] = '\t'; - _key_table[XKeysymToKeycode(_display, XK_Return)] = '\n'; - _key_table[XKeysymToKeycode(_display, XK_Print)] = KEY_PRINTSCREEN; - _key_table[XKeysymToKeycode(_display, XK_Delete)] = KEY_DELETE; - _key_table[XKeysymToKeycode(_display, XK_Pause)] = KEY_PAUSE; - _key_table[XKeysymToKeycode(_display, XK_Insert)] = KEY_INSERT; - _key_table[XKeysymToKeycode(_display, XK_Home)] = KEY_HOME; - _key_table[XKeysymToKeycode(_display, XK_End)] = KEY_END; - _key_table[XKeysymToKeycode(_display, XK_Page_Up)] = KEY_PAGEUP; - _key_table[XKeysymToKeycode(_display, XK_Page_Down)] = KEY_PAGEDOWN; - _key_table[XKeysymToKeycode(_display, XK_KP_0)] = KEY_KP_ + '0'; - _key_table[XKeysymToKeycode(_display, XK_KP_1)] = KEY_KP_ + '1'; - _key_table[XKeysymToKeycode(_display, XK_KP_2)] = KEY_KP_ + '2'; - _key_table[XKeysymToKeycode(_display, XK_KP_3)] = KEY_KP_ + '3'; - _key_table[XKeysymToKeycode(_display, XK_KP_4)] = KEY_KP_ + '4'; - _key_table[XKeysymToKeycode(_display, XK_KP_5)] = KEY_KP_ + '5'; - _key_table[XKeysymToKeycode(_display, XK_KP_6)] = KEY_KP_ + '6'; - _key_table[XKeysymToKeycode(_display, XK_KP_7)] = KEY_KP_ + '7'; - _key_table[XKeysymToKeycode(_display, XK_KP_8)] = KEY_KP_ + '8'; - _key_table[XKeysymToKeycode(_display, XK_KP_9)] = KEY_KP_ + '9'; - _key_table[XKeysymToKeycode(_display, XK_KP_Enter)] = KEY_KP_ + '\n'; - _key_table[XKeysymToKeycode(_display, XK_KP_Divide)] = KEY_KP_ + '/'; - _key_table[XKeysymToKeycode(_display, XK_KP_Multiply)] = KEY_KP_ + '*'; - _key_table[XKeysymToKeycode(_display, XK_KP_Add)] = KEY_KP_ + '+'; - _key_table[XKeysymToKeycode(_display, XK_KP_Subtract)] = KEY_KP_ + '-'; - _key_table[XKeysymToKeycode(_display, XK_KP_Decimal)] = KEY_KP_ + '.'; - _key_table[XKeysymToKeycode(_display, XK_KP_Separator)] = KEY_KP_ + ','; - - i32 screen = DefaultScreen(_display); - i32 depth = DefaultDepth (_display, screen); - Visual *visual = DefaultVisual(_display, screen); - - _gc = DefaultGC(_display, screen); - assert(_gc != NULL); - - XSetGraphicsExposures(_display, _gc, False); - - i32 _display_width = DisplayWidth (_display, screen); - i32 _display_height = DisplayHeight(_display, screen); - - i32 x = (_display_width - platform.frame_width) / 2; - i32 y = (_display_height - platform.frame_height) / 2; - - _window = XCreateWindow(_display, XDefaultRootWindow(_display), x, y, platform.frame_width, platform.frame_height, 0, depth, InputOutput, visual, CWEventMask, &(XSetWindowAttributes) { .event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | SubstructureNotifyMask, }); - - _im = XOpenIM(_display, NULL, NULL, NULL); - assert(_im != NULL); - - _ic = XCreateIC(_im, XNInputStyle, XIMPreeditNothing | XIMStatusNothing, XNClientWindow, _window, NULL); - assert(_ic != NULL); - - platform.pixels = _buffer; - platform.input = _input; - platform.clipboard = _clipboard_buffer; - - _image = (XImage) { - .width = platform.frame_width, - .height = platform.frame_height, - .depth = depth, - .xoffset = 0, - .format = ZPixmap, - .data = (c8 *) _buffer, - .byte_order = LSBFirst, - .bitmap_unit = 32, - .bitmap_bit_order = LSBFirst, - .bitmap_pad = 32, - .bits_per_pixel = 32, - .bytes_per_line = 4 * platform.frame_width, - .red_mask = 0xff0000, - .green_mask = 0x00ff00, - .blue_mask = 0x0000ff, - }; - - XInitImage(&_image); - - _wm_delete_window = XInternAtom(_display, "WM_DELETE_WINDOW", False); - _clipboard = XInternAtom(_display, "CLIPBOARD", False); - _targets = XInternAtom(_display, "TARGETS", False); - _utf8_string = XInternAtom(_display, "UTF8_STRING", False); - - XSetICFocus(_ic); - XSetWMProtocols(_display, _window, &_wm_delete_window, 1); - - XStoreName(_display, _window, platform.title); - XMapWindow(_display, _window); -} - -void p_cleanup(void) { - XDestroyWindow(_display, _window); - XCloseDisplay (_display); -} - -i32 p_handle_events(void) { - i32 num_events = 0; - - memset(platform.key_pressed, 0, sizeof platform.key_pressed); - - platform.input_size = 0; - platform.cursor_dx = 0; - platform.cursor_dy = 0; - platform.wheel_dy = 0; - - XEvent ev; - - b8 requested_clipboard = 0; - - while (XEventsQueued(_display, QueuedAlready) > 0) { - ++num_events; - - XNextEvent(_display, &ev); - XFilterEvent(&ev, _window); - - switch (ev.type) { - case DestroyNotify: - platform.done = 1; - break; - - case MotionNotify: - platform.cursor_dx += ev.xmotion.x - platform.cursor_x; - platform.cursor_dy += ev.xmotion.y - platform.cursor_y; - platform.cursor_x = ev.xmotion.x; - platform.cursor_y = ev.xmotion.y; - break; - - case ButtonPress: - platform.cursor_x = ev.xbutton.x; - platform.cursor_y = ev.xbutton.y; - switch (ev.xbutton.button) { - case Button1: - platform.key_down[BUTTON_LEFT] = 1; - ++platform.key_pressed[BUTTON_LEFT]; - break; - case Button2: - platform.key_down[BUTTON_MIDDLE] = 1; - ++platform.key_pressed[BUTTON_MIDDLE]; - break; - case Button3: - platform.key_down[BUTTON_RIGHT] = 1; - ++platform.key_pressed[BUTTON_RIGHT]; - break; - case Button4: ++platform.wheel_dy; break; - case Button5: --platform.wheel_dy; break; - default:; - } - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - break; - - case ButtonRelease: - platform.cursor_x = ev.xbutton.x; - platform.cursor_y = ev.xbutton.y; - switch (ev.xbutton.button) { - case Button1: platform.key_down[BUTTON_LEFT] = 0; break; - case Button2: platform.key_down[BUTTON_MIDDLE] = 0; break; - case Button3: platform.key_down[BUTTON_RIGHT] = 0; break; - default:; - } - break; - - case KeyPress: { - i16 k = _key_table[ev.xkey.keycode]; - platform.cursor_x = ev.xkey.x; - platform.cursor_y = ev.xkey.y; - platform.key_down [k] = 1; - platform.key_pressed[k]++; - platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); - platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); - platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); - platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); - platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); - platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); - if (platform.input_size < MAX_INPUT_SIZE) { - if (k < 32 || k >= 128) - platform.input[platform.input_size++] = (Input_Key) { - .ctrl = !!(ev.xkey.state & ControlMask), - .shift = !!(ev.xkey.state & ShiftMask), - .alt = !!(ev.xkey.state & Mod1Mask), - .caps = !!(ev.xkey.state & LockMask), - .num = !!(ev.xkey.state & Mod2Mask), - .scroll = !!(ev.xkey.state & Mod3Mask), - .key = k, - .c = 0, - }; - else { - c8 buf[16]; - i32 len = Xutf8LookupString(_ic, &ev.xkey, buf, sizeof buf - 1, NULL, NULL); - if (len > 0) - platform.input[platform.input_size++] = (Input_Key) { - .ctrl = !!(ev.xkey.state & ControlMask), - .shift = !!(ev.xkey.state & ShiftMask), - .alt = !!(ev.xkey.state & Mod1Mask), - .caps = !!(ev.xkey.state & LockMask), - .num = !!(ev.xkey.state & Mod2Mask), - .scroll = !!(ev.xkey.state & Mod3Mask), - .key = k, - .c = utf8_read(len, buf), - }; - } - } - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - } break; - - case KeyRelease: { - i16 k = _key_table[ev.xkey.keycode]; - platform.cursor_x = ev.xkey.x; - platform.cursor_y = ev.xkey.y; - platform.key_down [k] = 0; - platform.key_pressed[k]--; - platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); - platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); - platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); - platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); - platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); - platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); - } break; - - case SelectionRequest: - if (ev.xselectionrequest.requestor != _window && - ev.xselectionrequest.selection == _clipboard && - XGetSelectionOwner(_display, _clipboard) == _window) { - if (ev.xselectionrequest.property != None) { - if (ev.xselectionrequest.target == _targets) - XChangeProperty( - ev.xselectionrequest.display, - ev.xselectionrequest.requestor, - ev.xselectionrequest.property, - XA_ATOM, - 32, - PropModeReplace, - (u8 *) &_utf8_string, - 1 - ); - else if (ev.xselectionrequest.target == _utf8_string) - XChangeProperty( - ev.xselectionrequest.display, - ev.xselectionrequest.requestor, - ev.xselectionrequest.property, - ev.xselectionrequest.target, - 8, - PropModeReplace, - (u8 *) platform.clipboard, - platform.clipboard_size - ); - } - - XSendEvent(_display, ev.xselectionrequest.requestor, 0, 0, (XEvent *) &(XSelectionEvent) { - .type = SelectionNotify, - .serial = ev.xselectionrequest.serial, - .send_event = ev.xselectionrequest.send_event, - .display = ev.xselectionrequest.display, - .requestor = ev.xselectionrequest.requestor, - .selection = ev.xselectionrequest.selection, - .target = ev.xselectionrequest.target, - .property = ev.xselectionrequest.property, - .time = ev.xselectionrequest.time, - }); - } - break; - - case SelectionNotify: - if (ev.xselection.property != None) { - i64 len; - u8 *data; - - XGetWindowProperty( - _display, - _window, - _clipboard, - 0, - MAX_CLIPBOARD_SIZE, - False, - AnyPropertyType, - &(Atom) {0}, - &(int) {0}, - (unsigned long *) &len, - &(unsigned long) {0}, - &data - ); - - if (ev.xselection.target == _targets) { - _target = None; - Atom *list = (Atom *) data; - - for (i64 i = 0; i < len; i++) - if (list[i] == XA_STRING) - _target = XA_STRING; - else if (list[i] == _utf8_string) { - _target = _utf8_string; - break; - } - - if (_target != None) - XConvertSelection(_display, _clipboard, _target, _clipboard, _window, CurrentTime); - } else if (ev.xselection.target == _target) { - if (len > MAX_CLIPBOARD_SIZE) - len = MAX_CLIPBOARD_SIZE; - platform.clipboard_size = len; - if (len > 0) - memcpy(platform.clipboard, data, len); - } - - if (data) - XFree(data); - } - break; - - case EnterNotify: platform.has_cursor = 1; break; - case LeaveNotify: platform.has_cursor = 0; break; - case FocusOut: platform.has_focus = 0; break; - - case FocusIn: - platform.has_focus = 1; - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - break; - - case MappingNotify: - XRefreshKeyboardMapping(&ev.xmapping); - break; - - case ClientMessage: - if ((Atom) ev.xclient.data.l[0] == _wm_delete_window) - platform.done = 1; - break; - - default:; - } - } - - for (i64 k = 0; k < (i64) (sizeof platform.key_pressed / sizeof *platform.key_pressed); ++k) - platform.key_pressed[k] = platform.key_pressed[k] > 0 ? 1 : 0; - - XWindowAttributes attrs; - XGetWindowAttributes(_display, _window, &attrs); - - if ((platform.frame_width != attrs.width || platform.frame_height != attrs.height) && attrs.width * attrs.height * 4 <= (i32) sizeof _buffer) { - if (attrs.width > 0 && attrs.height > 0) { - _image.width = attrs.width; - _image.height = attrs.height; - _image.bytes_per_line = 4 * attrs.width; - } - - platform.frame_width = attrs.width; - platform.frame_height = attrs.height; - } - - return num_events; -} - -i32 p_wait_events(void) { - i32 num_events = 0; - - do { - num_events = p_handle_events(); - sched_yield(); - } while (num_events == 0); - - return num_events; -} - -void p_render_frame(void) { - if (platform.done) - return; - - XPutImage(_display, _window, _gc, &_image, 0, 0, 0, 0, platform.frame_width, platform.frame_height); - XFlush(_display); -} - -void p_clipboard_write(i64 size, c8 *data) { - assert(size <= MAX_CLIPBOARD_SIZE); - - XSetSelectionOwner(_display, _clipboard, _window, CurrentTime); - - platform.clipboard_size = size < MAX_CLIPBOARD_SIZE ? size : MAX_CLIPBOARD_SIZE; - if (platform.clipboard_size > 0) - memcpy(platform.clipboard, data, platform.clipboard_size); -} - -#endif - -// ================================================================ -// -// ALSA -// -// ================================================================ - -#ifdef __linux__ - -void p_handle_audio(i64 time_elapsed) { - (void) time_elapsed; - assert(0); -} - -void p_queue_sound(i64 delay, i64 num_samples, f32 *samples) { - (void) delay; - (void) num_samples; - (void) samples; - assert(0); -} - -#endif diff --git a/examples/landscape.c b/examples/landscape.c index 813ad21..4690875 100755 --- a/examples/landscape.c +++ b/examples/landscape.c @@ -3,31 +3,6 @@ #/ #/ landscape.c #/ -#/ This is a reduced system layer. -#/ It allows you to create a window, draw graphics in it, handle -#/ input events. -#/ -#/ ---------------------------------------------------------------- -#/ -#/ DESIGN PRINCIPLES -#/ -#/ - Minimalistic feature set. For graphics, you have access to the -#/ pixel buffer, and that's it. -#/ -#/ - No implicit control flow. No callbacks. You write your own -#/ main and call everything explicitly. But the number of things -#/ you have to call to do something is as little as possible. -#/ -#/ - Optimized to use in a single source file. -#/ Installation process? Ctrl+C, Ctrl+V, done. -#/ -#/ If you have an idea how to reduce the feature set further, -#/ let me know! -#/ -#/ ---------------------------------------------------------------- -#/ -#/ (C) 2024 Mitya Selivanov <guattari.tech>, MIT License -#/ #/ ================================================================ #/ #/ Self-compilation shell script @@ -49,159 +24,8 @@ gcc \ ./$BIN $@ && rm $BIN exit $? # */ #endif -// ================================================================ -// -// Basic declarations -// -// ================================================================ -#define _GNU_SOURCE - -typedef signed char i8; -typedef signed short i16; -typedef signed i32; -typedef signed long long i64; -typedef unsigned char u8; -typedef unsigned short u16; -typedef unsigned u32; -typedef unsigned long long u64; -typedef char c8; -typedef int c32; -typedef signed char b8; -typedef float f32; -typedef double f64; - -// ================================================================ -// -// PLATFORM API -// -// ================================================================ - -enum { - MAX_NUM_PIXELS = 10 * 1024 * 1024, - MAX_INPUT_SIZE = 256, - MAX_CLIPBOARD_SIZE = 10 * 1024 * 1024, - MAX_NUM_AUDIO_SAMPLES = 0, - MAX_NUM_SOCKETS = 0, - - AUDIO_NUM_CHANNELS = 2, - AUDIO_SAMPLE_RATE = 44100, - - IPv4 = 1, - IPv6 = 2, - - KEY_LEFT = 128, - KEY_RIGHT, - KEY_UP, - KEY_DOWN, - KEY_LCTRL, - KEY_RCTRL, - KEY_LSHIFT, - KEY_RSHIFT, - KEY_LALT, - KEY_RALT, - KEY_ESCAPE, - KEY_PRINTSCREEN, - KEY_DELETE, - KEY_PAUSE, - KEY_INSERT, - KEY_HOME, - KEY_END, - KEY_PAGEUP, - KEY_PAGEDOWN, - BUTTON_LEFT, - BUTTON_MIDDLE, - BUTTON_RIGHT, - MOD_CTRL, - MOD_SHIFT, - MOD_ALT, - MOD_CAPS, - MOD_NUM, - MOD_SCROLL, - KEY_F_, - KEY_KP_ = KEY_F_ + 64, -}; - -typedef struct { - b8 ctrl : 1; - b8 shift : 1; - b8 alt : 1; - b8 caps : 1; - b8 num : 1; - b8 scroll : 1; - u16 key; - c32 c; -} Input_Key; - -typedef struct { - c8 * title; - i32 frame_width; - i32 frame_height; - u32 * pixels; - i64 input_size; - Input_Key *input; - i64 clipboard_size; - c8 * clipboard; - b8 done; - b8 has_focus; - b8 has_cursor; - i32 cursor_x; - i32 cursor_y; - i32 cursor_dx; - i32 cursor_dy; - i64 wheel_dy; - b8 key_down[512]; - b8 key_pressed[512]; -} Platform; - -typedef struct { - u16 type; - u16 port; - union { - u8 v4_address[4]; - u8 v6_address[16]; - }; -} IP_Address; - -typedef i64 (*Thread_Proc)(void *user_data); - -// UTF-8 -i32 utf8_size(c32 c); -c32 utf8_read(i64 len, c8 *s); -i32 utf8_write(c32 c, c8 *buffer); - -// Window -void p_init(void); -void p_cleanup(void); -i32 p_handle_events(void); -i32 p_wait_events(void); -void p_render_frame(void); - -// Clipboard -void p_clipboard_write(i64 size, c8 *data); - -// Sound -void p_handle_audio(i64 time_elapsed); -void p_queue_sound(i64 delay, i64 num_samples, f32 *samples); - -// UDP sockets -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address); -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data); - -Platform platform = {0}; - -// ================================================================ -// -// GAME -// -// ================================================================ - -#include <stdio.h> -#include <stdlib.h> -#include <assert.h> -#include <string.h> -#include <time.h> -#include <math.h> +#include "../reduced_system_layer.c" #define EPS 1e-8 @@ -469,628 +293,3 @@ i32 main(i32 argc, c8 **argv) { p_cleanup(); return 0; } - -// ================================================================ -// -// PLATFORM IMPLEMENTATION -// -// ---------------------------------------------------------------- -// -// TODO -// - Sound -// - Sockets -// - Clipboard daemon -// -// ALSA -// https://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_min_8c-example.html -// -// ================================================================ - -#include <assert.h> -#include <string.h> - -c32 utf8_read(i64 len, c8 *s) { - if (len >= 1 && - (s[0] & 0x80) == 0) - return s[0]; - if (len >= 2 && - (s[0] & 0xe0) == 0xc0 && - (s[1] & 0xc0) == 0x80) - return (s[1] & 0x3f) - | ((s[0] & 0x1f) << 6); - if (len >= 3 && - (s[0] & 0xf0) == 0xe0 && - (s[1] & 0xc0) == 0x80 && - (s[2] & 0xc0) == 0x80) - return (s[2] & 0x3f) - | ((s[1] & 0x3f) << 6) - | ((s[0] & 0x0f) << 12); - if (len >= 4 && - (s[0] & 0xf8) == 0xf0 && - (s[1] & 0xc0) == 0x80 && - (s[2] & 0xc0) == 0x80 && - (s[3] & 0xc0) == 0x80) - return (s[3] & 0x3f) - | ((s[2] & 0x3f) << 6) - | ((s[1] & 0x3f) << 12) - | ((s[0] & 0x07) << 18); - return 0; -} - -i32 utf8_size(c32 c) { - if ((c & 0x00007f) == c) return 1; - if ((c & 0x0007ff) == c) return 2; - if ((c & 0x00ffff) == c) return 3; - if ((c & 0x1fffff) == c) return 4; - return 0; -} - -i32 utf8_write(c32 c, c8 *buffer) { - if ((c & 0x7f) == c) { - buffer[0] = (c8) c; - return 1; - } - - if ((c & 0x7ff) == c) { - buffer[0] = 0xc0 | ((c >> 6) & 0x1f); - buffer[1] = 0x80 | ( c & 0x3f); - return 2; - } - - if ((c & 0xffff) == c) { - buffer[0] = 0xc0 | ((c >> 12) & 0x0f); - buffer[1] = 0x80 | ((c >> 6) & 0x3f); - buffer[2] = 0x80 | ( c & 0x3f); - return 3; - } - - if ((c & 0x1fffff) == c) { - buffer[0] = 0xc0 | ((c >> 18) & 0x07); - buffer[1] = 0x80 | ((c >> 12) & 0x3f); - buffer[2] = 0x80 | ((c >> 6) & 0x3f); - buffer[3] = 0x80 | ( c & 0x3f); - return 4; - } - - return 0; -} - -// ================================================================ -// -// UDP sockets -// -// ================================================================ - -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address) { - (void) slot; - (void) address; - (void) size; - (void) data; - (void) remote_address; - assert(0); - return 0; -} - -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data) { - (void) slot; - (void) address; - (void) size; - (void) data; - assert(0); - return 0; -} - -// ================================================================ -// -// X11 -// -// ================================================================ - -#ifdef __linux__ - -#include <X11/Xlib.h> -#include <X11/Xutil.h> -#include <X11/Xatom.h> -#include <sched.h> -#include <time.h> - -static i16 _key_table[512] = {0}; -static u32 _buffer[MAX_NUM_PIXELS] = {0}; -static Input_Key _input[MAX_INPUT_SIZE] = {0}; -static c8 _clipboard_buffer[MAX_CLIPBOARD_SIZE] = {0}; -static XImage _image = {0}; -static Display * _display = NULL; -static GC _gc = NULL; -static XIM _im = NULL; -static XIC _ic = NULL; -static Window _window = 0; -static Atom _wm_delete_window = 0; -static Atom _clipboard = 0; -static Atom _targets = 0; -static Atom _utf8_string = 0; -static Atom _target = None; - -void p_init(void) { - _display = XOpenDisplay(NULL); - assert(_display != NULL); - - _key_table[XKeysymToKeycode(_display, XK_Left)] = KEY_LEFT; - _key_table[XKeysymToKeycode(_display, XK_Right)] = KEY_RIGHT; - _key_table[XKeysymToKeycode(_display, XK_Up)] = KEY_UP; - _key_table[XKeysymToKeycode(_display, XK_Down)] = KEY_DOWN; - _key_table[XKeysymToKeycode(_display, XK_1)] = '1'; - _key_table[XKeysymToKeycode(_display, XK_2)] = '2'; - _key_table[XKeysymToKeycode(_display, XK_3)] = '3'; - _key_table[XKeysymToKeycode(_display, XK_4)] = '4'; - _key_table[XKeysymToKeycode(_display, XK_5)] = '5'; - _key_table[XKeysymToKeycode(_display, XK_6)] = '6'; - _key_table[XKeysymToKeycode(_display, XK_7)] = '7'; - _key_table[XKeysymToKeycode(_display, XK_8)] = '8'; - _key_table[XKeysymToKeycode(_display, XK_9)] = '9'; - _key_table[XKeysymToKeycode(_display, XK_0)] = '0'; - _key_table[XKeysymToKeycode(_display, XK_A)] = 'a'; - _key_table[XKeysymToKeycode(_display, XK_B)] = 'b'; - _key_table[XKeysymToKeycode(_display, XK_C)] = 'c'; - _key_table[XKeysymToKeycode(_display, XK_D)] = 'd'; - _key_table[XKeysymToKeycode(_display, XK_E)] = 'e'; - _key_table[XKeysymToKeycode(_display, XK_F)] = 'f'; - _key_table[XKeysymToKeycode(_display, XK_G)] = 'g'; - _key_table[XKeysymToKeycode(_display, XK_H)] = 'h'; - _key_table[XKeysymToKeycode(_display, XK_I)] = 'i'; - _key_table[XKeysymToKeycode(_display, XK_J)] = 'j'; - _key_table[XKeysymToKeycode(_display, XK_K)] = 'k'; - _key_table[XKeysymToKeycode(_display, XK_L)] = 'l'; - _key_table[XKeysymToKeycode(_display, XK_M)] = 'm'; - _key_table[XKeysymToKeycode(_display, XK_N)] = 'n'; - _key_table[XKeysymToKeycode(_display, XK_O)] = 'o'; - _key_table[XKeysymToKeycode(_display, XK_P)] = 'p'; - _key_table[XKeysymToKeycode(_display, XK_Q)] = 'q'; - _key_table[XKeysymToKeycode(_display, XK_R)] = 'r'; - _key_table[XKeysymToKeycode(_display, XK_S)] = 's'; - _key_table[XKeysymToKeycode(_display, XK_T)] = 't'; - _key_table[XKeysymToKeycode(_display, XK_U)] = 'u'; - _key_table[XKeysymToKeycode(_display, XK_V)] = 'v'; - _key_table[XKeysymToKeycode(_display, XK_W)] = 'w'; - _key_table[XKeysymToKeycode(_display, XK_X)] = 'x'; - _key_table[XKeysymToKeycode(_display, XK_Y)] = 'y'; - _key_table[XKeysymToKeycode(_display, XK_Z)] = 'z'; - _key_table[XKeysymToKeycode(_display, XK_space)] = ' '; - _key_table[XKeysymToKeycode(_display, XK_braceleft)] = '['; - _key_table[XKeysymToKeycode(_display, XK_braceright)] = ']'; - _key_table[XKeysymToKeycode(_display, XK_colon)] = ';'; - _key_table[XKeysymToKeycode(_display, XK_quotedbl)] = '\''; - _key_table[XKeysymToKeycode(_display, XK_asciitilde)] = '`'; - _key_table[XKeysymToKeycode(_display, XK_backslash)] = '\\'; - _key_table[XKeysymToKeycode(_display, XK_comma)] = ','; - _key_table[XKeysymToKeycode(_display, XK_greater)] = '.'; - _key_table[XKeysymToKeycode(_display, XK_question)] = '/'; - _key_table[XKeysymToKeycode(_display, XK_minus)] = '-'; - _key_table[XKeysymToKeycode(_display, XK_equal)] = '='; - _key_table[XKeysymToKeycode(_display, XK_F1)] = KEY_F_ + 1; - _key_table[XKeysymToKeycode(_display, XK_F2)] = KEY_F_ + 2; - _key_table[XKeysymToKeycode(_display, XK_F3)] = KEY_F_ + 3; - _key_table[XKeysymToKeycode(_display, XK_F4)] = KEY_F_ + 4; - _key_table[XKeysymToKeycode(_display, XK_F5)] = KEY_F_ + 5; - _key_table[XKeysymToKeycode(_display, XK_F6)] = KEY_F_ + 6; - _key_table[XKeysymToKeycode(_display, XK_F7)] = KEY_F_ + 7; - _key_table[XKeysymToKeycode(_display, XK_F8)] = KEY_F_ + 8; - _key_table[XKeysymToKeycode(_display, XK_F9)] = KEY_F_ + 9; - _key_table[XKeysymToKeycode(_display, XK_F10)] = KEY_F_ + 10; - _key_table[XKeysymToKeycode(_display, XK_F11)] = KEY_F_ + 11; - _key_table[XKeysymToKeycode(_display, XK_F12)] = KEY_F_ + 12; - _key_table[XKeysymToKeycode(_display, XK_Control_L)] = KEY_LCTRL; - _key_table[XKeysymToKeycode(_display, XK_Control_R)] = KEY_RCTRL; - _key_table[XKeysymToKeycode(_display, XK_Shift_L)] = KEY_LSHIFT; - _key_table[XKeysymToKeycode(_display, XK_Shift_R)] = KEY_RSHIFT; - _key_table[XKeysymToKeycode(_display, XK_Alt_L)] = KEY_LALT; - _key_table[XKeysymToKeycode(_display, XK_Alt_R)] = KEY_RALT; - _key_table[XKeysymToKeycode(_display, XK_Escape)] = KEY_ESCAPE; - _key_table[XKeysymToKeycode(_display, XK_BackSpace)] = '\b'; - _key_table[XKeysymToKeycode(_display, XK_Tab)] = '\t'; - _key_table[XKeysymToKeycode(_display, XK_Return)] = '\n'; - _key_table[XKeysymToKeycode(_display, XK_Print)] = KEY_PRINTSCREEN; - _key_table[XKeysymToKeycode(_display, XK_Delete)] = KEY_DELETE; - _key_table[XKeysymToKeycode(_display, XK_Pause)] = KEY_PAUSE; - _key_table[XKeysymToKeycode(_display, XK_Insert)] = KEY_INSERT; - _key_table[XKeysymToKeycode(_display, XK_Home)] = KEY_HOME; - _key_table[XKeysymToKeycode(_display, XK_End)] = KEY_END; - _key_table[XKeysymToKeycode(_display, XK_Page_Up)] = KEY_PAGEUP; - _key_table[XKeysymToKeycode(_display, XK_Page_Down)] = KEY_PAGEDOWN; - _key_table[XKeysymToKeycode(_display, XK_KP_0)] = KEY_KP_ + '0'; - _key_table[XKeysymToKeycode(_display, XK_KP_1)] = KEY_KP_ + '1'; - _key_table[XKeysymToKeycode(_display, XK_KP_2)] = KEY_KP_ + '2'; - _key_table[XKeysymToKeycode(_display, XK_KP_3)] = KEY_KP_ + '3'; - _key_table[XKeysymToKeycode(_display, XK_KP_4)] = KEY_KP_ + '4'; - _key_table[XKeysymToKeycode(_display, XK_KP_5)] = KEY_KP_ + '5'; - _key_table[XKeysymToKeycode(_display, XK_KP_6)] = KEY_KP_ + '6'; - _key_table[XKeysymToKeycode(_display, XK_KP_7)] = KEY_KP_ + '7'; - _key_table[XKeysymToKeycode(_display, XK_KP_8)] = KEY_KP_ + '8'; - _key_table[XKeysymToKeycode(_display, XK_KP_9)] = KEY_KP_ + '9'; - _key_table[XKeysymToKeycode(_display, XK_KP_Enter)] = KEY_KP_ + '\n'; - _key_table[XKeysymToKeycode(_display, XK_KP_Divide)] = KEY_KP_ + '/'; - _key_table[XKeysymToKeycode(_display, XK_KP_Multiply)] = KEY_KP_ + '*'; - _key_table[XKeysymToKeycode(_display, XK_KP_Add)] = KEY_KP_ + '+'; - _key_table[XKeysymToKeycode(_display, XK_KP_Subtract)] = KEY_KP_ + '-'; - _key_table[XKeysymToKeycode(_display, XK_KP_Decimal)] = KEY_KP_ + '.'; - _key_table[XKeysymToKeycode(_display, XK_KP_Separator)] = KEY_KP_ + ','; - - i32 screen = DefaultScreen(_display); - i32 depth = DefaultDepth (_display, screen); - Visual *visual = DefaultVisual(_display, screen); - - _gc = DefaultGC(_display, screen); - assert(_gc != NULL); - - XSetGraphicsExposures(_display, _gc, False); - - i32 _display_width = DisplayWidth (_display, screen); - i32 _display_height = DisplayHeight(_display, screen); - - i32 x = (_display_width - platform.frame_width) / 2; - i32 y = (_display_height - platform.frame_height) / 2; - - _window = XCreateWindow(_display, XDefaultRootWindow(_display), x, y, platform.frame_width, platform.frame_height, 0, depth, InputOutput, visual, CWEventMask, &(XSetWindowAttributes) { .event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | SubstructureNotifyMask, }); - - _im = XOpenIM(_display, NULL, NULL, NULL); - assert(_im != NULL); - - _ic = XCreateIC(_im, XNInputStyle, XIMPreeditNothing | XIMStatusNothing, XNClientWindow, _window, NULL); - assert(_ic != NULL); - - platform.pixels = _buffer; - platform.input = _input; - platform.clipboard = _clipboard_buffer; - - _image = (XImage) { - .width = platform.frame_width, - .height = platform.frame_height, - .depth = depth, - .xoffset = 0, - .format = ZPixmap, - .data = (c8 *) _buffer, - .byte_order = LSBFirst, - .bitmap_unit = 32, - .bitmap_bit_order = LSBFirst, - .bitmap_pad = 32, - .bits_per_pixel = 32, - .bytes_per_line = 4 * platform.frame_width, - .red_mask = 0xff0000, - .green_mask = 0x00ff00, - .blue_mask = 0x0000ff, - }; - - XInitImage(&_image); - - _wm_delete_window = XInternAtom(_display, "WM_DELETE_WINDOW", False); - _clipboard = XInternAtom(_display, "CLIPBOARD", False); - _targets = XInternAtom(_display, "TARGETS", False); - _utf8_string = XInternAtom(_display, "UTF8_STRING", False); - - XSetICFocus(_ic); - XSetWMProtocols(_display, _window, &_wm_delete_window, 1); - - XStoreName(_display, _window, platform.title); - XMapWindow(_display, _window); -} - -void p_cleanup(void) { - XDestroyWindow(_display, _window); - XCloseDisplay (_display); -} - -i32 p_handle_events(void) { - i32 num_events = 0; - - memset(platform.key_pressed, 0, sizeof platform.key_pressed); - - platform.input_size = 0; - platform.cursor_dx = 0; - platform.cursor_dy = 0; - platform.wheel_dy = 0; - - XEvent ev; - - b8 requested_clipboard = 0; - - while (XEventsQueued(_display, QueuedAlready) > 0) { - ++num_events; - - XNextEvent(_display, &ev); - XFilterEvent(&ev, _window); - - switch (ev.type) { - case DestroyNotify: - platform.done = 1; - break; - - case MotionNotify: - platform.cursor_dx += ev.xmotion.x - platform.cursor_x; - platform.cursor_dy += ev.xmotion.y - platform.cursor_y; - platform.cursor_x = ev.xmotion.x; - platform.cursor_y = ev.xmotion.y; - break; - - case ButtonPress: - platform.cursor_x = ev.xbutton.x; - platform.cursor_y = ev.xbutton.y; - switch (ev.xbutton.button) { - case Button1: - platform.key_down[BUTTON_LEFT] = 1; - ++platform.key_pressed[BUTTON_LEFT]; - break; - case Button2: - platform.key_down[BUTTON_MIDDLE] = 1; - ++platform.key_pressed[BUTTON_MIDDLE]; - break; - case Button3: - platform.key_down[BUTTON_RIGHT] = 1; - ++platform.key_pressed[BUTTON_RIGHT]; - break; - case Button4: ++platform.wheel_dy; break; - case Button5: --platform.wheel_dy; break; - default:; - } - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - break; - - case ButtonRelease: - platform.cursor_x = ev.xbutton.x; - platform.cursor_y = ev.xbutton.y; - switch (ev.xbutton.button) { - case Button1: platform.key_down[BUTTON_LEFT] = 0; break; - case Button2: platform.key_down[BUTTON_MIDDLE] = 0; break; - case Button3: platform.key_down[BUTTON_RIGHT] = 0; break; - default:; - } - break; - - case KeyPress: { - i16 k = _key_table[ev.xkey.keycode]; - platform.cursor_x = ev.xkey.x; - platform.cursor_y = ev.xkey.y; - platform.key_down [k] = 1; - platform.key_pressed[k]++; - platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); - platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); - platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); - platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); - platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); - platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); - if (platform.input_size < MAX_INPUT_SIZE) { - if (k < 32 || k >= 128) - platform.input[platform.input_size++] = (Input_Key) { - .ctrl = !!(ev.xkey.state & ControlMask), - .shift = !!(ev.xkey.state & ShiftMask), - .alt = !!(ev.xkey.state & Mod1Mask), - .caps = !!(ev.xkey.state & LockMask), - .num = !!(ev.xkey.state & Mod2Mask), - .scroll = !!(ev.xkey.state & Mod3Mask), - .key = k, - .c = 0, - }; - else { - c8 buf[16]; - i32 len = Xutf8LookupString(_ic, &ev.xkey, buf, sizeof buf - 1, NULL, NULL); - if (len > 0) - platform.input[platform.input_size++] = (Input_Key) { - .ctrl = !!(ev.xkey.state & ControlMask), - .shift = !!(ev.xkey.state & ShiftMask), - .alt = !!(ev.xkey.state & Mod1Mask), - .caps = !!(ev.xkey.state & LockMask), - .num = !!(ev.xkey.state & Mod2Mask), - .scroll = !!(ev.xkey.state & Mod3Mask), - .key = k, - .c = utf8_read(len, buf), - }; - } - } - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - } break; - - case KeyRelease: { - i16 k = _key_table[ev.xkey.keycode]; - platform.cursor_x = ev.xkey.x; - platform.cursor_y = ev.xkey.y; - platform.key_down [k] = 0; - platform.key_pressed[k]--; - platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); - platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); - platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); - platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); - platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); - platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); - } break; - - case SelectionRequest: - if (ev.xselectionrequest.requestor != _window && - ev.xselectionrequest.selection == _clipboard && - XGetSelectionOwner(_display, _clipboard) == _window) { - if (ev.xselectionrequest.property != None) { - if (ev.xselectionrequest.target == _targets) - XChangeProperty( - ev.xselectionrequest.display, - ev.xselectionrequest.requestor, - ev.xselectionrequest.property, - XA_ATOM, - 32, - PropModeReplace, - (u8 *) &_utf8_string, - 1 - ); - else if (ev.xselectionrequest.target == _utf8_string) - XChangeProperty( - ev.xselectionrequest.display, - ev.xselectionrequest.requestor, - ev.xselectionrequest.property, - ev.xselectionrequest.target, - 8, - PropModeReplace, - (u8 *) platform.clipboard, - platform.clipboard_size - ); - } - - XSendEvent(_display, ev.xselectionrequest.requestor, 0, 0, (XEvent *) &(XSelectionEvent) { - .type = SelectionNotify, - .serial = ev.xselectionrequest.serial, - .send_event = ev.xselectionrequest.send_event, - .display = ev.xselectionrequest.display, - .requestor = ev.xselectionrequest.requestor, - .selection = ev.xselectionrequest.selection, - .target = ev.xselectionrequest.target, - .property = ev.xselectionrequest.property, - .time = ev.xselectionrequest.time, - }); - } - break; - - case SelectionNotify: - if (ev.xselection.property != None) { - i64 len; - u8 *data; - - XGetWindowProperty( - _display, - _window, - _clipboard, - 0, - MAX_CLIPBOARD_SIZE, - False, - AnyPropertyType, - &(Atom) {0}, - &(int) {0}, - (unsigned long *) &len, - &(unsigned long) {0}, - &data - ); - - if (ev.xselection.target == _targets) { - _target = None; - Atom *list = (Atom *) data; - - for (i64 i = 0; i < len; i++) - if (list[i] == XA_STRING) - _target = XA_STRING; - else if (list[i] == _utf8_string) { - _target = _utf8_string; - break; - } - - if (_target != None) - XConvertSelection(_display, _clipboard, _target, _clipboard, _window, CurrentTime); - } else if (ev.xselection.target == _target) { - if (len > MAX_CLIPBOARD_SIZE) - len = MAX_CLIPBOARD_SIZE; - platform.clipboard_size = len; - if (len > 0) - memcpy(platform.clipboard, data, len); - } - - if (data) - XFree(data); - } - break; - - case EnterNotify: platform.has_cursor = 1; break; - case LeaveNotify: platform.has_cursor = 0; break; - case FocusOut: platform.has_focus = 0; break; - - case FocusIn: - platform.has_focus = 1; - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - break; - - case MappingNotify: - XRefreshKeyboardMapping(&ev.xmapping); - break; - - case ClientMessage: - if ((Atom) ev.xclient.data.l[0] == _wm_delete_window) - platform.done = 1; - break; - - default:; - } - } - - for (i64 k = 0; k < (i64) (sizeof platform.key_pressed / sizeof *platform.key_pressed); ++k) - platform.key_pressed[k] = platform.key_pressed[k] > 0 ? 1 : 0; - - XWindowAttributes attrs; - XGetWindowAttributes(_display, _window, &attrs); - - if ((platform.frame_width != attrs.width || platform.frame_height != attrs.height) && attrs.width * attrs.height * 4 <= (i32) sizeof _buffer) { - if (attrs.width > 0 && attrs.height > 0) { - _image.width = attrs.width; - _image.height = attrs.height; - _image.bytes_per_line = 4 * attrs.width; - } - - platform.frame_width = attrs.width; - platform.frame_height = attrs.height; - } - - return num_events; -} - -i32 p_wait_events(void) { - i32 num_events = 0; - - do { - num_events = p_handle_events(); - sched_yield(); - } while (num_events == 0); - - return num_events; -} - -void p_render_frame(void) { - if (platform.done) - return; - - XPutImage(_display, _window, _gc, &_image, 0, 0, 0, 0, platform.frame_width, platform.frame_height); - XFlush(_display); -} - -void p_clipboard_write(i64 size, c8 *data) { - assert(size <= MAX_CLIPBOARD_SIZE); - - XSetSelectionOwner(_display, _clipboard, _window, CurrentTime); - - platform.clipboard_size = size < MAX_CLIPBOARD_SIZE ? size : MAX_CLIPBOARD_SIZE; - if (platform.clipboard_size > 0) - memcpy(platform.clipboard, data, platform.clipboard_size); -} - -#endif - -// ================================================================ -// -// ALSA -// -// ================================================================ - -#ifdef __linux__ - -void p_handle_audio(i64 time_elapsed) { - (void) time_elapsed; - assert(0); -} - -void p_queue_sound(i64 delay, i64 num_samples, f32 *samples) { - (void) delay; - (void) num_samples; - (void) samples; - assert(0); -} - -#endif diff --git a/examples/ui.c b/examples/ui.c index 810c9d6..bd508e8 100755 --- a/examples/ui.c +++ b/examples/ui.c @@ -3,31 +3,6 @@ #/ #/ ui.c #/ -#/ This is a reduced system layer. -#/ It allows you to create a window, draw graphics in it, handle -#/ input events. -#/ -#/ ---------------------------------------------------------------- -#/ -#/ DESIGN PRINCIPLES -#/ -#/ - Minimalistic feature set. For graphics, you have access to the -#/ pixel buffer, and that's it. -#/ -#/ - No implicit control flow. No callbacks. You write your own -#/ main and call everything explicitly. But the number of things -#/ you have to call to do something is as little as possible. -#/ -#/ - Optimized to use in a single source file. -#/ Installation process? Ctrl+C, Ctrl+V, done. -#/ -#/ If you have an idea how to reduce the feature set further, -#/ let me know! -#/ -#/ ---------------------------------------------------------------- -#/ -#/ (C) 2024 Mitya Selivanov <guattari.tech>, MIT License -#/ #/ ================================================================ #/ #/ Self-compilation shell script @@ -49,157 +24,8 @@ gcc \ ./$BIN $@ && rm $BIN exit $? # */ #endif -// ================================================================ -// -// Basic declarations -// -// ================================================================ - -#define _GNU_SOURCE - -typedef signed char i8; -typedef signed short i16; -typedef signed i32; -typedef signed long long i64; -typedef unsigned char u8; -typedef unsigned short u16; -typedef unsigned u32; -typedef unsigned long long u64; -typedef char c8; -typedef int c32; -typedef signed char b8; -typedef float f32; -typedef double f64; - -// ================================================================ -// -// PLATFORM API -// -// ================================================================ - -enum { - MAX_NUM_PIXELS = 10 * 1024 * 1024, - MAX_INPUT_SIZE = 256, - MAX_CLIPBOARD_SIZE = 10 * 1024 * 1024, - MAX_NUM_AUDIO_SAMPLES = 0, - MAX_NUM_SOCKETS = 0, - - AUDIO_NUM_CHANNELS = 2, - AUDIO_SAMPLE_RATE = 44100, - - IPv4 = 1, - IPv6 = 2, - - KEY_LEFT = 128, - KEY_RIGHT, - KEY_UP, - KEY_DOWN, - KEY_LCTRL, - KEY_RCTRL, - KEY_LSHIFT, - KEY_RSHIFT, - KEY_LALT, - KEY_RALT, - KEY_ESCAPE, - KEY_PRINTSCREEN, - KEY_DELETE, - KEY_PAUSE, - KEY_INSERT, - KEY_HOME, - KEY_END, - KEY_PAGEUP, - KEY_PAGEDOWN, - BUTTON_LEFT, - BUTTON_MIDDLE, - BUTTON_RIGHT, - MOD_CTRL, - MOD_SHIFT, - MOD_ALT, - MOD_CAPS, - MOD_NUM, - MOD_SCROLL, - KEY_F_, - KEY_KP_ = KEY_F_ + 64, -}; - -typedef struct { - b8 ctrl : 1; - b8 shift : 1; - b8 alt : 1; - b8 caps : 1; - b8 num : 1; - b8 scroll : 1; - u16 key; - c32 c; -} Input_Key; - -typedef struct { - c8 * title; - i32 frame_width; - i32 frame_height; - u32 * pixels; - i64 input_size; - Input_Key *input; - i64 clipboard_size; - c8 * clipboard; - b8 done; - b8 has_focus; - b8 has_cursor; - i32 cursor_x; - i32 cursor_y; - i32 cursor_dx; - i32 cursor_dy; - i64 wheel_dy; - b8 key_down[512]; - b8 key_pressed[512]; -} Platform; - -typedef struct { - u16 type; - u16 port; - union { - u8 v4_address[4]; - u8 v6_address[16]; - }; -} IP_Address; - -typedef i64 (*Thread_Proc)(void *user_data); -// UTF-8 -i32 utf8_size(c32 c); -c32 utf8_read(i64 len, c8 *s); -i32 utf8_write(c32 c, c8 *buffer); - -// Window -void p_init(void); -void p_cleanup(void); -i32 p_handle_events(void); -i32 p_wait_events(void); -void p_render_frame(void); - -// Clipboard -void p_clipboard_write(i64 size, c8 *data); - -// Sound -void p_handle_audio(i64 time_elapsed); -void p_queue_sound(i64 delay, i64 num_samples, f32 *samples); - -// UDP sockets -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address); -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data); - -Platform platform = {0}; - -// ================================================================ -// -// UI -// -// ================================================================ - -#include <stdio.h> -#include <stdlib.h> -#include <math.h> -#include <assert.h> +#include "../reduced_system_layer.c" #define CHAR_NUM_BITS_X 6 #define CHAR_NUM_BITS_Y 7 @@ -747,628 +573,3 @@ i32 main(i32 argc, c8 **argv) { p_cleanup(); return 0; } - -// ================================================================ -// -// PLATFORM IMPLEMENTATION -// -// ---------------------------------------------------------------- -// -// TODO -// - Sound -// - Sockets -// - Clipboard daemon -// -// ALSA -// https://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_min_8c-example.html -// -// ================================================================ - -#include <assert.h> -#include <string.h> - -c32 utf8_read(i64 len, c8 *s) { - if (len >= 1 && - (s[0] & 0x80) == 0) - return s[0]; - if (len >= 2 && - (s[0] & 0xe0) == 0xc0 && - (s[1] & 0xc0) == 0x80) - return (s[1] & 0x3f) - | ((s[0] & 0x1f) << 6); - if (len >= 3 && - (s[0] & 0xf0) == 0xe0 && - (s[1] & 0xc0) == 0x80 && - (s[2] & 0xc0) == 0x80) - return (s[2] & 0x3f) - | ((s[1] & 0x3f) << 6) - | ((s[0] & 0x0f) << 12); - if (len >= 4 && - (s[0] & 0xf8) == 0xf0 && - (s[1] & 0xc0) == 0x80 && - (s[2] & 0xc0) == 0x80 && - (s[3] & 0xc0) == 0x80) - return (s[3] & 0x3f) - | ((s[2] & 0x3f) << 6) - | ((s[1] & 0x3f) << 12) - | ((s[0] & 0x07) << 18); - return 0; -} - -i32 utf8_size(c32 c) { - if ((c & 0x00007f) == c) return 1; - if ((c & 0x0007ff) == c) return 2; - if ((c & 0x00ffff) == c) return 3; - if ((c & 0x1fffff) == c) return 4; - return 0; -} - -i32 utf8_write(c32 c, c8 *buffer) { - if ((c & 0x7f) == c) { - buffer[0] = (c8) c; - return 1; - } - - if ((c & 0x7ff) == c) { - buffer[0] = 0xc0 | ((c >> 6) & 0x1f); - buffer[1] = 0x80 | ( c & 0x3f); - return 2; - } - - if ((c & 0xffff) == c) { - buffer[0] = 0xc0 | ((c >> 12) & 0x0f); - buffer[1] = 0x80 | ((c >> 6) & 0x3f); - buffer[2] = 0x80 | ( c & 0x3f); - return 3; - } - - if ((c & 0x1fffff) == c) { - buffer[0] = 0xc0 | ((c >> 18) & 0x07); - buffer[1] = 0x80 | ((c >> 12) & 0x3f); - buffer[2] = 0x80 | ((c >> 6) & 0x3f); - buffer[3] = 0x80 | ( c & 0x3f); - return 4; - } - - return 0; -} - -// ================================================================ -// -// UDP sockets -// -// ================================================================ - -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address) { - (void) slot; - (void) address; - (void) size; - (void) data; - (void) remote_address; - assert(0); - return 0; -} - -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data) { - (void) slot; - (void) address; - (void) size; - (void) data; - assert(0); - return 0; -} - -// ================================================================ -// -// X11 -// -// ================================================================ - -#ifdef __linux__ - -#include <X11/Xlib.h> -#include <X11/Xutil.h> -#include <X11/Xatom.h> -#include <sched.h> -#include <time.h> - -static i16 _key_table[512] = {0}; -static u32 _buffer[MAX_NUM_PIXELS] = {0}; -static Input_Key _input[MAX_INPUT_SIZE] = {0}; -static c8 _clipboard_buffer[MAX_CLIPBOARD_SIZE] = {0}; -static XImage _image = {0}; -static Display * _display = NULL; -static GC _gc = NULL; -static XIM _im = NULL; -static XIC _ic = NULL; -static Window _window = 0; -static Atom _wm_delete_window = 0; -static Atom _clipboard = 0; -static Atom _targets = 0; -static Atom _utf8_string = 0; -static Atom _target = None; - -void p_init(void) { - _display = XOpenDisplay(NULL); - assert(_display != NULL); - - _key_table[XKeysymToKeycode(_display, XK_Left)] = KEY_LEFT; - _key_table[XKeysymToKeycode(_display, XK_Right)] = KEY_RIGHT; - _key_table[XKeysymToKeycode(_display, XK_Up)] = KEY_UP; - _key_table[XKeysymToKeycode(_display, XK_Down)] = KEY_DOWN; - _key_table[XKeysymToKeycode(_display, XK_1)] = '1'; - _key_table[XKeysymToKeycode(_display, XK_2)] = '2'; - _key_table[XKeysymToKeycode(_display, XK_3)] = '3'; - _key_table[XKeysymToKeycode(_display, XK_4)] = '4'; - _key_table[XKeysymToKeycode(_display, XK_5)] = '5'; - _key_table[XKeysymToKeycode(_display, XK_6)] = '6'; - _key_table[XKeysymToKeycode(_display, XK_7)] = '7'; - _key_table[XKeysymToKeycode(_display, XK_8)] = '8'; - _key_table[XKeysymToKeycode(_display, XK_9)] = '9'; - _key_table[XKeysymToKeycode(_display, XK_0)] = '0'; - _key_table[XKeysymToKeycode(_display, XK_A)] = 'a'; - _key_table[XKeysymToKeycode(_display, XK_B)] = 'b'; - _key_table[XKeysymToKeycode(_display, XK_C)] = 'c'; - _key_table[XKeysymToKeycode(_display, XK_D)] = 'd'; - _key_table[XKeysymToKeycode(_display, XK_E)] = 'e'; - _key_table[XKeysymToKeycode(_display, XK_F)] = 'f'; - _key_table[XKeysymToKeycode(_display, XK_G)] = 'g'; - _key_table[XKeysymToKeycode(_display, XK_H)] = 'h'; - _key_table[XKeysymToKeycode(_display, XK_I)] = 'i'; - _key_table[XKeysymToKeycode(_display, XK_J)] = 'j'; - _key_table[XKeysymToKeycode(_display, XK_K)] = 'k'; - _key_table[XKeysymToKeycode(_display, XK_L)] = 'l'; - _key_table[XKeysymToKeycode(_display, XK_M)] = 'm'; - _key_table[XKeysymToKeycode(_display, XK_N)] = 'n'; - _key_table[XKeysymToKeycode(_display, XK_O)] = 'o'; - _key_table[XKeysymToKeycode(_display, XK_P)] = 'p'; - _key_table[XKeysymToKeycode(_display, XK_Q)] = 'q'; - _key_table[XKeysymToKeycode(_display, XK_R)] = 'r'; - _key_table[XKeysymToKeycode(_display, XK_S)] = 's'; - _key_table[XKeysymToKeycode(_display, XK_T)] = 't'; - _key_table[XKeysymToKeycode(_display, XK_U)] = 'u'; - _key_table[XKeysymToKeycode(_display, XK_V)] = 'v'; - _key_table[XKeysymToKeycode(_display, XK_W)] = 'w'; - _key_table[XKeysymToKeycode(_display, XK_X)] = 'x'; - _key_table[XKeysymToKeycode(_display, XK_Y)] = 'y'; - _key_table[XKeysymToKeycode(_display, XK_Z)] = 'z'; - _key_table[XKeysymToKeycode(_display, XK_space)] = ' '; - _key_table[XKeysymToKeycode(_display, XK_braceleft)] = '['; - _key_table[XKeysymToKeycode(_display, XK_braceright)] = ']'; - _key_table[XKeysymToKeycode(_display, XK_colon)] = ';'; - _key_table[XKeysymToKeycode(_display, XK_quotedbl)] = '\''; - _key_table[XKeysymToKeycode(_display, XK_asciitilde)] = '`'; - _key_table[XKeysymToKeycode(_display, XK_backslash)] = '\\'; - _key_table[XKeysymToKeycode(_display, XK_comma)] = ','; - _key_table[XKeysymToKeycode(_display, XK_greater)] = '.'; - _key_table[XKeysymToKeycode(_display, XK_question)] = '/'; - _key_table[XKeysymToKeycode(_display, XK_minus)] = '-'; - _key_table[XKeysymToKeycode(_display, XK_equal)] = '='; - _key_table[XKeysymToKeycode(_display, XK_F1)] = KEY_F_ + 1; - _key_table[XKeysymToKeycode(_display, XK_F2)] = KEY_F_ + 2; - _key_table[XKeysymToKeycode(_display, XK_F3)] = KEY_F_ + 3; - _key_table[XKeysymToKeycode(_display, XK_F4)] = KEY_F_ + 4; - _key_table[XKeysymToKeycode(_display, XK_F5)] = KEY_F_ + 5; - _key_table[XKeysymToKeycode(_display, XK_F6)] = KEY_F_ + 6; - _key_table[XKeysymToKeycode(_display, XK_F7)] = KEY_F_ + 7; - _key_table[XKeysymToKeycode(_display, XK_F8)] = KEY_F_ + 8; - _key_table[XKeysymToKeycode(_display, XK_F9)] = KEY_F_ + 9; - _key_table[XKeysymToKeycode(_display, XK_F10)] = KEY_F_ + 10; - _key_table[XKeysymToKeycode(_display, XK_F11)] = KEY_F_ + 11; - _key_table[XKeysymToKeycode(_display, XK_F12)] = KEY_F_ + 12; - _key_table[XKeysymToKeycode(_display, XK_Control_L)] = KEY_LCTRL; - _key_table[XKeysymToKeycode(_display, XK_Control_R)] = KEY_RCTRL; - _key_table[XKeysymToKeycode(_display, XK_Shift_L)] = KEY_LSHIFT; - _key_table[XKeysymToKeycode(_display, XK_Shift_R)] = KEY_RSHIFT; - _key_table[XKeysymToKeycode(_display, XK_Alt_L)] = KEY_LALT; - _key_table[XKeysymToKeycode(_display, XK_Alt_R)] = KEY_RALT; - _key_table[XKeysymToKeycode(_display, XK_Escape)] = KEY_ESCAPE; - _key_table[XKeysymToKeycode(_display, XK_BackSpace)] = '\b'; - _key_table[XKeysymToKeycode(_display, XK_Tab)] = '\t'; - _key_table[XKeysymToKeycode(_display, XK_Return)] = '\n'; - _key_table[XKeysymToKeycode(_display, XK_Print)] = KEY_PRINTSCREEN; - _key_table[XKeysymToKeycode(_display, XK_Delete)] = KEY_DELETE; - _key_table[XKeysymToKeycode(_display, XK_Pause)] = KEY_PAUSE; - _key_table[XKeysymToKeycode(_display, XK_Insert)] = KEY_INSERT; - _key_table[XKeysymToKeycode(_display, XK_Home)] = KEY_HOME; - _key_table[XKeysymToKeycode(_display, XK_End)] = KEY_END; - _key_table[XKeysymToKeycode(_display, XK_Page_Up)] = KEY_PAGEUP; - _key_table[XKeysymToKeycode(_display, XK_Page_Down)] = KEY_PAGEDOWN; - _key_table[XKeysymToKeycode(_display, XK_KP_0)] = KEY_KP_ + '0'; - _key_table[XKeysymToKeycode(_display, XK_KP_1)] = KEY_KP_ + '1'; - _key_table[XKeysymToKeycode(_display, XK_KP_2)] = KEY_KP_ + '2'; - _key_table[XKeysymToKeycode(_display, XK_KP_3)] = KEY_KP_ + '3'; - _key_table[XKeysymToKeycode(_display, XK_KP_4)] = KEY_KP_ + '4'; - _key_table[XKeysymToKeycode(_display, XK_KP_5)] = KEY_KP_ + '5'; - _key_table[XKeysymToKeycode(_display, XK_KP_6)] = KEY_KP_ + '6'; - _key_table[XKeysymToKeycode(_display, XK_KP_7)] = KEY_KP_ + '7'; - _key_table[XKeysymToKeycode(_display, XK_KP_8)] = KEY_KP_ + '8'; - _key_table[XKeysymToKeycode(_display, XK_KP_9)] = KEY_KP_ + '9'; - _key_table[XKeysymToKeycode(_display, XK_KP_Enter)] = KEY_KP_ + '\n'; - _key_table[XKeysymToKeycode(_display, XK_KP_Divide)] = KEY_KP_ + '/'; - _key_table[XKeysymToKeycode(_display, XK_KP_Multiply)] = KEY_KP_ + '*'; - _key_table[XKeysymToKeycode(_display, XK_KP_Add)] = KEY_KP_ + '+'; - _key_table[XKeysymToKeycode(_display, XK_KP_Subtract)] = KEY_KP_ + '-'; - _key_table[XKeysymToKeycode(_display, XK_KP_Decimal)] = KEY_KP_ + '.'; - _key_table[XKeysymToKeycode(_display, XK_KP_Separator)] = KEY_KP_ + ','; - - i32 screen = DefaultScreen(_display); - i32 depth = DefaultDepth (_display, screen); - Visual *visual = DefaultVisual(_display, screen); - - _gc = DefaultGC(_display, screen); - assert(_gc != NULL); - - XSetGraphicsExposures(_display, _gc, False); - - i32 _display_width = DisplayWidth (_display, screen); - i32 _display_height = DisplayHeight(_display, screen); - - i32 x = (_display_width - platform.frame_width) / 2; - i32 y = (_display_height - platform.frame_height) / 2; - - _window = XCreateWindow(_display, XDefaultRootWindow(_display), x, y, platform.frame_width, platform.frame_height, 0, depth, InputOutput, visual, CWEventMask, &(XSetWindowAttributes) { .event_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | SubstructureNotifyMask, }); - - _im = XOpenIM(_display, NULL, NULL, NULL); - assert(_im != NULL); - - _ic = XCreateIC(_im, XNInputStyle, XIMPreeditNothing | XIMStatusNothing, XNClientWindow, _window, NULL); - assert(_ic != NULL); - - platform.pixels = _buffer; - platform.input = _input; - platform.clipboard = _clipboard_buffer; - - _image = (XImage) { - .width = platform.frame_width, - .height = platform.frame_height, - .depth = depth, - .xoffset = 0, - .format = ZPixmap, - .data = (c8 *) _buffer, - .byte_order = LSBFirst, - .bitmap_unit = 32, - .bitmap_bit_order = LSBFirst, - .bitmap_pad = 32, - .bits_per_pixel = 32, - .bytes_per_line = 4 * platform.frame_width, - .red_mask = 0xff0000, - .green_mask = 0x00ff00, - .blue_mask = 0x0000ff, - }; - - XInitImage(&_image); - - _wm_delete_window = XInternAtom(_display, "WM_DELETE_WINDOW", False); - _clipboard = XInternAtom(_display, "CLIPBOARD", False); - _targets = XInternAtom(_display, "TARGETS", False); - _utf8_string = XInternAtom(_display, "UTF8_STRING", False); - - XSetICFocus(_ic); - XSetWMProtocols(_display, _window, &_wm_delete_window, 1); - - XStoreName(_display, _window, platform.title); - XMapWindow(_display, _window); -} - -void p_cleanup(void) { - XDestroyWindow(_display, _window); - XCloseDisplay (_display); -} - -i32 p_handle_events(void) { - i32 num_events = 0; - - memset(platform.key_pressed, 0, sizeof platform.key_pressed); - - platform.input_size = 0; - platform.cursor_dx = 0; - platform.cursor_dy = 0; - platform.wheel_dy = 0; - - XEvent ev; - - b8 requested_clipboard = 0; - - while (XEventsQueued(_display, QueuedAlready) > 0) { - ++num_events; - - XNextEvent(_display, &ev); - XFilterEvent(&ev, _window); - - switch (ev.type) { - case DestroyNotify: - platform.done = 1; - break; - - case MotionNotify: - platform.cursor_dx += ev.xmotion.x - platform.cursor_x; - platform.cursor_dy += ev.xmotion.y - platform.cursor_y; - platform.cursor_x = ev.xmotion.x; - platform.cursor_y = ev.xmotion.y; - break; - - case ButtonPress: - platform.cursor_x = ev.xbutton.x; - platform.cursor_y = ev.xbutton.y; - switch (ev.xbutton.button) { - case Button1: - platform.key_down[BUTTON_LEFT] = 1; - ++platform.key_pressed[BUTTON_LEFT]; - break; - case Button2: - platform.key_down[BUTTON_MIDDLE] = 1; - ++platform.key_pressed[BUTTON_MIDDLE]; - break; - case Button3: - platform.key_down[BUTTON_RIGHT] = 1; - ++platform.key_pressed[BUTTON_RIGHT]; - break; - case Button4: ++platform.wheel_dy; break; - case Button5: --platform.wheel_dy; break; - default:; - } - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - break; - - case ButtonRelease: - platform.cursor_x = ev.xbutton.x; - platform.cursor_y = ev.xbutton.y; - switch (ev.xbutton.button) { - case Button1: platform.key_down[BUTTON_LEFT] = 0; break; - case Button2: platform.key_down[BUTTON_MIDDLE] = 0; break; - case Button3: platform.key_down[BUTTON_RIGHT] = 0; break; - default:; - } - break; - - case KeyPress: { - i16 k = _key_table[ev.xkey.keycode]; - platform.cursor_x = ev.xkey.x; - platform.cursor_y = ev.xkey.y; - platform.key_down [k] = 1; - platform.key_pressed[k]++; - platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); - platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); - platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); - platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); - platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); - platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); - if (platform.input_size < MAX_INPUT_SIZE) { - if (k < 32 || k >= 128) - platform.input[platform.input_size++] = (Input_Key) { - .ctrl = !!(ev.xkey.state & ControlMask), - .shift = !!(ev.xkey.state & ShiftMask), - .alt = !!(ev.xkey.state & Mod1Mask), - .caps = !!(ev.xkey.state & LockMask), - .num = !!(ev.xkey.state & Mod2Mask), - .scroll = !!(ev.xkey.state & Mod3Mask), - .key = k, - .c = 0, - }; - else { - c8 buf[16]; - i32 len = Xutf8LookupString(_ic, &ev.xkey, buf, sizeof buf - 1, NULL, NULL); - if (len > 0) - platform.input[platform.input_size++] = (Input_Key) { - .ctrl = !!(ev.xkey.state & ControlMask), - .shift = !!(ev.xkey.state & ShiftMask), - .alt = !!(ev.xkey.state & Mod1Mask), - .caps = !!(ev.xkey.state & LockMask), - .num = !!(ev.xkey.state & Mod2Mask), - .scroll = !!(ev.xkey.state & Mod3Mask), - .key = k, - .c = utf8_read(len, buf), - }; - } - } - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - } break; - - case KeyRelease: { - i16 k = _key_table[ev.xkey.keycode]; - platform.cursor_x = ev.xkey.x; - platform.cursor_y = ev.xkey.y; - platform.key_down [k] = 0; - platform.key_pressed[k]--; - platform.key_down[MOD_CTRL] = !!(ev.xkey.state & ControlMask); - platform.key_down[MOD_SHIFT] = !!(ev.xkey.state & ShiftMask); - platform.key_down[MOD_ALT] = !!(ev.xkey.state & Mod1Mask); - platform.key_down[MOD_CAPS] = !!(ev.xkey.state & LockMask); - platform.key_down[MOD_NUM] = !!(ev.xkey.state & Mod2Mask); - platform.key_down[MOD_SCROLL] = !!(ev.xkey.state & Mod3Mask); - } break; - - case SelectionRequest: - if (ev.xselectionrequest.requestor != _window && - ev.xselectionrequest.selection == _clipboard && - XGetSelectionOwner(_display, _clipboard) == _window) { - if (ev.xselectionrequest.property != None) { - if (ev.xselectionrequest.target == _targets) - XChangeProperty( - ev.xselectionrequest.display, - ev.xselectionrequest.requestor, - ev.xselectionrequest.property, - XA_ATOM, - 32, - PropModeReplace, - (u8 *) &_utf8_string, - 1 - ); - else if (ev.xselectionrequest.target == _utf8_string) - XChangeProperty( - ev.xselectionrequest.display, - ev.xselectionrequest.requestor, - ev.xselectionrequest.property, - ev.xselectionrequest.target, - 8, - PropModeReplace, - (u8 *) platform.clipboard, - platform.clipboard_size - ); - } - - XSendEvent(_display, ev.xselectionrequest.requestor, 0, 0, (XEvent *) &(XSelectionEvent) { - .type = SelectionNotify, - .serial = ev.xselectionrequest.serial, - .send_event = ev.xselectionrequest.send_event, - .display = ev.xselectionrequest.display, - .requestor = ev.xselectionrequest.requestor, - .selection = ev.xselectionrequest.selection, - .target = ev.xselectionrequest.target, - .property = ev.xselectionrequest.property, - .time = ev.xselectionrequest.time, - }); - } - break; - - case SelectionNotify: - if (ev.xselection.property != None) { - i64 len; - u8 *data; - - XGetWindowProperty( - _display, - _window, - _clipboard, - 0, - MAX_CLIPBOARD_SIZE, - False, - AnyPropertyType, - &(Atom) {0}, - &(int) {0}, - (unsigned long *) &len, - &(unsigned long) {0}, - &data - ); - - if (ev.xselection.target == _targets) { - _target = None; - Atom *list = (Atom *) data; - - for (i64 i = 0; i < len; i++) - if (list[i] == XA_STRING) - _target = XA_STRING; - else if (list[i] == _utf8_string) { - _target = _utf8_string; - break; - } - - if (_target != None) - XConvertSelection(_display, _clipboard, _target, _clipboard, _window, CurrentTime); - } else if (ev.xselection.target == _target) { - if (len > MAX_CLIPBOARD_SIZE) - len = MAX_CLIPBOARD_SIZE; - platform.clipboard_size = len; - if (len > 0) - memcpy(platform.clipboard, data, len); - } - - if (data) - XFree(data); - } - break; - - case EnterNotify: platform.has_cursor = 1; break; - case LeaveNotify: platform.has_cursor = 0; break; - case FocusOut: platform.has_focus = 0; break; - - case FocusIn: - platform.has_focus = 1; - if (!requested_clipboard) { - XConvertSelection(_display, _clipboard, _targets, _clipboard, _window, CurrentTime); - requested_clipboard = 1; - } - break; - - case MappingNotify: - XRefreshKeyboardMapping(&ev.xmapping); - break; - - case ClientMessage: - if ((Atom) ev.xclient.data.l[0] == _wm_delete_window) - platform.done = 1; - break; - - default:; - } - } - - for (i64 k = 0; k < (i64) (sizeof platform.key_pressed / sizeof *platform.key_pressed); ++k) - platform.key_pressed[k] = platform.key_pressed[k] > 0 ? 1 : 0; - - XWindowAttributes attrs; - XGetWindowAttributes(_display, _window, &attrs); - - if ((platform.frame_width != attrs.width || platform.frame_height != attrs.height) && attrs.width * attrs.height * 4 <= (i32) sizeof _buffer) { - if (attrs.width > 0 && attrs.height > 0) { - _image.width = attrs.width; - _image.height = attrs.height; - _image.bytes_per_line = 4 * attrs.width; - } - - platform.frame_width = attrs.width; - platform.frame_height = attrs.height; - } - - return num_events; -} - -i32 p_wait_events(void) { - i32 num_events = 0; - - do { - num_events = p_handle_events(); - sched_yield(); - } while (num_events == 0); - - return num_events; -} - -void p_render_frame(void) { - if (platform.done) - return; - - XPutImage(_display, _window, _gc, &_image, 0, 0, 0, 0, platform.frame_width, platform.frame_height); - XFlush(_display); -} - -void p_clipboard_write(i64 size, c8 *data) { - assert(size <= MAX_CLIPBOARD_SIZE); - - XSetSelectionOwner(_display, _clipboard, _window, CurrentTime); - - platform.clipboard_size = size < MAX_CLIPBOARD_SIZE ? size : MAX_CLIPBOARD_SIZE; - if (platform.clipboard_size > 0) - memcpy(platform.clipboard, data, platform.clipboard_size); -} - -#endif - -// ================================================================ -// -// ALSA -// -// ================================================================ - -#ifdef __linux__ - -void p_handle_audio(i64 time_elapsed) { - (void) time_elapsed; - assert(0); -} - -void p_queue_sound(i64 delay, i64 num_samples, f32 *samples) { - (void) delay; - (void) num_samples; - (void) samples; - assert(0); -} - -#endif diff --git a/reduced_system_layer.c b/reduced_system_layer.c index 379cc97..a200695 100755 --- a/reduced_system_layer.c +++ b/reduced_system_layer.c @@ -44,6 +44,7 @@ gcc \ -Wno-overlength-strings \ -O3 \ -fsanitize=undefined,address,leak -mshstk \ + -D REDUCED_SYSTEM_LAYER_EXAMPLE \ -lX11 -lm \ -o $BIN $SRC && \ ./$BIN $@ && rm $BIN @@ -57,6 +58,13 @@ exit $? # */ #define _GNU_SOURCE +#include <time.h> +#include <math.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <assert.h> + typedef signed char i8; typedef signed short i16; typedef signed i32; @@ -82,13 +90,13 @@ enum { MAX_INPUT_SIZE = 256, MAX_CLIPBOARD_SIZE = 10 * 1024 * 1024, MAX_NUM_AUDIO_SAMPLES = 0, - MAX_NUM_SOCKETS = 0, + MAX_NUM_SOCKETS = 64, AUDIO_NUM_CHANNELS = 2, AUDIO_SAMPLE_RATE = 44100, - IPv4 = 1, - IPv6 = 2, + IPv4_UDP = 1, + IPv6_UDP = 2, KEY_LEFT = 128, KEY_RIGHT, @@ -143,6 +151,7 @@ typedef struct { i64 clipboard_size; c8 * clipboard; b8 done; + b8 graceful_exit; b8 has_focus; b8 has_cursor; i32 cursor_x; @@ -155,11 +164,12 @@ typedef struct { } Platform; typedef struct { - u16 type; + u16 protocol; u16 port; union { - u8 v4_address[4]; - u8 v6_address[16]; + u32 v4_address_as_u32; + u8 v4_address[4]; + u8 v6_address[16]; }; } IP_Address; @@ -170,6 +180,11 @@ i32 utf8_size(c32 c); c32 utf8_read(i64 len, c8 *s); i32 utf8_write(c32 c, c8 *buffer); +// Time and sleep +i64 p_time(void); +void p_yield(void); +void p_sleep_for(i64 duration); + // Window void p_init(void); void p_cleanup(void); @@ -185,8 +200,8 @@ void p_handle_audio(i64 time_elapsed); void p_queue_sound(i64 delay, i64 num_samples, f32 *samples); // UDP sockets -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address); -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data); +i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, u16 *local_port, IP_Address *remote_address); +i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data, u16 *local_port); Platform platform = {0}; @@ -196,14 +211,16 @@ Platform platform = {0}; // // ================================================================ +#ifdef REDUCED_SYSTEM_LAYER_EXAMPLE + i32 main(i32 argc, c8 **argv) { (void) argc; (void) argv; platform = (Platform) { .title = "Reduced System Layer", - .frame_width = 960, - .frame_height = 720, + .frame_width = 1280, + .frame_height = 720, }; p_init(); @@ -217,6 +234,8 @@ i32 main(i32 argc, c8 **argv) { return 0; } +#endif + // ================================================================ // // PLATFORM IMPLEMENTATION @@ -225,7 +244,6 @@ i32 main(i32 argc, c8 **argv) { // // TODO // - Sound -// - Sockets // - Clipboard daemon // // ALSA @@ -233,9 +251,6 @@ i32 main(i32 argc, c8 **argv) { // // ================================================================ -#include <assert.h> -#include <string.h> - c32 utf8_read(i64 len, c8 *s) { if (len >= 1 && (s[0] & 0x80) == 0) @@ -304,29 +319,269 @@ i32 utf8_write(c32 c, c8 *buffer) { // ================================================================ // +// Unix +// +// ================================================================ + +#ifdef __unix__ + +#include <sched.h> +#include <unistd.h> + +i64 p_time(void) { + struct timespec t; + timespec_get(&t, TIME_UTC); + return 1000 * t.tv_sec + t.tv_nsec / 1000000; +} + +void p_yield(void) { + sched_yield(); +} + +void p_sleep_for(i64 duration) { + if (duration <= 0) { + usleep(0); + return; + } + + if (duration >= 1000) + // seconds + sleep(duration / 1000); + + if ((duration % 1000) > 0) + // microseconds + usleep((duration % 1000) * 1000); +} + +#endif + +// ================================================================ +// // UDP sockets // // ================================================================ -i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, IP_Address *remote_address) { - (void) slot; - (void) address; - (void) size; - (void) data; - (void) remote_address; - assert(0); - return 0; +#ifdef __unix__ + +#include <errno.h> +#include <signal.h> +#include <netinet/in.h> +#include <sys/socket.h> + +typedef struct { + b8 ready; + i32 socket; + u16 local_port; + IP_Address address; +} Socket_Slot; + +b8 _sockets_init = 0; +Socket_Slot _sockets[MAX_NUM_SOCKETS] = {0}; + +void sockets_initialize(void) { + if (_sockets_init) + return; + + signal(SIGPIPE, SIG_IGN); + _sockets_init = 1; } -i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data) { - (void) slot; - (void) address; - (void) size; - (void) data; - assert(0); - return 0; +void sockets_cleanup(void) { + for (i64 i = 0; i < MAX_NUM_SOCKETS; ++i) + if (_sockets[i].ready) { + close(_sockets[i].socket); + _sockets[i].ready = 0; + } +} + +b8 sockets_open(u16 slot, IP_Address address, u16 *local_port) { + sockets_initialize(); + + b8 change_address = + !_sockets[slot].ready + || _sockets[slot].address.protocol != address.protocol + || (address.port != 0 && _sockets[slot].local_port != address.port) + || (memcmp(_sockets[slot].address.v6_address, &(u8[sizeof address.v6_address]) {0}, sizeof address.v6_address) != 0 && + memcmp(_sockets[slot].address.v6_address, address.v6_address, sizeof address.v6_address) != 0); + + if (change_address && _sockets[slot].ready) { + close(_sockets[slot].socket); + _sockets[slot].ready = 0; + } + + struct sockaddr *p; + i32 p_len; + + struct sockaddr_in a4 = {0}; + struct sockaddr_in6 a6 = {0}; + + if (address.protocol == IPv4_UDP) { + p = (struct sockaddr *) &a4; + p_len = sizeof a4; + } else { + p = (struct sockaddr *) &a6; + p_len = sizeof a6; + } + + if (!_sockets[slot].ready) { + _sockets[slot].socket = socket(address.protocol == IPv4_UDP ? AF_INET : AF_INET6, SOCK_DGRAM, IPPROTO_UDP); + + if (_sockets[slot].socket == -1) { + fprintf(stderr, "ERROR: socket failed (errno %d)\n", errno); + return 0; + } + + if (address.protocol == IPv4_UDP) { + a4.sin_family = AF_INET; + a4.sin_port = htons(address.port); + a4.sin_addr.s_addr = address.v4_address_as_u32; + } else { + a6.sin6_family = AF_INET6; + a6.sin6_port = htons(address.port); + memcpy(a6.sin6_addr.s6_addr, address.v6_address, sizeof a6.sin6_addr.s6_addr); + } + + if (bind(_sockets[slot].socket, p, p_len) == -1) { + close(_sockets[slot].socket); + + fprintf(stderr, "ERROR: bind failed (errno %d)\n", errno); + return 0; + } + + if (getsockname(_sockets[slot].socket, p, &(socklen_t) {p_len}) == -1) { + close(_sockets[slot].socket); + + fprintf(stderr, "ERROR: getsockname failed (errno %d)\n", errno); + return 0; + } + + if (p->sa_family == AF_INET) + _sockets[slot].local_port = ntohs(a4.sin_port); + else + _sockets[slot].local_port = ntohs(a6.sin6_port); + + _sockets[slot].ready = 1; + _sockets[slot].address = address; + } + + if (local_port != NULL) + *local_port = _sockets[slot].local_port; + + return 1; +} + +i64 p_recv(u16 slot, IP_Address address, i64 size, u8 *data, u16 *local_port, IP_Address *remote_address) { + assert(slot < MAX_NUM_SOCKETS); + assert(address.protocol == IPv4_UDP || address.protocol == IPv6_UDP); + + if (!sockets_open(slot, address, local_port)) + return 0; + if (size <= 0) + return 0; + + struct sockaddr *p; + i32 p_len; + + struct sockaddr_in a4 = {0}; + struct sockaddr_in6 a6 = {0}; + + if (address.protocol == IPv4_UDP) { + p = (struct sockaddr *) &a4; + p_len = sizeof a4; + } else { + p = (struct sockaddr *) &a6; + p_len = sizeof a6; + } + + i64 recieved = recvfrom( + _sockets[slot].socket, + data, + size, + MSG_DONTWAIT, + p, + &(socklen_t) {p_len} + ); + + if (recieved < 0) { + if (errno == EAGAIN || errno == EWOULDBLOCK) + return 0; + + fprintf(stderr, "ERROR: recvfrom failed (errno %d)\n", errno); + return 0; + } + + if (remote_address != NULL) { + memset(remote_address, 0, sizeof *remote_address); + remote_address->protocol = address.protocol; + + if (address.protocol == IPv4_UDP) { + remote_address->port = ntohs(a4.sin_port); + remote_address->v4_address_as_u32 = a4.sin_addr.s_addr; + } else { + remote_address->port = ntohs(a6.sin6_port); + memcpy(remote_address->v6_address, a6.sin6_addr.s6_addr, sizeof remote_address->v6_address); + } + } + + return recieved; +} + +i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data, u16 *local_port) { + assert(slot < MAX_NUM_SOCKETS); + assert(address.protocol == IPv4_UDP || address.protocol == IPv6_UDP); + + IP_Address local_address = address; + local_address.port = 0; + + if (!sockets_open(slot, local_address, local_port)) + return 0; + if (size <= 0) + return 0; + + struct sockaddr *p; + i32 p_len; + + struct sockaddr_in a4 = {0}; + struct sockaddr_in6 a6 = {0}; + + if (address.protocol == IPv4_UDP) { + p = (struct sockaddr *) &a4; + p_len = sizeof a4; + + a4.sin_family = AF_INET; + a4.sin_port = htons(address.port); + a4.sin_addr.s_addr = address.v4_address_as_u32; + } else { + p = (struct sockaddr *) &a6; + p_len = sizeof a6; + + a6.sin6_family = AF_INET6; + a6.sin6_port = htons(address.port); + memcpy(a6.sin6_addr.s6_addr, address.v6_address, sizeof a6.sin6_addr.s6_addr); + } + + i64 sent = sendto( + _sockets[0].socket, + data, + size, + MSG_DONTWAIT, + p, + p_len + ); + + if (sent < 0) { + if (errno == EAGAIN || errno == EWOULDBLOCK) + return 0; + + fprintf(stderr, "ERROR: sendto failed (errno %d)\n", errno); + return 0; + } + + return sent; } +#endif + // ================================================================ // // X11 @@ -338,8 +593,6 @@ i64 p_send(u16 slot, IP_Address address, i64 size, u8 *data) { #include <X11/Xlib.h> #include <X11/Xutil.h> #include <X11/Xatom.h> -#include <sched.h> -#include <time.h> static i16 _key_table[512] = {0}; static u32 _buffer[MAX_NUM_PIXELS] = {0}; @@ -516,13 +769,25 @@ void p_init(void) { XSetICFocus(_ic); XSetWMProtocols(_display, _window, &_wm_delete_window, 1); - XStoreName(_display, _window, platform.title); + if (platform.title != NULL) + XStoreName(_display, _window, platform.title); + XMapWindow(_display, _window); } void p_cleanup(void) { - XDestroyWindow(_display, _window); - XCloseDisplay (_display); + if (!platform.graceful_exit) + return; + + if (_window != 0) + XDestroyWindow(_display, _window); + if (_display != NULL) + XCloseDisplay (_display); + + _display = NULL; + _window = 0; + + sockets_cleanup(); } i32 p_handle_events(void) { |