Skip to content

Commit 821441b

Browse files
authored
Merge pull request #142 from codam-coding-college/webasm
Add WASM support.
2 parents d695720 + 802ce12 commit 821441b

12 files changed

+668
-112
lines changed

.github/workflows/wasm.yml

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
# Simple workflow for deploying static content to GitHub Pages
2+
name: Deploy static content to Pages
3+
4+
on:
5+
push:
6+
branches: ["webasm"]
7+
workflow_dispatch:
8+
9+
permissions:
10+
contents: read
11+
pages: write
12+
id-token: write
13+
14+
# Allow only one concurrent deployment, skipping runs queued between the run in-progress and latest queued.
15+
# However, do NOT cancel in-progress runs as we want to allow these production deployments to complete.
16+
concurrency:
17+
group: "pages"
18+
cancel-in-progress: false
19+
20+
# Single deploy job since we're just deploying
21+
jobs:
22+
deploy:
23+
environment:
24+
name: github-pages
25+
url: ${{ steps.deployment.outputs.page_url }}
26+
runs-on: ubuntu-latest
27+
#TODO: add a build step to get the wasm file instead of commiting it.
28+
#Doesn't really matter atm since the git history is polluted anyway
29+
steps:
30+
- name: Checkout
31+
uses: actions/checkout@v4
32+
- name: Setup Pages
33+
uses: actions/configure-pages@v5
34+
- name: Upload artifact
35+
uses: actions/upload-pages-artifact@v3
36+
with:
37+
path: './web'
38+
- name: Deploy to GitHub Pages
39+
id: deployment
40+
uses: actions/deploy-pages@v4

CMakeLists.txt

