Compare commits
	
		
			No commits in common. "20f6e2f2dc80179661a7aa920467a388e690c71b" and "46dd8625120376fd538e4a23dd3de3b65c23ca32" have entirely different histories. 
		
	
	
		
			20f6e2f2dc
			...
			46dd862512
		
	
		
							
								
								
									
										5
									
								
								justfile
								
								
								
								
							
							
						
						
									
										5
									
								
								justfile
								
								
								
								
							|  | @ -12,6 +12,5 @@ run: build | ||||||
| 
 | 
 | ||||||
| transpile_shaders_metal: | transpile_shaders_metal: | ||||||
|     mkdir -p bin/transpiled_shaders |     mkdir -p bin/transpiled_shaders | ||||||
| 
 |     naga shaders/vertex.wgsl bin/transpiled_shaders/vertex.metal --metal-version 1.2 | ||||||
|     xcrun -sdk macosx metal -o bin/transpiled_shaders/text_atlas.ir -c shaders/text_atlas.metal |     naga shaders/fragment.wgsl bin/transpiled_shaders/fragment.metal --metal-version 1.2 | ||||||
|     xcrun -sdk macosx metallib -o bin/shaders.metallib bin/transpiled_shaders/text_atlas.ir |  | ||||||
|  |  | ||||||
|  | @ -0,0 +1,21 @@ | ||||||
|  | #version 330 core | ||||||
|  | struct VertexOutput { | ||||||
|  |     vec4 position; | ||||||
|  |     vec2 tex_coord; | ||||||
|  | }; | ||||||
|  | uniform sampler2D _group_0_binding_0_fs; | ||||||
|  | 
 | ||||||
|  | smooth in vec2 _vs2fs_location0; | ||||||
|  | layout(location = 0) out vec4 _fs2p_location0; | ||||||
|  | 
 | ||||||
