#ifndef SOKOL_GL_INCLUDED /* sokol_gl.h -- OpenGL 1.x style rendering on top of sokol_gfx.h Project URL: https://github.com/floooh/sokol Do this: #define SOKOL_GL_IMPL before you include this file in *one* C or C++ file to create the implementation. The following defines are used by the implementation to select the platform-specific embedded shader code (these are the same defines as used by sokol_gfx.h and sokol_app.h): SOKOL_GLCORE33 SOKOL_GLES2 SOKOL_GLES3 SOKOL_D3D11 SOKOL_METAL ...optionally provide the following macros to override defaults: SOKOL_ASSERT(c) - your own assert macro (default: assert(c)) SOKOL_MALLOC(s) - your own malloc function (default: malloc(s)) SOKOL_FREE(p) - your own free function (default: free(p)) SOKOL_API_DECL - public function declaration prefix (default: extern) SOKOL_API_IMPL - public function implementation prefix (default: -) SOKOL_LOG(msg) - your own logging function (default: puts(msg)) SOKOL_UNREACHABLE() - a guard macro for unreachable code (default: assert(false)) If sokol_gl.h is compiled as a DLL, define the following before including the declaration or implementation: SOKOL_DLL On Windows, SOKOL_DLL will define SOKOL_API_DECL as __declspec(dllexport) or __declspec(dllimport) as needed. Include the following headers before including sokol_gl.h: sokol_gfx.h Matrix functions have been taken from MESA and Regal. FEATURE OVERVIEW: ================= sokol_gl.h implements a subset of the OpenGLES 1.x feature set useful for when you just want to quickly render a bunch of colored triangles or lines without having to mess with buffers and shaders. The current feature set is mostly useful for debug visualizations and simple UI-style 2D rendering: What's implemented: - vertex components: - position (x, y, z) - 2D texture coords (u, v) - color (r, g, b, a) - primitive types: - triangle list and strip - line list and strip - quad list (TODO: quad strips) - point list (TODO: point size) - one texture layer (no multi-texturing) - viewport and scissor-rect with selectable origin (top-left or bottom-left) - all GL 1.x matrix stack functions, and additionally equivalent functions for gluPerspective and gluLookat Notable GLES 1.x features that are *NOT* implemented: - vertex lighting (this is the most likely GL feature that might be added later) - vertex arrays (although providing whole chunks of vertex data at once might be a useful feature for a later version) - texture coordinate generation - point size and line width - all pixel store functions - no ALPHA_TEST - no clear functions (clearing is handled by the sokol-gfx render pass) - fog Notable differences to GL: - No "enum soup" for render states etc, instead there's a 'pipeline stack', this is similar to GL's matrix stack, but for pipeline-state-objects. The pipeline object at the top of the pipeline stack defines the active set of render states - All angles are in radians, not degrees (note the sgl_rad() and sgl_deg() conversion functions) - No enable/disable state for scissor test, this is always enabled STEP BY STEP: ============= --- To initialize sokol-gl, call: sgl_setup(const sgl_desc_t* desc) NOTE that sgl_setup() must be called *after* initializing sokol-gfx (via sg_setup). This is because sgl_setup() needs to create sokol-gfx resource objects. sgl_setup() needs to know the attributes of the sokol-gfx render pass where sokol-gl rendering will happen through the passed-in sgl_desc_t struct: sg_pixel_format color_format - color pixel format of render pass sg_pixel_format depth_format - depth pixel format of render pass int sample_count - MSAA sample count of render pass These values have the same defaults as sokol_gfx.h and sokol_app.h, to use the default values, leave them zero-initialized. You can adjust the maximum number of vertices and drawing commands per frame through the members: int max_vertices - default is 65536 int max_commands - default is 16384 You can adjust the size of the internal pipeline state object pool with: int pipeline_pool_size - default is 64 Finally you can change the face winding for front-facing triangles and quads: sg_face_winding face_winding - default is SG_FACEWINDING_CCW The default winding for front faces is counter-clock-wise. This is the same as OpenGL's default, but different from sokol-gfx. --- Optionally create pipeline-state-objects if you need render state that differs from sokol-gl's default state: sgl_pipeline pip = sgl_make_pipeline(const sg_pipeline_desc* desc) The similarity with sokol_gfx.h's sg_pipeline type and sg_make_pipeline() function is intended. sgl_make_pipeline() also takes a standard sokol-gfx sg_pipeline_desc object to describe the render state, but without: - shader - vertex layout - color- and depth-pixel-formats - primitive type (lines, triangles, ...) - MSAA sample count Those will be filled in by sgl_make_pipeline(). Note that each call to sgl_make_pipeline() needs to create several sokol-gfx pipeline objects (one for each primitive type). --- if you need to destroy sgl_pipeline objects before sgl_shutdown(): sgl_destroy_pipeline(sgl_pipeline pip) --- After sgl_setup() you can call any of the sokol-gl functions anywhere in a frame, *except* sgl_draw(). The 'vanilla' functions will only change internal sokol-gl state, and not call any sokol-gfx functions. --- Unlike OpenGL, sokol-gl has a function to reset internal state to a known default. This is useful at the start of a sequence of rendering operations: void sgl_defaults(void) This will set the following default state: - current texture coordinate to u=0.0f, v=0.0f - current color to white (rgba all 1.0f) - unbind the current texture and texturing will be disabled - *all* matrices will be set to identity (also the projection matrix) - the default render state will be set by loading the 'default pipeline' into the top of the pipeline stack The current matrix- and pipeline-stack-depths will not be changed by sgl_defaults(). --- change the currently active renderstate through the pipeline-stack functions, this works similar to the traditional GL matrix stack: ...load the default pipeline state on the top of the pipeline stack: sgl_default_pipeline() ...load a specific pipeline on the top of the pipeline stack: sgl_load_pipeline(sgl_pipeline pip) ...push and pop the pipeline stack: sgl_push_pipeline() sgl_pop_pipeline() --- control texturing with: sgl_enable_texture() sgl_disable_texture() sgl_texture(sg_image img) --- set the current viewport and scissor rect with: sgl_viewport(int x, int y, int w, int h, bool origin_top_left) sgl_scissor_rect(int x, int y, int w, int h, bool origin_top_left) ...these calls add a new command to the internal command queue, so that the viewport or scissor rect are set at the right time relative to other sokol-gl calls. --- adjust the transform matrices, matrix manipulation works just like the OpenGL matrix stack: ...set the current matrix mode: sgl_matrix_mode_modelview() sgl_matrix_mode_projection() sgl_matrix_mode_texture() ...load the identity matrix into the current matrix: sgl_load_identity() ...translate, rotate and scale the current matrix: sgl_translate(float x, float y, float z) sgl_rotate(float angle_rad, float x, float y, float z) sgl_scale(float x, float y, float z) NOTE that all angles in sokol-gl are in radians, not in degree. Convert between radians and degree with the helper functions: float sgl_rad(float deg) - degrees to radians float sgl_deg(float rad) - radians to degrees ...directly load the current matrix from a float[16] array: sgl_load_matrix(const float m[16]) sgl_load_transpose_matrix(const float m[16]) ...directly multiply the current matrix from a float[16] array: sgl_mult_matrix(const float m[16]) sgl_mult_transpose_matrix(const float m[16]) The memory layout of those float[16] arrays is the same as in OpenGL. ...more matrix functions: sgl_frustum(float left, float right, float bottom, float top, float near, float far) sgl_ortho(float left, float right, float bottom, float top, float near, float far) sgl_perspective(float fov_y, float aspect, float near, float far) sgl_lookat(float eye_x, float eye_y, float eye_z, float center_x, float center_y, float center_z, float up_x, float up_y, float up_z) These functions work the same as glFrustum(), glOrtho(), gluPerspective() and gluLookAt(). ...and finally to push / pop the current matrix stack: sgl_push_matrix(void) sgl_pop_matrix(void) Again, these work the same as glPushMatrix() and glPopMatrix(). --- perform primitive rendering: ...set the current texture coordinate and color 'registers' with: sgl_t2f(float u, float v) - set current texture coordinate sgl_c*(...) - set current color There are several functions for setting the color (as float values, unsigned byte values, packed as unsigned 32-bit integer, with and without alpha). NOTE that these are the only functions that can be called both inside sgl_begin_*() / sgl_end() and outside. ...start a primitive vertex sequence with: sgl_begin_points() sgl_begin_lines() sgl_begin_line_strip() sgl_begin_triangles() sgl_begin_triangle_strip() sgl_begin_quads() ...after sgl_begin_*() specifiy vertices: sgl_v*(...) sgl_v*_t*(...) sgl_v*_c*(...) sgl_v*_t*_c*(...) These functions write a new vertex to sokol-gl's internal vertex buffer, optionally with texture-coords and color. If the texture coordinate and/or color is missing, it will be taken from the current texture-coord and color 'register'. ...finally, after specifying vertices, call: sgl_end() This will record a new draw command in sokol-gl's internal command list, or it will extend the previous draw command if no relevant state has changed since the last sgl_begin/end pair. --- inside a sokol-gfx rendering pass, call: sgl_draw() This will render everything that has been recorded since the last call to sgl_draw() through sokol-gfx, and will 'rewind' the internal vertex-, uniform- and command-buffers. --- sokol-gl tracks a single internal error code which can be queried with sgl_error_t sgl_error(void) ...which can return the following error codes: SGL_NO_ERROR - all OK, no error occurred since last sgl_draw() SGL_ERROR_VERTICES_FULL - internal vertex buffer is full (checked in sgl_end()) SGL_ERROR_UNIFORMS_FULL - the internal uniforms buffer is full (checked in sgl_end()) SGL_ERROR_COMMANDS_FULL - the internal command buffer is full (checked in sgl_end()) SGL_ERROR_STACK_OVERFLOW - matrix- or pipeline-stack overflow SGL_ERROR_STACK_UNDERFLOW - matrix- or pipeline-stack underflow ...if sokol-gl is in an error-state, sgl_draw() will skip any rendering, and reset the error code to SGL_NO_ERROR. UNDER THE HOOD: =============== sokol_gl.h works by recording vertex data and rendering commands into memory buffers, and then drawing the recorded commands via sokol_gfx.h The only functions which call into sokol_gfx.h are: - sgl_setup() - sgl_shutdown() - sgl_draw() sgl_setup() must be called after initializing sokol-gfx. sgl_shutdown() must be called before shutting down sokol-gfx. sgl_draw() must be called once per frame inside a sokol-gfx render pass. All other sokol-gl function can be called anywhere in a frame, since they just record data into memory buffers owned by sokol-gl. What happens in: sgl_setup(): - 3 memory buffers are allocated, one for vertex data, one for uniform data, and one for commands - sokol-gfx resources are created: a (dynamic) vertex buffer, a shader object (using embedded shader source or byte code), and an 8x8 all-white default texture One vertex is 24 bytes: - float3 position - float2 texture coords - uint32_t color One uniform block is 128 bytes: - mat4 model-view-projection matrix - mat4 texture matrix One draw command is ca. 24 bytes for the actual command code plus command arguments. Each sgl_end() consumes one command, and one uniform block (only when the matrices have changed). The required size for one sgl_begin/end pair is (at most): (152 + 24 * num_verts) bytes sgl_shutdown(): - all sokol-gfx resources (buffer, shader, default-texture and all pipeline objects) are destroyed - the 3 memory buffers are freed sgl_draw(): - copy all recorded vertex data into the dynamic sokol-gfx buffer via a call to sg_update_buffer() - for each recorded command: - if it's a viewport command, call sg_apply_viewport() - if it's a scissor-rect command, call sg_apply_scissor_rect() - if it's a draw command: - depending on what has changed since the last draw command, call sg_apply_pipeline(), sg_apply_bindings() and sg_apply_uniforms() - finally call sg_draw() All other functions only modify the internally tracked state, add data to the vertex, uniform and command buffers, or manipulate the matrix stack. ON DRAW COMMAND MERGING ======================= Not every call to sgl_end() will automatically record a new draw command. If possible, the previous draw command will simply be extended, resulting in fewer actual draw calls later in sgl_draw(). A draw command will be merged with the previous command if "no relevant state has changed" since the last sgl_end(), meaning: - no calls to sgl_apply_viewport() and sgl_apply_scissor_rect() - the primitive type hasn't changed - the primitive type isn't a 'strip type' (no line or triangle strip) - the pipeline state object hasn't changed - none of the matrices has changed - none of the texture state has changed Merging a draw command simply means that the number of vertices to render in the previous draw command will be incremented by the number of vertices in the new draw command. LICENSE ======= zlib/libpng license Copyright (c) 2018 Andre Weissflog This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software. Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions: 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. 3. This notice may not be removed or altered from any source distribution. */ #define SOKOL_GL_INCLUDED (1) #include <stdint.h> #include <stdbool.h> #if !defined(SOKOL_GFX_INCLUDED) #error "Please include sokol_gfx.h before sokol_gl.h" #endif #ifndef SOKOL_API_DECL #if defined(_WIN32) && defined(SOKOL_DLL) && defined(SOKOL_IMPL) #define SOKOL_API_DECL __declspec(dllexport) #elif defined(_WIN32) && defined(SOKOL_DLL) #define SOKOL_API_DECL __declspec(dllimport) #else #define SOKOL_API_DECL extern #endif #endif #ifdef __cplusplus extern "C" { #endif /* sokol_gl pipeline handle (created with sgl_make_pipeline()) */ typedef struct sgl_pipeline { uint32_t id; } sgl_pipeline; /* sgl_error_t Errors are reset each frame after calling sgl_draw(), get the last error code with sgl_error() */ typedef enum sgl_error_t { SGL_NO_ERROR = 0, SGL_ERROR_VERTICES_FULL, SGL_ERROR_UNIFORMS_FULL, SGL_ERROR_COMMANDS_FULL, SGL_ERROR_STACK_OVERFLOW, SGL_ERROR_STACK_UNDERFLOW, } sgl_error_t; typedef struct sgl_desc_t { int max_vertices; /* size for vertex buffer */ int max_commands; /* size of uniform- and command-buffers */ int pipeline_pool_size; /* size of the internal pipeline pool, default is 64 */ sg_pixel_format color_format; sg_pixel_format depth_format; int sample_count; sg_face_winding face_winding; /* default front face winding is CCW */ } sgl_desc_t; /* setup/shutdown/misc */ SOKOL_API_DECL void sgl_setup(const sgl_desc_t* desc); SOKOL_API_DECL void sgl_shutdown(void); SOKOL_API_DECL sgl_error_t sgl_error(void); SOKOL_API_DECL void sgl_defaults(void); SOKOL_API_DECL float sgl_rad(float deg); SOKOL_API_DECL float sgl_deg(float rad); /* create and destroy pipeline objects */ SOKOL_API_DECL sgl_pipeline sgl_make_pipeline(const sg_pipeline_desc* desc); SOKOL_API_DECL void sgl_destroy_pipeline(sgl_pipeline pip); /* render state functions */ SOKOL_API_DECL void sgl_viewport(int x, int y, int w, int h, bool origin_top_left); SOKOL_API_DECL void sgl_scissor_rect(int x, int y, int w, int h, bool origin_top_left); SOKOL_API_DECL void sgl_enable_texture(void); SOKOL_API_DECL void sgl_disable_texture(void); SOKOL_API_DECL void sgl_texture(sg_image img); /* pipeline stack functions */ SOKOL_API_DECL void sgl_default_pipeline(void); SOKOL_API_DECL void sgl_load_pipeline(sgl_pipeline pip); SOKOL_API_DECL void sgl_push_pipeline(void); SOKOL_API_DECL void sgl_pop_pipeline(void); /* matrix stack functions */ SOKOL_API_DECL void sgl_matrix_mode_modelview(void); SOKOL_API_DECL void sgl_matrix_mode_projection(void); SOKOL_API_DECL void sgl_matrix_mode_texture(void); SOKOL_API_DECL void sgl_load_identity(void); SOKOL_API_DECL void sgl_load_matrix(const float m[16]); SOKOL_API_DECL void sgl_load_transpose_matrix(const float m[16]); SOKOL_API_DECL void sgl_mult_matrix(const float m[16]); SOKOL_API_DECL void sgl_mult_transpose_matrix(const float m[16]); SOKOL_API_DECL void sgl_rotate(float angle_rad, float x, float y, float z); SOKOL_API_DECL void sgl_scale(float x, float y, float z); SOKOL_API_DECL void sgl_translate(float x, float y, float z); SOKOL_API_DECL void sgl_frustum(float l, float r, float b, float t, float n, float f); SOKOL_API_DECL void sgl_ortho(float l, float r, float b, float t, float n, float f); SOKOL_API_DECL void sgl_perspective(float fov_y, float aspect, float z_near, float z_far); SOKOL_API_DECL void sgl_lookat(float eye_x, float eye_y, float eye_z, float center_x, float center_y, float center_z, float up_x, float up_y, float up_z); SOKOL_API_DECL void sgl_push_matrix(void); SOKOL_API_DECL void sgl_pop_matrix(void); /* these functions only set the internal 'current texcoord / color' (valid inside or outside begin/end) */ SOKOL_API_DECL void sgl_t2f(float u, float v); SOKOL_API_DECL void sgl_c3f(float r, float g, float b); SOKOL_API_DECL void sgl_c4f(float r, float g, float b, float a); SOKOL_API_DECL void sgl_c3b(uint8_t r, uint8_t g, uint8_t b); SOKOL_API_DECL void sgl_c4b(uint8_t r, uint8_t g, uint8_t b, uint8_t a); SOKOL_API_DECL void sgl_c1i(uint32_t rgba); /* define primitives, each begin/end is one draw command */ SOKOL_API_DECL void sgl_begin_points(void); SOKOL_API_DECL void sgl_begin_lines(void); SOKOL_API_DECL void sgl_begin_line_strip(void); SOKOL_API_DECL void sgl_begin_triangles(void); SOKOL_API_DECL void sgl_begin_triangle_strip(void); SOKOL_API_DECL void sgl_begin_quads(void); SOKOL_API_DECL void sgl_v2f(float x, float y); SOKOL_API_DECL void sgl_v3f(float x, float y, float z); SOKOL_API_DECL void sgl_v2f_t2f(float x, float y, float u, float v); SOKOL_API_DECL void sgl_v3f_t2f(float x, float y, float z, float u, float v); SOKOL_API_DECL void sgl_v2f_c3f(float x, float y, float r, float g, float b); SOKOL_API_DECL void sgl_v2f_c3b(float x, float y, uint8_t r, uint8_t g, uint8_t b); SOKOL_API_DECL void sgl_v2f_c4f(float x, float y, float r, float g, float b, float a); SOKOL_API_DECL void sgl_v2f_c4b(float x, float y, uint8_t r, uint8_t g, uint8_t b, uint8_t a); SOKOL_API_DECL void sgl_v2f_c1i(float x, float y, uint32_t rgba); SOKOL_API_DECL void sgl_v3f_c3f(float x, float y, float z, float r, float g, float b); SOKOL_API_DECL void sgl_v3f_c3b(float x, float y, float z, uint8_t r, uint8_t g, uint8_t b); SOKOL_API_DECL void sgl_v3f_c4f(float x, float y, float z, float r, float g, float b, float a); SOKOL_API_DECL void sgl_v3f_c4b(float x, float y, float z, uint8_t r, uint8_t g, uint8_t b, uint8_t a); SOKOL_API_DECL void sgl_v3f_c1i(float x, float y, float z, uint32_t rgba); SOKOL_API_DECL void sgl_v2f_t2f_c3f(float x, float y, float u, float v, float r, float g, float b); SOKOL_API_DECL void sgl_v2f_t2f_c3b(float x, float y, float u, float v, uint8_t r, uint8_t g, uint8_t b); SOKOL_API_DECL void sgl_v2f_t2f_c4f(float x, float y, float u, float v, float r, float g, float b, float a); SOKOL_API_DECL void sgl_v2f_t2f_c4b(float x, float y, float u, float v, uint8_t r, uint8_t g, uint8_t b, uint8_t a); SOKOL_API_DECL void sgl_v2f_t2f_c1i(float x, float y, float u, float v, uint32_t rgba); SOKOL_API_DECL void sgl_v3f_t2f_c3f(float x, float y, float z, float u, float v, float r, float g, float b); SOKOL_API_DECL void sgl_v3f_t2f_c3b(float x, float y, float z, float u, float v, uint8_t r, uint8_t g, uint8_t b); SOKOL_API_DECL void sgl_v3f_t2f_c4f(float x, float y, float z, float u, float v, float r, float g, float b, float a); SOKOL_API_DECL void sgl_v3f_t2f_c4b(float x, float y, float z, float u, float v, uint8_t r, uint8_t g, uint8_t b, uint8_t a); SOKOL_API_DECL void sgl_v3f_t2f_c1i(float x, float y, float z, float u, float v, uint32_t rgba); SOKOL_API_DECL void sgl_end(void); /* render everything */ SOKOL_API_DECL void sgl_draw(void); #ifdef __cplusplus } /* extern "C" */ #endif #endif /* SOKOL_GL_INCLUDED */ /*-- IMPLEMENTATION ----------------------------------------------------------*/ #ifdef SOKOL_GL_IMPL #define SOKOL_GL_IMPL_INCLUDED (1) #include <stddef.h> /* offsetof */ #include <string.h> /* memset */ #include <math.h> /* M_PI, sqrtf, sinf, cosf */ #ifndef M_PI #define M_PI 3.14159265358979323846264338327 #endif #ifndef SOKOL_API_IMPL #define SOKOL_API_IMPL #endif #ifndef SOKOL_DEBUG #ifndef NDEBUG #define SOKOL_DEBUG (1) #endif #endif #ifndef SOKOL_ASSERT #include <assert.h> #define SOKOL_ASSERT(c) assert(c) #endif #ifndef SOKOL_MALLOC #include <stdlib.h> #define SOKOL_MALLOC(s) malloc(s) #define SOKOL_FREE(p) free(p) #endif #ifndef SOKOL_LOG #ifdef SOKOL_DEBUG #include <stdio.h> #define SOKOL_LOG(s) { SOKOL_ASSERT(s); puts(s); } #else #define SOKOL_LOG(s) #endif #endif #ifndef SOKOL_UNREACHABLE #define SOKOL_UNREACHABLE SOKOL_ASSERT(false) #endif #define _sgl_def(val, def) (((val) == 0) ? (def) : (val)) #define _SGL_INIT_COOKIE (0xABCDABCD) #if defined(SOKOL_GLCORE33) static const char* _sgl_vs_src = "#version 330\n" "uniform mat4 mvp;\n" "uniform mat4 tm;\n" "in vec4 position;\n" "in vec2 texcoord0;\n" "in vec4 color0;\n" "out vec4 uv;\n" "out vec4 color;\n" "void main() {\n" " gl_Position = mvp * position;\n" " uv = tm * vec4(texcoord0, 0.0, 1.0);\n" " color = color0;\n" "}\n"; static const char* _sgl_fs_src = "#version 330\n" "uniform sampler2D tex;\n" "in vec4 uv;\n" "in vec4 color;\n" "out vec4 frag_color;\n" "void main() {\n" " frag_color = texture(tex, uv.xy) * color;\n" "}\n"; #elif defined(SOKOL_GLES2) || defined(SOKOL_GLES3) static const char* _sgl_vs_src = "uniform mat4 mvp;\n" "uniform mat4 tm;\n" "attribute vec4 position;\n" "attribute vec2 texcoord0;\n" "attribute vec4 color0;\n" "varying vec4 uv;\n" "varying vec4 color;\n" "void main() {\n" " gl_Position = mvp * position;\n" " uv = tm * vec4(texcoord0, 0.0, 1.0);\n" " color = color0;\n" "}\n"; static const char* _sgl_fs_src = "precision mediump float;\n" "uniform sampler2D tex;\n" "varying vec4 uv;\n" "varying vec4 color;\n" "void main() {\n" " gl_FragColor = texture2D(tex, uv.xy) * color;\n" "}\n"; #elif defined(SOKOL_METAL) static const char* _sgl_vs_src = "#include <metal_stdlib>\n" "using namespace metal;\n" "struct params_t {\n" " float4x4 mvp;\n" " float4x4 tm;\n" "};\n" "struct vs_in {\n" " float4 pos [[attribute(0)]];\n" " float2 uv [[attribute(1)]];\n" " float4 color [[attribute(2)]];\n" "};\n" "struct vs_out {\n" " float4 pos [[position]];\n" " float4 uv;\n" " float4 color;\n" "};\n" "vertex vs_out _main(vs_in in [[stage_in]], constant params_t& params [[buffer(0)]]) {\n" " vs_out out;\n" " out.pos = params.mvp * in.pos;\n" " out.uv = params.tm * float4(in.uv, 0.0, 1.0);\n" " out.color = in.color;\n" " return out;\n" "}\n"; static const char* _sgl_fs_src = "#include <metal_stdlib>\n" "using namespace metal;\n" "struct fs_in {\n" " float4 uv;\n" " float4 color;\n" "};\n" "fragment float4 _main(fs_in in [[stage_in]], texture2d<float> tex [[texture(0)]], sampler smp [[sampler(0)]]) {\n" " return tex.sample(smp, in.uv.xy) * in.color;\n" "}\n"; #elif defined(SOKOL_D3D11) /* Shader blobs for D3D11, compiled with: fxc.exe /T vs_5_0 /Fh vs.h /Gec /O3 vs.hlsl fxc.exe /T ps_5_0 /Fh fs.h /Gec /O3 fs.hlsl Vertex shader source: cbuffer params: register(b0) { float4x4 mvp; float4x4 tm; }; struct vs_in { float4 pos: POSITION; float2 uv: TEXCOORD0; float4 color: COLOR0; }; struct vs_out { float4 uv: TEXCOORD0; float4 color: COLOR0; float4 pos: SV_Position; }; vs_out main(vs_in inp) { vs_out outp; outp.pos = mul(mvp, inp.pos); outp.uv = mul(tm, float4(inp.uv, 0.0, 1.0)); outp.color = inp.color; return outp; }; Pixel shader source: Texture2D<float4> tex: register(t0); sampler smp: register(s0); float4 main(float4 uv: TEXCOORD0, float4 color: COLOR0): SV_Target0 { return tex.Sample(smp, uv.xy) * color; } */ static const uint8_t _sgl_vs_bin[] = { 68, 88, 66, 67, 239, 161, 1, 229, 179, 68, 206, 40, 34, 15, 57, 169, 103, 117, 134, 191, 1, 0, 0, 0, 120, 4, 0, 0, 5, 0, 0, 0, 52, 0, 0, 0, 104, 1, 0, 0, 216, 1, 0, 0, 76, 2, 0, 0, 220, 3, 0, 0, 82, 68, 69, 70, 44, 1, 0, 0, 1, 0, 0, 0, 100, 0, 0, 0, 1, 0, 0, 0, 60, 0, 0, 0, 0, 5, 254, 255, 0, 145, 0, 0, 3, 1, 0, 0, 82, 68, 49, 49, 60, 0, 0, 0, 24, 0, 0, 0, 32, 0, 0, 0, 40, 0, 0, 0, 36, 0, 0, 0, 12, 0, 0, 0, 0, 0, 0, 0, 92, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 112, 97, 114, 97, 109, 115, 0, 171, 92, 0, 0, 0, 2, 0, 0, 0, 124, 0, 0, 0, 128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 204, 0, 0, 0, 0, 0, 0, 0, 64, 0, 0, 0, 2, 0, 0, 0, 220, 0, 0, 0, 0, 0, 0, 0, 255, 255, 255, 255, 0, 0, 0, 0, 255, 255, 255, 255, 0, 0, 0, 0, 0, 1, 0, 0, 64, 0, 0, 0, 64, 0, 0, 0, 2, 0, 0, 0, 220, 0, 0, 0, 0, 0, 0, 0, 255, 255, 255, 255, 0, 0, 0, 0, 255, 255, 255, 255, 0, 0, 0, 0, 109, 118, 112, 0, 102, 108, 111, 97, 116, 52, 120, 52, 0, 171, 171, 171, 3, 0, 3, 0, 4, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 208, 0, 0, 0, 116, 109, 0, 77, 105, 99, 114, 111, 115, 111, 102, 116, 32, 40, 82, 41, 32, 72, 76, 83, 76, 32, 83, 104, 97, 100, 101, 114, 32, 67, 111, 109, 112, 105, 108, 101, 114, 32, 49, 48, 46, 49, 0, 171, 73, 83, 71, 78, 104, 0, 0, 0, 3, 0, 0, 0, 8, 0, 0, 0, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 0, 0, 0, 0, 15, 15, 0, 0, 89, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 1, 0, 0, 0, 3, 3, 0, 0, 98, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 2, 0, 0, 0, 15, 15, 0, 0, 80, 79, 83, 73, 84, 73, 79, 78, 0, 84, 69, 88, 67, 79, 79, 82, 68, 0, 67, 79, 76, 79, 82, 0, 79, 83, 71, 78, 108, 0, 0, 0, 3, 0, 0, 0, 8, 0, 0, 0, 80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 0, 0, 0, 0, 15, 0, 0, 0, 89, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 1, 0, 0, 0, 15, 0, 0, 0, 95, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 3, 0, 0, 0, 2, 0, 0, 0, 15, 0, 0, 0, 84, 69, 88, 67, 79, 79, 82, 68, 0, 67, 79, 76, 79, 82, 0, 83, 86, 95, 80, 111, 115, 105, 116, 105, 111, 110, 0, 171, 83, 72, 69, 88, 136, 1, 0, 0, 80, 0, 1, 0, 98, 0, 0, 0, 106, 8, 0, 1, 89, 0, 0, 4, 70, 142, 32, 0, 0, 0, 0, 0, 8, 0, 0, 0, 95, 0, 0, 3, 242, 16, 16, 0, 0, 0, 0, 0, 95, 0, 0, 3, 50, 16, 16, 0, 1, 0, 0, 0, 95, 0, 0, 3, 242, 16, 16, 0, 2, 0, 0, 0, 101, 0, 0, 3, 242, 32, 16, 0, 0, 0, 0, 0, 101, 0, 0, 3, 242, 32, 16, 0, 1, 0, 0, 0, 103, 0, 0, 4, 242, 32, 16, 0, 2, 0, 0, 0, 1, 0, 0, 0, 104, 0, 0, 2, 1, 0, 0, 0, 56, 0, 0, 8, 242, 0, 16, 0, 0, 0, 0, 0, 86, 21, 16, 0, 1, 0, 0, 0, 70, 142, 32, 0, 0, 0, 0, 0, 5, 0, 0, 0, 50, 0, 0, 10, 242, 0, 16, 0, 0, 0, 0, 0, 70, 142, 32, 0, 0, 0, 0, 0, 4, 0, 0, 0, 6, 16, 16, 0, 1, 0, 0, 0, 70, 14, 16, 0, 0, 0, 0, 0, 0, 0, 0, 8, 242, 32, 16, 0, 0, 0, 0, 0, 70, 14, 16, 0, 0, 0, 0, 0, 70, 142, 32, 0, 0, 0, 0, 0, 7, 0, 0, 0, 54, 0, 0, 5, 242, 32, 16, 0, 1, 0, 0, 0, 70, 30, 16, 0, 2, 0, 0, 0, 56, 0, 0, 8, 242, 0, 16, 0, 0, 0, 0, 0, 86, 21, 16, 0, 0, 0, 0, 0, 70, 142, 32, 0, 0, 0, 0, 0, 1, 0, 0, 0, 50, 0, 0, 10, 242, 0, 16, 0, 0, 0, 0, 0, 70, 142, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 6, 16, 16, 0, 0, 0, 0, 0, 70, 14, 16, 0, 0, 0, 0, 0, 50, 0, 0, 10, 242, 0, 16, 0, 0, 0, 0, 0, 70, 142, 32, 0, 0, 0, 0, 0, 2, 0, 0, 0, 166, 26, 16, 0, 0, 0, 0, 0, 70, 14, 16, 0, 0, 0, 0, 0, 50, 0, 0, 10, 242, 32, 16, 0, 2, 0, 0, 0, 70, 142, 32, 0, 0, 0, 0, 0, 3, 0, 0, 0, 246, 31, 16, 0, 0, 0, 0, 0, 70, 14, 16, 0, 0, 0, 0, 0, 62, 0, 0, 1, 83, 84, 65, 84, 148, 0, 0, 0, 9, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 6, 0, 0, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; static uint8_t _sgl_fs_bin[] = { 68, 88, 66, 67, 145, 182, 34, 101, 114, 183, 46, 3, 176, 243, 147, 199, 109, 42, 196, 114, 1, 0, 0, 0, 176, 2, 0, 0, 5, 0, 0, 0, 52, 0, 0, 0, 232, 0, 0, 0, 56, 1, 0, 0, 108, 1, 0, 0, 20, 2, 0, 0, 82, 68, 69, 70, 172, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 0, 0, 0, 60, 0, 0, 0, 0, 5, 255, 255, 0, 145, 0, 0, 132, 0, 0, 0, 82, 68, 49, 49, 60, 0, 0, 0, 24, 0, 0, 0, 32, 0, 0, 0, 40, 0, 0, 0, 36, 0, 0, 0, 12, 0, 0, 0, 0, 0, 0, 0, 124, 0, 0, 0, 3, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 128, 0, 0, 0, 2, 0, 0, 0, 5, 0, 0, 0, 4, 0, 0, 0, 255, 255, 255, 255, 0, 0, 0, 0, 1, 0, 0, 0, 13, 0, 0, 0, 115, 109, 112, 0, 116, 101, 120, 0, 77, 105, 99, 114, 111, 115, 111, 102, 116, 32, 40, 82, 41, 32, 72, 76, 83, 76, 32, 83, 104, 97, 100, 101, 114, 32, 67, 111, 109, 112, 105, 108, 101, 114, 32, 49, 48, 46, 49, 0, 73, 83, 71, 78, 72, 0, 0, 0, 2, 0, 0, 0, 8, 0, 0, 0, 56, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 0, 0, 0, 0, 15, 3, 0, 0, 65, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 1, 0, 0, 0, 15, 15, 0, 0, 84, 69, 88, 67, 79, 79, 82, 68, 0, 67, 79, 76, 79, 82, 0, 171, 79, 83, 71, 78, 44, 0, 0, 0, 1, 0, 0, 0, 8, 0, 0, 0, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 0, 0, 0, 0, 15, 0, 0, 0, 83, 86, 95, 84, 97, 114, 103, 101, 116, 0, 171, 171, 83, 72, 69, 88, 160, 0, 0, 0, 80, 0, 0, 0, 40, 0, 0, 0, 106, 8, 0, 1, 90, 0, 0, 3, 0, 96, 16, 0, 0, 0, 0, 0, 88, 24, 0, 4, 0, 112, 16, 0, 0, 0, 0, 0, 85, 85, 0, 0, 98, 16, 0, 3, 50, 16, 16, 0, 0, 0, 0, 0, 98, 16, 0, 3, 242, 16, 16, 0, 1, 0, 0, 0, 101, 0, 0, 3, 242, 32, 16, 0, 0, 0, 0, 0, 104, 0, 0, 2, 1, 0, 0, 0, 69, 0, 0, 139, 194, 0, 0, 128, 67, 85, 21, 0, 242, 0, 16, 0, 0, 0, 0, 0, 70, 16, 16, 0, 0, 0, 0, 0, 70, 126, 16, 0, 0, 0, 0, 0, 0, 96, 16, 0, 0, 0, 0, 0, 56, 0, 0, 7, 242, 32, 16, 0, 0, 0, 0, 0, 70, 14, 16, 0, 0, 0, 0, 0, 70, 30, 16, 0, 1, 0, 0, 0, 62, 0, 0, 1, 83, 84, 65, 84, 148, 0, 0, 0, 3, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; #elif defined(SOKOL_DUMMY_BACKEND) static const char* _sgl_vs_src = ""; static const char* _sgl_fs_src = ""; #endif typedef enum { SGL_PRIMITIVETYPE_POINTS = 0, SGL_PRIMITIVETYPE_LINES, SGL_PRIMITIVETYPE_LINE_STRIP, SGL_PRIMITIVETYPE_TRIANGLES, SGL_PRIMITIVETYPE_TRIANGLE_STRIP, SGL_PRIMITIVETYPE_QUADS, SGL_NUM_PRIMITIVE_TYPES, } _sgl_primitive_type_t; typedef struct { uint32_t id; sg_resource_state state; } _sgl_slot_t; typedef struct { int size; int queue_top; uint32_t* gen_ctrs; int* free_queue; } _sgl_pool_t; typedef struct { _sgl_slot_t slot; sg_pipeline pip[SGL_NUM_PRIMITIVE_TYPES]; } _sgl_pipeline_t; typedef struct { _sgl_pool_t pool; _sgl_pipeline_t* pips; } _sgl_pipeline_pool_t; typedef enum { SGL_MATRIXMODE_MODELVIEW, SGL_MATRIXMODE_PROJECTION, SGL_MATRIXMODE_TEXTURE, SGL_NUM_MATRIXMODES } _sgl_matrix_mode_t; typedef struct { float pos[3]; float uv[2]; uint32_t rgba; } _sgl_vertex_t; typedef struct { float v[4][4]; } _sgl_matrix_t; typedef struct { _sgl_matrix_t mvp; /* model-view-projection matrix */ _sgl_matrix_t tm; /* texture matrix */ } _sgl_uniform_t; typedef enum { SGL_COMMAND_DRAW, SGL_COMMAND_VIEWPORT, SGL_COMMAND_SCISSOR_RECT, } _sgl_command_type_t; typedef struct { sg_pipeline pip; sg_image img; int base_vertex; int num_vertices; int uniform_index; } _sgl_draw_args_t; typedef struct { int x, y, w, h; bool origin_top_left; } _sgl_viewport_args_t; typedef struct { int x, y, w, h; bool origin_top_left; } _sgl_scissor_rect_args_t; typedef union { _sgl_draw_args_t draw; _sgl_viewport_args_t viewport; _sgl_scissor_rect_args_t scissor_rect; } _sgl_args_t; typedef struct { _sgl_command_type_t cmd; _sgl_args_t args; } _sgl_command_t; #define _SGL_INVALID_SLOT_INDEX (0) #define _SGL_MAX_STACK_DEPTH (64) #define _SGL_DEFAULT_PIPELINE_POOL_SIZE (64) #define _SGL_DEFAULT_MAX_VERTICES (1<<16) #define _SGL_DEFAULT_MAX_COMMANDS (1<<14) #define _SGL_SLOT_SHIFT (16) #define _SGL_MAX_POOL_SIZE (1<<_SGL_SLOT_SHIFT) #define _SGL_SLOT_MASK (_SGL_MAX_POOL_SIZE-1) typedef struct { uint32_t init_cookie; sgl_desc_t desc; int num_vertices; int num_uniforms; int num_commands; int cur_vertex; int cur_uniform; int cur_command; _sgl_vertex_t* vertices; _sgl_uniform_t* uniforms; _sgl_command_t* commands; /* state tracking */ int base_vertex; int vtx_count; /* number of times vtx function has been called, used for non-triangle primitives */ sgl_error_t error; bool in_begin; float u, v; uint32_t rgba; _sgl_primitive_type_t cur_prim_type; sg_image cur_img; bool texturing_enabled; bool matrix_dirty; /* reset in sgl_end(), set in any of the matrix stack functions */ /* sokol-gfx resources */ sg_buffer vbuf; sg_image def_img; /* a default white texture */ sg_shader shd; sg_bindings bind; sgl_pipeline def_pip; _sgl_pipeline_pool_t pip_pool; /* pipeline stack */ int pip_tos; sgl_pipeline pip_stack[_SGL_MAX_STACK_DEPTH]; /* matrix stacks */ _sgl_matrix_mode_t cur_matrix_mode; int matrix_tos[SGL_NUM_MATRIXMODES]; _sgl_matrix_t matrix_stack[SGL_NUM_MATRIXMODES][_SGL_MAX_STACK_DEPTH]; } _sgl_t; static _sgl_t _sgl; /*== PRIVATE FUNCTIONS =======================================================*/ static void _sgl_init_pool(_sgl_pool_t* pool, int num) { SOKOL_ASSERT(pool && (num >= 1)); /* slot 0 is reserved for the 'invalid id', so bump the pool size by 1 */ pool->size = num + 1; pool->queue_top = 0; /* generation counters indexable by pool slot index, slot 0 is reserved */ size_t gen_ctrs_size = sizeof(uint32_t) * pool->size; pool->gen_ctrs = (uint32_t*) SOKOL_MALLOC(gen_ctrs_size); SOKOL_ASSERT(pool->gen_ctrs); memset(pool->gen_ctrs, 0, gen_ctrs_size); /* it's not a bug to only reserve 'num' here */ pool->free_queue = (int*) SOKOL_MALLOC(sizeof(int)*num); SOKOL_ASSERT(pool->free_queue); /* never allocate the zero-th pool item since the invalid id is 0 */ for (int i = pool->size-1; i >= 1; i--) { pool->free_queue[pool->queue_top++] = i; } } static void _sgl_discard_pool(_sgl_pool_t* pool) { SOKOL_ASSERT(pool); SOKOL_ASSERT(pool->free_queue); SOKOL_FREE(pool->free_queue); pool->free_queue = 0; SOKOL_ASSERT(pool->gen_ctrs); SOKOL_FREE(pool->gen_ctrs); pool->gen_ctrs = 0; pool->size = 0; pool->queue_top = 0; } static int _sgl_pool_alloc_index(_sgl_pool_t* pool) { SOKOL_ASSERT(pool); SOKOL_ASSERT(pool->free_queue); if (pool->queue_top > 0) { int slot_index = pool->free_queue[--pool->queue_top]; SOKOL_ASSERT((slot_index > 0) && (slot_index < pool->size)); return slot_index; } else { /* pool exhausted */ return _SGL_INVALID_SLOT_INDEX; } } static void _sgl_pool_free_index(_sgl_pool_t* pool, int slot_index) { SOKOL_ASSERT((slot_index > _SGL_INVALID_SLOT_INDEX) && (slot_index < pool->size)); SOKOL_ASSERT(pool); SOKOL_ASSERT(pool->free_queue); SOKOL_ASSERT(pool->queue_top < pool->size); #ifdef SOKOL_DEBUG /* debug check against double-free */ for (int i = 0; i < pool->queue_top; i++) { SOKOL_ASSERT(pool->free_queue[i] != slot_index); } #endif pool->free_queue[pool->queue_top++] = slot_index; SOKOL_ASSERT(pool->queue_top <= (pool->size-1)); } static void _sgl_reset_pipeline(_sgl_pipeline_t* pip) { SOKOL_ASSERT(pip); memset(pip, 0, sizeof(_sgl_pipeline_t)); } static void _sgl_setup_pipeline_pool(const sgl_desc_t* desc) { SOKOL_ASSERT(desc); /* note: the pools here will have an additional item, since slot 0 is reserved */ SOKOL_ASSERT((desc->pipeline_pool_size > 0) && (desc->pipeline_pool_size < _SGL_MAX_POOL_SIZE)); _sgl_init_pool(&_sgl.pip_pool.pool, desc->pipeline_pool_size); size_t pool_byte_size = sizeof(_sgl_pipeline_t) * _sgl.pip_pool.pool.size; _sgl.pip_pool.pips = (_sgl_pipeline_t*) SOKOL_MALLOC(pool_byte_size); SOKOL_ASSERT(_sgl.pip_pool.pips); memset(_sgl.pip_pool.pips, 0, pool_byte_size); } static void _sgl_discard_pipeline_pool(void) { SOKOL_FREE(_sgl.pip_pool.pips); _sgl.pip_pool.pips = 0; _sgl_discard_pool(&_sgl.pip_pool.pool); } /* allocate the slot at slot_index: - bump the slot's generation counter - create a resource id from the generation counter and slot index - set the slot's id to this id - set the slot's state to ALLOC - return the resource id */ static uint32_t _sgl_slot_alloc(_sgl_pool_t* pool, _sgl_slot_t* slot, int slot_index) { /* FIXME: add handling for an overflowing generation counter, for now, just overflow (another option is to disable the slot) */ SOKOL_ASSERT(pool && pool->gen_ctrs); SOKOL_ASSERT((slot_index > _SGL_INVALID_SLOT_INDEX) && (slot_index < pool->size)); SOKOL_ASSERT((slot->state == SG_RESOURCESTATE_INITIAL) && (slot->id == SG_INVALID_ID)); uint32_t ctr = ++pool->gen_ctrs[slot_index]; slot->id = (ctr<<_SGL_SLOT_SHIFT)|(slot_index & _SGL_SLOT_MASK); slot->state = SG_RESOURCESTATE_ALLOC; return slot->id; } /* extract slot index from id */ static int _sgl_slot_index(uint32_t id) { int slot_index = (int) (id & _SGL_SLOT_MASK); SOKOL_ASSERT(_SGL_INVALID_SLOT_INDEX != slot_index); return slot_index; } /* get pipeline pointer without id-check */ static _sgl_pipeline_t* _sgl_pipeline_at(uint32_t pip_id) { SOKOL_ASSERT(SG_INVALID_ID != pip_id); int slot_index = _sgl_slot_index(pip_id); SOKOL_ASSERT((slot_index > _SGL_INVALID_SLOT_INDEX) && (slot_index < _sgl.pip_pool.pool.size)); return &_sgl.pip_pool.pips[slot_index]; } /* get pipeline pointer with id-check, returns 0 if no match */ static _sgl_pipeline_t* _sgl_lookup_pipeline(uint32_t pip_id) { if (SG_INVALID_ID != pip_id) { _sgl_pipeline_t* pip = _sgl_pipeline_at(pip_id); if (pip->slot.id == pip_id) { return pip; } } return 0; } static sgl_pipeline _sgl_alloc_pipeline(void) { sgl_pipeline res; int slot_index = _sgl_pool_alloc_index(&_sgl.pip_pool.pool); if (_SGL_INVALID_SLOT_INDEX != slot_index) { res.id =_sgl_slot_alloc(&_sgl.pip_pool.pool, &_sgl.pip_pool.pips[slot_index].slot, slot_index); } else { /* pool is exhausted */ res.id = SG_INVALID_ID; } return res; } static void _sgl_init_pipeline(sgl_pipeline pip_id, const sg_pipeline_desc* in_desc) { SOKOL_ASSERT((pip_id.id != SG_INVALID_ID) && in_desc); /* create a new desc with 'patched' shader and pixel format state */ sg_pipeline_desc desc = *in_desc; desc.layout.buffers[0].stride = sizeof(_sgl_vertex_t); { sg_vertex_attr_desc* pos = &desc.layout.attrs[0]; pos->offset = offsetof(_sgl_vertex_t, pos); pos->format = SG_VERTEXFORMAT_FLOAT3; } { sg_vertex_attr_desc* uv = &desc.layout.attrs[1]; uv->offset = offsetof(_sgl_vertex_t, uv); uv->format = SG_VERTEXFORMAT_FLOAT2; } { sg_vertex_attr_desc* rgba = &desc.layout.attrs[2]; rgba->offset = offsetof(_sgl_vertex_t, rgba); rgba->format = SG_VERTEXFORMAT_UBYTE4N; } if (in_desc->shader.id == SG_INVALID_ID) { desc.shader = _sgl.shd; } desc.index_type = SG_INDEXTYPE_NONE; desc.blend.color_format = _sgl.desc.color_format; desc.blend.depth_format = _sgl.desc.depth_format; desc.rasterizer.sample_count = _sgl.desc.sample_count; if (desc.rasterizer.face_winding == _SG_FACEWINDING_DEFAULT) { desc.rasterizer.face_winding = _sgl.desc.face_winding; } if (desc.blend.color_write_mask == _SG_COLORMASK_DEFAULT) { desc.blend.color_write_mask = SG_COLORMASK_RGB; } _sgl_pipeline_t* pip = _sgl_lookup_pipeline(pip_id.id); SOKOL_ASSERT(pip && (pip->slot.state == SG_RESOURCESTATE_ALLOC)); pip->slot.state = SG_RESOURCESTATE_VALID; for (int i = 0; i < SGL_NUM_PRIMITIVE_TYPES; i++) { switch (i) { case SGL_PRIMITIVETYPE_POINTS: desc.primitive_type = SG_PRIMITIVETYPE_POINTS; break; case SGL_PRIMITIVETYPE_LINES: desc.primitive_type = SG_PRIMITIVETYPE_LINES; break; case SGL_PRIMITIVETYPE_LINE_STRIP: desc.primitive_type = SG_PRIMITIVETYPE_LINE_STRIP; break; case SGL_PRIMITIVETYPE_TRIANGLES: desc.primitive_type = SG_PRIMITIVETYPE_TRIANGLES; break; case SGL_PRIMITIVETYPE_TRIANGLE_STRIP: case SGL_PRIMITIVETYPE_QUADS: desc.primitive_type = SG_PRIMITIVETYPE_TRIANGLE_STRIP; break; } if (SGL_PRIMITIVETYPE_QUADS == i) { /* quads are emulated via triangles, use the same pipeline object */ pip->pip[i] = pip->pip[SGL_PRIMITIVETYPE_TRIANGLES]; } else { pip->pip[i] = sg_make_pipeline(&desc); if (pip->pip[i].id == SG_INVALID_ID) { SOKOL_LOG("sokol_gl.h: failed to create pipeline object"); pip->slot.state = SG_RESOURCESTATE_FAILED; } } } } static sgl_pipeline _sgl_make_pipeline(const sg_pipeline_desc* desc) { SOKOL_ASSERT(desc); sgl_pipeline pip_id = _sgl_alloc_pipeline(); if (pip_id.id != SG_INVALID_ID) { _sgl_init_pipeline(pip_id, desc); } else { SOKOL_LOG("sokol_gl.h: pipeline pool exhausted!"); } return pip_id; } static void _sgl_destroy_pipeline(sgl_pipeline pip_id) { _sgl_pipeline_t* pip = _sgl_lookup_pipeline(pip_id.id); if (pip) { for (int i = 0; i < SGL_NUM_PRIMITIVE_TYPES; i++) { if (i != SGL_PRIMITIVETYPE_QUADS) { sg_destroy_pipeline(pip->pip[i]); } } _sgl_reset_pipeline(pip); _sgl_pool_free_index(&_sgl.pip_pool.pool, _sgl_slot_index(pip_id.id)); } } static sg_pipeline _sgl_get_pipeline(sgl_pipeline pip_id, _sgl_primitive_type_t prim_type) { _sgl_pipeline_t* pip = _sgl_lookup_pipeline(pip_id.id); if (pip) { return pip->pip[prim_type]; } else { sg_pipeline dummy_pip; dummy_pip.id = SG_INVALID_ID; return dummy_pip; } } static inline void _sgl_begin(_sgl_primitive_type_t mode) { _sgl.in_begin = true; _sgl.base_vertex = _sgl.cur_vertex; _sgl.vtx_count = 0; _sgl.cur_prim_type = mode; } static void _sgl_rewind(void) { _sgl.base_vertex = 0; _sgl.cur_vertex = 0; _sgl.cur_uniform = 0; _sgl.cur_command = 0; _sgl.error = SGL_NO_ERROR; _sgl.matrix_dirty = true; } static inline _sgl_vertex_t* _sgl_next_vertex(void) { if (_sgl.cur_vertex < _sgl.num_vertices) { return &_sgl.vertices[_sgl.cur_vertex++]; } else { _sgl.error = SGL_ERROR_VERTICES_FULL; return 0; } } static inline _sgl_uniform_t* _sgl_next_uniform(void) { if (_sgl.cur_uniform < _sgl.num_uniforms) { return &_sgl.uniforms[_sgl.cur_uniform++]; } else { _sgl.error = SGL_ERROR_UNIFORMS_FULL; return 0; } } static inline _sgl_command_t* _sgl_prev_command(void) { if (_sgl.cur_command > 0) { return &_sgl.commands[_sgl.cur_command - 1]; } else { return 0; } } static inline _sgl_command_t* _sgl_next_command(void) { if (_sgl.cur_command < _sgl.num_commands) { return &_sgl.commands[_sgl.cur_command++]; } else { _sgl.error = SGL_ERROR_COMMANDS_FULL; return 0; } } static inline uint32_t _sgl_pack_rgbab(uint8_t r, uint8_t g, uint8_t b, uint8_t a) { return (uint32_t)(((uint32_t)a<<24)|((uint32_t)b<<16)|((uint32_t)g<<8)|r); } static inline float _sgl_clamp(float v, float lo, float hi) { if (v < lo) return lo; else if (v > hi) return hi; else return v; } static inline uint32_t _sgl_pack_rgbaf(float r, float g, float b, float a) { uint8_t r_u8 = (uint8_t) (_sgl_clamp(r, 0.0f, 1.0f) * 255.0f); uint8_t g_u8 = (uint8_t) (_sgl_clamp(g, 0.0f, 1.0f) * 255.0f); uint8_t b_u8 = (uint8_t) (_sgl_clamp(b, 0.0f, 1.0f) * 255.0f); uint8_t a_u8 = (uint8_t) (_sgl_clamp(a, 0.0f, 1.0f) * 255.0f); return _sgl_pack_rgbab(r_u8, g_u8, b_u8, a_u8); } static inline void _sgl_vtx(float x, float y, float z, float u, float v, uint32_t rgba) { SOKOL_ASSERT(_sgl.in_begin); _sgl_vertex_t* vtx; /* handle non-native primitive types */ if ((_sgl.cur_prim_type == SGL_PRIMITIVETYPE_QUADS) && ((_sgl.vtx_count & 3) == 3)) { /* for quads, before writing the last quad vertex, reuse the first and third vertex to start the second triangle in the quad */ vtx = _sgl_next_vertex(); if (vtx) { *vtx = *(vtx - 3); } vtx = _sgl_next_vertex(); if (vtx) { *vtx = *(vtx - 2); } } vtx = _sgl_next_vertex(); if (vtx) { vtx->pos[0] = x; vtx->pos[1] = y; vtx->pos[2] = z; vtx->uv[0] = u; vtx->uv[1] = v; vtx->rgba = rgba; } _sgl.vtx_count++; } static void _sgl_identity(_sgl_matrix_t* m) { for (int c = 0; c < 4; c++) { for (int r = 0; r < 4; r++) { m->v[c][r] = (r == c) ? 1.0f : 0.0f; } } } static void _sgl_transpose(_sgl_matrix_t* dst, const _sgl_matrix_t* m) { SOKOL_ASSERT(dst != m); for (int c = 0; c < 4; c++) { for (int r = 0; r < 4; r++) { dst->v[r][c] = m->v[c][r]; } } } /* _sgl_rotate, _sgl_frustum, _sgl_ortho from MESA m_matric.c */ static void _sgl_matmul4(_sgl_matrix_t* p, const _sgl_matrix_t* a, const _sgl_matrix_t* b) { for (int r = 0; r < 4; r++) { float ai0=a->v[0][r], ai1=a->v[1][r], ai2=a->v[2][r], ai3=a->v[3][r]; p->v[0][r] = ai0*b->v[0][0] + ai1*b->v[0][1] + ai2*b->v[0][2] + ai3*b->v[0][3]; p->v[1][r] = ai0*b->v[1][0] + ai1*b->v[1][1] + ai2*b->v[1][2] + ai3*b->v[1][3]; p->v[2][r] = ai0*b->v[2][0] + ai1*b->v[2][1] + ai2*b->v[2][2] + ai3*b->v[2][3]; p->v[3][r] = ai0*b->v[3][0] + ai1*b->v[3][1] + ai2*b->v[3][2] + ai3*b->v[3][3]; } } static void _sgl_mul(_sgl_matrix_t* dst, const _sgl_matrix_t* m) { _sgl_matmul4(dst, dst, m); } static void _sgl_rotate(_sgl_matrix_t* dst, float a, float x, float y, float z) { float s = sinf(a); float c = cosf(a); float mag = sqrtf(x*x + y*y + z*z); if (mag < 1.0e-4F) { return; } x /= mag; y /= mag; z /= mag; float xx = x * x; float yy = y * y; float zz = z * z; float xy = x * y; float yz = y * z; float zx = z * x; float xs = x * s; float ys = y * s; float zs = z * s; float one_c = 1.0f - c; _sgl_matrix_t m; m.v[0][0] = (one_c * xx) + c; m.v[1][0] = (one_c * xy) - zs; m.v[2][0] = (one_c * zx) + ys; m.v[3][0] = 0.0f; m.v[0][1] = (one_c * xy) + zs; m.v[1][1] = (one_c * yy) + c; m.v[2][1] = (one_c * yz) - xs; m.v[3][1] = 0.0f; m.v[0][2] = (one_c * zx) - ys; m.v[1][2] = (one_c * yz) + xs; m.v[2][2] = (one_c * zz) + c; m.v[3][2] = 0.0f; m.v[0][3] = 0.0f; m.v[1][3] = 0.0f; m.v[2][3] = 0.0f; m.v[3][3] = 1.0f; _sgl_mul(dst, &m); } static void _sgl_scale(_sgl_matrix_t* dst, float x, float y, float z) { for (int r = 0; r < 4; r++) { dst->v[0][r] *= x; dst->v[1][r] *= y; dst->v[2][r] *= z; } } static void _sgl_translate(_sgl_matrix_t* dst, float x, float y, float z) { for (int r = 0; r < 4; r++) { dst->v[3][r] = dst->v[0][r]*x + dst->v[1][r]*y + dst->v[2][r]*z + dst->v[3][r]; } } static void _sgl_frustum(_sgl_matrix_t* dst, float left, float right, float bottom, float top, float znear, float zfar) { float x = (2.0f * znear) / (right - left); float y = (2.0f * znear) / (top - bottom); float a = (right + left) / (right - left); float b = (top + bottom) / (top - bottom); float c = -(zfar + znear) / (zfar - znear); float d = -(2.0f * zfar * znear) / (zfar - znear); _sgl_matrix_t m; m.v[0][0] = x; m.v[0][1] = 0.0f; m.v[0][2] = 0.0f; m.v[0][3] = 0.0f; m.v[1][0] = 0.0f; m.v[1][1] = y; m.v[1][2] = 0.0f; m.v[1][3] = 0.0f; m.v[2][0] = a; m.v[2][1] = b; m.v[2][2] = c; m.v[2][3] = -1.0f; m.v[3][0] = 0.0f; m.v[3][1] = 0.0f; m.v[3][2] = d; m.v[3][3] = 0.0f; _sgl_mul(dst, &m); } static void _sgl_ortho(_sgl_matrix_t* dst, float left, float right, float bottom, float top, float znear, float zfar) { _sgl_matrix_t m; m.v[0][0] = 2.0f / (right - left); m.v[1][0] = 0.0f; m.v[2][0] = 0.0f; m.v[3][0] = -(right + left) / (right - left); m.v[0][1] = 0.0f; m.v[1][1] = 2.0f / (top - bottom); m.v[2][1] = 0.0f; m.v[3][1] = -(top + bottom) / (top - bottom); m.v[0][2] = 0.0f; m.v[1][2] = 0.0f; m.v[2][2] = -2.0f / (zfar - znear); m.v[3][2] = -(zfar + znear) / (zfar - znear); m.v[0][3] = 0.0f; m.v[1][3] = 0.0f; m.v[2][3] = 0.0f; m.v[3][3] = 1.0f; _sgl_mul(dst, &m); } /* _sgl_perspective, _sgl_lookat from Regal project.c */ static void _sgl_perspective(_sgl_matrix_t* dst, float fovy, float aspect, float znear, float zfar) { float sine = sinf(fovy / 2.0f); float delta_z = zfar - znear; if ((delta_z == 0.0f) || (sine == 0.0f) || (aspect == 0.0f)) { return; } float cotan = cosf(fovy / 2.0f) / sine; _sgl_matrix_t m; _sgl_identity(&m); m.v[0][0] = cotan / aspect; m.v[1][1] = cotan; m.v[2][2] = -(zfar + znear) / delta_z; m.v[2][3] = -1.0f; m.v[3][2] = -2.0f * znear * zfar / delta_z; m.v[3][3] = 0.0f; _sgl_mul(dst, &m); } static void _sgl_normalize(float v[3]) { float r = sqrtf(v[0]*v[0] + v[1]*v[1] + v[2]*v[2]); if (r == 0.0f) { return; } v[0] /= r; v[1] /= r; v[2] /= r; } static void _sgl_cross(float v1[3], float v2[3], float res[3]) { res[0] = v1[1]*v2[2] - v1[2]*v2[1]; res[1] = v1[2]*v2[0] - v1[0]*v2[2]; res[2] = v1[0]*v2[1] - v1[1]*v2[0]; } static void _sgl_lookat(_sgl_matrix_t* dst, float eye_x, float eye_y, float eye_z, float center_x, float center_y, float center_z, float up_x, float up_y, float up_z) { float fwd[3], side[3], up[3]; fwd[0] = center_x - eye_x; fwd[1] = center_y - eye_y; fwd[2] = center_z - eye_z; up[0] = up_x; up[1] = up_y; up[2] = up_z; _sgl_normalize(fwd); _sgl_cross(fwd, up, side); _sgl_normalize(side); _sgl_cross(side, fwd, up); _sgl_matrix_t m; _sgl_identity(&m); m.v[0][0] = side[0]; m.v[1][0] = side[1]; m.v[2][0] = side[2]; m.v[0][1] = up[0]; m.v[1][1] = up[1]; m.v[2][1] = up[2]; m.v[0][2] = -fwd[0]; m.v[1][2] = -fwd[1]; m.v[2][2] = -fwd[2]; _sgl_mul(dst, &m); _sgl_translate(dst, -eye_x, -eye_y, -eye_z); } /* current top-of-stack projection matrix */ static inline _sgl_matrix_t* _sgl_matrix_projection(void) { return &_sgl.matrix_stack[SGL_MATRIXMODE_PROJECTION][_sgl.matrix_tos[SGL_MATRIXMODE_PROJECTION]]; } /* get top-of-stack modelview matrix */ static inline _sgl_matrix_t* _sgl_matrix_modelview(void) { return &_sgl.matrix_stack[SGL_MATRIXMODE_MODELVIEW][_sgl.matrix_tos[SGL_MATRIXMODE_MODELVIEW]]; } /* get top-of-stack texture matrix */ static inline _sgl_matrix_t* _sgl_matrix_texture(void) { return &_sgl.matrix_stack[SGL_MATRIXMODE_TEXTURE][_sgl.matrix_tos[SGL_MATRIXMODE_TEXTURE]]; } /* get pointer to current top-of-stack of current matrix mode */ static inline _sgl_matrix_t* _sgl_matrix(void) { return &_sgl.matrix_stack[_sgl.cur_matrix_mode][_sgl.matrix_tos[_sgl.cur_matrix_mode]]; } /*== PUBLIC FUNCTIONS ========================================================*/ SOKOL_API_IMPL void sgl_setup(const sgl_desc_t* desc) { SOKOL_ASSERT(desc); memset(&_sgl, 0, sizeof(_sgl)); _sgl.init_cookie = _SGL_INIT_COOKIE; _sgl.desc = *desc; _sgl.desc.pipeline_pool_size = _sgl_def(_sgl.desc.pipeline_pool_size, _SGL_DEFAULT_PIPELINE_POOL_SIZE); _sgl.desc.max_vertices = _sgl_def(_sgl.desc.max_vertices, _SGL_DEFAULT_MAX_VERTICES); _sgl.desc.max_commands = _sgl_def(_sgl.desc.max_commands, _SGL_DEFAULT_MAX_COMMANDS); _sgl.desc.face_winding = _sgl_def(_sgl.desc.face_winding, SG_FACEWINDING_CCW); /* allocate buffers and pools */ _sgl.num_vertices = _sgl.desc.max_vertices; _sgl.num_uniforms = _sgl.desc.max_commands; _sgl.num_commands = _sgl.num_uniforms; _sgl.vertices = (_sgl_vertex_t*) SOKOL_MALLOC(_sgl.num_vertices * sizeof(_sgl_vertex_t)); SOKOL_ASSERT(_sgl.vertices); _sgl.uniforms = (_sgl_uniform_t*) SOKOL_MALLOC(_sgl.num_uniforms * sizeof(_sgl_uniform_t)); SOKOL_ASSERT(_sgl.uniforms); _sgl.commands = (_sgl_command_t*) SOKOL_MALLOC(_sgl.num_commands * sizeof(_sgl_command_t)); SOKOL_ASSERT(_sgl.commands); _sgl_setup_pipeline_pool(&_sgl.desc); /* create sokol-gfx resource objects */ sg_push_debug_group("sokol-gl"); sg_buffer_desc vbuf_desc; memset(&vbuf_desc, 0, sizeof(vbuf_desc)); vbuf_desc.size = _sgl.num_vertices * sizeof(_sgl_vertex_t); vbuf_desc.type = SG_BUFFERTYPE_VERTEXBUFFER; vbuf_desc.usage = SG_USAGE_STREAM; vbuf_desc.label = "sgl-vertex-buffer"; _sgl.vbuf = sg_make_buffer(&vbuf_desc); SOKOL_ASSERT(SG_INVALID_ID != _sgl.vbuf.id); uint32_t pixels[64]; for (int i = 0; i < 64; i++) { pixels[i] = 0xFFFFFFFF; } sg_image_desc img_desc; memset(&img_desc, 0, sizeof(img_desc)); img_desc.type = SG_IMAGETYPE_2D; img_desc.width = 8; img_desc.height = 8; img_desc.num_mipmaps = 1; img_desc.pixel_format = SG_PIXELFORMAT_RGBA8; img_desc.min_filter = SG_FILTER_NEAREST; img_desc.mag_filter = SG_FILTER_NEAREST; img_desc.content.subimage[0][0].ptr = pixels; img_desc.content.subimage[0][0].size = sizeof(pixels); img_desc.label = "sgl-default-texture"; _sgl.def_img = sg_make_image(&img_desc); SOKOL_ASSERT(SG_INVALID_ID != _sgl.def_img.id); _sgl.cur_img = _sgl.def_img; sg_shader_desc shd_desc; memset(&shd_desc, 0, sizeof(shd_desc)); shd_desc.attrs[0].name = "position"; shd_desc.attrs[1].name = "texcoord0"; shd_desc.attrs[2].name = "color0"; shd_desc.attrs[0].sem_name = "POSITION"; shd_desc.attrs[1].sem_name = "TEXCOORD"; shd_desc.attrs[2].sem_name = "COLOR"; sg_shader_uniform_block_desc* ub = &shd_desc.vs.uniform_blocks[0]; ub->size = sizeof(_sgl_uniform_t); ub->uniforms[0].name = "mvp"; ub->uniforms[0].type = SG_UNIFORMTYPE_MAT4; ub->uniforms[1].name = "tm"; ub->uniforms[1].type = SG_UNIFORMTYPE_MAT4; shd_desc.fs.images[0].name = "tex"; shd_desc.fs.images[0].type = SG_IMAGETYPE_2D; #if defined(SOKOL_D3D11) shd_desc.vs.byte_code = _sgl_vs_bin; shd_desc.vs.byte_code_size = sizeof(_sgl_vs_bin); shd_desc.fs.byte_code = _sgl_fs_bin; shd_desc.fs.byte_code_size = sizeof(_sgl_fs_bin); #else shd_desc.vs.source = _sgl_vs_src; shd_desc.fs.source = _sgl_fs_src; #endif shd_desc.label = "sgl-shader"; _sgl.shd = sg_make_shader(&shd_desc); /* create default pipeline object */ sg_pipeline_desc def_pip_desc; memset(&def_pip_desc, 0, sizeof(def_pip_desc)); def_pip_desc.depth_stencil.depth_write_enabled = true; _sgl.def_pip = _sgl_make_pipeline(&def_pip_desc); sg_pop_debug_group(); /* default state */ _sgl.rgba = 0xFFFFFFFF; for (int i = 0; i < SGL_NUM_MATRIXMODES; i++) { _sgl_identity(&_sgl.matrix_stack[i][0]); } _sgl.pip_stack[0] = _sgl.def_pip; _sgl.matrix_dirty = true; } SOKOL_API_IMPL void sgl_shutdown(void) { SOKOL_ASSERT(_sgl.init_cookie == 0xABCDABCD); SOKOL_FREE(_sgl.vertices); _sgl.vertices = 0; SOKOL_FREE(_sgl.uniforms); _sgl.uniforms = 0; SOKOL_FREE(_sgl.commands); _sgl.commands = 0; sg_destroy_buffer(_sgl.vbuf); sg_destroy_image(_sgl.def_img); sg_destroy_shader(_sgl.shd); _sgl_destroy_pipeline(_sgl.def_pip); // FIXME: need to destroy ALL valid pipeline objects in pool here _sgl_discard_pipeline_pool(); _sgl.init_cookie = 0; } SOKOL_API_IMPL sgl_error_t sgl_error(void) { return _sgl.error; } SOKOL_API_IMPL float sgl_rad(float deg) { return (deg * (float)M_PI) / 180.0f; } SOKOL_API_IMPL float sgl_deg(float rad) { return (rad * 180.0f) / (float)M_PI; } SOKOL_API_IMPL sgl_pipeline sgl_make_pipeline(const sg_pipeline_desc* desc) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); return _sgl_make_pipeline(desc); } SOKOL_API_IMPL void sgl_destroy_pipeline(sgl_pipeline pip_id) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl_destroy_pipeline(pip_id); } SOKOL_API_IMPL void sgl_load_pipeline(sgl_pipeline pip_id) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT((_sgl.pip_tos >= 0) && (_sgl.pip_tos < _SGL_MAX_STACK_DEPTH)); _sgl.pip_stack[_sgl.pip_tos] = pip_id; } SOKOL_API_IMPL void sgl_default_pipeline(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT((_sgl.pip_tos >= 0) && (_sgl.pip_tos < _SGL_MAX_STACK_DEPTH)); _sgl.pip_stack[_sgl.pip_tos] = _sgl.def_pip; } SOKOL_API_IMPL void sgl_push_pipeline(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); if (_sgl.pip_tos < (_SGL_MAX_STACK_DEPTH - 1)) { _sgl.pip_tos++; _sgl.pip_stack[_sgl.pip_tos] = _sgl.pip_stack[_sgl.pip_tos-1]; } else { _sgl.error = SGL_ERROR_STACK_OVERFLOW; } } SOKOL_API_IMPL void sgl_pop_pipeline(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); if (_sgl.pip_tos > 0) { _sgl.pip_tos--; } else { _sgl.error = SGL_ERROR_STACK_UNDERFLOW; } } SOKOL_API_IMPL void sgl_defaults(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl.u = 0.0f; _sgl.v = 0.0f; _sgl.rgba = 0xFFFFFFFF; _sgl.texturing_enabled = false; _sgl.cur_img = _sgl.def_img; sgl_default_pipeline(); _sgl_identity(_sgl_matrix_texture()); _sgl_identity(_sgl_matrix_modelview()); _sgl_identity(_sgl_matrix_projection()); _sgl.cur_matrix_mode = SGL_MATRIXMODE_MODELVIEW; _sgl.matrix_dirty = true; } SOKOL_API_IMPL void sgl_viewport(int x, int y, int w, int h, bool origin_top_left) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl_command_t* cmd = _sgl_next_command(); if (cmd) { cmd->cmd = SGL_COMMAND_VIEWPORT; cmd->args.viewport.x = x; cmd->args.viewport.y = y; cmd->args.viewport.w = w; cmd->args.viewport.h = h; cmd->args.viewport.origin_top_left = origin_top_left; } } SOKOL_API_IMPL void sgl_scissor_rect(int x, int y, int w, int h, bool origin_top_left) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl_command_t* cmd = _sgl_next_command(); if (cmd) { cmd->cmd = SGL_COMMAND_SCISSOR_RECT; cmd->args.scissor_rect.x = x; cmd->args.scissor_rect.y = y; cmd->args.scissor_rect.w = w; cmd->args.scissor_rect.h = h; cmd->args.scissor_rect.origin_top_left = origin_top_left; } } SOKOL_API_IMPL void sgl_enable_texture(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl.texturing_enabled = true; } SOKOL_API_IMPL void sgl_disable_texture(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl.texturing_enabled = false; } SOKOL_API_IMPL void sgl_texture(sg_image img) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); if (SG_INVALID_ID != img.id) { _sgl.cur_img = img; } else { _sgl.cur_img = _sgl.def_img; } } SOKOL_API_IMPL void sgl_begin_points(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl_begin(SGL_PRIMITIVETYPE_POINTS); } SOKOL_API_IMPL void sgl_begin_lines(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl_begin(SGL_PRIMITIVETYPE_LINES); } SOKOL_API_IMPL void sgl_begin_line_strip(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl_begin(SGL_PRIMITIVETYPE_LINE_STRIP); } SOKOL_API_IMPL void sgl_begin_triangles(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl_begin(SGL_PRIMITIVETYPE_TRIANGLES); } SOKOL_API_IMPL void sgl_begin_triangle_strip(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl_begin(SGL_PRIMITIVETYPE_TRIANGLE_STRIP); } SOKOL_API_IMPL void sgl_begin_quads(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(!_sgl.in_begin); _sgl_begin(SGL_PRIMITIVETYPE_QUADS); } SOKOL_API_IMPL void sgl_end(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT(_sgl.in_begin); _sgl.in_begin = false; if (_sgl.base_vertex == _sgl.cur_vertex) { return; } bool matrix_dirty = _sgl.matrix_dirty; if (matrix_dirty) { _sgl.matrix_dirty = false; _sgl_uniform_t* uni = _sgl_next_uniform(); if (uni) { _sgl_matmul4(&uni->mvp, _sgl_matrix_projection(), _sgl_matrix_modelview()); uni->tm = *_sgl_matrix_texture(); } } /* check if command can be merged with previous command */ sg_pipeline pip = _sgl_get_pipeline(_sgl.pip_stack[_sgl.pip_tos], _sgl.cur_prim_type); sg_image img = _sgl.texturing_enabled ? _sgl.cur_img : _sgl.def_img; _sgl_command_t* prev_cmd = _sgl_prev_command(); bool merge_cmd = false; if (prev_cmd) { if ((prev_cmd->cmd == SGL_COMMAND_DRAW) && (_sgl.cur_prim_type != SGL_PRIMITIVETYPE_LINE_STRIP) && (_sgl.cur_prim_type != SGL_PRIMITIVETYPE_TRIANGLE_STRIP) && !matrix_dirty && (prev_cmd->args.draw.img.id == img.id) && (prev_cmd->args.draw.pip.id == pip.id)) { merge_cmd = true; } } if (merge_cmd) { /* draw command can be merged with the previous command */ prev_cmd->args.draw.num_vertices += _sgl.cur_vertex - _sgl.base_vertex; } else { /* append a new draw command */ _sgl_command_t* cmd = _sgl_next_command(); if (cmd) { SOKOL_ASSERT(_sgl.cur_uniform > 0); cmd->cmd = SGL_COMMAND_DRAW; cmd->args.draw.img = img; cmd->args.draw.pip = _sgl_get_pipeline(_sgl.pip_stack[_sgl.pip_tos], _sgl.cur_prim_type); cmd->args.draw.base_vertex = _sgl.base_vertex; cmd->args.draw.num_vertices = _sgl.cur_vertex - _sgl.base_vertex; cmd->args.draw.uniform_index = _sgl.cur_uniform - 1; } } } SOKOL_API_IMPL void sgl_t2f(float u, float v) { _sgl.u = u; _sgl.v = v; } SOKOL_API_IMPL void sgl_c3f(float r, float g, float b) { _sgl.rgba = _sgl_pack_rgbaf(r, g, b, 1.0f); } SOKOL_API_IMPL void sgl_c4f(float r, float g, float b, float a) { _sgl.rgba = _sgl_pack_rgbaf(r, g, b, a); } SOKOL_API_IMPL void sgl_c3b(uint8_t r, uint8_t g, uint8_t b) { _sgl.rgba = _sgl_pack_rgbab(r, g, b, 255); } SOKOL_API_IMPL void sgl_c4b(uint8_t r, uint8_t g, uint8_t b, uint8_t a) { _sgl.rgba = _sgl_pack_rgbab(r, g, b, a); } SOKOL_API_IMPL void sgl_c1i(uint32_t rgba) { _sgl.rgba = rgba; } SOKOL_API_IMPL void sgl_v2f(float x, float y) { _sgl_vtx(x, y, 0.0f, _sgl.u, _sgl.v, _sgl.rgba); } SOKOL_API_IMPL void sgl_v3f(float x, float y, float z) { _sgl_vtx(x, y, z, _sgl.u, _sgl.v, _sgl.rgba); } SOKOL_API_IMPL void sgl_v2f_t2f(float x, float y, float u, float v) { _sgl_vtx(x, y, 0.0f, u, v, _sgl.rgba); } SOKOL_API_IMPL void sgl_v3f_t2f(float x, float y, float z, float u, float v) { _sgl_vtx(x, y, z, u, v, _sgl.rgba); } SOKOL_API_IMPL void sgl_v2f_c3f(float x, float y, float r, float g, float b) { _sgl_vtx(x, y, 0.0f, _sgl.u, _sgl.v, _sgl_pack_rgbaf(r, g, b, 1.0f)); } SOKOL_API_IMPL void sgl_v2f_c3b(float x, float y, uint8_t r, uint8_t g, uint8_t b) { _sgl_vtx(x, y, 0.0f, _sgl.u, _sgl.v, _sgl_pack_rgbab(r, g, b, 255)); } SOKOL_API_IMPL void sgl_v2f_c4f(float x, float y, float r, float g, float b, float a) { _sgl_vtx(x, y, 0.0f, _sgl.u, _sgl.v, _sgl_pack_rgbaf(r, g, b, a)); } SOKOL_API_IMPL void sgl_v2f_c4b(float x, float y, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { _sgl_vtx(x, y, 0.0f, _sgl.u, _sgl.v, _sgl_pack_rgbab(r, g, b, a)); } SOKOL_API_IMPL void sgl_v2f_c1i(float x, float y, uint32_t rgba) { _sgl_vtx(x, y, 0.0f, _sgl.u, _sgl.v, rgba); } SOKOL_API_IMPL void sgl_v3f_c3f(float x, float y, float z, float r, float g, float b) { _sgl_vtx(x, y, z, _sgl.u, _sgl.v, _sgl_pack_rgbaf(r, g, b, 1.0f)); } SOKOL_API_IMPL void sgl_v3f_c3b(float x, float y, float z, uint8_t r, uint8_t g, uint8_t b) { _sgl_vtx(x, y, z, _sgl.u, _sgl.v, _sgl_pack_rgbab(r, g, b, 255)); } SOKOL_API_IMPL void sgl_v3f_c4f(float x, float y, float z, float r, float g, float b, float a) { _sgl_vtx(x, y, z, _sgl.u, _sgl.v, _sgl_pack_rgbaf(r, g, b, a)); } SOKOL_API_IMPL void sgl_v3f_c4b(float x, float y, float z, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { _sgl_vtx(x, y, z, _sgl.u, _sgl.v, _sgl_pack_rgbab(r, g, b, a)); } SOKOL_API_IMPL void sgl_v3f_c1i(float x, float y, float z, uint32_t rgba) { _sgl_vtx(x, y, z, _sgl.u, _sgl.v, rgba); } SOKOL_API_IMPL void sgl_v2f_t2f_c3f(float x, float y, float u, float v, float r, float g, float b) { _sgl_vtx(x, y, 0.0f, u, v, _sgl_pack_rgbaf(r, g, b, 1.0f)); } SOKOL_API_IMPL void sgl_v2f_t2f_c3b(float x, float y, float u, float v, uint8_t r, uint8_t g, uint8_t b) { _sgl_vtx(x, y, 0.0f, u, v, _sgl_pack_rgbab(r, g, b, 255)); } SOKOL_API_IMPL void sgl_v2f_t2f_c4f(float x, float y, float u, float v, float r, float g, float b, float a) { _sgl_vtx(x, y, 0.0f, u, v, _sgl_pack_rgbaf(r, g, b, a)); } SOKOL_API_IMPL void sgl_v2f_t2f_c4b(float x, float y, float u, float v, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { _sgl_vtx(x, y, 0.0f, u, v, _sgl_pack_rgbab(r, g, b, a)); } SOKOL_API_IMPL void sgl_v2f_t2f_c1i(float x, float y, float u, float v, uint32_t rgba) { _sgl_vtx(x, y, 0.0f, u, v, rgba); } SOKOL_API_IMPL void sgl_v3f_t2f_c3f(float x, float y, float z, float u, float v, float r, float g, float b) { _sgl_vtx(x, y, z, u, v, _sgl_pack_rgbaf(r, g, b, 1.0f)); } SOKOL_API_IMPL void sgl_v3f_t2f_c3b(float x, float y, float z, float u, float v, uint8_t r, uint8_t g, uint8_t b) { _sgl_vtx(x, y, z, u, v, _sgl_pack_rgbab(r, g, b, 255)); } SOKOL_API_IMPL void sgl_v3f_t2f_c4f(float x, float y, float z, float u, float v, float r, float g, float b, float a) { _sgl_vtx(x, y, z, u, v, _sgl_pack_rgbaf(r, g, b, a)); } SOKOL_API_IMPL void sgl_v3f_t2f_c4b(float x, float y, float z, float u, float v, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { _sgl_vtx(x, y, z, u, v, _sgl_pack_rgbab(r, g, b, a)); } SOKOL_API_IMPL void sgl_v3f_t2f_c1i(float x, float y, float z, float u, float v, uint32_t rgba) { _sgl_vtx(x, y, z, u, v, rgba); } SOKOL_API_IMPL void sgl_matrix_mode_modelview(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.cur_matrix_mode = SGL_MATRIXMODE_MODELVIEW; } SOKOL_API_IMPL void sgl_matrix_mode_projection(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.cur_matrix_mode = SGL_MATRIXMODE_PROJECTION; } SOKOL_API_IMPL void sgl_matrix_mode_texture(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.cur_matrix_mode = SGL_MATRIXMODE_TEXTURE; } SOKOL_API_IMPL void sgl_load_identity(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_identity(_sgl_matrix()); } SOKOL_API_IMPL void sgl_load_matrix(const float m[16]) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; memcpy(&_sgl_matrix()->v[0][0], &m[0], 64); } SOKOL_API_IMPL void sgl_load_transpose_matrix(const float m[16]) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_transpose(_sgl_matrix(), (const _sgl_matrix_t*) &m[0]); } SOKOL_API_IMPL void sgl_mult_matrix(const float m[16]) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; const _sgl_matrix_t* m0 = (const _sgl_matrix_t*) &m[0]; _sgl_mul(_sgl_matrix(), m0); } SOKOL_API_IMPL void sgl_mult_transpose_matrix(const float m[16]) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_matrix_t m0; _sgl_transpose(&m0, (const _sgl_matrix_t*) &m[0]); _sgl_mul(_sgl_matrix(), &m0); } SOKOL_API_IMPL void sgl_rotate(float angle_rad, float x, float y, float z) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_rotate(_sgl_matrix(), angle_rad, x, y, z); } SOKOL_API_IMPL void sgl_scale(float x, float y, float z) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_scale(_sgl_matrix(), x, y, z); } SOKOL_API_IMPL void sgl_translate(float x, float y, float z) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_translate(_sgl_matrix(), x, y, z); } SOKOL_API_IMPL void sgl_frustum(float l, float r, float b, float t, float n, float f) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_frustum(_sgl_matrix(), l, r, b, t, n, f); } SOKOL_API_IMPL void sgl_ortho(float l, float r, float b, float t, float n, float f) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_ortho(_sgl_matrix(), l, r, b, t, n, f); } SOKOL_API_IMPL void sgl_perspective(float fov_y, float aspect, float z_near, float z_far) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_perspective(_sgl_matrix(), fov_y, aspect, z_near, z_far); } SOKOL_API_IMPL void sgl_lookat(float eye_x, float eye_y, float eye_z, float center_x, float center_y, float center_z, float up_x, float up_y, float up_z) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); _sgl.matrix_dirty = true; _sgl_lookat(_sgl_matrix(), eye_x, eye_y, eye_z, center_x, center_y, center_z, up_x, up_y, up_z); } SOKOL_API_DECL void sgl_push_matrix(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT((_sgl.cur_matrix_mode >= 0) && (_sgl.cur_matrix_mode < SGL_NUM_MATRIXMODES)); _sgl.matrix_dirty = true; if (_sgl.matrix_tos[_sgl.cur_matrix_mode] < (_SGL_MAX_STACK_DEPTH - 1)) { const _sgl_matrix_t* src = _sgl_matrix(); _sgl.matrix_tos[_sgl.cur_matrix_mode]++; _sgl_matrix_t* dst = _sgl_matrix(); *dst = *src; } else { _sgl.error = SGL_ERROR_STACK_OVERFLOW; } } SOKOL_API_DECL void sgl_pop_matrix(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); SOKOL_ASSERT((_sgl.cur_matrix_mode >= 0) && (_sgl.cur_matrix_mode < SGL_NUM_MATRIXMODES)); _sgl.matrix_dirty = true; if (_sgl.matrix_tos[_sgl.cur_matrix_mode] > 0) { _sgl.matrix_tos[_sgl.cur_matrix_mode]--; } else { _sgl.error = SGL_ERROR_STACK_UNDERFLOW; } } /* this renders the accumulated draw commands via sokol-gfx */ SOKOL_API_IMPL void sgl_draw(void) { SOKOL_ASSERT(_SGL_INIT_COOKIE == _sgl.init_cookie); if ((_sgl.error == SGL_NO_ERROR) && (_sgl.cur_vertex > 0) && (_sgl.cur_command > 0)) { uint32_t cur_pip_id = SG_INVALID_ID; uint32_t cur_img_id = SG_INVALID_ID; int cur_uniform_index = -1; sg_push_debug_group("sokol-gl"); sg_update_buffer(_sgl.vbuf, _sgl.vertices, _sgl.cur_vertex * sizeof(_sgl_vertex_t)); _sgl.bind.vertex_buffers[0] = _sgl.vbuf; for (int i = 0; i < _sgl.cur_command; i++) { const _sgl_command_t* cmd = &_sgl.commands[i]; switch (cmd->cmd) { case SGL_COMMAND_VIEWPORT: { const _sgl_viewport_args_t* args = &cmd->args.viewport; sg_apply_viewport(args->x, args->y, args->w, args->h, args->origin_top_left); } break; case SGL_COMMAND_SCISSOR_RECT: { const _sgl_scissor_rect_args_t* args = &cmd->args.scissor_rect; sg_apply_scissor_rect(args->x, args->y, args->w, args->h, args->origin_top_left); } break; case SGL_COMMAND_DRAW: { const _sgl_draw_args_t* args = &cmd->args.draw; if (args->pip.id != cur_pip_id) { sg_apply_pipeline(args->pip); cur_pip_id = args->pip.id; /* when pipeline changes, also need to re-apply uniforms and bindings */ cur_img_id = SG_INVALID_ID; cur_uniform_index = -1; } if (cur_img_id != args->img.id) { _sgl.bind.fs_images[0] = args->img; sg_apply_bindings(&_sgl.bind); cur_img_id = args->img.id; } if (cur_uniform_index != args->uniform_index) { sg_apply_uniforms(SG_SHADERSTAGE_VS, 0, &_sgl.uniforms[args->uniform_index], sizeof(_sgl_uniform_t)); cur_uniform_index = args->uniform_index; } /* FIXME: what if number of vertices doesn't match the primitive type? */ sg_draw(args->base_vertex, args->num_vertices, 1); } break; } } sg_pop_debug_group(); } _sgl_rewind(); } #endif /* SOKOL_GL_IMPL */