+44-35
Original file line numberDiff line numberDiff line change
@@ -34,14 +34,13 @@ set(BUILD_TESTS OFF CACHE BOOL "Build the tests to verify the integrity of the l
3434
add_definitions(-D LODEPNG_NO_COMPILE_ENCODER)
3535
add_definitions(-D LODEPNG_NO_COMPILE_ANCILLARY_CHUNKS)
3636

37-
if(UNIX)
38-
set(CCSHADER ${PROJECT_SOURCE_DIR}/tools/compile_shader.sh)
37+
if(UNIX AND NOT EMSCRIPTEN)
3938
add_compile_options(
4039
-Wextra
4140
-Wall
4241
-Werror
4342
-Wunreachable-code
44-
43+
4544
# Some low priority warnings that are annoying.
4645
-Wno-char-subscripts
4746
-Wno-sign-compare
@@ -58,31 +57,38 @@ if(UNIX)
5857
endif(DEBUG)
5958
else()
6059
# TODO: Figure out what we need for windows.
61-
set(CCSHADER ${PROJECT_SOURCE_DIR}/tools/compile_shader.bat)
6260
endif()
6361

6462
# Build specific files
6563
# @see https://cmake.org/cmake/help/latest/command/add_custom_command.html
6664
# -----------------------------------------------------------------------------
6765

66+
if (UNIX)
67+
set(CCSHADER ${TOOLS_DIR}/compile_shader.sh)
68+
else()
69+
set(CCSHADER ${TOOLS_DIR}/compile_shader.bat)
70+
endif()
71+
72+
# Add custom command for fragment shader
6873
add_custom_command(
69-
COMMENT "Building fragment shader"
70-
DEPENDS ${PROJECT_SOURCE_DIR}/shaders/default.frag
71-
OUTPUT mlx_frag_shader.c
72-
COMMAND ${CCSHADER} ${PROJECT_SOURCE_DIR}/shaders/default.frag > mlx_frag_shader.c
73-
VERBATIM
74-
PRE_BUILD
75-
USES_TERMINAL
74+
COMMENT "Building fragment shader"
75+
DEPENDS ${PROJECT_SOURCE_DIR}/shaders/default.frag
76+
OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/mlx_frag_shader.c
77+
COMMAND ${CCSHADER} ${PROJECT_SOURCE_DIR}/shaders/default.frag ${EMSCRIPTEN} > ${CMAKE_CURRENT_BINARY_DIR}/mlx_frag_shader.c
78+
VERBATIM
79+
PRE_BUILD
80+
USES_TERMINAL
7681
)
7782

83+
# Add custom command for vertex shader
7884
add_custom_command(
79-
COMMENT "Building vertex shader"
80-
DEPENDS ${PROJECT_SOURCE_DIR}/shaders/default.vert
81-
OUTPUT mlx_vert_shader.c
82-
COMMAND ${CCSHADER} ${PROJECT_SOURCE_DIR}/shaders/default.vert > mlx_vert_shader.c
83-
VERBATIM
84-
PRE_BUILD
85-
USES_TERMINAL
85+
COMMENT "Building vertex shader"
86+
DEPENDS ${PROJECT_SOURCE_DIR}/shaders/default.vert
87+
OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/mlx_vert_shader.c
88+
COMMAND ${CCSHADER} ${PROJECT_SOURCE_DIR}/shaders/default.vert ${EMSCRIPTEN} > ${CMAKE_CURRENT_BINARY_DIR}/mlx_vert_shader.c
89+
VERBATIM
90+
PRE_BUILD
91+
USES_TERMINAL
8692
)
8793

8894
# Sources
@@ -125,29 +131,32 @@ target_include_directories(mlx42 PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/include)
125131
# Dependencies
126132
# -----------------------------------------------------------------------------
127133

128-
find_package(glfw3)
129134
find_package(OpenGL REQUIRED)
130135

131-
target_link_libraries(mlx42 OpenGL::GL)
132-
if (NOT glfw3_FOUND AND GLFW_FETCH)
133-
message(STATUS "Install GLFW to suppress this message")
134-
message(STATUS "Please wait, fetching GLFW ...")
135-
include(${CMAKE_DIR}/LinkGLFW.cmake)
136-
LinkGLFW(mlx42)
137-
elseif(NOT glfw3_FOUND AND NOT GLFW_FETCH)
138-
message(FATAL_ERROR "Unable to build: GLFW can't be found nor fetched.")
139-
endif()
140-
141-
if (glfw3_FOUND)
142-
target_link_libraries(mlx42 ${GLFW3_LIBRARY})
143-
endif()
144-
if(APPLE)
145-
target_link_libraries(mlx42 "-framework Cocoa" "-framework IOKit")
136+
if(EMSCRIPTEN)
137+
target_link_libraries(mlx42 "-s USE_GLFW=3" "-s FULL_ES3=1")
138+
else()
139+
target_link_libraries(mlx42 OpenGL::GL)
140+
find_package(glfw3)
141+
if (glfw3_FOUND)
142+
target_link_libraries(mlx42 ${GLFW3_LIBRARY})
143+
endif()
144+
if (NOT glfw3_FOUND AND GLFW_FETCH)
145+
message(STATUS "Install GLFW to suppress this message")
146+
message(STATUS "Please wait, fetching GLFW ...")
147+
include(${CMAKE_DIR}/LinkGLFW.cmake)
148+
LinkGLFW(mlx42)
149+
elseif(NOT glfw3_FOUND AND NOT GLFW_FETCH)
150+
message(FATAL_ERROR "Unable to build: GLFW can't be found nor fetched.")
151+
endif()
152+
if(APPLE)
153+
target_link_libraries(mlx42 "-framework Cocoa" "-framework IOKit")
154+
endif()
146155
endif()
147156

148157
# Testing
149158
# -----------------------------------------------------------------------------
150-
# Only build tests if we are the main project or explicitly told to, make sure
159+
# Only build tests if we are the main project or explicitly told to, make sure
151160
# tests are not built when mlx42 is included as a subproject, use MLX42_BUILD_TESTS to overwrite this
152161
# use cmake -DBUILD_TESTS=ON/-DMLX42_BUILD_TESTS=ON to build tests
153162

README.md

+10-2
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,9 @@ MLX42 is a performant, easy to use, cross-platform, minimal windowing graphics l
1616

1717
It provides primitive tools to draw textures onto the window as well as modifying them at runtime as they get displayed on the window.
1818

19+
> [!IMPORTANT]
20+
> At times it may seem like no updates have taken place for a long time. This is expected, the project / lib is considered completed and requires minimal updates. Bug fixes are still guaranteed and the project is still being actively maintained.
21+
1922
# Features ✨
2023

2124
MLX42 comes with a plethora of features that make using it actually a joy instead of a chore.
@@ -39,8 +42,13 @@ It is built on OpenGL and uses batched rendering to speed up the rendering proce
3942
## Open source && Community driven 🌐
4043
This project is being actively maintained by Codam as well as students from the 42 Network. This gives students the direct opportunity to learn more about the library itself as well as fix any potential bugs instead of merely accepting them.
4144

42-
> [!IMPORTANT]
43-
> At times it may seem like no updates have taken place for a long time. This is expected, the project / lib is considered completed and requires minimal updates. Bug fixes are still guaranteed and the project is still being actively maintained.
45+
## Emscripten Compatibility 🚀
46+
MLX42 introduces compatibility with [Emscripten](https://emscripten.org/), allowing MLX42 to run in web browsers through WebAssembly. This modification were made possible thanks to [@PepeLevi](https://github.com/PepeLevi/MLX42_emcc), credits to him for his fork and contributions.
47+
48+
### Highlights
49+
- **Emscripten Support**: Compile MLX42 with Emscripten, enabling graphical applications to run in a web environment.
50+
- **WebAssembly Compatibility**: Ensures that MLX42 can be utilized in modern web browsers, expanding its usability beyond traditional desktop environments.
51+
- **Updated Documentation**: Provided guidance on how to build and run MLX42 projects using Emscripten.
4452

4553
---
4654

src/mlx_init.c

+65-52
Original file line numberDiff line numberDiff line change
@@ -70,7 +70,7 @@ static bool mlx_create_buffers(mlx_t* mlx)
7070
/**
7171
* Compiles the given shader source code of a given shader type.
7272
* Returns shader object via param.
73-
*
73+
*
7474
* @param code The shader source code.
7575
* @param Type GL_VERTEX_SHADER, GL_FRAGMENT_SHADER, GL_GEOMETRY_SHADER, ...
7676
* @return Non-zero on success, else 0.
@@ -81,7 +81,7 @@ static uint32_t mlx_compile_shader(const char* code, int32_t type)
8181
int32_t success;
8282
char infolog[512] = {0};
8383

84-
if (!code || (shader = glCreateShader(type)) == 0)
84+
if (!code || (shader = glCreateShader(type)) == 0)
8585
return (0);
8686

8787
GLint len = strlen(code);
@@ -100,53 +100,56 @@ static uint32_t mlx_compile_shader(const char* code, int32_t type)
100100

101101
static bool mlx_init_render(mlx_t* mlx)
102102
{
103-
uint32_t vshader = 0;
104-
uint32_t fshader = 0;
105-
char infolog[512] = {0};
106-
mlx_ctx_t* mlxctx = mlx->context;
107-
108-
glfwMakeContextCurrent(mlx->window);
109-
glfwSetFramebufferSizeCallback(mlx->window, framebuffer_callback);
110-
glfwSetWindowUserPointer(mlx->window, mlx);
111-
glfwSwapInterval(MLX_SWAP_INTERVAL);
112-
113-
// Load all OpenGL function pointers
114-
if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress))
115-
return (mlx_error(MLX_GLADFAIL));
116-
117-
if (!(vshader = mlx_compile_shader(vert_shader, GL_VERTEX_SHADER)))
118-
return (mlx_error(MLX_VERTFAIL));
119-
if (!(fshader = mlx_compile_shader(frag_shader, GL_FRAGMENT_SHADER)))
120-
return (mlx_error(MLX_FRAGFAIL));
121-
if (!(mlxctx->shaderprogram = glCreateProgram()))
122-
{
123-
glDeleteShader(fshader);
124-
glDeleteShader(vshader);
125-
return (mlx_error(MLX_SHDRFAIL));
126-
}
127-
glAttachShader(mlxctx->shaderprogram, vshader);
128-
glAttachShader(mlxctx->shaderprogram, fshader);
129-
glLinkProgram(mlxctx->shaderprogram);
130-
131-
glDeleteShader(vshader);
132-
glDeleteShader(fshader);
133-
glDetachShader(mlxctx->shaderprogram, vshader);
134-
glDetachShader(mlxctx->shaderprogram, fshader);
135-
136-
int32_t success;
137-
glGetProgramiv(mlxctx->shaderprogram, GL_LINK_STATUS, &success);
138-
if (!success)
139-
{
140-
glGetProgramInfoLog(mlxctx->shaderprogram, sizeof(infolog), NULL, infolog);
141-
fprintf(stderr, "%s", infolog);
142-
return (mlx_error(MLX_SHDRFAIL));
143-
}
144-
glUseProgram(mlxctx->shaderprogram);
145-
146-
for (size_t i = 0; i < 16; i++)
147-
mlxctx->bound_textures[i] = 0;
148-
149-
return (true);
103+
uint32_t vshader = 0;
104+
uint32_t fshader = 0;
105+
char infolog[512] = {0};
106+
mlx_ctx_t* mlxctx = mlx->context;
107+
108+
glfwMakeContextCurrent(mlx->window);
109+
glfwSetFramebufferSizeCallback(mlx->window, framebuffer_callback);
110+
glfwSetWindowUserPointer(mlx->window, mlx);
111+
glfwSwapInterval(MLX_SWAP_INTERVAL);
112+
113+
// Load all OpenGL function pointers
114+
if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress))
115+
return (mlx_error(MLX_GLADFAIL));
116+
if (!(vshader = mlx_compile_shader(vert_shader, GL_VERTEX_SHADER)))
117+
return (mlx_error(MLX_VERTFAIL));
118+
if (!(fshader = mlx_compile_shader(frag_shader, GL_FRAGMENT_SHADER)))
119+
return (mlx_error(MLX_FRAGFAIL));;
120+
if (!(mlxctx->shaderprogram = glCreateProgram()))
121+
{
122+
glDeleteShader(fshader);
123+
glDeleteShader(vshader);
124+
return (mlx_error(MLX_SHDRFAIL));
125+
}
126+
glAttachShader(mlxctx->shaderprogram, vshader);
127+
glAttachShader(mlxctx->shaderprogram, fshader);
128+
glLinkProgram(mlxctx->shaderprogram);
129+
130+
int32_t success;
131+
glGetProgramiv(mlxctx->shaderprogram, GL_LINK_STATUS, &success);
132+
if (!success)
133+
{
134+
glGetProgramInfoLog(mlxctx->shaderprogram, sizeof(infolog), NULL, infolog);
135+
fprintf(stderr, "%s", infolog);
136+
glDeleteProgram(mlxctx->shaderprogram);
137+
glDeleteShader(vshader);
138+
glDeleteShader(fshader);
139+
return (mlx_error(MLX_SHDRFAIL));
140+
}
141+
142+
// Detach shaders after linking but before deleting them
143+
glDetachShader(mlxctx->shaderprogram, vshader);
144+
glDetachShader(mlxctx->shaderprogram, fshader);
145+
146+
// Delete shaders
147+
glDeleteShader(vshader);
148+
glDeleteShader(fshader);
149+
glUseProgram(mlxctx->shaderprogram);
150+
for (size_t i = 0; i < 16; i++)
151+
mlxctx->bound_textures[i] = 0;
152+
return (true);
150153
}
151154

152155
//= Public =//
@@ -174,25 +177,35 @@ mlx_t* mlx_init(int32_t width, int32_t height, const char* title, bool resize)
174177
return (free(mlx), (void*)mlx_error(MLX_MEMFAIL));
175178

176179
mlx_ctx_t* const mlxctx = mlx->context;
180+
mlx->window = NULL;
177181
mlx->width = width;
178182
mlx->height = height;
179183
mlxctx->initialWidth = width;
180184
mlxctx->initialHeight = height;
181185

186+
#ifdef EMSCRIPTEN
187+
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 2);
188+
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 0);
189+
glfwWindowHint(GLFW_DECORATED, GLFW_TRUE);
190+
glfwWindowHint(GLFW_VISIBLE, GLFW_TRUE);
191+
glfwWindowHint(GLFW_CLIENT_API, GLFW_OPENGL_ES_API);
192+
glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
193+
#else
182194
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
183195
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
184196
glfwWindowHint(GLFW_MAXIMIZED, mlx_settings[MLX_MAXIMIZED]);
185197
glfwWindowHint(GLFW_DECORATED, mlx_settings[MLX_DECORATED]);
186198
glfwWindowHint(GLFW_VISIBLE, !mlx_settings[MLX_HEADLESS]);
187-
glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
199+
#endif
188200
#ifdef __APPLE__
189201
glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);
190202
#endif
191-
glfwWindowHint(GLFW_RESIZABLE, resize);
203+
glfwWindowHint(GLFW_RESIZABLE, resize ? GLFW_TRUE : GLFW_FALSE);
192204
if (!(mlx->window = glfwCreateWindow(width, height, title, mlx_settings[MLX_FULLSCREEN] ? glfwGetPrimaryMonitor() : NULL, NULL)))
193-
return (mlx_terminate(mlx), (void*)mlx_error(MLX_WINFAIL));
205+
return (glfwTerminate(), (void*)mlx_error(MLX_WINFAIL));
194206
if (!mlx_init_render(mlx) || !mlx_create_buffers(mlx))
195207
return (mlx_terminate(mlx), NULL);
208+
glfwMakeContextCurrent(mlx->window);
196209
return (mlx);
197210
}
198211

src/mlx_loop.c

+12-2
Original file line numberDiff line numberDiff line change
@@ -90,17 +90,25 @@ bool mlx_loop_hook(mlx_t* mlx, void (*f)(void*), void* param)
9090
}
9191

9292
// glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);
93+
/**
94+
* In Emscripten the lood is defined differently, there the this function
95+
* is passed to the while loop instead
96+
*/
9397
void mlx_loop(mlx_t* mlx)
9498
{
9599
MLX_NONNULL(mlx);
96100

101+
#ifdef EMSCRIPTEN
102+
static double start, oldstart = 0;
103+
#else
97104
double start, oldstart = 0;
98-
while (!glfwWindowShouldClose(mlx->window))
105+
while (!glfwWindowShouldClose(mlx->window))
99106
{
107+
#endif
100108
start = glfwGetTime();
101109
mlx->delta_time = start - oldstart;
102110
oldstart = start;
103-
111+
104112
glClearColor(0.2f, 0.2f, 0.2f, 1.0f);
105113
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
106114
glfwGetWindowSize(mlx->window, &(mlx->width), &(mlx->height));
@@ -114,5 +122,7 @@ void mlx_loop(mlx_t* mlx)
114122

115123
glfwSwapBuffers(mlx->window);
116124
glfwPollEvents();
125+
#ifndef EMSCRIPTEN
117126
}
127+
#endif
118128
}

0 commit comments

Comments
 (0)