|  | void main() { | ||||||
|  |     VertexOutput input_ = VertexOutput(gl_FragCoord, _vs2fs_location0); | ||||||
|  |     float text_color = 0.0; | ||||||
|  |     vec4 _e8 = texture(_group_0_binding_0_fs, vec2(vec2(input_.tex_coord.x, input_.tex_coord.y))); | ||||||
|  |     text_color = _e8.x; | ||||||
|  |     float _e11 = text_color; | ||||||
|  |     float _e17 = text_color; | ||||||
|  |     _fs2p_location0 = vec4((_e11 * vec3(1.0, 1.0, 1.0)), _e17); | ||||||
|  |     return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @ -0,0 +1,16 @@ | ||||||
|  | struct VertexOutput { | ||||||
|  |     @builtin(position) position: vec4<f32>, | ||||||
|  |     @location(0) tex_coord: vec2<f32>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @group(0) @binding(0) | ||||||
|  | var texture: texture_2d<f32>; | ||||||
|  | @group(0) @binding(1) | ||||||
|  | var texture_sampler: sampler; | ||||||
|  | 
 | ||||||
|  | @fragment | ||||||
|  | fn fs_main(input: VertexOutput) -> @location(0) vec4<f32> { | ||||||
|  |     var text_color = textureSample(texture, texture_sampler, vec2<f32>(input.tex_coord.x, input.tex_coord.y)).r; | ||||||
|  | 
 | ||||||
|  |     return vec4<f32>(text_color * vec3<f32>(1., 1., 1.), text_color); | ||||||
|  | } | ||||||
|  | @ -1,66 +0,0 @@ | ||||||
| #include <metal_stdlib> |  | ||||||
| 
 |  | ||||||
| using namespace metal; |  | ||||||
| 
 |  | ||||||
| struct VertexInput { |  | ||||||
|     float2 position; |  | ||||||
|     float2 tex_coord; |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| struct VertexOutput { |  | ||||||
|     float4 position [[position]]; |  | ||||||
|     float2 tex_coord; |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| struct Glyph { |  | ||||||
|     float2 atlas_position; |  | ||||||
|     float2 size; |  | ||||||
|     float2 target_position; |  | ||||||
|     float y_offset; |  | ||||||
|     float _haha_alignment; |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| struct UniformParams { |  | ||||||
|     float2 screen_size; |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| float4 to_device_position(float2 position, float2 size) { |  | ||||||
|     return float4(((position / size) * 2.0) - float2(1.0), 1.0, 1.0); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| vertex VertexOutput |  | ||||||
| vs_main( |  | ||||||
|         uint vertex_id [[vertex_id]], |  | ||||||
|         uint glyph_id [[instance_id]], |  | ||||||
|         constant VertexInput *vertices [[buffer(0)]], |  | ||||||
|         constant Glyph *glyphs [[buffer(1)]], |  | ||||||
|         constant UniformParams ¶ms [[buffer(2)]] |  | ||||||
| ) |  | ||||||
| { |  | ||||||
|     VertexOutput out; |  | ||||||
| 
 |  | ||||||
|     Glyph glyph = glyphs[glyph_id]; |  | ||||||
| 
 |  | ||||||
|     float2 scaled_size = ((vertices[vertex_id].position + 1.0) / 2.0) * (glyph.size/2.0); |  | ||||||
|     float2 scaled_size_2 = ((vertices[vertex_id].position + 1.0) / 2.0) * (glyph.size); |  | ||||||
|     float2 glyph_pos = scaled_size + glyph.target_position + float2(0, glyph.y_offset/2.0+32); |  | ||||||
|     float4 device_position = to_device_position(glyph_pos, params.screen_size); |  | ||||||
|     float2 atlas_position = (scaled_size_2 + glyph.atlas_position) / 1024.0; |  | ||||||
| 
 |  | ||||||
|     device_position.y = -device_position.y; |  | ||||||
| 
 |  | ||||||
|     out.position = device_position; |  | ||||||
|     out.tex_coord = atlas_position; |  | ||||||
| 
 |  | ||||||
|     return out; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| fragment float4 fs_main(VertexOutput in [[stage_in]], |  | ||||||
|                             texture2d<float, access::sample> texture [[texture(0)]]) |  | ||||||
| { |  | ||||||
|     constexpr sampler texture_sampler (mag_filter::linear, min_filter::linear); |  | ||||||
| 
 |  | ||||||
|     float text_color = texture.sample(texture_sampler, in.tex_coord).r; |  | ||||||
|     return float4(text_color * float3(1,1,1), text_color); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
|  | @ -0,0 +1,49 @@ | ||||||
|  | #version 330 core | ||||||
|  | uniform uint naga_vs_first_instance; | ||||||
|  | 
 | ||||||
|  | struct VertexInput { | ||||||
|  |     vec3 position; | ||||||
|  |     vec2 tex_coord; | ||||||
|  |     vec2 atlas_position; | ||||||
|  |     vec2 size; | ||||||
|  |     vec2 target_position; | ||||||
|  |     float y_offset; | ||||||
|  |     uint glyph_id; | ||||||
|  | }; | ||||||
|  | struct VertexOutput { | ||||||
|  |     vec4 position; | ||||||
|  |     vec2 tex_coord; | ||||||
|  | }; | ||||||
|  | uniform vec4 screen_size; | ||||||
|  | 
 | ||||||
|  | layout(location = 0) in vec3 _p2vs_location0; | ||||||
|  | layout(location = 1) in vec2 _p2vs_location1; | ||||||
|  | layout(location = 2) in vec2 _p2vs_location2; | ||||||
|  | layout(location = 3) in vec2 _p2vs_location3; | ||||||
|  | layout(location = 4) in vec2 _p2vs_location4; | ||||||
|  | layout(location = 5) in float _p2vs_location5; | ||||||
|  | smooth out vec2 _vs2fs_location0; | ||||||
|  | 
 | ||||||
|  | vec4 to_device_position(vec2 position, vec2 size) { | ||||||
|  |     return vec4((((position / size) * 2.0) - vec2(1.0)), 1.0, 1.0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void main() { | ||||||
|  |     VertexInput input_ = VertexInput(_p2vs_location0, _p2vs_location1, _p2vs_location2, _p2vs_location3, _p2vs_location4, _p2vs_location5, (uint(gl_InstanceID) + naga_vs_first_instance)); | ||||||
|  |     VertexOutput out_ = VertexOutput(vec4(0.0), vec2(0.0)); | ||||||
|  |     vec4 vertex_pos = vec4(0.0); | ||||||
|  |     vec2 atlas_position = vec2(0.0); | ||||||
|  |     vec4 _e28 = to_device_position((((((input_.position.xy + vec2(1.0)) / vec2(2.0)) * (input_.size / vec2(2.0))) + input_.target_position) + vec2(0.0, ((input_.y_offset / 2.0) + 32.0))), screen_size.xy); | ||||||
|  |     vertex_pos = _e28; | ||||||
|  |     atlas_position = (((((input_.position.xy + vec2(1.0)) / vec2(2.0)) * input_.size) + input_.atlas_position) / vec2(1024.0)); | ||||||
|  |     vec4 _e47 = vertex_pos; | ||||||
|  |     out_.position = _e47; | ||||||
|  |     vec2 _e49 = atlas_position; | ||||||
|  |     out_.tex_coord = _e49; | ||||||
|  |     VertexOutput _e50 = out_; | ||||||
|  |     gl_Position = _e50.position; | ||||||
|  |     _vs2fs_location0 = _e50.tex_coord; | ||||||
|  |     gl_Position.yz = vec2(-gl_Position.y, gl_Position.z * 2.0 - gl_Position.w); | ||||||
|  |     return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @ -0,0 +1,42 @@ | ||||||
|  | struct VertexInput { | ||||||
|  |     @location(0) position: vec3<f32>, | ||||||
|  |     @location(1) tex_coord: vec2<f32>, | ||||||
|  | 
 | ||||||
|  |     @location(2) atlas_position: vec2<f32>, | ||||||
|  |     @location(3) size: vec2<f32>, | ||||||
|  |     @location(4) target_position: vec2<f32>, | ||||||
|  |     @location(5) y_offset: f32, | ||||||
|  | 
 | ||||||
|  |     @builtin(instance_index) glyph_id: u32, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct VertexOutput { | ||||||
|  |     @builtin(position) position: vec4<f32>, | ||||||
|  |     @location(0) tex_coord: vec2<f32>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct Params { | ||||||
|  |     screen_size: vec4<f32>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn to_device_position(position: vec2<f32>, size: vec2<f32>) -> vec4<f32> { | ||||||
|  |     return vec4<f32>((((position / size) * 2.) - 1.), 1., 1.); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @group(0) @binding(0) | ||||||
|  | var<uniform> params: Params; | ||||||
|  | 
 | ||||||
|  | @vertex | ||||||
|  | fn vs_main(input: VertexInput) -> VertexOutput { | ||||||
|  |     var out: VertexOutput; | ||||||
|  | 
 | ||||||
|  |     var vertex_pos = to_device_position(((input.position.xy + 1.) / 2.) * (input.size/2.0) + input.target_position + vec2<f32>(0., (input.y_offset/2.0)+32), params.screen_size.xy); | ||||||
|  |     vertex_pos.y = -vertex_pos.y; | ||||||
|  |     var atlas_position = (((input.position.xy + 1.) / 2.) * input.size + input.atlas_position) / vec2<f32>(1024); | ||||||
|  | 
 | ||||||
|  |     out.position = vertex_pos; | ||||||
|  |     out.tex_coord = atlas_position; | ||||||
|  | 
 | ||||||
|  |     return out; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @ -1,6 +1,5 @@ | ||||||
| -DED_UI_IMPLEMENTATION | -DED_UI_IMPLEMENTATION | ||||||
| -DED_HT_IMPLEMENTATION | -DED_HT_IMPLEMENTATION | ||||||
| -DED_STRING_IMPLEMENTATION | -DED_STRING_IMPLEMENTATION | ||||||
| -DED_GFX_IMPLEMENTATION |  | ||||||
| -I../vendor/ | -I../vendor/ | ||||||
| -ObjC | -ObjC | ||||||
|  |  | ||||||
							
								
								
									
										362
									
								
								src/gfx.h
								
								
								
								
							
							
						
						
									
										362
									
								
								src/gfx.h
								
								
								
								
							|  | @ -1,362 +0,0 @@ | ||||||
| // Graphics layer abstraction.
 |  | ||||||
| 
 |  | ||||||
| #ifndef ED_GFX_INCLUDED |  | ||||||
| #define ED_GFX_INCLUDED |  | ||||||
| #include <AppKit/AppKit.h> |  | ||||||
| #include <Foundation/Foundation.h> |  | ||||||
| #include <CoreGraphics/CoreGraphics.h> |  | ||||||
| #include <Metal/Metal.h> |  | ||||||
| #include <QuartzCore/QuartzCore.h> |  | ||||||
| #include <QuartzCore/CoreAnimation.h> |  | ||||||
| #include <stdint.h> |  | ||||||
| 
 |  | ||||||
| #include "ed_array.h" |  | ||||||
| 
 |  | ||||||
| bool keep_running = true; |  | ||||||
| 
 |  | ||||||
| @interface EDGFXView : NSView |  | ||||||
| @end |  | ||||||
| @interface AppDelegate : NSObject<NSApplicationDelegate> |  | ||||||
| @end |  | ||||||
| @interface WindowDelegate : NSObject<NSWindowDelegate> |  | ||||||
| @end |  | ||||||
| 
 |  | ||||||
| #define wrapIdArray(T) typedef id<T> _ ## T;\ |  | ||||||
| arrayTemplate(_ ## T); |  | ||||||
| 
 |  | ||||||
| wrapIdArray(MTLRenderPipelineState); |  | ||||||
| wrapIdArray(MTLBuffer); |  | ||||||
| wrapIdArray(MTLTexture); |  | ||||||
| 
 |  | ||||||
| #if defined(__APPLE__) |  | ||||||
| typedef struct { |  | ||||||
|     NSApplication *application; |  | ||||||
|     NSWindow *window; |  | ||||||
|     EDGFXView *view; |  | ||||||
|     bool keep_running; |  | ||||||
| 
 |  | ||||||
|     // Metal objects
 |  | ||||||
|     id<MTLDevice> device; |  | ||||||
|     CAMetalLayer *metal_layer; |  | ||||||
|     id<MTLLibrary> library; |  | ||||||
|     id<MTLCommandQueue> command_queue; |  | ||||||
| 
 |  | ||||||
|     array(_MTLRenderPipelineState) pipelines; |  | ||||||
|     array(_MTLBuffer) buffers; |  | ||||||
|     array(_MTLTexture) textures; |  | ||||||
| } _metal_gfx_context; |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| typedef void (*_gfx_frame_func)(); |  | ||||||
| typedef struct { |  | ||||||
| #if defined(__APPLE__) |  | ||||||
|     _metal_gfx_context backend; |  | ||||||
| #else |  | ||||||
| #error "Unsupported platform" |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
|     uint32_t frame_width; |  | ||||||
|     uint32_t frame_height; |  | ||||||
|     _gfx_frame_func frame_func; |  | ||||||
| } gfx_context_t; |  | ||||||
| static gfx_context_t _gfx_context; |  | ||||||
| 
 |  | ||||||
| #ifdef ED_GFX_IMPLEMENTATION |  | ||||||
| 
 |  | ||||||
| #if defined(__APPLE__) |  | ||||||
| static void _metal_gfx_present(_metal_gfx_context *cx); |  | ||||||
| static void _metal_gfx_send_events(_metal_gfx_context *cx); |  | ||||||
| 
 |  | ||||||
| @implementation AppDelegate |  | ||||||
| - (NSApplicationTerminateReply)applicationShouldTerminate:(NSApplication *)sender { |  | ||||||
|     keep_running = false; |  | ||||||
| 
 |  | ||||||
|     return NSTerminateCancel; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| - (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)sender { |  | ||||||
|     keep_running = false; |  | ||||||
| 
 |  | ||||||
|     return NO; |  | ||||||
| } |  | ||||||
| @end |  | ||||||
| 
 |  | ||||||
| @implementation WindowDelegate |  | ||||||
| - (BOOL)windowShouldClose:(NSApplication *)sender { |  | ||||||
|     keep_running = false; |  | ||||||
|     return YES; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| - (void)windowDidResize:(NSNotification *)notification { |  | ||||||
|     NSLog(@"did resize\n"); |  | ||||||
| 
 |  | ||||||
|     _gfx_context.frame_width = _gfx_context.backend.window.contentView.frame.size.width; |  | ||||||
|     _gfx_context.frame_height = _gfx_context.backend.window.contentView.frame.size.height; |  | ||||||
|     [_gfx_context.backend.metal_layer setDrawableSize:CGSizeMake(_gfx_context.frame_width, _gfx_context.frame_height)]; |  | ||||||
| 
 |  | ||||||
|     _metal_gfx_present(&_gfx_context.backend); |  | ||||||
| } |  | ||||||
| @end |  | ||||||
| 
 |  | ||||||
| @implementation EDGFXView |  | ||||||
| - (BOOL)isOpaque { |  | ||||||
|     return YES; |  | ||||||
| } |  | ||||||
| - (void)updateLayer { |  | ||||||
|     _metal_gfx_present(&_gfx_context.backend); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| - (BOOL)wantsLayer { |  | ||||||
|     return YES; |  | ||||||
| } |  | ||||||
| - (BOOL)wantsUpdateLayer { |  | ||||||
|     return YES; |  | ||||||
| } |  | ||||||
| - (NSViewLayerContentsRedrawPolicy)layerContentsRedrawPolicy { |  | ||||||
|     return NSViewLayerContentsRedrawOnSetNeedsDisplay; |  | ||||||
| } |  | ||||||
| @end |  | ||||||
| 
 |  | ||||||
| static _metal_gfx_context _metal_gfx_init_context(uint32_t width, uint32_t height) { |  | ||||||
|     NSApplication *application = [NSApplication sharedApplication]; |  | ||||||
|     if (application == NULL) { |  | ||||||
|         fprintf(stderr, "NSApplication:sharedApplication failed\n"); |  | ||||||
|         exit(1); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     NSString *title = @"editor - [C is Illegal]"; |  | ||||||
| 
 |  | ||||||
|     NSRect rect = NSMakeRect(0, 0, width, height); |  | ||||||
|     NSWindow *window = [[NSWindow alloc] initWithContentRect:rect |  | ||||||
|                                                    styleMask:NSWindowStyleMaskTitled|NSWindowStyleMaskClosable|NSWindowStyleMaskMiniaturizable|NSWindowStyleMaskResizable |  | ||||||
|                                                      backing:NSBackingStoreBuffered |  | ||||||
|                                                        defer:NO]; |  | ||||||
|     EDGFXView *view = [[EDGFXView alloc] initWithFrame:rect]; |  | ||||||
|     [view updateTrackingAreas]; |  | ||||||
|     [window setTitle:title]; |  | ||||||
|     [window setContentView:view]; |  | ||||||
|     [window setDelegate:[[WindowDelegate alloc] init]]; |  | ||||||
|     [window makeKeyAndOrderFront:NULL]; |  | ||||||
| 
 |  | ||||||
|     // TODO: make this work
 |  | ||||||
|     // if (application.mainMenu == NULL) {
 |  | ||||||
|     //     NSMenu *menu = [[NSMenu alloc] initWithTitle:@"an_editor"];
 |  | ||||||
|     //     if (menu == NULL) {
 |  | ||||||
|     //         fprintf(stderr, "failed to create application menu\n");
 |  | ||||||
|     //         exit(1);
 |  | ||||||
|     //     }
 |  | ||||||
|     //     application.mainMenu = menu;
 |  | ||||||
|     // }
 |  | ||||||
|     [application setDelegate:[[AppDelegate alloc] init]]; |  | ||||||
|     [application setActivationPolicy:NSApplicationActivationPolicyRegular]; |  | ||||||
|     [application setPresentationOptions:NSApplicationPresentationDefault]; |  | ||||||
|     [application finishLaunching]; |  | ||||||
| 
 |  | ||||||
|     id<MTLDevice> device = MTLCreateSystemDefaultDevice(); |  | ||||||
| 
 |  | ||||||
|     CAMetalLayer *metal_layer = [CAMetalLayer layer]; |  | ||||||
|     metal_layer.device = device; |  | ||||||
|     metal_layer.pixelFormat = MTLPixelFormatRGBA8Unorm; |  | ||||||
|     metal_layer.frame = CGRectMake(0, 0, width, height); |  | ||||||
|     metal_layer.needsDisplayOnBoundsChange = YES; |  | ||||||
|     metal_layer.presentsWithTransaction = YES; |  | ||||||
|     metal_layer.autoresizingMask = kCALayerWidthSizable|kCALayerHeightSizable; |  | ||||||
|     view.wantsLayer = YES; |  | ||||||
|     [view.layer addSublayer:metal_layer]; |  | ||||||
|     view.layerContentsRedrawPolicy = NSViewLayerContentsRedrawOnSetNeedsDisplay; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     NSError *libraryError = NULL; |  | ||||||
|     NSURL *libraryURL = [[NSBundle mainBundle] URLForResource:@"./shaders" withExtension:@"metallib"]; |  | ||||||
|     if (libraryURL == NULL) { |  | ||||||
|         fprintf(stderr, "Couldn't find library file\n"); |  | ||||||
|         exit(1); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     id<MTLLibrary> library = [device newLibraryWithURL:libraryURL error:&libraryError]; |  | ||||||
| 
 |  | ||||||
|     if (library == NULL) { |  | ||||||
|         if (libraryError.description != NULL) { |  | ||||||
|             NSLog(@"Error description: %@\n", libraryError.description); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         exit(1); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     id<MTLCommandQueue> command_queue = [device newCommandQueue]; |  | ||||||
|     id<MTLFunction> vertex_func = [library newFunctionWithName:@"vs_main"]; |  | ||||||
|     id<MTLFunction> fragment_func = [library newFunctionWithName:@"fs_main"]; |  | ||||||
| 
 |  | ||||||
|     MTLRenderPipelineDescriptor *pipeline_descriptor = [[MTLRenderPipelineDescriptor alloc] init]; |  | ||||||
|     [pipeline_descriptor setVertexFunction:vertex_func]; |  | ||||||
|     [pipeline_descriptor setFragmentFunction:fragment_func]; |  | ||||||
|     pipeline_descriptor.colorAttachments[0].pixelFormat = MTLPixelFormatRGBA8Unorm; |  | ||||||
| 
 |  | ||||||
|     NSError *pipeline_error = NULL; |  | ||||||
|     array(_MTLRenderPipelineState) pipelines = newArray(_MTLRenderPipelineState, 2); |  | ||||||
|     pushArray(_MTLRenderPipelineState, &pipelines, [device newRenderPipelineStateWithDescriptor:pipeline_descriptor error:&pipeline_error]); |  | ||||||
| 
 |  | ||||||
|     array(_MTLBuffer) buffers = newArray(_MTLBuffer, 8); |  | ||||||
|     array(_MTLTexture) textures = newArray(_MTLTexture, 8); |  | ||||||
| 
 |  | ||||||
|     if (pipeline_error != NULL) { |  | ||||||
|         if (pipeline_error.description != NULL) { |  | ||||||
|             NSLog(@"Error description: %@\n", pipeline_error.description); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         exit(1); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return (_metal_gfx_context) { |  | ||||||
|         .application = application, |  | ||||||
|         .window = window, |  | ||||||
|         .view = view, |  | ||||||
|         .keep_running = true, |  | ||||||
| 
 |  | ||||||
|         .device = device, |  | ||||||
|         .metal_layer = metal_layer, |  | ||||||
|         .library = library, |  | ||||||
|         .command_queue = command_queue, |  | ||||||
|         .pipelines = pipelines, |  | ||||||
|         .buffers = buffers, |  | ||||||
|         .textures = textures, |  | ||||||
|     }; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void _metal_gfx_send_events(_metal_gfx_context *cx) { |  | ||||||
|     NSEvent *event = [cx->application nextEventMatchingMask:NSEventMaskAny |  | ||||||
|         untilDate:[NSDate distantPast] inMode:NSDefaultRunLoopMode dequeue:YES]; |  | ||||||
| 
 |  | ||||||
|     [cx->application sendEvent:event]; |  | ||||||
|     [cx->application updateWindows]; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void _metal_gfx_present(_metal_gfx_context *cx) { |  | ||||||
|     _gfx_context.frame_func(); |  | ||||||
| 
 |  | ||||||
|     id<CAMetalDrawable> drawable = [cx->metal_layer nextDrawable]; |  | ||||||
| 
 |  | ||||||
|     id<MTLCommandBuffer> command_buffer = [cx->command_queue commandBuffer]; |  | ||||||
|     MTLRenderPassDescriptor *render_pass_desc = [MTLRenderPassDescriptor renderPassDescriptor]; |  | ||||||
|     render_pass_desc.colorAttachments[0].texture = drawable.texture; |  | ||||||
|     render_pass_desc.colorAttachments[0].loadAction = MTLLoadActionClear; |  | ||||||
|     render_pass_desc.colorAttachments[0].clearColor = MTLClearColorMake(0,0,0,1); |  | ||||||
|     render_pass_desc.colorAttachments[0].storeAction = MTLStoreActionStore; |  | ||||||
| 
 |  | ||||||
|     id<MTLRenderCommandEncoder> encoder = [command_buffer renderCommandEncoderWithDescriptor:render_pass_desc]; |  | ||||||
|     [encoder setRenderPipelineState:cx->pipelines.data[0]]; |  | ||||||
| 
 |  | ||||||
|     // FIXME: allow these to be described by the user instead of hardcoded
 |  | ||||||
|     [encoder setVertexBuffer:cx->buffers.data[0] offset:0 atIndex:0]; |  | ||||||
|     [encoder setVertexBuffer:cx->buffers.data[2] offset:0 atIndex:1]; |  | ||||||
|     [encoder setVertexBuffer:cx->buffers.data[3] offset:0 atIndex:2]; |  | ||||||
|     [encoder setFragmentTexture:cx->textures.data[0] atIndex:0]; |  | ||||||
|     [encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle indexCount:6 indexType:MTLIndexTypeUInt16 indexBuffer:cx->buffers.data[1] indexBufferOffset:0 instanceCount:36]; |  | ||||||
|     [encoder endEncoding]; |  | ||||||
| 
 |  | ||||||
|     [command_buffer presentDrawable:drawable]; |  | ||||||
|     [command_buffer commit]; |  | ||||||
| 
 |  | ||||||
|     [command_buffer waitUntilScheduled]; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| size_t _metal_gfx_push_texture_buffer(_metal_gfx_context *cx, uint32_t width, uint32_t height, const void *data, size_t len) { |  | ||||||
|     MTLTextureDescriptor *texture_desc = [MTLTextureDescriptor texture2DDescriptorWithPixelFormat:MTLPixelFormatR8Unorm width:width height:height mipmapped:false]; |  | ||||||
|     _MTLTexture texture = [cx->device newTextureWithDescriptor:texture_desc]; |  | ||||||
| 
 |  | ||||||
|     MTLRegion region = MTLRegionMake2D(0, 0, width, height); |  | ||||||
|     [texture replaceRegion:region mipmapLevel:0 slice:0 withBytes:data bytesPerRow:width*sizeof(uint8_t) bytesPerImage:len]; |  | ||||||
| 
 |  | ||||||
|     pushArray(_MTLTexture, &cx->textures, texture); |  | ||||||
|     return cx->textures.size-1; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void _metal_gfx_resize_texture_buffer(_metal_gfx_context *cx, uint32_t width, size_t texture_index, uint32_t height) { |  | ||||||
|     [cx->textures.data[texture_index] setPurgeableState:MTLPurgeableStateEmpty]; |  | ||||||
|     [cx->textures.data[texture_index] release]; |  | ||||||
| 
 |  | ||||||
|     MTLTextureDescriptor *texture_desc = [MTLTextureDescriptor texture2DDescriptorWithPixelFormat:MTLPixelFormatR8Unorm width:width height:height mipmapped:false]; |  | ||||||
|     cx->textures.data[texture_index] = [cx->device newTextureWithDescriptor:texture_desc]; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| size_t _metal_gfx_push_vertex_buffer(_metal_gfx_context *cx, const void *data, size_t len) { |  | ||||||
|     pushArray(_MTLBuffer, &cx->buffers, [cx->device newBufferWithBytes:data |  | ||||||
|                                                                 length:len |  | ||||||
|                                                                options:MTLResourceStorageModeShared] |  | ||||||
|     ); |  | ||||||
| 
 |  | ||||||
|     return cx->buffers.size-1; |  | ||||||
| } |  | ||||||
| size_t _metal_gfx_allocate_vertex_buffer(_metal_gfx_context *cx, size_t len) { |  | ||||||
|     pushArray(_MTLBuffer, &cx->buffers, [cx->device newBufferWithLength:len options:MTLResourceStorageModeShared] |  | ||||||
|     ); |  | ||||||
| 
 |  | ||||||
|     return cx->buffers.size-1; |  | ||||||
| } |  | ||||||
| void _metal_gfx_update_buffer(_metal_gfx_context *cx, size_t buffer_index, const void *data, size_t len) { |  | ||||||
|     void *buffer_contents = [cx->buffers.data[buffer_index] contents]; |  | ||||||
| 
 |  | ||||||
|     // FIXME: actually check to see if this will fit in the buffer
 |  | ||||||
|     memcpy(buffer_contents, data, len); |  | ||||||
| } |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| void * gfx_init_context(_gfx_frame_func frame_func, uint32_t width, uint32_t height) { |  | ||||||
|     __auto_type backend = |  | ||||||
| #if defined(__APPLE__) |  | ||||||
|     _metal_gfx_init_context(width, height); |  | ||||||
| #else |  | ||||||
| #error "Unsupported graphics backend" |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
|     _gfx_context.backend = backend; |  | ||||||
|     _gfx_context.frame_func = frame_func; |  | ||||||
|     _gfx_context.frame_width = width; |  | ||||||
|     _gfx_context.frame_height = height; |  | ||||||
| 
 |  | ||||||
|     return &_gfx_context; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void gfx_run_events(gfx_context_t *cx) { |  | ||||||
| #if defined(__APPLE__) |  | ||||||
|     return _metal_gfx_send_events(&cx->backend); |  | ||||||
| #else |  | ||||||
| #error "Unsupported graphics backend" |  | ||||||
| #endif |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| size_t gfx_push_texture_buffer(gfx_context_t *cx, uint32_t width, uint32_t height, const void *data, size_t len) { |  | ||||||
| #if defined(__APPLE__) |  | ||||||
|     return _metal_gfx_push_texture_buffer(&cx->backend, width, height, data, len); |  | ||||||
| #else |  | ||||||
| #error "Unsupported graphics backend" |  | ||||||
| #endif |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| size_t gfx_push_vertex_buffer(gfx_context_t *cx, const void *data, size_t len) { |  | ||||||
| #if defined(__APPLE__) |  | ||||||
|     return _metal_gfx_push_vertex_buffer(&cx->backend, data, len); |  | ||||||
| #else |  | ||||||
| #error "Unsupported graphics backend" |  | ||||||
| #endif |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| size_t gfx_allocate_vertex_buffer(gfx_context_t *cx, size_t len) { |  | ||||||
| #if defined(__APPLE__) |  | ||||||
|     return _metal_gfx_allocate_vertex_buffer(&cx->backend, len); |  | ||||||
| #else |  | ||||||
| #error "Unsupported graphics backend" |  | ||||||
| #endif |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // FIXME: abstract different backends
 |  | ||||||
| void gfx_update_buffer(gfx_context_t *cx, size_t buffer_index, const void *data, size_t len) { |  | ||||||
| #if defined(__APPLE__) |  | ||||||
|     return _metal_gfx_update_buffer(&cx->backend, buffer_index, data, len); |  | ||||||
| #else |  | ||||||
| #error "Unsupported graphics backend" |  | ||||||
| #endif |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif |  | ||||||
| #endif |  | ||||||
							
								
								
									
										14
									
								
								src/ht.h
								
								
								
								
							
							
						
						
									
										14
									
								
								src/ht.h
								
								
								
								
							|  | @ -96,20 +96,6 @@ void *ht_get(ed_ht *ht, string key) { | ||||||
|     return NULL; |     return NULL; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ht_remove(ed_ht *ht, string key) { |  | ||||||
|     ht_hash_t hash = ht_hash(key, ht->capacity); |  | ||||||
| 
 |  | ||||||
|     for (size_t i=hash; i<ht->capacity; ++i) { |  | ||||||
|         if (ht->key_slots[i].key.data == NULL) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (string_eq(ht->key_slots[i].key, key)) { |  | ||||||
|             ht->key_slots[i] = (ed_ht_slot) { 0 }; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void ht_destroy(ed_ht *ht) { | void ht_destroy(ed_ht *ht) { | ||||||
|     // TODO: destroy the hash table
 |     // TODO: destroy the hash table
 | ||||||
|     free(ht->key_slots); |     free(ht->key_slots); | ||||||
|  |  | ||||||
							
								
								
									
										279
									
								
								src/main.c
								
								
								
								
							
							
						
						
									
										279
									
								
								src/main.c
								
								
								
								
							|  | @ -1,13 +1,36 @@ | ||||||
| #include <stdint.h> |  | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| #include <stdlib.h> | #include <stdlib.h> | ||||||
| #include <inttypes.h> | #include <inttypes.h> | ||||||
| #include <stdbool.h> | 
 | ||||||
|  | // #define ARENA_IMPLEMENTATION
 | ||||||
|  | // #include <tsoding/arena.h>
 | ||||||
|  | 
 | ||||||
|  | #define SOKOL_DEBUG | ||||||
|  | 
 | ||||||
|  | #define SOKOL_APP_IMPL | ||||||
|  | #define SOKOL_GFX_IMPL | ||||||
|  | #define SOKOL_GLUE_IMPL | ||||||
|  | #define SOKOL_FETCH_IMPL | ||||||
|  | #define SOKOL_LOG_IMPL | ||||||
|  | 
 | ||||||
|  | // TODO: condition compilation
 | ||||||
|  | #if defined (__APPLE__) | ||||||
|  |     #define SOKOL_METAL | ||||||
|  | #elif defined (__linux__) || defined (__unix__) | ||||||
|  |     #define SOKOL_GLCORE33 | ||||||
|  | #else | ||||||
|  |     #error "Unsupported platform for shaders" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #include <sokol/sokol_log.h> | ||||||
|  | #include <sokol/sokol_gfx.h> | ||||||
|  | #include <sokol/sokol_app.h> | ||||||
|  | #include <sokol/sokol_glue.h> | ||||||
|  | #include <sokol/sokol_fetch.h> | ||||||
| 
 | 
 | ||||||
| #define STB_TRUETYPE_IMPLEMENTATION | #define STB_TRUETYPE_IMPLEMENTATION | ||||||
| #include <stb/std_truetype.h> | #include <stb/std_truetype.h> | ||||||
| 
 | 
 | ||||||
| #define ED_GFX_IMPLEMENTATION |  | ||||||
| #define ED_STRING_IMPLEMENTATION | #define ED_STRING_IMPLEMENTATION | ||||||
| #define ED_HT_IMPLEMENTATION | #define ED_HT_IMPLEMENTATION | ||||||
| #define ED_UI_IMPLEMENTATION | #define ED_UI_IMPLEMENTATION | ||||||
|  | @ -15,7 +38,7 @@ | ||||||
| #include "ht.h" | #include "ht.h" | ||||||
| #include "ui.h" | #include "ui.h" | ||||||
| #include "ed_array.h" | #include "ed_array.h" | ||||||
| #include "gfx.h" | 
 | ||||||
| 
 | 
 | ||||||
| // static Arena default_arena = {0};
 | // static Arena default_arena = {0};
 | ||||||
| // static Arena temporary_arena = {0};
 | // static Arena temporary_arena = {0};
 | ||||||
|  | @ -39,15 +62,23 @@ typedef struct { | ||||||
|     float size[2]; |     float size[2]; | ||||||
|     float position[2]; |     float position[2]; | ||||||
|     float y_offset; |     float y_offset; | ||||||
|     float _haha_alignment; |  | ||||||
| } GpuGlyph; | } GpuGlyph; | ||||||
| arrayTemplate(GpuGlyph); | arrayTemplate(GpuGlyph); | ||||||
| 
 | 
 | ||||||
| typedef struct { | typedef struct { | ||||||
|     float screen_size[2]; |     float screen_size[4]; | ||||||
| } GpuUniformParams; | } GpuUniformParams; | ||||||
| 
 | 
 | ||||||
| static struct { | static struct { | ||||||
|  |     sg_pass_action pass_action; | ||||||
|  |     sg_pipeline pip; | ||||||
|  |     sg_bindings bind; | ||||||
|  | 
 | ||||||
|  |     sg_image text_atlas_image; | ||||||
|  |     sg_sampler text_atlas_sampler; | ||||||
|  | 
 | ||||||
|  |     sg_shader_desc scratch_shader_desc; | ||||||
|  | 
 | ||||||
|     array(GpuUiRect) gpu_ui_rects; |     array(GpuUiRect) gpu_ui_rects; | ||||||
|     array(GpuGlyph) gpu_glyphs; |     array(GpuGlyph) gpu_glyphs; | ||||||
|     array(GpuGlyph) glyph_cache; |     array(GpuGlyph) glyph_cache; | ||||||
|  | @ -55,13 +86,13 @@ static struct { | ||||||
|     bool should_exit; |     bool should_exit; | ||||||
| 
 | 
 | ||||||
|     ui_context ui_cx; |     ui_context ui_cx; | ||||||
|     gfx_context_t *gfx_cx; |  | ||||||
| } state; | } state; | ||||||
| 
 | 
 | ||||||
| void queue_text(string text, float position[2]) { | void queue_text(string text, float position[2]) { | ||||||
|     float x = 0; |     float x = 0; | ||||||
|     for (size_t i=0; i < text.len; ++i) { |     for (size_t i=0; i < text.len; ++i) { | ||||||
|         if (text.data[i] >= 32) { |         if (text.data[i] >= 32) { | ||||||
|  |             //GpuGlyph glyph = *((GpuGlyph *)(state.glyph_cache.data+((text.data[i] - 32) * sizeof(GpuGlyph))));
 | ||||||
|             GpuGlyph glyph = state.glyph_cache.data[text.data[i] - 32]; |             GpuGlyph glyph = state.glyph_cache.data[text.data[i] - 32]; | ||||||
| 
 | 
 | ||||||
|             glyph.position[0] = x+position[0]; |             glyph.position[0] = x+position[0]; | ||||||
|  | @ -73,25 +104,43 @@ void queue_text(string text, float position[2]) { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ed_init(_gfx_frame_func frame_func) { | void vertex_shader_loaded(const sfetch_response_t *response) { | ||||||
|     state.gfx_cx = gfx_init_context(frame_func, 640, 480); |     if (response->fetched) { | ||||||
|     state.ui_cx = ui_init_context(); |         state.scratch_shader_desc.vs = (sg_shader_stage_desc) { | ||||||
|  |             .source = response->data.ptr, | ||||||
|  |             .entry = "vs_main", | ||||||
|  |             .uniform_blocks[0] = { | ||||||
|  |                 .size = sizeof(GpuUniformParams), | ||||||
|  |                 .layout = SG_UNIFORMLAYOUT_STD140, | ||||||
|  |                 .uniforms = { | ||||||
|  |                     [0] = { .name = "screen_size", .type = SG_UNIFORMTYPE_FLOAT4 }, | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  |     } else if (response->failed) { | ||||||
|  |         fprintf(stderr, "failed to load vertex shader\n"); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|     float vertices[] = { | void fragment_shader_loaded(const sfetch_response_t *response) { | ||||||
|         // positions       texture coords
 |     if (response->fetched) { | ||||||
|         -1.0f,  1.0f,       0.0f, 0.0f, |         state.scratch_shader_desc.fs = (sg_shader_stage_desc){ | ||||||
|          1.0f,  1.0f,       1.0f, 0.0f, |             .source = response->data.ptr, | ||||||
|          1.0f, -1.0f,       1.0f, 1.0f, |             .entry = "fs_main", | ||||||
|         -1.0f, -1.0f,       0.0f, 1.0f, |             .images[0].used = true, | ||||||
|     }; |             .samplers[0].used = true, | ||||||
|     const uint16_t indices[] = { 0, 1, 2,  0, 2, 3 }; |             .image_sampler_pairs[0] = { .glsl_name = "_group_0_binding_0_fs", .used = true, .image_slot = 0, .sampler_slot = 0 }, | ||||||
| 
 |         }; | ||||||
|     // NOTE: the order of these matter
 |         state.scratch_shader_desc.fs.source = response->data.ptr; | ||||||
|     gfx_push_vertex_buffer(state.gfx_cx, vertices, sizeof(vertices)); |         state.scratch_shader_desc.fs.entry = "fs_main"; | ||||||
|     gfx_push_vertex_buffer(state.gfx_cx, indices, sizeof(indices)); |     } else if (response->failed) { | ||||||
|     gfx_allocate_vertex_buffer(state.gfx_cx, state.gpu_glyphs.capacity * sizeof(GpuGlyph)); |         fprintf(stderr, "failed to load vertex shader\n"); | ||||||
|     gfx_allocate_vertex_buffer(state.gfx_cx, sizeof(GpuUniformParams)); |         exit(1); | ||||||
|  |     } | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|  | void ed_init() { | ||||||
|     uint8_t ttf_buffer[1<<20]; |     uint8_t ttf_buffer[1<<20]; | ||||||
|     // TODO: grab default font from the system
 |     // TODO: grab default font from the system
 | ||||||
|     FILE *ttf_file = fopen("./bin/JetBrainsMono-Medium.ttf", "rb"); |     FILE *ttf_file = fopen("./bin/JetBrainsMono-Medium.ttf", "rb"); | ||||||
|  | @ -105,6 +154,61 @@ void ed_init(_gfx_frame_func frame_func) { | ||||||
|     stbtt_fontinfo font; |     stbtt_fontinfo font; | ||||||
|     stbtt_InitFont(&font, ttf_buffer, stbtt_GetFontOffsetForIndex(ttf_buffer, 0)); |     stbtt_InitFont(&font, ttf_buffer, stbtt_GetFontOffsetForIndex(ttf_buffer, 0)); | ||||||
| 
 | 
 | ||||||
|  |     sg_setup(&(sg_desc) { | ||||||
|  |         .environment = sglue_environment(), | ||||||
|  |         .logger.func = slog_func, | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     float vertices[] = { | ||||||
|  |         // positions            texture coords
 | ||||||
|  |         -1.0f,  1.0f, 1.0f,     0.0f, 0.0f, | ||||||
|  |          1.0f,  1.0f, 1.0f,     1.0f, 0.0f, | ||||||
|  |          1.0f, -1.0f, 1.0f,     1.0f, 1.0f, | ||||||
|  |         -1.0f, -1.0f, 1.0f,     0.0f, 1.0f, | ||||||
|  |     }; | ||||||
|  |     state.bind.vertex_buffers[0] = sg_make_buffer(&(sg_buffer_desc) { | ||||||
|  |         .data = SG_RANGE(vertices) | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     const uint16_t indices[] = { 0, 1, 2,  0, 2, 3 }; | ||||||
|  |     state.bind.index_buffer = sg_make_buffer(&(sg_buffer_desc) { | ||||||
|  |         .type = SG_BUFFERTYPE_INDEXBUFFER, | ||||||
|  |         .data = SG_RANGE(indices) | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     sfetch_setup(&(sfetch_desc_t){ .logger.func = slog_func }); | ||||||
|  | 
 | ||||||
|  |     char vs_source[8000] = { 0 }; | ||||||
|  |     char fs_source[8000] = { 0 }; | ||||||
|  | 
 | ||||||
|  |     sfetch_handle_t vs_handle = sfetch_send(&(sfetch_request_t) { | ||||||
|  | #if defined (__APPLE__) | ||||||
|  |         .path = "./bin/transpiled_shaders/vertex.metal", | ||||||
|  | #elif defined (__linux__) || defined (__unix__) | ||||||
|  |         .path = "./shaders/vertex.vert", | ||||||
|  | #else | ||||||
|  | #error "Unsupported platform for shaders" | ||||||
|  | #endif | ||||||
|  |         .callback = vertex_shader_loaded, | ||||||
|  |         .buffer = { .ptr = vs_source, .size = sizeof(vs_source) }, | ||||||
|  |     }); | ||||||
|  |     sfetch_handle_t fs_handle = sfetch_send(&(sfetch_request_t) { | ||||||
|  | #if defined (__APPLE__) | ||||||
|  |         .path = "./bin/transpiled_shaders/fragment.metal", | ||||||
|  | #elif defined (__linux__) || defined (__unix__) | ||||||
|  |         .path = "./shaders/fragment.frag", | ||||||
|  | #else | ||||||
|  | #error "Unsupported platform for shaders" | ||||||
|  | #endif | ||||||
|  |         .callback = fragment_shader_loaded, | ||||||
|  |         .buffer = { .ptr = fs_source, .size = sizeof(fs_source) }, | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     // block until files are loaded
 | ||||||
|  |     while (sfetch_handle_valid(vs_handle) || sfetch_handle_valid(fs_handle)) { | ||||||
|  |         sfetch_dowork(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     const int font_bitmap_size = 1024; |     const int font_bitmap_size = 1024; | ||||||
|     const int rasterized_font_height = 64; |     const int rasterized_font_height = 64; | ||||||
|     uint8_t *font_bitmap = context_alloc(font_bitmap_size*font_bitmap_size * sizeof(uint8_t)); |     uint8_t *font_bitmap = context_alloc(font_bitmap_size*font_bitmap_size * sizeof(uint8_t)); | ||||||
|  | @ -163,49 +267,134 @@ void ed_init(_gfx_frame_func frame_func) { | ||||||
|         x += width; |         x += width; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     gfx_push_texture_buffer(state.gfx_cx, font_bitmap_size, font_bitmap_size, font_bitmap, font_bitmap_size*font_bitmap_size * sizeof(uint8_t)); |     state.bind.vertex_buffers[1] = sg_make_buffer(&(sg_buffer_desc) { | ||||||
|  |         .size = state.gpu_glyphs.capacity * sizeof(GpuGlyph), | ||||||
|  |         .usage = SG_USAGE_STREAM, | ||||||
|  |         .label = "glyph buffer" | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     state.text_atlas_sampler = sg_make_sampler(&(sg_sampler_desc) { .mag_filter = SG_FILTER_LINEAR }); | ||||||
|  |     state.text_atlas_image = sg_make_image(&(sg_image_desc) { | ||||||
|  |         .width = font_bitmap_size, | ||||||
|  |         .height = font_bitmap_size, | ||||||
|  |         .pixel_format = SG_PIXELFORMAT_R8, | ||||||
|  |         .data.subimage[0][0] = { .ptr = font_bitmap, .size = font_bitmap_size*font_bitmap_size * sizeof(uint8_t) }, | ||||||
|  |     }); | ||||||
|  |     state.bind.fs.images[0] = state.text_atlas_image; | ||||||
|  |     state.bind.fs.samplers[0] = state.text_atlas_sampler; | ||||||
|  | 
 | ||||||
|  |     state.pass_action = (sg_pass_action) { | ||||||
|  |         .colors[0] = { | ||||||
|  |             .load_action = SG_LOADACTION_CLEAR, | ||||||
|  |             .clear_value = { 0.0f, 0.0f, 0.0f, 1.0f } | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |     sg_shader shd = sg_make_shader(&state.scratch_shader_desc); | ||||||
|  |     state.pip = sg_make_pipeline(&(sg_pipeline_desc) { | ||||||
|  |         .shader = shd, | ||||||
|  |         .index_type = SG_INDEXTYPE_UINT16, | ||||||
|  |         .layout = { | ||||||
|  |             .buffers[1].step_func = SG_VERTEXSTEP_PER_INSTANCE, | ||||||
|  |             .attrs = { | ||||||
|  |                 [0] = { .format=SG_VERTEXFORMAT_FLOAT3, .buffer_index = 0 }, | ||||||
|  |                 [1] = { .format=SG_VERTEXFORMAT_FLOAT2, .buffer_index = 0 }, | ||||||
|  | 
 | ||||||
|  |                 [2] = { .format=SG_VERTEXFORMAT_FLOAT2, .buffer_index = 1 }, | ||||||
|  |                 [3] = { .format=SG_VERTEXFORMAT_FLOAT2, .buffer_index = 1 }, | ||||||
|  |                 [4] = { .format=SG_VERTEXFORMAT_FLOAT2, .buffer_index = 1 }, | ||||||
|  |                 [5] = { .format=SG_VERTEXFORMAT_FLOAT, .buffer_index = 1 }, | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     // queue_text(_String("But what even is text! []!@#$%^&*()_=+"), (float[]){ 0, 0 });
 | ||||||
|  |     // queue_text(_String("v0.1.0"), (float[]){ 32, 128 });
 | ||||||
|  |     // queue_text(_String("an_editor - what even"), (float[]){ 32, 256 });
 | ||||||
|  | 
 | ||||||
|  |     state.ui_cx = ui_init_context(); | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| void ed_frame() { | void ed_frame() { | ||||||
|     state.ui_cx.frame_elements.data[0].size.computed_size[0] = state.gfx_cx->frame_width; |     string label = _String("Number 1"); | ||||||
|     state.ui_cx.frame_elements.data[0].size.computed_size[1] = state.gfx_cx->frame_height; |     ui_element(&state.ui_cx, label); | ||||||
| 
 |  | ||||||
|     ui_element(&state.ui_cx, _String("Number 1")); |  | ||||||
|     ui_element(&state.ui_cx, _String("ui element 2")); |     ui_element(&state.ui_cx, _String("ui element 2")); | ||||||
|     ui_element(&state.ui_cx, _String("ui element 3")); |     ui_element(&state.ui_cx, _String("ui element 3")); | ||||||
| 
 | 
 | ||||||
|     ui_compute_layout(&state.ui_cx, 0); |     ui_compute_layout(&state.ui_cx, 0); | ||||||
|  |     state.ui_cx.frame_elements.data[0].size.computed_size[0] = sapp_width(); | ||||||
|  |     state.ui_cx.frame_elements.data[0].size.computed_size[1] = sapp_height(); | ||||||
|  |     ui_update_cache(&state.ui_cx, 0); | ||||||
| 
 | 
 | ||||||
|     state.gpu_glyphs.size = 0; |     state.gpu_glyphs.size = 0; | ||||||
|     for (size_t i = 0; i < state.ui_cx.frame_elements.size; ++i) { |     for (size_t i = 0; i < state.ui_cx.cached_elements.capacity; ++i) { | ||||||
|         string text = state.ui_cx.frame_elements.data[i].key; |         if (state.ui_cx.cached_elements.key_slots[i].key.data != NULL) { | ||||||
|         ui_element_frame_data *elm = &state.ui_cx.frame_elements.data[i]; |             string text = state.ui_cx.cached_elements.key_slots[i].key; | ||||||
| 
 | 
 | ||||||
|         queue_text(text, (float[]){ (float)elm->size.computed_pos[0], (float)elm->size.computed_pos[1] }); |             ui_element_cache_data *value = ht_get(&state.ui_cx.cached_elements, text); | ||||||
|  |             if (value) { | ||||||
|  |                 queue_text(text, (float[]){ (float)value->size.computed_pos[0], (float)value->size.computed_pos[1] }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     ui_update_cache(&state.ui_cx, 0); |  | ||||||
|     ui_prune(&state.ui_cx); |  | ||||||
| 
 |  | ||||||
|     if (state.gpu_glyphs.size > 0) { |     if (state.gpu_glyphs.size > 0) { | ||||||
|         gfx_update_buffer(state.gfx_cx, 2, state.gpu_glyphs.data, state.gpu_glyphs.size * sizeof(GpuGlyph)); |         sg_update_buffer(state.bind.vertex_buffers[1], &(sg_range) { | ||||||
|         fprintf(stderr, "updated glyph buffer: %zu\n", state.gpu_glyphs.size); |             .ptr = state.gpu_glyphs.data, | ||||||
|  |             .size = state.gpu_glyphs.size * sizeof(GpuGlyph) | ||||||
|  |         }); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     GpuUniformParams gpu_uniform_params = { |     GpuUniformParams gpu_uniform_params = { | ||||||
|         .screen_size = { |         .screen_size = { | ||||||
|             (float)state.gfx_cx->frame_width, |             sapp_widthf(), | ||||||
|             (float)state.gfx_cx->frame_height, |             sapp_heightf(), | ||||||
|  |             0, | ||||||
|  |             0 | ||||||
|         }, |         }, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     gfx_update_buffer(state.gfx_cx, 3, &gpu_uniform_params, sizeof(GpuUniformParams)); |     sg_begin_pass(&(sg_pass) { .action = state.pass_action, .swapchain = sglue_swapchain() }); | ||||||
|  |     { | ||||||
|  |         sg_apply_pipeline(state.pip); | ||||||
|  |         sg_apply_bindings(&state.bind); | ||||||
|  |         sg_apply_uniforms(SG_SHADERSTAGE_VS, 0, &(sg_range) { .ptr = &gpu_uniform_params, .size = sizeof(GpuUniformParams) }); | ||||||
|  |         sg_draw(0, 6, state.gpu_glyphs.size); | ||||||
|  |     } | ||||||
|  |     sg_end_pass(); | ||||||
|  |     sg_commit(); | ||||||
| } | } | ||||||
|  | void ed_cleanup() { | ||||||
|  |     sfetch_shutdown(); | ||||||
|  |     sg_shutdown(); | ||||||
|  | } | ||||||
|  | void ed_event(const sapp_event *event) { | ||||||
|  |     switch (event->type) { | ||||||
|  |         case SAPP_EVENTTYPE_MOUSE_DOWN: | ||||||
|  |             if (event->mouse_button == SAPP_MOUSEBUTTON_LEFT) { | ||||||
|  |                 sapp_lock_mouse(true); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
| 
 | 
 | ||||||
| int main(int argc, char* argv[]) { |         case SAPP_EVENTTYPE_MOUSE_UP: | ||||||
|     ed_init(ed_frame); |             if (event->mouse_button == SAPP_MOUSEBUTTON_LEFT) { | ||||||
|  |                 sapp_lock_mouse(false); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
| 
 | 
 | ||||||
|     while(keep_running) { |         default: | ||||||
|         gfx_run_events(state.gfx_cx); |             break; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | sapp_desc sokol_main(int argc, char *argv[]) { | ||||||
|  |     return (sapp_desc) { | ||||||
|  |         .width = 640, | ||||||
|  |         .height = 480, | ||||||
|  |         .init_cb = ed_init, | ||||||
|  |         .frame_cb = ed_frame, | ||||||
|  |         .cleanup_cb = ed_cleanup, | ||||||
|  |         .event_cb = ed_event, | ||||||
|  |         .icon.sokol_default = true, | ||||||
|  |         .logger.func = slog_func, | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										30
									
								
								src/ui.h
								
								
								
								
							
							
						
						
									
										30
									
								
								src/ui.h
								
								
								
								
							|  | @ -24,7 +24,6 @@ | ||||||
| #define _prev_ref(index) (_elm(_prev(index))) | #define _prev_ref(index) (_elm(_prev(index))) | ||||||
| #define _parent_ref(index) (_elm(_parent(index))) | #define _parent_ref(index) (_elm(_parent(index))) | ||||||
| 
 | 
 | ||||||
| #include <stdio.h> |  | ||||||
| #include <stddef.h> | #include <stddef.h> | ||||||
| #include <stdint.h> | #include <stdint.h> | ||||||
| #include <stdbool.h> | #include <stdbool.h> | ||||||
|  | @ -361,13 +360,6 @@ void ui_update_cache(ui_context *cx, size_t element_index) { | ||||||
|         do { |         do { | ||||||
|             __auto_type child = _elm(child_index); |             __auto_type child = _elm(child_index); | ||||||
| 
 | 
 | ||||||
|             size_t last_instantiated_index = cx->frame_index; |  | ||||||
| 
 |  | ||||||
|             ui_element_cache_data *cache; |  | ||||||
|             if ((cache = ht_get(&cx->cached_elements, child->key))) { |  | ||||||
|                 last_instantiated_index = cache->last_instantiated_index; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             ht_set(&cx->cached_elements, child->key, &(ui_element_cache_data) { |             ht_set(&cx->cached_elements, child->key, &(ui_element_cache_data) { | ||||||
|                 .label = child->label, |                 .label = child->label, | ||||||
|                 .size = { |                 .size = { | ||||||
|  | @ -375,8 +367,8 @@ void ui_update_cache(ui_context *cx, size_t element_index) { | ||||||
|                     .semantic_size = { child->size.semantic_size[0], child->size.semantic_size[1] }, |                     .semantic_size = { child->size.semantic_size[0], child->size.semantic_size[1] }, | ||||||
|                     .computed_size = { child->size.computed_size[0], child->size.computed_size[1] }, |                     .computed_size = { child->size.computed_size[0], child->size.computed_size[1] }, | ||||||
|                     .computed_pos = { child->size.computed_pos[0], child->size.computed_pos[1] }, |                     .computed_pos = { child->size.computed_pos[0], child->size.computed_pos[1] }, | ||||||
|                 }, |                 } | ||||||
|                 .last_instantiated_index = last_instantiated_index, |                 // FIXME: don't mangle last_instantiated_index
 | ||||||
|             }); |             }); | ||||||
| 
 | 
 | ||||||
|             ui_update_cache(cx, child_index); |             ui_update_cache(cx, child_index); | ||||||
|  | @ -392,23 +384,5 @@ void ui_update_cache(ui_context *cx, size_t element_index) { | ||||||
|     cx->current_parent = 0; |     cx->current_parent = 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void ui_prune(ui_context *cx) { |  | ||||||
|     for (size_t i = 0; i < cx->cached_elements.capacity; ++i) { |  | ||||||
|         if (cx->cached_elements.key_slots[i].key.data != NULL) { |  | ||||||
|             string key = cx->cached_elements.key_slots[i].key; |  | ||||||
| 
 |  | ||||||
|             // if this element hasn't been created in the past 5 frames, remove it
 |  | ||||||
|             ui_element_cache_data *cached = ht_get(&cx->cached_elements, key); |  | ||||||
|             if (cached && cached->last_instantiated_index < cx->frame_index-5) { |  | ||||||
|                 // fprintf(stderr, "removing %.*s from cache, cache index: %zu, frame index: %zu\n", (int)key.len, key.data, cached->last_instantiated_index, cx->frame_index);
 |  | ||||||
| 
 |  | ||||||
|                 ht_remove(&cx->cached_elements, key); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     cx->frame_index += 1; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif | #endif | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -0,0 +1,162 @@ | ||||||
|  | #if defined(SOKOL_IMPL) && !defined(SOKOL_GLUE_IMPL) | ||||||
|  | #define SOKOL_GLUE_IMPL | ||||||
|  | #endif | ||||||
|  | #ifndef SOKOL_GLUE_INCLUDED | ||||||
|  | /*
 | ||||||
|  |     sokol_glue.h -- glue helper functions for sokol headers | ||||||
|  | 
 | ||||||
|  |     Project URL: https://github.com/floooh/sokol
 | ||||||
|  | 
 | ||||||
|  |     Do this: | ||||||
|  |         #define SOKOL_IMPL or | ||||||
|  |         #define SOKOL_GLUE_IMPL | ||||||
|  |     before you include this file in *one* C or C++ file to create the | ||||||
|  |     implementation. | ||||||
|  | 
 | ||||||
|  |     ...optionally provide the following macros to override defaults: | ||||||
|  | 
 | ||||||
|  |     SOKOL_ASSERT(c)     - your own assert macro (default: assert(c)) | ||||||
|  |     SOKOL_GLUE_API_DECL - public function declaration prefix (default: extern) | ||||||
|  |     SOKOL_API_DECL      - same as SOKOL_GLUE_API_DECL | ||||||
|  |     SOKOL_API_IMPL      - public function implementation prefix (default: -) | ||||||
|  | 
 | ||||||
|  |     If sokol_glue.h is compiled as a DLL, define the following before | ||||||
|  |     including the declaration or implementation: | ||||||
|  | 
 | ||||||
|  |     SOKOL_DLL | ||||||
|  | 
 | ||||||
|  |     On Windows, SOKOL_DLL will define SOKOL_GLUE_API_DECL as __declspec(dllexport) | ||||||
|  |     or __declspec(dllimport) as needed. | ||||||
|  | 
 | ||||||
|  |     OVERVIEW | ||||||
|  |     ======== | ||||||
|  |     sokol_glue.h provides glue helper functions between sokol_gfx.h and sokol_app.h, | ||||||
|  |     so that sokol_gfx.h doesn't need to depend on sokol_app.h but can be | ||||||
|  |     used with different window system glue libraries. | ||||||
|  | 
 | ||||||
|  |     PROVIDED FUNCTIONS | ||||||
|  |     ================== | ||||||
|  | 
 | ||||||
|  |     sg_environment sglue_environment(void) | ||||||
|  | 
 | ||||||
|  |         Returns an sg_environment struct initialized by calling sokol_app.h | ||||||
|  |         functions. Use this in the sg_setup() call like this: | ||||||
|  | 
 | ||||||
|  |         sg_setup(&(sg_desc){ | ||||||
|  |             .environment = sglue_enviornment(), | ||||||
|  |             ... | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |     sg_swapchain sglue_swapchain(void) | ||||||
|  | 
 | ||||||
|  |         Returns an sg_swapchain struct initialized by calling sokol_app.h | ||||||
|  |         functions. Use this in sg_begin_pass() for a 'swapchain pass' like | ||||||
|  |         this: | ||||||
|  | 
 | ||||||
|  |         sg_begin_pass(&(sg_pass){ .swapchain = sglue_swapchain(), ... }); | ||||||
|  | 
 | ||||||
|  |     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_GLUE_INCLUDED | ||||||
|  | 
 | ||||||
|  | #if defined(SOKOL_API_DECL) && !defined(SOKOL_GLUE_API_DECL) | ||||||
|  | #define SOKOL_GLUE_API_DECL SOKOL_API_DECL | ||||||
|  | #endif | ||||||
|  | #ifndef SOKOL_GLUE_API_DECL | ||||||
|  | #if defined(_WIN32) && defined(SOKOL_DLL) && defined(SOKOL_GLUE_IMPL) | ||||||
|  | #define SOKOL_GLUE_API_DECL __declspec(dllexport) | ||||||
|  | #elif defined(_WIN32) && defined(SOKOL_DLL) | ||||||
|  | #define SOKOL_GLUE_API_DECL __declspec(dllimport) | ||||||
|  | #else | ||||||
|  | #define SOKOL_GLUE_API_DECL extern | ||||||
|  | #endif | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifndef SOKOL_GFX_INCLUDED | ||||||
|  | #error "Please include sokol_gfx.h before sokol_glue.h" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifdef __cplusplus | ||||||
|  | extern "C" { | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | SOKOL_GLUE_API_DECL sg_environment sglue_environment(void); | ||||||
|  | SOKOL_GLUE_API_DECL sg_swapchain sglue_swapchain(void); | ||||||
|  | 
 | ||||||
|  | #ifdef __cplusplus | ||||||
|  | } /* extern "C" */ | ||||||
|  | #endif | ||||||
|  | #endif /* SOKOL_GLUE_INCLUDED */ | ||||||
|  | 
 | ||||||
|  | /*-- IMPLEMENTATION ----------------------------------------------------------*/ | ||||||
|  | #ifdef SOKOL_GLUE_IMPL | ||||||
|  | #define SOKOL_GLUE_IMPL_INCLUDED (1) | ||||||
|  | #include <string.h> /* memset */ | ||||||
|  | 
 | ||||||
|  | #ifndef SOKOL_APP_INCLUDED | ||||||
|  | #error "Please include sokol_app.h before the sokol_glue.h implementation" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifndef SOKOL_API_IMPL | ||||||
|  | #define SOKOL_API_IMPL | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | SOKOL_API_IMPL sg_environment sglue_environment(void) { | ||||||
|  |     sg_environment env; | ||||||
|  |     memset(&env, 0, sizeof(env)); | ||||||
|  |     env.defaults.color_format = (sg_pixel_format) sapp_color_format(); | ||||||
|  |     env.defaults.depth_format = (sg_pixel_format) sapp_depth_format(); | ||||||
|  |     env.defaults.sample_count = sapp_sample_count(); | ||||||
|  |     env.metal.device = sapp_metal_get_device(); | ||||||
|  |     env.d3d11.device = sapp_d3d11_get_device(); | ||||||
|  |     env.d3d11.device_context = sapp_d3d11_get_device_context(); | ||||||
|  |     env.wgpu.device = sapp_wgpu_get_device(); | ||||||
|  |     return env; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | SOKOL_API_IMPL sg_swapchain sglue_swapchain(void) { | ||||||
|  |     sg_swapchain swapchain; | ||||||
|  |     memset(&swapchain, 0, sizeof(swapchain)); | ||||||
|  |     swapchain.width = sapp_width(); | ||||||
|  |     swapchain.height = sapp_height(); | ||||||
|  |     swapchain.sample_count = sapp_sample_count(); | ||||||
|  |     swapchain.color_format = (sg_pixel_format)sapp_color_format(); | ||||||
|  |     swapchain.depth_format = (sg_pixel_format)sapp_depth_format(); | ||||||
|  |     swapchain.metal.current_drawable = sapp_metal_get_current_drawable(); | ||||||
|  |     swapchain.metal.depth_stencil_texture = sapp_metal_get_depth_stencil_texture(); | ||||||
|  |     swapchain.metal.msaa_color_texture = sapp_metal_get_msaa_color_texture(); | ||||||
|  |     swapchain.d3d11.render_view = sapp_d3d11_get_render_view(); | ||||||
|  |     swapchain.d3d11.resolve_view = sapp_d3d11_get_resolve_view(); | ||||||
|  |     swapchain.d3d11.depth_stencil_view = sapp_d3d11_get_depth_stencil_view(); | ||||||
|  |     swapchain.wgpu.render_view = sapp_wgpu_get_render_view(); | ||||||
|  |     swapchain.wgpu.resolve_view = sapp_wgpu_get_resolve_view(); | ||||||
|  |     swapchain.wgpu.depth_stencil_view = sapp_wgpu_get_depth_stencil_view(); | ||||||
|  |     swapchain.gl.framebuffer = sapp_gl_get_framebuffer(); | ||||||
|  |     return swapchain; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif /* SOKOL_GLUE_IMPL */ | ||||||
|  | @ -0,0 +1,343 @@ | ||||||
|  | #if defined(SOKOL_IMPL) && !defined(SOKOL_LOG_IMPL) | ||||||
|  | #define SOKOL_LOG_IMPL | ||||||
|  | #endif | ||||||
|  | #ifndef SOKOL_LOG_INCLUDED | ||||||
|  | /*
 | ||||||
|  |     sokol_log.h -- common logging callback for sokol headers | ||||||
|  | 
 | ||||||
|  |     Project URL: https://github.com/floooh/sokol
 | ||||||
|  | 
 | ||||||
|  |     Example code: https://github.com/floooh/sokol-samples
 | ||||||
|  | 
 | ||||||
|  |     Do this: | ||||||
|  |         #define SOKOL_IMPL or | ||||||
|  |         #define SOKOL_LOG_IMPL | ||||||
|  |     before you include this file in *one* C or C++ file to create the | ||||||
|  |     implementation. | ||||||
|  | 
 | ||||||
|  |     Optionally provide the following defines when building the implementation: | ||||||
|  | 
 | ||||||
|  |     SOKOL_ASSERT(c)             - your own assert macro (default: assert(c)) | ||||||
|  |     SOKOL_UNREACHABLE()         - a guard macro for unreachable code (default: assert(false)) | ||||||
|  |     SOKOL_LOG_API_DECL          - public function declaration prefix (default: extern) | ||||||
|  |     SOKOL_API_DECL              - same as SOKOL_GFX_API_DECL | ||||||
|  |     SOKOL_API_IMPL              - public function implementation prefix (default: -) | ||||||
|  | 
 | ||||||
|  |     Optionally define the following for verbose output: | ||||||
|  | 
 | ||||||
|  |     SOKOL_DEBUG         - by default this is defined if _DEBUG is defined | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     OVERVIEW | ||||||
|  |     ======== | ||||||
|  |     sokol_log.h provides a default logging callback for other sokol headers. | ||||||
|  | 
 | ||||||
|  |     To use the default log callback, just include sokol_log.h and provide | ||||||
|  |     a function pointer to the 'slog_func' function when setting up the | ||||||
|  |     sokol library: | ||||||
|  | 
 | ||||||
|  |     For instance with sokol_audio.h: | ||||||
|  | 
 | ||||||
|  |         #include "sokol_log.h" | ||||||
|  |         ... | ||||||
|  |         saudio_setup(&(saudio_desc){ .logger.func = slog_func }); | ||||||
|  | 
 | ||||||
|  |     Logging output goes to stderr and/or a platform specific logging subsystem | ||||||
|  |     (which means that in some scenarios you might see logging messages duplicated): | ||||||
|  | 
 | ||||||
|  |         - Windows: stderr + OutputDebugStringA() | ||||||
|  |         - macOS/iOS/Linux: stderr + syslog() | ||||||
|  |         - Emscripten: console.info()/warn()/error() | ||||||
|  |         - Android: __android_log_write() | ||||||
|  | 
 | ||||||
|  |     On Windows with sokol_app.h also note the runtime config items to make | ||||||
|  |     stdout/stderr output visible on the console for WinMain() applications | ||||||
|  |     via sapp_desc.win32_console_attach or sapp_desc.win32_console_create, | ||||||
|  |     however when running in a debugger on Windows, the logging output should | ||||||
|  |     show up on the debug output UI panel. | ||||||
|  | 
 | ||||||
|  |     In debug mode, a log message might look like this: | ||||||
|  | 
 | ||||||
|  |         [sspine][error][id:12] /Users/floh/projects/sokol/util/sokol_spine.h:3472:0: | ||||||
|  |             SKELETON_DESC_NO_ATLAS: no atlas object provided in sspine_skeleton_desc.atlas | ||||||
|  | 
 | ||||||
|  |     The source path and line number is formatted like compiler errors, in some IDEs (like VSCode) | ||||||
|  |     such error messages are clickable. | ||||||
|  | 
 | ||||||
|  |     In release mode, logging is less verbose as to not bloat the executable with string data, but you still get | ||||||
|  |     enough information to identify the type and location of an error: | ||||||
|  | 
 | ||||||
|  |         [sspine][error][id:12][line:3472] | ||||||
|  | 
 | ||||||
|  |     RULES FOR WRITING YOUR OWN LOGGING FUNCTION | ||||||
|  |     =========================================== | ||||||
|  |     - must be re-entrant because it might be called from different threads | ||||||
|  |     - must treat **all** provided string pointers as optional (can be null) | ||||||
|  |     - don't store the string pointers, copy the string data instead | ||||||
|  |     - must not return for log level panic | ||||||
|  | 
 | ||||||
|  |     LICENSE | ||||||
|  |     ======= | ||||||
|  |     zlib/libpng license | ||||||
|  | 
 | ||||||
|  |     Copyright (c) 2023 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_LOG_INCLUDED (1) | ||||||
|  | #include <stdint.h> | ||||||
|  | 
 | ||||||
|  | #if defined(SOKOL_API_DECL) && !defined(SOKOL_LOG_API_DECL) | ||||||
|  | #define SOKOL_LOG_API_DECL SOKOL_API_DECL | ||||||
|  | #endif | ||||||
|  | #ifndef SOKOL_LOG_API_DECL | ||||||
|  | #if defined(_WIN32) && defined(SOKOL_DLL) && defined(SOKOL_LOG_IMPL) | ||||||
|  | #define SOKOL_LOG_API_DECL __declspec(dllexport) | ||||||
|  | #elif defined(_WIN32) && defined(SOKOL_DLL) | ||||||
|  | #define SOKOL_LOG_API_DECL __declspec(dllimport) | ||||||
|  | #else | ||||||
|  | #define SOKOL_LOG_API_DECL extern | ||||||
|  | #endif | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifdef __cplusplus | ||||||
|  | extern "C" { | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |     Plug this function into the 'logger.func' struct item when initializing any of the sokol | ||||||
|  |     headers. For instance for sokol_audio.h it would loom like this: | ||||||
|  | 
 | ||||||
|  |     saudio_setup(&(saudio_desc){ | ||||||
|  |         .logger = { | ||||||
|  |             .func = slog_func | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | */ | ||||||
|  | SOKOL_LOG_API_DECL void slog_func(const char* tag, uint32_t log_level, uint32_t log_item, const char* message, uint32_t line_nr, const char* filename, void* user_data); | ||||||
|  | 
 | ||||||
|  | #ifdef __cplusplus | ||||||
|  | } // extern "C"
 | ||||||
|  | #endif | ||||||
|  | #endif // SOKOL_LOG_INCLUDED
 | ||||||
|  | 
 | ||||||
|  | // ██ ███    ███ ██████  ██      ███████ ███    ███ ███████ ███    ██ ████████  █████  ████████ ██  ██████  ███    ██
 | ||||||
|  | // ██ ████  ████ ██   ██ ██      ██      ████  ████ ██      ████   ██    ██    ██   ██    ██    ██ ██    ██ ████   ██
 | ||||||
|  | // ██ ██ ████ ██ ██████  ██      █████   ██ ████ ██ █████   ██ ██  ██    ██    ███████    ██    ██ ██    ██ ██ ██  ██
 | ||||||
|  | // ██ ██  ██  ██ ██      ██      ██      ██  ██  ██ ██      ██  ██ ██    ██    ██   ██    ██    ██ ██    ██ ██  ██ ██
 | ||||||
|  | // ██ ██      ██ ██      ███████ ███████ ██      ██ ███████ ██   ████    ██    ██   ██    ██    ██  ██████  ██   ████
 | ||||||
|  | //
 | ||||||
|  | // >>implementation
 | ||||||
|  | #ifdef SOKOL_LOG_IMPL | ||||||
|  | #define SOKOL_LOG_IMPL_INCLUDED (1) | ||||||
|  | 
 | ||||||
|  | #ifndef SOKOL_API_IMPL | ||||||
|  |     #define SOKOL_API_IMPL | ||||||
|  | #endif | ||||||
|  | #ifndef SOKOL_DEBUG | ||||||
|  |     #ifndef NDEBUG | ||||||
|  |         #define SOKOL_DEBUG | ||||||
|  |     #endif | ||||||
|  | #endif | ||||||
|  | #ifndef SOKOL_ASSERT | ||||||
|  |     #include <assert.h> | ||||||
|  |     #define SOKOL_ASSERT(c) assert(c) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifndef _SOKOL_PRIVATE | ||||||
|  |     #if defined(__GNUC__) || defined(__clang__) | ||||||
|  |         #define _SOKOL_PRIVATE __attribute__((unused)) static | ||||||
|  |     #else | ||||||
|  |         #define _SOKOL_PRIVATE static | ||||||
|  |     #endif | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifndef _SOKOL_UNUSED | ||||||
|  |     #define _SOKOL_UNUSED(x) (void)(x) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | // platform detection
 | ||||||
|  | #if defined(__APPLE__) | ||||||
|  |     #define _SLOG_APPLE (1) | ||||||
|  | #elif defined(__EMSCRIPTEN__) | ||||||
|  |     #define _SLOG_EMSCRIPTEN (1) | ||||||
|  | #elif defined(_WIN32) | ||||||
|  |     #define _SLOG_WINDOWS (1) | ||||||
|  | #elif defined(__ANDROID__) | ||||||
|  |     #define _SLOG_ANDROID (1) | ||||||
|  | #elif defined(__linux__) || defined(__unix__) | ||||||
|  |     #define _SLOG_LINUX (1) | ||||||
|  | #else | ||||||
|  | #error "sokol_log.h: unknown platform" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #include <stdlib.h> // abort | ||||||
|  | #include <stdio.h>  // fputs | ||||||
|  | #include <stddef.h> // size_t | ||||||
|  | 
 | ||||||
|  | #if defined(_SLOG_EMSCRIPTEN) | ||||||
|  | #include <emscripten/emscripten.h> | ||||||
|  | #elif defined(_SLOG_WINDOWS) | ||||||
|  | #ifndef WIN32_LEAN_AND_MEAN | ||||||
|  |     #define WIN32_LEAN_AND_MEAN | ||||||
|  | #endif | ||||||
|  | #ifndef NOMINMAX | ||||||
|  |     #define NOMINMAX | ||||||
|  | #endif | ||||||
|  | #include <windows.h> | ||||||
|  | #elif defined(_SLOG_ANDROID) | ||||||
|  | #include <android/log.h> | ||||||
|  | #elif defined(_SLOG_LINUX) || defined(_SLOG_APPLE) | ||||||
|  | #include <syslog.h> | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | // size of line buffer (on stack!) in bytes including terminating zero
 | ||||||
|  | #define _SLOG_LINE_LENGTH (512) | ||||||
|  | 
 | ||||||
|  | _SOKOL_PRIVATE char* _slog_append(const char* str, char* dst, char* end) { | ||||||
|  |     if (str) { | ||||||
|  |         char c; | ||||||
|  |         while (((c = *str++) != 0) && (dst < (end - 1))) { | ||||||
|  |             *dst++ = c; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     *dst = 0; | ||||||
|  |     return dst; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | _SOKOL_PRIVATE char* _slog_itoa(uint32_t x, char* buf, size_t buf_size) { | ||||||
|  |     const size_t max_digits_and_null = 11; | ||||||
|  |     if (buf_size < max_digits_and_null) { | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  |     char* p = buf + max_digits_and_null; | ||||||
|  |     *--p = 0; | ||||||
|  |     do { | ||||||
|  |         *--p = '0' + (x % 10); | ||||||
|  |         x /= 10; | ||||||
|  |     } while (x != 0); | ||||||
|  |     return p; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #if defined(_SLOG_EMSCRIPTEN) | ||||||
|  | EM_JS(void, slog_js_log, (uint32_t level, const char* c_str), { | ||||||
|  |     const str = UTF8ToString(c_str); | ||||||
|  |     switch (level) { | ||||||
|  |         case 0: console.error(str); break; | ||||||
|  |         case 1: console.error(str); break; | ||||||
|  |         case 2: console.warn(str); break; | ||||||
|  |         default: console.info(str); break; | ||||||
|  |     } | ||||||
|  | }); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | SOKOL_API_IMPL void slog_func(const char* tag, uint32_t log_level, uint32_t log_item, const char* message, uint32_t line_nr, const char* filename, void* user_data) { | ||||||
|  |     _SOKOL_UNUSED(user_data); | ||||||
|  | 
 | ||||||
|  |     const char* log_level_str; | ||||||
|  |     switch (log_level) { | ||||||
|  |         case 0: log_level_str = "panic"; break; | ||||||
|  |         case 1: log_level_str = "error"; break; | ||||||
|  |         case 2: log_level_str = "warning"; break; | ||||||
|  |         default: log_level_str = "info"; break; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // build log output line
 | ||||||
|  |     char line_buf[_SLOG_LINE_LENGTH]; | ||||||
|  |     char* str = line_buf; | ||||||
|  |     char* end = line_buf + sizeof(line_buf); | ||||||
|  |     char num_buf[32]; | ||||||
|  |     if (tag) { | ||||||
|  |         str = _slog_append("[", str, end); | ||||||
|  |         str = _slog_append(tag, str, end); | ||||||
|  |         str = _slog_append("]", str, end); | ||||||
|  |     } | ||||||
|  |     str = _slog_append("[", str, end); | ||||||
|  |     str = _slog_append(log_level_str, str, end); | ||||||
|  |     str = _slog_append("]", str, end); | ||||||
|  |     str = _slog_append("[id:", str, end); | ||||||
|  |     str = _slog_append(_slog_itoa(log_item, num_buf, sizeof(num_buf)), str, end); | ||||||
|  |     str = _slog_append("]", str, end); | ||||||
|  |     // if a filename is provided, build a clickable log message that's compatible with compiler error messages
 | ||||||
|  |     if (filename) { | ||||||
|  |         str = _slog_append(" ", str, end); | ||||||
|  |         #if defined(_MSC_VER) | ||||||
|  |             // MSVC compiler error format
 | ||||||
|  |             str = _slog_append(filename, str, end); | ||||||
|  |             str = _slog_append("(", str, end); | ||||||
|  |             str = _slog_append(_slog_itoa(line_nr, num_buf, sizeof(num_buf)), str, end); | ||||||
|  |             str = _slog_append("): ", str, end); | ||||||
|  |         #else | ||||||
|  |             // gcc/clang compiler error format
 | ||||||
|  |             str = _slog_append(filename, str, end); | ||||||
|  |             str = _slog_append(":", str, end); | ||||||
|  |             str = _slog_append(_slog_itoa(line_nr, num_buf, sizeof(num_buf)), str, end); | ||||||
|  |             str = _slog_append(":0: ", str, end); | ||||||
|  |         #endif | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |         str = _slog_append("[line:", str, end); | ||||||
|  |         str = _slog_append(_slog_itoa(line_nr, num_buf, sizeof(num_buf)), str, end); | ||||||
|  |         str = _slog_append("] ", str, end); | ||||||
|  |     } | ||||||
|  |     if (message) { | ||||||
|  |         str = _slog_append("\n\t", str, end); | ||||||
|  |         str = _slog_append(message, str, end); | ||||||
|  |     } | ||||||
|  |     str = _slog_append("\n\n", str, end); | ||||||
|  |     if (0 == log_level) { | ||||||
|  |         str = _slog_append("ABORTING because of [panic]\n", str, end); | ||||||
|  |         (void)str; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // print to stderr?
 | ||||||
|  |     #if defined(_SLOG_LINUX) || defined(_SLOG_WINDOWS) || defined(_SLOG_APPLE) | ||||||
|  |         fputs(line_buf, stderr); | ||||||
|  |     #endif | ||||||
|  | 
 | ||||||
|  |     // platform specific logging calls
 | ||||||
|  |     #if defined(_SLOG_WINDOWS) | ||||||
|  |         OutputDebugStringA(line_buf); | ||||||
|  |     #elif defined(_SLOG_ANDROID) | ||||||
|  |         int prio; | ||||||
|  |         switch (log_level) { | ||||||
|  |             case 0: prio = ANDROID_LOG_FATAL; break; | ||||||
|  |             case 1: prio = ANDROID_LOG_ERROR; break; | ||||||
|  |             case 2: prio = ANDROID_LOG_WARN; break; | ||||||
|  |             default: prio = ANDROID_LOG_INFO; break; | ||||||
|  |         } | ||||||
|  |         __android_log_write(prio, "SOKOL", line_buf); | ||||||
|  |     #elif defined(_SLOG_EMSCRIPTEN) | ||||||
|  |         slog_js_log(log_level, line_buf); | ||||||
|  |     #elif defined(_SLOG_LINUX) || defined(_SLOG_APPLE) | ||||||
|  |         int prio; | ||||||
|  |         switch (log_level) { | ||||||
|  |             case 0: prio = LOG_CRIT; break; | ||||||
|  |             case 1: prio = LOG_ERR; break; | ||||||
|  |             case 2: prio = LOG_WARNING; break; | ||||||
|  |             default: prio = LOG_INFO; break; | ||||||
|  |         } | ||||||
|  |         syslog(prio, "%s", line_buf); | ||||||
|  |     #endif | ||||||
|  |     if (0 == log_level) { | ||||||
|  |         abort(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | #endif // SOKOL_LOG_IMPL
 | ||||||
		Loading…
	
		Reference in New Issue