diff options
| author | 3gg <3gg@shellblade.net> | 2024-01-18 19:33:18 -0800 | 
|---|---|---|
| committer | 3gg <3gg@shellblade.net> | 2024-01-18 19:33:18 -0800 | 
| commit | fc883e0b0449509ba2e1c5d14d187feee098ab34 (patch) | |
| tree | 83dec5ce272cf07ddf7855a44413253210438490 | |
| parent | cef3385c2bee0b098a7795548345a9281ace008e (diff) | |
Simplify game callbacks.
| -rw-r--r-- | gfx-app/include/gfx/gfx_app.h | 39 | ||||
| -rw-r--r-- | gltfview/CMakeLists.txt | 3 | ||||
| -rw-r--r-- | gltfview/src/game.c | 52 | ||||
| -rw-r--r-- | gltfview/src/game.h | 34 | ||||
| -rw-r--r-- | gltfview/src/main.c | 67 | 
5 files changed, 82 insertions, 113 deletions
| diff --git a/gfx-app/include/gfx/gfx_app.h b/gfx-app/include/gfx/gfx_app.h index 0033bde..3c544fa 100644 --- a/gfx-app/include/gfx/gfx_app.h +++ b/gfx-app/include/gfx/gfx_app.h | |||
| @@ -2,6 +2,7 @@ | |||
| 2 | 2 | ||
| 3 | #include <stdbool.h> | 3 | #include <stdbool.h> | 
| 4 | 4 | ||
| 5 | /// Application settings. | ||
| 5 | typedef struct GfxAppDesc { | 6 | typedef struct GfxAppDesc { | 
| 6 | int argc; // Number of application arguments. | 7 | int argc; // Number of application arguments. | 
| 7 | const char** argv; // Application arguments. | 8 | const char** argv; // Application arguments. | 
| @@ -12,12 +13,19 @@ typedef struct GfxAppDesc { | |||
| 12 | const char* title; // Window title. | 13 | const char* title; // Window title. | 
| 13 | } GfxAppDesc; | 14 | } GfxAppDesc; | 
| 14 | 15 | ||
| 16 | typedef bool (*GfxAppInit)(const GfxAppDesc*, void** app_state); | ||
| 17 | typedef void (*GfxAppUpdate)(void* app_state, double t, double dt); | ||
| 18 | typedef void (*GfxAppRender)(void* app_state); | ||
| 19 | typedef void (*GfxAppResize)(void* app_state, int width, int height); | ||
| 20 | typedef void (*GfxAppShutdown)(void* app_state); | ||
| 21 | |||
| 22 | /// Application callback functions. | ||
| 15 | typedef struct GfxAppCallbacks { | 23 | typedef struct GfxAppCallbacks { | 
| 16 | bool (*init)(const GfxAppDesc*, void** app_state); | 24 | GfxAppInit init; | 
| 17 | void (*update)(void* app_state, double t, double dt); | 25 | GfxAppUpdate update; | 
| 18 | void (*render)(void* app_state); | 26 | GfxAppRender render; | 
| 19 | void (*resize)(void* app_state, int width, int height); | 27 | GfxAppResize resize; | 
| 20 | void (*shutdown)(void* app_state); | 28 | GfxAppShutdown shutdown; | 
| 21 | } GfxAppCallbacks; | 29 | } GfxAppCallbacks; | 
| 22 | 30 | ||
| 23 | /// Create a window with an OpenGL context and run the main loop. | 31 | /// Create a window with an OpenGL context and run the main loop. | 
| @@ -25,3 +33,24 @@ bool gfx_app_run(const GfxAppDesc*, const GfxAppCallbacks*); | |||
| 25 | 33 | ||
| 26 | /// Get the mouse coordinates relative to the app's window. | 34 | /// Get the mouse coordinates relative to the app's window. | 
| 27 | void gfx_app_get_mouse_position(double* x, double* y); | 35 | void gfx_app_get_mouse_position(double* x, double* y); | 
| 36 | |||
| 37 | /// Define a main function that initializes and puts the application in a loop. | ||
| 38 | /// See also: gfx_app_run(). | ||
| 39 | #define GFX_APP_MAIN(WIDTH, HEIGHT, MAX_FPS) \ | ||
| 40 | int main(int argc, const char** argv) { \ | ||
| 41 | gfx_app_run( \ | ||
| 42 | &(GfxAppDesc){ \ | ||
| 43 | .argc = argc, \ | ||
| 44 | .argv = argv, \ | ||
| 45 | .width = WIDTH, \ | ||
| 46 | .height = HEIGHT, \ | ||
| 47 | .max_fps = MAX_FPS, \ | ||
| 48 | .update_delta_time = MAX_FPS > 0 ? 1.0 / (double)MAX_FPS : 0.0}, \ | ||
| 49 | &(GfxAppCallbacks){ \ | ||
| 50 | .init = (GfxAppInit)app_init, \ | ||
| 51 | .update = (GfxAppUpdate)app_update, \ | ||
| 52 | .render = (GfxAppRender)app_render, \ | ||
| 53 | .resize = (GfxAppResize)app_resize, \ | ||
| 54 | .shutdown = (GfxAppShutdown)app_end}); \ | ||
| 55 | return 0; \ | ||
| 56 | } | ||
| diff --git a/gltfview/CMakeLists.txt b/gltfview/CMakeLists.txt index de745ce..98c3b47 100644 --- a/gltfview/CMakeLists.txt +++ b/gltfview/CMakeLists.txt | |||
| @@ -5,8 +5,7 @@ add_subdirectory(src/plugins) | |||
| 5 | project(gltfview) | 5 | project(gltfview) | 
| 6 | 6 | ||
| 7 | add_executable(gltfview | 7 | add_executable(gltfview | 
| 8 | src/game.c | 8 | src/game.c) | 
| 9 | src/main.c) | ||
| 10 | 9 | ||
| 11 | target_include_directories(gltfview PRIVATE | 10 | target_include_directories(gltfview PRIVATE | 
| 12 | src/) | 11 | src/) | 
| diff --git a/gltfview/src/game.c b/gltfview/src/game.c index 5ca9ad4..c331190 100644 --- a/gltfview/src/game.c +++ b/gltfview/src/game.c | |||
| @@ -7,11 +7,17 @@ | |||
| 7 | #define _GNU_SOURCE 200112L // For readlink() | 7 | #define _GNU_SOURCE 200112L // For readlink() | 
| 8 | 8 | ||
| 9 | #include "game.h" | 9 | #include "game.h" | 
| 10 | |||
| 10 | #include "plugins/plugin.h" | 11 | #include "plugins/plugin.h" | 
| 11 | 12 | ||
| 13 | #include <gfx/gfx.h> | ||
| 14 | #include <gfx/gfx_app.h> | ||
| 12 | #include <gfx/render_backend.h> | 15 | #include <gfx/render_backend.h> | 
| 16 | #include <gfx/renderer.h> | ||
| 13 | #include <gfx/scene/camera.h> | 17 | #include <gfx/scene/camera.h> | 
| 18 | #include <gfx/scene/node.h> | ||
| 14 | #include <gfx/scene/object.h> | 19 | #include <gfx/scene/object.h> | 
| 20 | #include <gfx/scene/scene.h> | ||
| 15 | 21 | ||
| 16 | #include <error.h> | 22 | #include <error.h> | 
| 17 | #include <log/log.h> | 23 | #include <log/log.h> | 
| @@ -21,6 +27,7 @@ | |||
| 21 | #include <assert.h> | 27 | #include <assert.h> | 
| 22 | #include <stdbool.h> | 28 | #include <stdbool.h> | 
| 23 | #include <stdio.h> | 29 | #include <stdio.h> | 
| 30 | #include <stdlib.h> | ||
| 24 | 31 | ||
| 25 | #include <linux/limits.h> | 32 | #include <linux/limits.h> | 
| 26 | 33 | ||
| @@ -28,18 +35,32 @@ | |||
| 28 | 35 | ||
| 29 | #undef _GNU_SOURCE | 36 | #undef _GNU_SOURCE | 
| 30 | 37 | ||
| 31 | // Plugin to load if no plugin is provided. | 38 | static const int WIDTH = 1350; | 
| 32 | static const char* DEFAULT_PLUGIN = "texture_view"; | 39 | static const int HEIGHT = 900; | 
| 40 | static const int MAX_FPS = 60; | ||
| 33 | 41 | ||
| 34 | bool game_new(Game* game, int argc, const char** argv) { | 42 | void app_end(Game* game); | 
| 35 | assert(game); | 43 | |
| 44 | bool app_init(const GfxAppDesc* desc, void** app_state) { | ||
| 45 | assert(desc); | ||
| 46 | |||
| 47 | if (desc->argc <= 1) { | ||
| 48 | LOGE("Usage: %s <plugin> [plugin args]", desc->argv[0]); | ||
| 49 | return false; | ||
| 50 | } | ||
| 51 | |||
| 52 | Game* game = calloc(1, sizeof(Game)); | ||
| 53 | if (!game) { | ||
| 54 | LOGE("Failed to allocate game state"); | ||
| 55 | return false; | ||
| 56 | } | ||
| 36 | 57 | ||
| 37 | // Syntax: game [plugin] <plugin args> | 58 | // Syntax: game <plugin> [plugin args] | 
| 38 | // | 59 | // | 
| 39 | // Here we consume the [plugin] arg so that plugins receive the remainder | 60 | // Here we consume the <plugin> arg so that plugins receive the remainder | 
| 40 | // args starting from 0. | 61 | // args starting from 0. | 
| 41 | game->argc = argc - 1; | 62 | game->argc = desc->argc - 1; | 
| 42 | game->argv = argv + 1; | 63 | game->argv = desc->argv + 1; | 
| 43 | 64 | ||
| 44 | char exe_path_buf[NAME_MAX] = {0}; | 65 | char exe_path_buf[NAME_MAX] = {0}; | 
| 45 | if (readlink("/proc/self/exe", exe_path_buf, sizeof(exe_path_buf)) == -1) { | 66 | if (readlink("/proc/self/exe", exe_path_buf, sizeof(exe_path_buf)) == -1) { | 
| @@ -59,7 +80,7 @@ bool game_new(Game* game, int argc, const char** argv) { | |||
| 59 | goto cleanup; | 80 | goto cleanup; | 
| 60 | } | 81 | } | 
| 61 | 82 | ||
| 62 | const char* plugin = argc > 1 ? argv[1] : DEFAULT_PLUGIN; | 83 | const char* plugin = desc->argv[1]; | 
| 63 | if (!(game->plugin = load_plugin(game->plugin_engine, plugin))) { | 84 | if (!(game->plugin = load_plugin(game->plugin_engine, plugin))) { | 
| 64 | goto cleanup; | 85 | goto cleanup; | 
| 65 | } | 86 | } | 
| @@ -91,15 +112,16 @@ bool game_new(Game* game, int argc, const char** argv) { | |||
| 91 | } | 112 | } | 
| 92 | } | 113 | } | 
| 93 | 114 | ||
| 115 | *app_state = game; | ||
| 94 | return true; | 116 | return true; | 
| 95 | 117 | ||
| 96 | cleanup: | 118 | cleanup: | 
| 97 | LOGE("Gfx error: %s", get_error()); | 119 | LOGE("Gfx error: %s", get_error()); | 
| 98 | game_end(game); | 120 | app_end(game); | 
| 99 | return false; | 121 | return false; | 
| 100 | } | 122 | } | 
| 101 | 123 | ||
| 102 | void game_end(Game* game) { | 124 | void app_end(Game* game) { | 
| 103 | assert(game); | 125 | assert(game); | 
| 104 | if (game->gfx) { | 126 | if (game->gfx) { | 
| 105 | gfx_destroy(&game->gfx); | 127 | gfx_destroy(&game->gfx); | 
| @@ -112,7 +134,7 @@ void game_end(Game* game) { | |||
| 112 | } | 134 | } | 
| 113 | } | 135 | } | 
| 114 | 136 | ||
| 115 | void game_update(Game* game, double t, double dt) { | 137 | void app_update(Game* game, double t, double dt) { | 
| 116 | plugin_engine_update(game->plugin_engine); | 138 | plugin_engine_update(game->plugin_engine); | 
| 117 | if (plugin_reloaded(game->plugin) && | 139 | if (plugin_reloaded(game->plugin) && | 
| 118 | plugin_resolve(game->plugin, plugin_init, "init")) { | 140 | plugin_resolve(game->plugin, plugin_init, "init")) { | 
| @@ -129,7 +151,7 @@ void game_update(Game* game, double t, double dt) { | |||
| 129 | } | 151 | } | 
| 130 | } | 152 | } | 
| 131 | 153 | ||
| 132 | void game_render(const Game* game) { | 154 | void app_render(const Game* game) { | 
| 133 | RenderBackend* render_backend = gfx_get_render_backend(game->gfx); | 155 | RenderBackend* render_backend = gfx_get_render_backend(game->gfx); | 
| 134 | Renderer* renderer = gfx_get_renderer(game->gfx); | 156 | Renderer* renderer = gfx_get_renderer(game->gfx); | 
| 135 | 157 | ||
| @@ -149,7 +171,7 @@ void game_render(const Game* game) { | |||
| 149 | gfx_end_frame(render_backend); | 171 | gfx_end_frame(render_backend); | 
| 150 | } | 172 | } | 
| 151 | 173 | ||
| 152 | void game_set_viewport(Game* game, int width, int height) { | 174 | void app_resize(Game* game, int width, int height) { | 
| 153 | RenderBackend* render_backend = gfx_get_render_backend(game->gfx); | 175 | RenderBackend* render_backend = gfx_get_render_backend(game->gfx); | 
| 154 | gfx_set_viewport(render_backend, width, height); | 176 | gfx_set_viewport(render_backend, width, height); | 
| 155 | 177 | ||
| @@ -162,3 +184,5 @@ void game_set_viewport(Game* game, int width, int height) { | |||
| 162 | Camera* camera = gfx_get_camera_camera(game->camera); | 184 | Camera* camera = gfx_get_camera_camera(game->camera); | 
| 163 | camera->projection = projection; | 185 | camera->projection = projection; | 
| 164 | } | 186 | } | 
| 187 | |||
| 188 | GFX_APP_MAIN(WIDTH, HEIGHT, MAX_FPS); | ||
| diff --git a/gltfview/src/game.h b/gltfview/src/game.h index 2a7b7ef..53725c7 100644 --- a/gltfview/src/game.h +++ b/gltfview/src/game.h | |||
| @@ -1,19 +1,13 @@ | |||
| 1 | /* | ||
| 2 | * Header file defining the game state, included by plugins. | ||
| 3 | */ | ||
| 1 | #pragma once | 4 | #pragma once | 
| 2 | 5 | ||
| 3 | #include <gfx/gfx.h> | ||
| 4 | #include <gfx/render_backend.h> | ||
| 5 | #include <gfx/renderer.h> | ||
| 6 | #include <gfx/scene/camera.h> | ||
| 7 | #include <gfx/scene/node.h> | ||
| 8 | #include <gfx/scene/scene.h> | ||
| 9 | |||
| 10 | #include <stdbool.h> | ||
| 11 | |||
| 12 | typedef struct Plugin Plugin; | ||
| 13 | typedef struct PluginEngine PluginEngine; | 6 | typedef struct PluginEngine PluginEngine; | 
| 14 | 7 | typedef struct Plugin Plugin; | |
| 15 | /// The delta time the game should be updated with. | 8 | typedef struct Gfx Gfx; | 
| 16 | static const double game_dt = 1.0 / 60.0; | 9 | typedef struct Scene Scene; | 
| 10 | typedef struct SceneCamera SceneCamera; | ||
| 17 | 11 | ||
| 18 | /// Game state. | 12 | /// Game state. | 
| 19 | typedef struct { | 13 | typedef struct { | 
| @@ -22,16 +16,6 @@ typedef struct { | |||
| 22 | PluginEngine* plugin_engine; | 16 | PluginEngine* plugin_engine; | 
| 23 | Plugin* plugin; | 17 | Plugin* plugin; | 
| 24 | Gfx* gfx; | 18 | Gfx* gfx; | 
| 25 | Scene* scene; | 19 | Scene* scene; // TODO: Move scene graph to plugin? | 
| 26 | SceneCamera* camera; | 20 | SceneCamera* camera; // TODO: Move too. | 
| 27 | } Game; | 21 | } Game; | 
| 28 | |||
| 29 | bool game_new(Game*, int argc, const char** argv); | ||
| 30 | |||
| 31 | void game_end(Game*); | ||
| 32 | |||
| 33 | void game_update(Game*, double t, double dt); | ||
| 34 | |||
| 35 | void game_render(const Game*); | ||
| 36 | |||
| 37 | void game_set_viewport(Game*, int width, int height); | ||
| diff --git a/gltfview/src/main.c b/gltfview/src/main.c deleted file mode 100644 index f4863b4..0000000 --- a/gltfview/src/main.c +++ /dev/null | |||
| @@ -1,67 +0,0 @@ | |||
| 1 | #include "game.h" | ||
| 2 | |||
| 3 | #include <gfx/gfx_app.h> | ||
| 4 | #include <log/log.h> | ||
| 5 | |||
| 6 | #include <stdlib.h> | ||
| 7 | |||
| 8 | static bool init(const GfxAppDesc* desc, void** app_state) { | ||
| 9 | Game* game = calloc(1, sizeof(Game)); | ||
| 10 | if (!game) { | ||
| 11 | LOGE("Failed to allocate game state"); | ||
| 12 | return false; | ||
| 13 | } | ||
| 14 | if (!game_new(game, desc->argc, desc->argv)) { | ||
| 15 | LOGE("Failed to initialize game"); | ||
| 16 | return false; | ||
| 17 | } | ||
| 18 | *app_state = game; | ||
| 19 | return true; | ||
| 20 | } | ||
| 21 | |||
| 22 | static void shutdown(void* app_state) { | ||
| 23 | assert(app_state); | ||
| 24 | Game* game = (Game*)(app_state); | ||
| 25 | game_end(game); | ||
| 26 | } | ||
| 27 | |||
| 28 | static void update(void* app_state, double t, double dt) { | ||
| 29 | assert(app_state); | ||
| 30 | Game* game = (Game*)(app_state); | ||
| 31 | game_update(game, t, dt); | ||
| 32 | } | ||
| 33 | |||
| 34 | static void render(void* app_state) { | ||
| 35 | assert(app_state); | ||
| 36 | Game* game = (Game*)(app_state); | ||
| 37 | game_render(game); | ||
| 38 | } | ||
| 39 | |||
| 40 | static void resize(void* app_state, int width, int height) { | ||
| 41 | assert(app_state); | ||
| 42 | Game* game = (Game*)(app_state); | ||
| 43 | game_set_viewport(game, width, height); | ||
| 44 | } | ||
| 45 | |||
| 46 | int main(int argc, const char** argv) { | ||
| 47 | const int initial_width = 1350; | ||
| 48 | const int initial_height = 900; | ||
| 49 | const int max_fps = 60; | ||
| 50 | |||
| 51 | gfx_app_run( | ||
| 52 | &(GfxAppDesc){ | ||
| 53 | .argc = argc, | ||
| 54 | .argv = argv, | ||
| 55 | .width = initial_width, | ||
| 56 | .height = initial_height, | ||
| 57 | .max_fps = max_fps, | ||
| 58 | .update_delta_time = max_fps > 0 ? 1.0 / (double)max_fps : 0.0}, | ||
| 59 | &(GfxAppCallbacks){ | ||
| 60 | .init = init, | ||
| 61 | .update = update, | ||
| 62 | .render = render, | ||
| 63 | .resize = resize, | ||
| 64 | .shutdown = shutdown}); | ||
| 65 | |||
| 66 | return 0; | ||
| 67 | } | ||
