[graphics] Allocate canvases and fonts from graphics subsystem's arena
This commit is contained in:
parent
6bf989564d
commit
28f5af32fa
|
@ -114,14 +114,13 @@ bool mg_egl_surface_get_hidden(mg_surface_data* interface)
|
|||
return(mp_layer_get_hidden(&surface->layer));
|
||||
}
|
||||
|
||||
mg_surface mg_egl_surface_create_for_window(mp_window window)
|
||||
mg_surface_data* mg_egl_surface_create_for_window(mp_window window)
|
||||
{
|
||||
mg_surface res = mg_surface_nil();
|
||||
|
||||
mg_egl_surface* surface = 0;
|
||||
mp_window_data* windowData = mp_window_ptr_from_handle(window);
|
||||
if(windowData)
|
||||
{
|
||||
mg_egl_surface* surface = malloc_type(mg_egl_surface);
|
||||
surface = malloc_type(mg_egl_surface);
|
||||
memset(surface, 0, sizeof(mg_egl_surface));
|
||||
|
||||
surface->interface.backend = MG_BACKEND_GLES;
|
||||
|
@ -182,8 +181,6 @@ mg_surface mg_egl_surface_create_for_window(mp_window window)
|
|||
mg_gl_load_gles(&surface->api, (mg_gl_load_proc)eglGetProcAddress);
|
||||
|
||||
eglSwapInterval(surface->eglDisplay, 1);
|
||||
|
||||
res = mg_surface_handle_alloc((mg_surface_data*)surface);
|
||||
}
|
||||
return(res);
|
||||
return((mg_surface_data*)surface);
|
||||
}
|
||||
|
|
|
@ -9,9 +9,9 @@
|
|||
#ifndef __EGL_SURFACE_H_
|
||||
#define __EGL_SURFACE_H_
|
||||
|
||||
#include"graphics.h"
|
||||
#include"graphics_internal.h"
|
||||
#include"mp_app.h"
|
||||
|
||||
mg_surface mg_egl_surface_create_for_window(mp_window window);
|
||||
mg_surface_data* mg_egl_surface_create_for_window(mp_window window);
|
||||
|
||||
#endif // __EGL_SURFACE_H_
|
||||
|
|
153
src/graphics.c
153
src/graphics.c
|
@ -120,12 +120,6 @@ typedef struct mg_glyph_data
|
|||
|
||||
} mg_glyph_data;
|
||||
|
||||
enum
|
||||
{
|
||||
MG_STREAM_MAX_COUNT = 128,
|
||||
MG_IMAGE_MAX_COUNT = 128
|
||||
};
|
||||
|
||||
enum
|
||||
{
|
||||
MG_MATRIX_STACK_MAX_DEPTH = 64,
|
||||
|
@ -137,7 +131,6 @@ enum
|
|||
typedef struct mg_font_data
|
||||
{
|
||||
list_elt freeListElt;
|
||||
u32 generation;
|
||||
|
||||
u32 rangeCount;
|
||||
u32 glyphCount;
|
||||
|
@ -180,14 +173,15 @@ typedef struct mg_data
|
|||
mem_arena handleArena;
|
||||
list_info handleFreeList;
|
||||
|
||||
mem_arena resourceArena;
|
||||
list_info canvasFreeList;
|
||||
list_info fontFreeList;
|
||||
|
||||
} mg_data;
|
||||
|
||||
typedef struct mg_canvas_data
|
||||
{
|
||||
list_elt freeListElt;
|
||||
u32 generation;
|
||||
|
||||
u64 frameCounter;
|
||||
|
||||
mg_attributes attributes;
|
||||
bool textFlip;
|
||||
|
@ -206,12 +200,6 @@ typedef struct mg_canvas_data
|
|||
u32 primitiveCount;
|
||||
mg_primitive primitives[MG_MAX_PRIMITIVE_COUNT];
|
||||
|
||||
/*
|
||||
vec2 atlasPos;
|
||||
u32 atlasLineHeight;
|
||||
mg_image blankImage;
|
||||
*/
|
||||
|
||||
//NOTE: these are used at render time
|
||||
mp_rect clip;
|
||||
mg_mat2x3 transform;
|
||||
|
@ -235,9 +223,8 @@ void mg_init()
|
|||
if(!__mgData.init)
|
||||
{
|
||||
mem_arena_init(&__mgData.handleArena);
|
||||
mem_arena_init(&__mgData.resourceArena);
|
||||
__mgData.init = true;
|
||||
|
||||
//...
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -434,7 +421,12 @@ bool mg_surface_is_nil(mg_surface surface) { return(surface.h == 0); }
|
|||
|
||||
mg_surface mg_surface_create_for_window(mp_window window, mg_backend_id backend)
|
||||
{
|
||||
mg_surface surface = mg_surface_nil();
|
||||
if(__mgData.init)
|
||||
{
|
||||
mg_init();
|
||||
}
|
||||
mg_surface surfaceHandle = mg_surface_nil();
|
||||
mg_surface_data* surface = 0;
|
||||
|
||||
switch(backend)
|
||||
{
|
||||
|
@ -459,7 +451,11 @@ mg_surface mg_surface_create_for_window(mp_window window, mg_backend_id backend)
|
|||
default:
|
||||
break;
|
||||
}
|
||||
return(surface);
|
||||
if(surface)
|
||||
{
|
||||
surfaceHandle = mg_surface_handle_alloc(surface);
|
||||
}
|
||||
return(surfaceHandle);
|
||||
}
|
||||
|
||||
void mg_surface_destroy(mg_surface handle)
|
||||
|
@ -2166,53 +2162,66 @@ bool mg_font_is_nil(mg_font font) { return(font.h == 0); }
|
|||
|
||||
mg_font mg_font_create_from_memory(u32 size, byte* buffer, u32 rangeCount, unicode_range* ranges)
|
||||
{
|
||||
mg_font_data* fontData = malloc_type(mg_font_data);
|
||||
mg_font font = mg_font_handle_alloc(fontData);
|
||||
if(!__mgData.init)
|
||||
{
|
||||
mg_init();
|
||||
}
|
||||
mg_font fontHandle = mg_font_nil();
|
||||
|
||||
mg_font_data* font = ListPopEntry(&__mgData.fontFreeList, mg_font_data, freeListElt);
|
||||
if(!font)
|
||||
{
|
||||
font = mem_arena_alloc_type(&__mgData.resourceArena, mg_font_data);
|
||||
}
|
||||
if(font)
|
||||
{
|
||||
memset(font, 0, sizeof(mg_font_data));
|
||||
fontHandle = mg_font_handle_alloc(font);
|
||||
|
||||
stbtt_fontinfo stbttFontInfo;
|
||||
stbtt_InitFont(&stbttFontInfo, buffer, 0);
|
||||
|
||||
//NOTE(martin): load font metrics data
|
||||
fontData->unitsPerEm = 1./stbtt_ScaleForMappingEmToPixels(&stbttFontInfo, 1);
|
||||
font->unitsPerEm = 1./stbtt_ScaleForMappingEmToPixels(&stbttFontInfo, 1);
|
||||
|
||||
int ascent, descent, lineGap, x0, x1, y0, y1;
|
||||
stbtt_GetFontVMetrics(&stbttFontInfo, &ascent, &descent, &lineGap);
|
||||
stbtt_GetFontBoundingBox(&stbttFontInfo, &x0, &y0, &x1, &y1);
|
||||
|
||||
fontData->extents.ascent = ascent;
|
||||
fontData->extents.descent = -descent;
|
||||
fontData->extents.leading = lineGap;
|
||||
fontData->extents.width = x1 - x0;
|
||||
font->extents.ascent = ascent;
|
||||
font->extents.descent = -descent;
|
||||
font->extents.leading = lineGap;
|
||||
font->extents.width = x1 - x0;
|
||||
|
||||
stbtt_GetCodepointBox(&stbttFontInfo, 'x', &x0, &y0, &x1, &y1);
|
||||
fontData->extents.xHeight = y1 - y0;
|
||||
font->extents.xHeight = y1 - y0;
|
||||
|
||||
stbtt_GetCodepointBox(&stbttFontInfo, 'M', &x0, &y0, &x1, &y1);
|
||||
fontData->extents.capHeight = y1 - y0;
|
||||
font->extents.capHeight = y1 - y0;
|
||||
|
||||
//NOTE(martin): load codepoint ranges
|
||||
fontData->rangeCount = rangeCount;
|
||||
fontData->glyphMap = malloc_array(mg_glyph_map_entry, rangeCount);
|
||||
fontData->glyphCount = 0;
|
||||
font->rangeCount = rangeCount;
|
||||
font->glyphMap = malloc_array(mg_glyph_map_entry, rangeCount);
|
||||
font->glyphCount = 0;
|
||||
|
||||
for(int i=0; i<rangeCount; i++)
|
||||
{
|
||||
//NOTE(martin): initialize the map entry.
|
||||
// The glyph indices are offseted by 1, to reserve 0 as an invalid glyph index.
|
||||
fontData->glyphMap[i].range = ranges[i];
|
||||
fontData->glyphMap[i].firstGlyphIndex = fontData->glyphCount + 1;
|
||||
fontData->glyphCount += ranges[i].count;
|
||||
font->glyphMap[i].range = ranges[i];
|
||||
font->glyphMap[i].firstGlyphIndex = font->glyphCount + 1;
|
||||
font->glyphCount += ranges[i].count;
|
||||
}
|
||||
|
||||
fontData->glyphs = malloc_array(mg_glyph_data, fontData->glyphCount);
|
||||
font->glyphs = malloc_array(mg_glyph_data, font->glyphCount);
|
||||
|
||||
//NOTE(martin): first do a count of outlines
|
||||
int outlineCount = 0;
|
||||
for(int rangeIndex=0; rangeIndex<rangeCount; rangeIndex++)
|
||||
{
|
||||
utf32 codePoint = fontData->glyphMap[rangeIndex].range.firstCodePoint;
|
||||
u32 firstGlyphIndex = fontData->glyphMap[rangeIndex].firstGlyphIndex;
|
||||
u32 endGlyphIndex = firstGlyphIndex + fontData->glyphMap[rangeIndex].range.count;
|
||||
utf32 codePoint = font->glyphMap[rangeIndex].range.firstCodePoint;
|
||||
u32 firstGlyphIndex = font->glyphMap[rangeIndex].firstGlyphIndex;
|
||||
u32 endGlyphIndex = firstGlyphIndex + font->glyphMap[rangeIndex].range.count;
|
||||
|
||||
for(int glyphIndex = firstGlyphIndex;
|
||||
glyphIndex < endGlyphIndex; glyphIndex++)
|
||||
|
@ -2232,20 +2241,20 @@ mg_font mg_font_create_from_memory(u32 size, byte* buffer, u32 rangeCount, unico
|
|||
}
|
||||
}
|
||||
//NOTE(martin): allocate outlines
|
||||
fontData->outlines = malloc_array(mg_path_elt, outlineCount);
|
||||
fontData->outlineCount = 0;
|
||||
font->outlines = malloc_array(mg_path_elt, outlineCount);
|
||||
font->outlineCount = 0;
|
||||
|
||||
//NOTE(martin): load metrics and outlines
|
||||
for(int rangeIndex=0; rangeIndex<rangeCount; rangeIndex++)
|
||||
{
|
||||
utf32 codePoint = fontData->glyphMap[rangeIndex].range.firstCodePoint;
|
||||
u32 firstGlyphIndex = fontData->glyphMap[rangeIndex].firstGlyphIndex;
|
||||
u32 endGlyphIndex = firstGlyphIndex + fontData->glyphMap[rangeIndex].range.count;
|
||||
utf32 codePoint = font->glyphMap[rangeIndex].range.firstCodePoint;
|
||||
u32 firstGlyphIndex = font->glyphMap[rangeIndex].firstGlyphIndex;
|
||||
u32 endGlyphIndex = firstGlyphIndex + font->glyphMap[rangeIndex].range.count;
|
||||
|
||||
for(int glyphIndex = firstGlyphIndex;
|
||||
glyphIndex < endGlyphIndex; glyphIndex++)
|
||||
{
|
||||
mg_glyph_data* glyph = &(fontData->glyphs[glyphIndex-1]);
|
||||
mg_glyph_data* glyph = &(font->glyphs[glyphIndex-1]);
|
||||
|
||||
int stbttGlyphIndex = stbtt_FindGlyphIndex(&stbttFontInfo, codePoint);
|
||||
if(stbttGlyphIndex == 0)
|
||||
|
@ -2277,12 +2286,12 @@ mg_font mg_font_create_from_memory(u32 size, byte* buffer, u32 rangeCount, unico
|
|||
stbtt_vertex* vertices = 0;
|
||||
int vertexCount = stbtt_GetGlyphShape(&stbttFontInfo, stbttGlyphIndex, &vertices);
|
||||
|
||||
glyph->pathDescriptor = (mg_path_descriptor){.startIndex = fontData->outlineCount,
|
||||
glyph->pathDescriptor = (mg_path_descriptor){.startIndex = font->outlineCount,
|
||||
.count = vertexCount,
|
||||
.startPoint = {0, 0}};
|
||||
|
||||
mg_path_elt* elements = fontData->outlines + fontData->outlineCount;
|
||||
fontData->outlineCount += vertexCount;
|
||||
mg_path_elt* elements = font->outlines + font->outlineCount;
|
||||
font->outlineCount += vertexCount;
|
||||
vec2 currentPos = {0, 0};
|
||||
|
||||
for(int vertIndex = 0; vertIndex < vertexCount; vertIndex++)
|
||||
|
@ -2326,7 +2335,8 @@ mg_font mg_font_create_from_memory(u32 size, byte* buffer, u32 rangeCount, unico
|
|||
codePoint++;
|
||||
}
|
||||
}
|
||||
return(font);
|
||||
}
|
||||
return(fontHandle);
|
||||
}
|
||||
|
||||
void mg_font_destroy(mg_font fontHandle)
|
||||
|
@ -2337,7 +2347,8 @@ void mg_font_destroy(mg_font fontHandle)
|
|||
free(fontData->glyphMap);
|
||||
free(fontData->glyphs);
|
||||
free(fontData->outlines);
|
||||
free(fontData);
|
||||
|
||||
ListPush(&__mgData.fontFreeList, &fontData->freeListElt);
|
||||
mg_handle_recycle(fontHandle.h);
|
||||
}
|
||||
}
|
||||
|
@ -2598,7 +2609,6 @@ mp_rect mg_text_bounding_box(mg_font font, f32 fontSize, str8 text)
|
|||
mg_canvas mg_canvas_nil() { return((mg_canvas){.h = 0}); }
|
||||
bool mg_canvas_is_nil(mg_canvas canvas) { return(canvas.h == 0); }
|
||||
|
||||
|
||||
#if MG_COMPILE_BACKEND_METAL
|
||||
mg_canvas_backend* mg_mtl_canvas_create(mg_surface surface);
|
||||
#endif
|
||||
|
@ -2609,7 +2619,7 @@ bool mg_canvas_is_nil(mg_canvas canvas) { return(canvas.h == 0); }
|
|||
|
||||
mg_canvas mg_canvas_create(mg_surface surface)
|
||||
{
|
||||
mg_canvas canvas = mg_canvas_nil();
|
||||
mg_canvas canvasHandle = mg_canvas_nil();
|
||||
mg_surface_data* surfaceData = mg_surface_data_from_handle(surface);
|
||||
if(surfaceData)
|
||||
{
|
||||
|
@ -2634,32 +2644,31 @@ mg_canvas mg_canvas_create(mg_surface surface)
|
|||
|
||||
if(backend)
|
||||
{
|
||||
mg_canvas_data* canvasData = malloc_type(mg_canvas_data);
|
||||
memset(canvasData, 0, sizeof(mg_canvas_data));
|
||||
mg_canvas_data* canvas = ListPopEntry(&__mgData.canvasFreeList, mg_canvas_data, freeListElt);
|
||||
if(!canvas)
|
||||
{
|
||||
canvas = mem_arena_alloc_type(&__mgData.resourceArena, mg_canvas_data);
|
||||
}
|
||||
if(canvas)
|
||||
{
|
||||
memset(canvas, 0, sizeof(mg_canvas_data));
|
||||
}
|
||||
|
||||
canvasData->backend = backend;
|
||||
canvas->backend = backend;
|
||||
|
||||
canvasData->primitiveCount = 0;
|
||||
canvasData->path.startIndex = 0;
|
||||
canvasData->path.count = 0;
|
||||
canvasData->nextShapeIndex = 0;
|
||||
canvasData->attributes.color = (mg_color){0, 0, 0, 1};
|
||||
canvasData->attributes.tolerance = 1;
|
||||
canvasData->attributes.width = 10;
|
||||
canvasData->attributes.clip = (mp_rect){-FLT_MAX/2, -FLT_MAX/2, FLT_MAX, FLT_MAX};
|
||||
canvas->attributes.color = (mg_color){0, 0, 0, 1};
|
||||
canvas->attributes.tolerance = 1;
|
||||
canvas->attributes.width = 10;
|
||||
canvas->attributes.clip = (mp_rect){-FLT_MAX/2, -FLT_MAX/2, FLT_MAX, FLT_MAX};
|
||||
|
||||
canvasData->transform = (mg_mat2x3){{1, 0, 0,
|
||||
0, 1, 0}};
|
||||
|
||||
|
||||
canvas = mg_canvas_handle_alloc(canvasData);
|
||||
mg_canvas_set_current(canvas);
|
||||
canvasHandle = mg_canvas_handle_alloc(canvas);
|
||||
mg_canvas_set_current(canvasHandle);
|
||||
|
||||
//TODO: move that in mg_canvas_set_current() if needed?
|
||||
mg_surface_prepare(surface);
|
||||
}
|
||||
}
|
||||
return(canvas);
|
||||
return(canvasHandle);
|
||||
}
|
||||
|
||||
void mg_canvas_destroy(mg_canvas handle)
|
||||
|
@ -2677,7 +2686,7 @@ void mg_canvas_destroy(mg_canvas handle)
|
|||
{
|
||||
canvas->backend->destroy(canvas->backend);
|
||||
}
|
||||
free(canvas);
|
||||
ListPush(&__mgData.canvasFreeList, &canvas->freeListElt);
|
||||
mg_handle_recycle(handle.h);
|
||||
}
|
||||
}
|
||||
|
@ -2918,8 +2927,6 @@ void mg_flush()
|
|||
canvas->primitiveCount = 0;
|
||||
canvas->path.startIndex = 0;
|
||||
canvas->path.count = 0;
|
||||
|
||||
canvas->frameCounter++;
|
||||
}
|
||||
|
||||
//------------------------------------------------------------------------------------------
|
||||
|
|
|
@ -9,13 +9,13 @@
|
|||
#ifndef __MTL_SURFACE_H_
|
||||
#define __MTL_SURFACE_H_
|
||||
|
||||
#include"graphics.h"
|
||||
#include"graphics_internal.h"
|
||||
|
||||
#ifdef __OBJC__
|
||||
#import<Metal/Metal.h>
|
||||
#endif
|
||||
|
||||
mg_surface mg_mtl_surface_create_for_window(mp_window window);
|
||||
mg_surface_data* mg_mtl_surface_create_for_window(mp_window window);
|
||||
|
||||
void* mg_mtl_surface_render_encoder(mg_surface surface);
|
||||
void* mg_mtl_surface_compute_encoder(mg_surface surface);
|
||||
|
|
|
@ -165,16 +165,13 @@ bool mg_mtl_surface_get_hidden(mg_surface_data* interface)
|
|||
//TODO fix that according to real scaling, depending on the monitor settings
|
||||
static const f32 MG_MTL_SURFACE_CONTENTS_SCALING = 2;
|
||||
|
||||
mg_surface mg_mtl_surface_create_for_window(mp_window window)
|
||||
mg_surface_data* mg_mtl_surface_create_for_window(mp_window window)
|
||||
{
|
||||
mg_mtl_surface* surface = 0;
|
||||
mp_window_data* windowData = mp_window_ptr_from_handle(window);
|
||||
if(!windowData)
|
||||
if(windowData)
|
||||
{
|
||||
return(mg_surface_nil());
|
||||
}
|
||||
else
|
||||
{
|
||||
mg_mtl_surface* surface = (mg_mtl_surface*)malloc(sizeof(mg_mtl_surface));
|
||||
surface = (mg_mtl_surface*)malloc(sizeof(mg_mtl_surface));
|
||||
|
||||
//NOTE(martin): setup interface functions
|
||||
surface->interface.backend = MG_BACKEND_METAL;
|
||||
|
@ -235,10 +232,8 @@ mg_surface mg_mtl_surface_create_for_window(mp_window window)
|
|||
surface->drawable = nil;
|
||||
surface->commandBuffer = nil;
|
||||
}
|
||||
|
||||
mg_surface handle = mg_surface_handle_alloc((mg_surface_data*)surface);
|
||||
return(handle);
|
||||
}
|
||||
return((mg_surface_data*)surface);
|
||||
}
|
||||
|
||||
void* mg_mtl_surface_layer(mg_surface surface)
|
||||
|
|
Loading…
Reference in New Issue