Unity build 0-dependency clean up
This commit is contained in:
parent
85480c11aa
commit
d74eeb6d98
155
.github/workflows/ci.yml
vendored
155
.github/workflows/ci.yml
vendored
@ -10,13 +10,18 @@ on:
|
||||
schedule:
|
||||
- cron: "0 0 * * *"
|
||||
|
||||
env:
|
||||
BB_CI: 1
|
||||
|
||||
jobs:
|
||||
generate-config:
|
||||
runs-on: ubuntu-24.04
|
||||
permissions: write-all
|
||||
outputs:
|
||||
BIRTH_GITHUB_TARGETS: ${{ steps.generate-config.outputs.BIRTH_GITHUB_TARGETS }}
|
||||
BIRTH_BUILD_TYPES: ${{ steps.generate-config.outputs.BIRTH_BUILD_TYPES }}
|
||||
BIRTH_CMAKE_BUILD_TYPES: ${{ steps.generate-config.outputs.BIRTH_CMAKE_BUILD_TYPES }}
|
||||
BIRTH_COMPILERS: ${{ steps.generate-config.outputs.BIRTH_COMPILERS }}
|
||||
BIRTH_MACOS_IMAGE: ${{ steps.generate-config.outputs.BIRTH_LINUX_IMAGE }}
|
||||
BIRTH_LINUX_IMAGE: ${{ steps.generate-config.outputs.BIRTH_LINUX_IMAGE }}
|
||||
BIRTH_WINDOWS_IMAGE: ${{ steps.generate-config.outputs.BIRTH_WINDOWS_IMAGE }}
|
||||
@ -26,7 +31,7 @@ jobs:
|
||||
uses: actions/checkout@v4
|
||||
- name: Generate config
|
||||
id: generate-config
|
||||
uses: birth-software/github-config@v3
|
||||
uses: birth-software/github-config@v4
|
||||
- name: Create tag
|
||||
if: github.ref == 'refs/heads/main'
|
||||
shell: bash
|
||||
@ -52,8 +57,24 @@ jobs:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
os: ${{ fromJSON(needs.generate-config.outputs.BIRTH_GITHUB_TARGETS) }}
|
||||
CMAKE_BUILD_TYPE: ${{ fromJSON(needs.generate-config.outputs.BIRTH_CMAKE_BUILD_TYPES) }}
|
||||
os: [ ubuntu-latest, macos-latest, windows-latest ]
|
||||
C_COMPILER: ${{ fromJSON(needs.generate-config.outputs.BIRTH_COMPILERS) }}
|
||||
BIRTH_BUILD_TYPE: ${{ fromJSON(needs.generate-config.outputs.BIRTH_BUILD_TYPES) }}
|
||||
exclude:
|
||||
- C_COMPILER: gcc
|
||||
os: macos-latest
|
||||
- C_COMPILER: gcc
|
||||
os: windows-latest
|
||||
- C_COMPILER: cl
|
||||
os: macos-latest
|
||||
- C_COMPILER: cl
|
||||
os: ubuntu-latest
|
||||
- C_COMPILER: tcc
|
||||
BIRTH_BUILD_TYPE: release_safe
|
||||
- C_COMPILER: tcc
|
||||
BIRTH_BUILD_TYPE: release_fast
|
||||
- C_COMPILER: tcc
|
||||
BIRTH_BUILD_TYPE: release_small
|
||||
runs-on: ${{ matrix.os }}
|
||||
env:
|
||||
BIRTH_LINUX_IMAGE: ${{ needs.generate-config.outputs.BIRTH_LINUX_IMAGE }}
|
||||
@ -61,96 +82,48 @@ jobs:
|
||||
BIRTH_WINDOWS_IMAGE: ${{ needs.generate-config.outputs.BIRTH_WINDOWS_IMAGE }}
|
||||
RELEASE_TAG_NAME: ${{ needs.generate-config.outputs.RELEASE_TAG_NAME }}
|
||||
steps:
|
||||
- name: Maximize build space
|
||||
if: matrix.os == env.BIRTH_LINUX_IMAGE
|
||||
uses: AdityaGarg8/remove-unwanted-software@v4.1
|
||||
with:
|
||||
remove-android: 'true'
|
||||
remove-dotnet: 'true'
|
||||
remove-haskell: 'true'
|
||||
remove-codeql: 'true'
|
||||
remove-docker-images: 'true'
|
||||
remove-large-packages: 'true'
|
||||
remove-cached-tools: 'true'
|
||||
remove-swapfile: 'true'
|
||||
- uses: actions/checkout@v4
|
||||
- uses: birth-software/github-setup@v3
|
||||
- name: Fetch LLVM
|
||||
id: fetch_llvm
|
||||
run: ./fetch-llvm.sh
|
||||
- name: Fetch dependencies
|
||||
if: matrix.os == 'ubuntu-latest'
|
||||
run: |
|
||||
if [[ "${{matrix.C_COMPILER}}" == "tcc" ]]; then
|
||||
wget https://github.com/birth-software/tinycc/releases/download/dev/tcc-x86_64-linux.7z
|
||||
7z x tcc-x86_64-linux.7z
|
||||
echo $PWD/tinycc/bin >> $GITHUB_PATH
|
||||
fi
|
||||
- name: Fetch dependencies
|
||||
if: matrix.os == 'macos-latest'
|
||||
run: |
|
||||
if [[ "${{matrix.C_COMPILER}}" == "tcc" ]]; then
|
||||
wget https://github.com/birth-software/tinycc/releases/download/dev/tcc-aarch64-macos.7z
|
||||
7z x tcc-aarch64-macos.7z
|
||||
echo $PWD/tinycc/bin >> $GITHUB_PATH
|
||||
fi
|
||||
- name: Fetch dependencies
|
||||
if: matrix.os == 'windows-latest'
|
||||
shell: bash
|
||||
env:
|
||||
CMAKE_BUILD_TYPE: ${{matrix.CMAKE_BUILD_TYPE}}
|
||||
run: |
|
||||
set -eux
|
||||
if [[ "${{matrix.C_COMPILER}}" == "tcc" ]]; then
|
||||
curl -L https://github.com/birth-software/tinycc/releases/download/dev/tcc-x86_64-windows.7z --output tcc-x86_64-windows.7z
|
||||
7z x tcc-x86_64-windows.7z
|
||||
TCC_PATH=$(cygpath -w "$PWD/tinycc")
|
||||
echo $TCC_PATH >> $GITHUB_PATH
|
||||
fi
|
||||
- name: Build
|
||||
id: build
|
||||
run: ./project.sh
|
||||
shell: bash
|
||||
if: matrix.os != 'windows-latest'
|
||||
env:
|
||||
CMAKE_BUILD_TYPE: ${{matrix.CMAKE_BUILD_TYPE}}
|
||||
CMAKE_PREFIX_PATH: ${{steps.fetch_llvm.outputs.CMAKE_PREFIX_PATH}}
|
||||
BB_IS_CI: true
|
||||
- name: Test
|
||||
run: ./project.sh test all
|
||||
shell: bash
|
||||
CC: ${{matrix.C_COMPILER}}
|
||||
BB_BUILD_TYPE: ${{matrix.BIRTH_BUILD_TYPE}}
|
||||
run: ./build.sh
|
||||
- name: Build
|
||||
if: matrix.os == 'windows-latest'
|
||||
shell: cmd
|
||||
env:
|
||||
CMAKE_BUILD_TYPE: ${{matrix.CMAKE_BUILD_TYPE}}
|
||||
CMAKE_PREFIX_PATH: ${{steps.fetch_llvm.outputs.CMAKE_PREFIX_PATH}}
|
||||
BB_IS_CI: true
|
||||
- name: Package
|
||||
id: package
|
||||
if: ${{ always() && steps.build.outcome == 'success' }}
|
||||
run: ./package.sh
|
||||
shell: bash
|
||||
env:
|
||||
CMAKE_BUILD_TYPE: ${{matrix.CMAKE_BUILD_TYPE}}
|
||||
BB_IS_CI: true
|
||||
- name: Upload artifact
|
||||
if: ${{ always() && steps.package.outcome == 'success' }}
|
||||
uses: actions/upload-artifact@v4
|
||||
with:
|
||||
name: ${{env.BLOAT_BUSTER_RELEASE_NAME_BASE}}
|
||||
if-no-files-found: 'error'
|
||||
path: ${{env.BLOAT_BUSTER_RELEASE_PATH_BASE}}.7z*
|
||||
- name: Release - Attempt 1
|
||||
id: release_attempt_1
|
||||
uses: softprops/action-gh-release@v2
|
||||
if: github.ref == 'refs/heads/main'
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
with:
|
||||
name: "Development build"
|
||||
tag_name: ${{ env.RELEASE_TAG_NAME }}
|
||||
fail_on_unmatched_files: true
|
||||
make_latest: true
|
||||
files: |
|
||||
${{ env.BLOAT_BUSTER_RELEASE_PATH_BASE }}.7z
|
||||
${{ env.BLOAT_BUSTER_RELEASE_PATH_BASE }}.7z.b2sum
|
||||
continue-on-error: true
|
||||
- name: Release - Attempt 2
|
||||
id: release_attempt_2
|
||||
if: ${{ steps.release_attempt_1.outcome == 'failure' && github.ref == 'refs/heads/main' }}
|
||||
uses: softprops/action-gh-release@v2
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
with:
|
||||
name: "Development build"
|
||||
tag_name: ${{ env.RELEASE_TAG_NAME }}
|
||||
fail_on_unmatched_files: true
|
||||
make_latest: true
|
||||
files: |
|
||||
${{ env.BLOAT_BUSTER_RELEASE_PATH_BASE }}.7z
|
||||
${{ env.BLOAT_BUSTER_RELEASE_PATH_BASE }}.7z.b2sum
|
||||
continue-on-error: true
|
||||
- name: Release - Attempt 3
|
||||
if: ${{ steps.release_attempt_2.outcome == 'failure' && github.ref == 'refs/heads/main' }}
|
||||
uses: softprops/action-gh-release@v2
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
with:
|
||||
name: "Development build"
|
||||
tag_name: ${{ env.RELEASE_TAG_NAME }}
|
||||
fail_on_unmatched_files: true
|
||||
make_latest: true
|
||||
files: |
|
||||
${{ env.BLOAT_BUSTER_RELEASE_PATH_BASE }}.7z
|
||||
${{ env.BLOAT_BUSTER_RELEASE_PATH_BASE }}.7z.b2sum
|
||||
CC: ${{matrix.C_COMPILER}}
|
||||
BB_BUILD_TYPE: ${{matrix.BIRTH_BUILD_TYPE}}
|
||||
run: |
|
||||
call "C:\Program Files\Microsoft Visual Studio\2022\Enterprise\VC\Auxiliary\Build\vcvarsall.bat" x64 || exit /b 1
|
||||
call build.bat || exit /b 1
|
||||
- name: Run
|
||||
run: ./cache/bb
|
||||
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -6,3 +6,5 @@ project
|
||||
/project.dSYM/
|
||||
/.cache/
|
||||
imgui.ini
|
||||
/cache/
|
||||
.DS_Store
|
||||
|
281
bootstrap/bloat-buster/bb.c
Normal file
281
bootstrap/bloat-buster/bb.c
Normal file
@ -0,0 +1,281 @@
|
||||
#if 0
|
||||
#include <std/base.h>
|
||||
#include <std/project.h>
|
||||
#include <std/os.h>
|
||||
#include <std/virtual_buffer.h>
|
||||
#include <std/windowing.h>
|
||||
#include <std/rendering.h>
|
||||
#include <std/ui_core.h>
|
||||
#include <std/ui_builder.h>
|
||||
|
||||
#include <std/base.c>
|
||||
#include <std/os.c>
|
||||
#include <std/virtual_buffer.c>
|
||||
#include <std/windowing.c>
|
||||
#include <std/rendering.c>
|
||||
#include <std/ui_core.c>
|
||||
#include <std/ui_builder.c>
|
||||
|
||||
#define default_font_height (24)
|
||||
global_variable u32 proportional_font_height = default_font_height;
|
||||
global_variable u32 monospace_font_height = default_font_height;
|
||||
|
||||
fn TextureIndex white_texture_create(Arena* arena, Renderer* renderer)
|
||||
{
|
||||
u32 white_texture_width = 1024;
|
||||
u32 white_texture_height = white_texture_width;
|
||||
let(white_texture_buffer, arena_allocate(arena, u32, white_texture_width * white_texture_height));
|
||||
memset(white_texture_buffer, 0xff, white_texture_width * white_texture_height * sizeof(u32));
|
||||
|
||||
let(white_texture, renderer_texture_create(renderer, (TextureMemory) {
|
||||
.pointer = white_texture_buffer,
|
||||
.width = white_texture_width,
|
||||
.height = white_texture_height,
|
||||
.depth = 1,
|
||||
.format = TEXTURE_FORMAT_R8G8B8A8_SRGB,
|
||||
}));
|
||||
|
||||
return white_texture;
|
||||
}
|
||||
|
||||
STRUCT(BBPanel)
|
||||
{
|
||||
BBPanel* first;
|
||||
BBPanel* last;
|
||||
BBPanel* next;
|
||||
BBPanel* previous;
|
||||
BBPanel* parent;
|
||||
f32 parent_percentage;
|
||||
Axis2 split_axis;
|
||||
};
|
||||
|
||||
STRUCT(BBWindow)
|
||||
{
|
||||
WindowingInstance* handle;
|
||||
RenderWindow* render;
|
||||
BBWindow* previous;
|
||||
BBWindow* next;
|
||||
BBPanel* root_panel;
|
||||
UI_State* ui;
|
||||
};
|
||||
|
||||
STRUCT(BBGUIState)
|
||||
{
|
||||
Arena* arena;
|
||||
Timestamp last_frame_timestamp;
|
||||
BBWindow* first_window;
|
||||
BBWindow* last_window;
|
||||
Renderer* renderer;
|
||||
// TODO: should this not be thread local?
|
||||
WindowingEventQueue event_queue;
|
||||
};
|
||||
global_variable BBGUIState state;
|
||||
|
||||
fn void ui_top_bar()
|
||||
{
|
||||
ui_push(pref_height, ui_em(1, 1));
|
||||
{
|
||||
ui_push(child_layout_axis, AXIS2_X);
|
||||
let(top_bar, ui_widget_make((UI_WidgetFlags) {
|
||||
}, strlit("top_bar")));
|
||||
ui_push(parent, top_bar);
|
||||
{
|
||||
ui_button(strlit("Button 1"));
|
||||
ui_button(strlit("Button 2"));
|
||||
ui_button(strlit("Button 3"));
|
||||
}
|
||||
ui_pop(parent);
|
||||
ui_pop(child_layout_axis);
|
||||
}
|
||||
ui_pop(pref_height);
|
||||
}
|
||||
|
||||
STRUCT(UI_Node)
|
||||
{
|
||||
String name;
|
||||
String type;
|
||||
String value;
|
||||
String namespace;
|
||||
String function;
|
||||
};
|
||||
|
||||
fn void ui_node(UI_Node node)
|
||||
{
|
||||
let(node_widget, ui_widget_make_format((UI_WidgetFlags) {
|
||||
.draw_background = 1,
|
||||
.draw_text = 1,
|
||||
}, "{s} : {s} = {s}##{s}{s}", node.name, node.type, node.value, node.function, node.namespace));
|
||||
}
|
||||
|
||||
fn void app_update()
|
||||
{
|
||||
let(frame_end, os_timestamp());
|
||||
windowing_poll_events(/* &state.event_queue */);
|
||||
let(frame_ms, os_resolve_timestamps(state.last_frame_timestamp, frame_end, TIME_UNIT_MILLISECONDS));
|
||||
state.last_frame_timestamp = frame_end;
|
||||
|
||||
Renderer* renderer = state.renderer;
|
||||
|
||||
BBWindow* window = state.first_window;
|
||||
while (likely(window))
|
||||
{
|
||||
let(previous, window->previous);
|
||||
let(next, window->next);
|
||||
|
||||
let(render_window, window->render);
|
||||
renderer_window_frame_begin(renderer, render_window);
|
||||
|
||||
ui_state_select(window->ui);
|
||||
|
||||
if (likely(ui_build_begin(window->handle, frame_ms, &state.event_queue)))
|
||||
{
|
||||
ui_push(font_size, default_font_height);
|
||||
|
||||
ui_top_bar();
|
||||
ui_push(child_layout_axis, AXIS2_X);
|
||||
let(workspace_widget, ui_widget_make_format((UI_WidgetFlags) {}, "workspace{u64}", window->handle));
|
||||
ui_push(parent, workspace_widget);
|
||||
{
|
||||
// Node visualizer
|
||||
ui_push(child_layout_axis, AXIS2_Y);
|
||||
let(node_visualizer_widget, ui_widget_make_format((UI_WidgetFlags) {
|
||||
.draw_background = 1,
|
||||
}, "node_visualizer{u64}", window->handle));
|
||||
|
||||
ui_push(parent, node_visualizer_widget);
|
||||
{
|
||||
ui_node((UI_Node) {
|
||||
.name = strlit("a"),
|
||||
.type = strlit("s32"),
|
||||
.value = strlit("1"),
|
||||
.namespace = strlit("foo"),
|
||||
.function = strlit("main"),
|
||||
});
|
||||
ui_node((UI_Node) {
|
||||
.name = strlit("b"),
|
||||
.type = strlit("s32"),
|
||||
.value = strlit("2"),
|
||||
.namespace = strlit("foo"),
|
||||
.function = strlit("main"),
|
||||
});
|
||||
}
|
||||
ui_pop(parent);
|
||||
ui_pop(child_layout_axis);
|
||||
|
||||
// Side-panel stub
|
||||
ui_button(strlit("Options"));
|
||||
}
|
||||
ui_pop(parent);
|
||||
ui_pop(child_layout_axis);
|
||||
|
||||
ui_build_end();
|
||||
|
||||
ui_draw();
|
||||
|
||||
ui_pop(font_size);
|
||||
|
||||
renderer_window_frame_end(renderer, render_window);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (previous)
|
||||
{
|
||||
previous->next = next;
|
||||
}
|
||||
|
||||
if (next)
|
||||
{
|
||||
next->previous = previous;
|
||||
}
|
||||
|
||||
if (state.first_window == window)
|
||||
{
|
||||
state.first_window = next;
|
||||
}
|
||||
|
||||
if (state.last_window == window)
|
||||
{
|
||||
state.last_window = previous;
|
||||
}
|
||||
}
|
||||
|
||||
window = next;
|
||||
}
|
||||
}
|
||||
|
||||
fn void window_refresh_callback(WindowingInstance* window, void* context)
|
||||
{
|
||||
unused(window);
|
||||
unused(context);
|
||||
app_update();
|
||||
}
|
||||
|
||||
int main()
|
||||
{
|
||||
state.arena = arena_initialize_default(MB(2));
|
||||
if (!windowing_initialize())
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
|
||||
state.renderer = rendering_initialize(state.arena);
|
||||
if (!state.renderer)
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
|
||||
WindowingInstantiate window_create_options = {
|
||||
.name = strlit("Bloat Buster"),
|
||||
.size = { .width = 1600, .height = 900 },
|
||||
};
|
||||
state.first_window = state.last_window = arena_allocate(state.arena, BBWindow, 1);
|
||||
state.first_window->handle = windowing_instantiate(window_create_options);
|
||||
|
||||
state.first_window->render = rendering_initialize_window(state.renderer, state.first_window->handle);
|
||||
|
||||
state.first_window->ui = ui_state_allocate(state.renderer, state.first_window->render);
|
||||
state.first_window->root_panel = arena_allocate(state.arena, BBPanel, 1);
|
||||
state.first_window->root_panel->parent_percentage = 1.0f;
|
||||
state.first_window->root_panel->split_axis = AXIS2_X;
|
||||
|
||||
#ifndef __APPLE__
|
||||
window_rect_texture_update_begin(state.first_window->render);
|
||||
|
||||
let(white_texture, white_texture_create(state.arena, state.renderer));
|
||||
TextureAtlasCreate monospace_font_create = {
|
||||
#ifdef _WIN32
|
||||
.font_path = strlit("C:/Users/David/Downloads/Fira_Sans/FiraSans-Regular.ttf"),
|
||||
#elif defined(__linux__)
|
||||
.font_path = strlit("/usr/share/fonts/TTF/FiraSans-Regular.ttf"),
|
||||
#elif defined(__APPLE__)
|
||||
.font_path = strlit("/Users/david/Library/Fonts/FiraSans-Regular.ttf"),
|
||||
#else
|
||||
.font_path = strlit("WRONG_PATH"),
|
||||
#endif
|
||||
.text_height = monospace_font_height,
|
||||
};
|
||||
let(monospace_font, font_texture_atlas_create(state.arena, state.renderer, monospace_font_create));
|
||||
let(proportional_font, monospace_font);
|
||||
|
||||
window_queue_rect_texture_update(state.first_window->render, RECT_TEXTURE_SLOT_WHITE, white_texture);
|
||||
renderer_queue_font_update(state.renderer, state.first_window->render, RENDER_FONT_TYPE_MONOSPACE, monospace_font);
|
||||
renderer_queue_font_update(state.renderer, state.first_window->render, RENDER_FONT_TYPE_PROPORTIONAL, proportional_font);
|
||||
|
||||
window_rect_texture_update_end(state.renderer, state.first_window->render);
|
||||
#endif
|
||||
|
||||
state.last_frame_timestamp = os_timestamp();
|
||||
|
||||
while (state.first_window)
|
||||
{
|
||||
app_update();
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
#else
|
||||
int main()
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
#endif
|
@ -1,343 +0,0 @@
|
||||
#pragma once
|
||||
|
||||
#ifdef NDEBUG
|
||||
#define BB_DEBUG 0
|
||||
#else
|
||||
#define BB_DEBUG 1
|
||||
#endif
|
||||
|
||||
#ifdef STATIC
|
||||
#define LINK_LIBC 0
|
||||
#else
|
||||
#define LINK_LIBC 1
|
||||
#endif
|
||||
|
||||
#include <stdint.h>
|
||||
#include <stdarg.h>
|
||||
#include <stddef.h>
|
||||
#include <stdlib.h>
|
||||
#if defined(__x86_64__)
|
||||
#include <immintrin.h>
|
||||
#endif
|
||||
#include <math.h>
|
||||
|
||||
typedef uint8_t u8;
|
||||
typedef uint16_t u16;
|
||||
typedef uint32_t u32;
|
||||
typedef uint64_t u64;
|
||||
typedef __uint128_t u128;
|
||||
typedef unsigned int uint;
|
||||
|
||||
typedef int8_t s8;
|
||||
typedef int16_t s16;
|
||||
typedef int32_t s32;
|
||||
typedef int64_t s64;
|
||||
typedef __int128_t s128;
|
||||
|
||||
typedef size_t usize;
|
||||
|
||||
typedef _Float16 f16;
|
||||
typedef float f32;
|
||||
typedef double f64;
|
||||
|
||||
typedef u32 Hash32;
|
||||
typedef u64 Hash64;
|
||||
|
||||
#define STRUCT_FORWARD_DECL(S) typedef struct S S
|
||||
#define STRUCT(S) STRUCT_FORWARD_DECL(S); struct S
|
||||
#define UNION_FORWARD_DECL(U) typedef union U U
|
||||
#define UNION(U) UNION_FORWARD_DECL(U); union U
|
||||
|
||||
typedef enum Corner
|
||||
{
|
||||
CORNER_00,
|
||||
CORNER_01,
|
||||
CORNER_10,
|
||||
CORNER_11,
|
||||
CORNER_COUNT,
|
||||
} Corner;
|
||||
|
||||
typedef float float2 __attribute__((ext_vector_type(2)));
|
||||
typedef float float3 __attribute__((ext_vector_type(3)));
|
||||
typedef float float4 __attribute__((ext_vector_type(4)));
|
||||
typedef float2 vec2;
|
||||
typedef float3 vec3;
|
||||
typedef float4 vec4;
|
||||
|
||||
typedef u32 uint2 __attribute__((ext_vector_type(2)));
|
||||
typedef u32 uint3 __attribute__((ext_vector_type(3)));
|
||||
typedef u32 uint4 __attribute__((ext_vector_type(4)));
|
||||
|
||||
UNION(F32Interval2)
|
||||
{
|
||||
struct
|
||||
{
|
||||
float2 min;
|
||||
float2 max;
|
||||
};
|
||||
struct
|
||||
{
|
||||
float2 p0;
|
||||
float2 p1;
|
||||
};
|
||||
struct
|
||||
{
|
||||
f32 x0;
|
||||
f32 y0;
|
||||
f32 x1;
|
||||
f32 y1;
|
||||
};
|
||||
float2 v[2];
|
||||
};
|
||||
|
||||
static_assert(sizeof(F32Interval2) == 4 * sizeof(f32));
|
||||
|
||||
typedef enum Axis2
|
||||
{
|
||||
AXIS2_X,
|
||||
AXIS2_Y,
|
||||
AXIS2_COUNT,
|
||||
} Axis2;
|
||||
|
||||
#ifdef __cplusplus
|
||||
#define EXPORT extern "C"
|
||||
#else
|
||||
#define EXPORT
|
||||
#endif
|
||||
|
||||
#if defined(__cplusplus) && defined(__linux__)
|
||||
#define NO_EXCEPT __THROW
|
||||
#else
|
||||
#define NO_EXCEPT
|
||||
#endif
|
||||
|
||||
|
||||
#define Slice(T) Slice_ ## T
|
||||
#define SliceP(T) SliceP_ ## T
|
||||
#define declare_slice_ex(T, StructName) STRUCT(StructName) \
|
||||
{\
|
||||
T* pointer;\
|
||||
u64 length;\
|
||||
}
|
||||
|
||||
#define declare_slice(T) declare_slice_ex(T, Slice(T))
|
||||
#define declare_slice_p(T) declare_slice_ex(T*, SliceP(T))
|
||||
|
||||
declare_slice(u8);
|
||||
declare_slice(u16);
|
||||
declare_slice(u32);
|
||||
declare_slice(u64);
|
||||
declare_slice(s8);
|
||||
declare_slice(s16);
|
||||
declare_slice(s32);
|
||||
declare_slice(s64);
|
||||
|
||||
declare_slice_p(char);
|
||||
declare_slice_p(void);
|
||||
|
||||
typedef Slice(u8) String;
|
||||
declare_slice(String);
|
||||
|
||||
#define NamedEnumMemberEnum(e, enum_member) e ## _ ## enum_member
|
||||
#define NamedEnumMemberString(e, enum_member) strlit(#enum_member)
|
||||
|
||||
typedef SliceP(char) CStringSlice;
|
||||
|
||||
#ifdef _WIN32
|
||||
typedef void* FileDescriptor;
|
||||
#else
|
||||
typedef int FileDescriptor;
|
||||
#endif
|
||||
|
||||
#define FOR_N(it, start, end) \
|
||||
for (u32 it = (start), end__ = (end); it < end__; ++it)
|
||||
|
||||
#define FOR_REV_N(it, start, end) \
|
||||
for (u32 it = (end), start__ = (start); (it--) > start__;)
|
||||
|
||||
#define FOR_BIT(it, start, bits) \
|
||||
for (typeof(bits) _bits_ = (bits), it = (start); _bits_; _bits_ >>= 1, ++it) if (_bits_ & 1)
|
||||
|
||||
#define FOREACH_SET(it, set) \
|
||||
FOR_N(_i, 0, ((set)->arr.capacity + 63) / 64) FOR_BIT(it, _i*64, (set)->arr.pointer[_i])
|
||||
|
||||
#define MIN(a, b) (((a) < (b)) ? (a) : (b))
|
||||
#define MAX(a, b) (((a) > (b)) ? (a) : (b))
|
||||
#define CLAMP(a, x, b) (((a)>(x))?(a):((b)<(x))?(b):(x))
|
||||
|
||||
|
||||
#ifndef INFINITY
|
||||
#define INFINITY __builtin_inff()
|
||||
#endif
|
||||
#ifndef NAN
|
||||
#define NAN __builtin_nanf("")
|
||||
#endif
|
||||
#define fn static
|
||||
#define method __attribute__((visibility("internal")))
|
||||
#define global_variable static
|
||||
#define forceinline __attribute__((always_inline))
|
||||
#define likely(x) __builtin_expect(!!(x), 1)
|
||||
#define unlikely(x) __builtin_expect(!!(x), 0)
|
||||
#define breakpoint() __builtin_debugtrap()
|
||||
#define failed_execution() my_panic("Failed execution at {cstr}:{u32}\n", __FILE__, __LINE__)
|
||||
|
||||
#define trap() __builtin_trap()
|
||||
#define array_length(arr) sizeof(arr) / sizeof((arr)[0])
|
||||
#define KB(n) ((n) * 1024)
|
||||
#define MB(n) ((n) * 1024 * 1024)
|
||||
#define GB(n) ((u64)(n) * 1024 * 1024 * 1024)
|
||||
#define TB(n) ((u64)(n) * 1024 * 1024 * 1024 * 1024)
|
||||
#define unused(x) (void)(x)
|
||||
#define may_be_unused __attribute__((unused))
|
||||
#define truncate_value(Destination, source) (Destination)(source)
|
||||
#define cast_to(Destination, Source, source) cast_ ## Source ## _to_ ## Destination (source, __FILE__, __LINE__)
|
||||
#define size_until_end(T, field_name) (sizeof(T) - offsetof(T, field_name))
|
||||
#define SWAP(a, b) \
|
||||
do {\
|
||||
auto temp = a;\
|
||||
a = b;\
|
||||
b = temp;\
|
||||
} while (0)
|
||||
|
||||
#define slice_from_pointer_range(T, start, end) (Slice(T)) { .pointer = start, .length = (u64)(end - start), }
|
||||
|
||||
#define strlit_len(s) (sizeof(s) - 1)
|
||||
#define strlit(s) (String){ .pointer = (u8*)(s), .length = strlit_len(s), }
|
||||
#define ch_to_str(ch) (String){ .pointer = &ch, .length = 1 }
|
||||
#define array_to_slice(arr) { .pointer = (arr), .length = array_length(arr) }
|
||||
#define array_to_bytes(arr) { .pointer = (u8*)(arr), .length = sizeof(arr) }
|
||||
#define pointer_to_bytes(p) (String) { .pointer = (u8*)(p), .length = sizeof(*p) }
|
||||
#define scalar_to_bytes(s) pointer_to_bytes(&(s))
|
||||
#define string_to_c(s) ((char*)((s).pointer))
|
||||
#define cstr(s) ((String) { .pointer = (u8*)(s), .length = strlen((char*)s), } )
|
||||
|
||||
#define case_to_name(prefix, e) case prefix ## e: return strlit(#e)
|
||||
|
||||
const may_be_unused global_variable u8 brace_open = '{';
|
||||
const may_be_unused global_variable u8 brace_close = '}';
|
||||
|
||||
const may_be_unused global_variable u8 parenthesis_open = '(';
|
||||
const may_be_unused global_variable u8 parenthesis_close = ')';
|
||||
|
||||
const may_be_unused global_variable u8 bracket_open = '[';
|
||||
const may_be_unused global_variable u8 bracket_close = ']';
|
||||
|
||||
#define s_get(s, i) (s).pointer[i]
|
||||
#define s_get_pointer(s, i) &((s).pointer[i])
|
||||
#define s_get_slice(T, s, start, end) (Slice(T)){ .pointer = ((s).pointer) + (start), .length = (end) - (start) }
|
||||
#define s_equal(a, b) ((a).length == (b).length && memcmp((a).pointer, (b).pointer, sizeof(*((a).pointer)) * (a).length) == 0)
|
||||
|
||||
#if BB_DEBUG
|
||||
#define assert(x) if (unlikely(!(x))) { my_panic("Assert failed: \"" # x "\" at {cstr}:{u32}\n", __FILE__, __LINE__); }
|
||||
#else
|
||||
#define assert(x) unlikely(!(x))
|
||||
#endif
|
||||
|
||||
#ifndef __cplusplus
|
||||
// Undefine unreachable if needed to provide a more safe-guard implementation
|
||||
#ifdef unreachable
|
||||
#undef unreachable
|
||||
#endif
|
||||
#if BB_DEBUG
|
||||
#define unreachable() my_panic("Unreachable triggered\n", __FILE__, __LINE__)
|
||||
#else
|
||||
#define unreachable() __builtin_unreachable()
|
||||
#endif
|
||||
|
||||
#define static_assert(x) _Static_assert((x), "Static assert failed!")
|
||||
#define alignof(x) _Alignof(x)
|
||||
#define auto __auto_type
|
||||
#else
|
||||
#define restrict __restrict
|
||||
#endif
|
||||
|
||||
#define todo() my_panic("TODO at {cstr}:{u32}\n", __FILE__, __LINE__)
|
||||
|
||||
EXPORT u64 align_forward(u64 value, u64 alignment);
|
||||
EXPORT u64 align_backward(u64 value, u64 alignment);
|
||||
EXPORT u8 log2_alignment(u64 alignment);
|
||||
EXPORT u8 is_power_of_two(u64 value);
|
||||
EXPORT u8 first_bit_set_32(u32 value);
|
||||
EXPORT u64 first_bit_set_64(u64 value);
|
||||
|
||||
EXPORT void* memcpy(void* const restrict dst, const void* const restrict src, usize size) NO_EXCEPT;
|
||||
EXPORT void* memmove(void* const dst, const void* const src, usize n) NO_EXCEPT;
|
||||
EXPORT void* memset(void* dst, int n, usize size) NO_EXCEPT;
|
||||
EXPORT int memcmp(const void* a, const void* b, usize n) NO_EXCEPT;
|
||||
EXPORT usize strlen (const char* c_string) NO_EXCEPT;
|
||||
EXPORT int strcmp(const char* s1, const char* s2) NO_EXCEPT;
|
||||
EXPORT int strncmp(const char* s1, const char* s2, usize length) NO_EXCEPT;
|
||||
|
||||
EXPORT u8 cast_u32_to_u8(u32 source, const char* name, int line);
|
||||
EXPORT u16 cast_u32_to_u16(u32 source, const char* name, int line);
|
||||
EXPORT s16 cast_u32_to_s16(u32 source, const char* name, int line);
|
||||
EXPORT s32 cast_u32_to_s32(u32 source, const char* name, int line);
|
||||
EXPORT u8 cast_u64_to_u8(u64 source, const char* name, int line);
|
||||
EXPORT u16 cast_u64_to_u16(u64 source, const char* name, int line);
|
||||
EXPORT u32 cast_u64_to_u32(u64 source, const char* name, int line);
|
||||
EXPORT s32 cast_u64_to_s32(u64 source, const char* name, int line);
|
||||
EXPORT s64 cast_u64_to_s64(u64 source, const char* name, int line);
|
||||
EXPORT u8 cast_s32_to_u8(s32 source, const char* name, int line);
|
||||
EXPORT u16 cast_s32_to_u16(s32 source, const char* name, int line);
|
||||
EXPORT u32 cast_s32_to_u32(s32 source, const char* name, int line);
|
||||
EXPORT u64 cast_s32_to_u64(s32 source, const char* name, int line);
|
||||
EXPORT s16 cast_s32_to_s16(s32 source, const char* name, int line);
|
||||
EXPORT u16 cast_s64_to_u16(s64 source, const char* name, int line);
|
||||
EXPORT u32 cast_s64_to_u32(s64 source, const char* name, int line);
|
||||
EXPORT u64 cast_s64_to_u64(s64 source, const char* name, int line);
|
||||
EXPORT s32 cast_s64_to_s32(s64 source, const char* name, int line);
|
||||
|
||||
EXPORT u32 format_decimal(String buffer, u64 decimal);
|
||||
EXPORT u32 format_hexadecimal(String buffer, u64 hexadecimal);
|
||||
EXPORT u64 format_float(String buffer, f64 value_double);
|
||||
|
||||
EXPORT u64 is_decimal_digit(u8 ch);
|
||||
EXPORT u32 is_space(u8 ch, u8 next_ch);
|
||||
EXPORT u8 get_next_ch_safe(String string, u64 index);
|
||||
EXPORT u64 is_identifier_start(u8 ch);
|
||||
EXPORT u64 is_identifier_ch(u8 ch);
|
||||
EXPORT u64 is_alphabetic(u8 ch);
|
||||
|
||||
EXPORT u64 parse_decimal(String string);
|
||||
|
||||
global_variable const Hash64 fnv_offset = 14695981039346656037ull;
|
||||
global_variable const u64 fnv_prime = 1099511628211ull;
|
||||
|
||||
EXPORT Hash32 hash32_fib_end(Hash32 hash);
|
||||
EXPORT Hash32 hash64_fib_end(Hash64 hash);
|
||||
|
||||
EXPORT Hash64 hash_byte(Hash64 source, u8 ch);
|
||||
EXPORT Hash64 hash_bytes(String bytes);
|
||||
EXPORT Hash32 hash64_to_hash32(Hash64 hash64);
|
||||
|
||||
EXPORT u64 round_up_to_next_power_of_2(u64 n);
|
||||
|
||||
STRUCT(TextureIndex)
|
||||
{
|
||||
u32 value;
|
||||
};
|
||||
|
||||
EXPORT void print(const char* format, ...);
|
||||
EXPORT u8 os_is_being_debugged();
|
||||
|
||||
fn u64 safe_flag(u64 value, u64 flag)
|
||||
{
|
||||
u64 result = value & ((u64)0 - flag);
|
||||
return result;
|
||||
}
|
||||
|
||||
#define member_from_offset(pointer, type, memory_offset) (*(type*)((u8*)pointer + memory_offset))
|
||||
#define offset_of(T, member) __builtin_offsetof(T, member)
|
||||
|
||||
#define my_panic(...) do \
|
||||
{\
|
||||
print(__VA_ARGS__);\
|
||||
if (os_is_being_debugged())\
|
||||
{\
|
||||
trap();\
|
||||
}\
|
||||
else\
|
||||
{\
|
||||
exit(1);\
|
||||
}\
|
||||
} while (0)
|
@ -1,6 +0,0 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/base.h>
|
||||
|
||||
EXPORT String format_string(String buffer, const char* format, ...);
|
||||
EXPORT String format_string_va(String buffer, const char* format, va_list args);
|
@ -1,120 +0,0 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/base.h>
|
||||
|
||||
typedef enum TimeUnit
|
||||
{
|
||||
TIME_UNIT_NANOSECONDS,
|
||||
TIME_UNIT_MICROSECONDS,
|
||||
TIME_UNIT_MILLISECONDS,
|
||||
TIME_UNIT_SECONDS,
|
||||
} TimeUnit;
|
||||
|
||||
|
||||
STRUCT(Timestamp)
|
||||
{
|
||||
u128 value;
|
||||
};
|
||||
|
||||
STRUCT(OSFileOpenFlags)
|
||||
{
|
||||
u32 truncate:1;
|
||||
u32 executable:1;
|
||||
u32 write:1;
|
||||
u32 read:1;
|
||||
u32 create:1;
|
||||
u32 directory:1;
|
||||
};
|
||||
|
||||
STRUCT(OSFilePermissions)
|
||||
{
|
||||
u8 readable:1;
|
||||
u8 writable:1;
|
||||
u8 executable:1;
|
||||
};
|
||||
|
||||
STRUCT(OSReserveProtectionFlags)
|
||||
{
|
||||
u32 read:1;
|
||||
u32 write:1;
|
||||
u32 execute:1;
|
||||
u32 reserved:29;
|
||||
};
|
||||
|
||||
STRUCT(OSReserveMapFlags)
|
||||
{
|
||||
u32 priv:1;
|
||||
u32 anon:1;
|
||||
u32 noreserve:1;
|
||||
u32 reserved:29;
|
||||
};
|
||||
|
||||
STRUCT(Arena)
|
||||
{
|
||||
u64 reserved_size;
|
||||
u64 position;
|
||||
u64 os_position;
|
||||
u64 granularity;
|
||||
u8 reserved[4 * 8];
|
||||
};
|
||||
|
||||
STRUCT(FileWriteOptions)
|
||||
{
|
||||
String path;
|
||||
String content;
|
||||
u8 executable;
|
||||
};
|
||||
|
||||
#if __APPLE__
|
||||
const global_variable u64 page_size = KB(16);
|
||||
#else
|
||||
const global_variable u64 page_size = KB(4);
|
||||
#endif
|
||||
|
||||
global_variable u64 minimum_granularity = page_size;
|
||||
// global_variable u64 middle_granularity = MB(2);
|
||||
global_variable u64 default_size = GB(4);
|
||||
|
||||
EXPORT void print(const char* format, ...);
|
||||
EXPORT void run_command(Arena* arena, CStringSlice arguments, char* envp[]);
|
||||
EXPORT String file_read(Arena* arena, String path);
|
||||
EXPORT void file_write(FileWriteOptions options);
|
||||
|
||||
EXPORT String path_dir(String string);
|
||||
EXPORT String path_base(String string);
|
||||
EXPORT String path_no_extension(String string);
|
||||
|
||||
|
||||
EXPORT Arena* arena_init(u64 reserved_size, u64 granularity, u64 initial_size);
|
||||
EXPORT Arena* arena_init_default(u64 initial_size);
|
||||
EXPORT void arena_clear(Arena* arena);
|
||||
EXPORT String arena_join_string(Arena* arena, Slice(String) pieces);
|
||||
EXPORT u8* arena_allocate_bytes(Arena* arena, u64 size, u64 alignment);
|
||||
EXPORT void arena_reset(Arena* arena);
|
||||
|
||||
#define arena_allocate(arena, T, count) (T*)(arena_allocate_bytes(arena, sizeof(T) * count, alignof(T)))
|
||||
#define arena_allocate_slice(arena, T, count) (Slice(T)){ .pointer = arena_allocate(arena, T, count), .length = count }
|
||||
|
||||
EXPORT u8* os_reserve(u64 base, u64 size, OSReserveProtectionFlags protection, OSReserveMapFlags map);
|
||||
EXPORT void os_commit(void* address, u64 size);
|
||||
|
||||
EXPORT u8 os_file_descriptor_is_valid(FileDescriptor fd);
|
||||
EXPORT FileDescriptor os_file_open(String path, OSFileOpenFlags flags, OSFilePermissions permissions);
|
||||
EXPORT void os_file_close(FileDescriptor fd);
|
||||
EXPORT u64 os_file_get_size(FileDescriptor fd);
|
||||
EXPORT void os_file_write(FileDescriptor fd, String content);
|
||||
EXPORT FileDescriptor os_stdout_get();
|
||||
EXPORT void os_directory_make(String path);
|
||||
|
||||
EXPORT void calibrate_cpu_timer();
|
||||
|
||||
EXPORT void print_string(String string);
|
||||
|
||||
EXPORT Timestamp os_timestamp();
|
||||
EXPORT f64 os_resolve_timestamps(Timestamp start, Timestamp end, TimeUnit time_unit);
|
||||
EXPORT u8 os_is_being_debugged();
|
||||
|
||||
#if _WIN32
|
||||
typedef void* HANDLE;
|
||||
EXPORT HANDLE os_windows_get_module_handle();
|
||||
#endif
|
@ -1,182 +0,0 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/base.h>
|
||||
|
||||
typedef struct Renderer Renderer;
|
||||
|
||||
#include <std/window.h>
|
||||
#include <std/font_provider.h>
|
||||
|
||||
typedef struct RenderWindow RenderWindow;
|
||||
typedef struct Pipeline Pipeline;
|
||||
|
||||
STRUCT(RectDraw)
|
||||
{
|
||||
F32Interval2 vertex;
|
||||
F32Interval2 texture;
|
||||
vec4 colors[4];
|
||||
u32 texture_index;
|
||||
};
|
||||
|
||||
#include "../std/shaders/rect.inc"
|
||||
typedef struct RectVertex RectVertex;
|
||||
decl_vb(RectVertex);
|
||||
|
||||
typedef enum BBPipeline
|
||||
{
|
||||
BB_PIPELINE_RECT,
|
||||
BB_PIPELINE_COUNT,
|
||||
} BBPipeline;
|
||||
|
||||
typedef enum RenderFontType
|
||||
{
|
||||
RENDER_FONT_TYPE_MONOSPACE,
|
||||
RENDER_FONT_TYPE_PROPORTIONAL,
|
||||
RENDER_FONT_TYPE_COUNT,
|
||||
} RenderFontType;
|
||||
|
||||
typedef enum RectTextureSlot
|
||||
{
|
||||
RECT_TEXTURE_SLOT_WHITE,
|
||||
RECT_TEXTURE_SLOT_MONOSPACE_FONT,
|
||||
RECT_TEXTURE_SLOT_PROPORTIONAL_FONT,
|
||||
RECT_TEXTURE_SLOT_COUNT
|
||||
} RectTextureSlot;
|
||||
|
||||
typedef enum IndexType : u8
|
||||
{
|
||||
INDEX_TYPE_U32,
|
||||
} IndexType;
|
||||
|
||||
typedef enum TextureFormat : u8
|
||||
{
|
||||
TEXTURE_FORMAT_R8_UNORM,
|
||||
TEXTURE_FORMAT_R8G8B8A8_SRGB,
|
||||
} TextureFormat;
|
||||
|
||||
STRUCT(TextureMemory)
|
||||
{
|
||||
void* pointer;
|
||||
u32 width;
|
||||
u32 height;
|
||||
u32 depth;
|
||||
TextureFormat format;
|
||||
};
|
||||
|
||||
typedef enum ShaderStage : u8
|
||||
{
|
||||
SHADER_STAGE_VERTEX,
|
||||
SHADER_STAGE_FRAGMENT,
|
||||
} ShaderStage;
|
||||
|
||||
STRUCT(PipelineCreate)
|
||||
{
|
||||
Slice(u16) shader_source_indices;
|
||||
u16 layout_index;
|
||||
};
|
||||
declare_slice(PipelineCreate);
|
||||
|
||||
STRUCT(PushConstantRange)
|
||||
{
|
||||
u16 offset;
|
||||
u16 size;
|
||||
ShaderStage stage;
|
||||
};
|
||||
declare_slice(PushConstantRange);
|
||||
|
||||
typedef enum DescriptorType : u8
|
||||
{
|
||||
DESCRIPTOR_TYPE_IMAGE_PLUS_SAMPLER,
|
||||
DESCRIPTOR_TYPE_COUNT,
|
||||
} DescriptorType;
|
||||
|
||||
STRUCT(DescriptorSetLayoutBinding)
|
||||
{
|
||||
u8 binding;
|
||||
DescriptorType type;
|
||||
ShaderStage stage;
|
||||
u8 count;
|
||||
};
|
||||
declare_slice(DescriptorSetLayoutBinding);
|
||||
|
||||
STRUCT(DescriptorSetLayoutCreate)
|
||||
{
|
||||
Slice(DescriptorSetLayoutBinding) bindings;
|
||||
};
|
||||
declare_slice(DescriptorSetLayoutCreate);
|
||||
|
||||
STRUCT(PipelineLayoutCreate)
|
||||
{
|
||||
Slice(PushConstantRange) push_constant_ranges;
|
||||
Slice(DescriptorSetLayoutCreate) descriptor_set_layouts;
|
||||
};
|
||||
declare_slice(PipelineLayoutCreate);
|
||||
|
||||
STRUCT(GraphicsPipelinesCreate)
|
||||
{
|
||||
Slice(String) shader_binaries;
|
||||
Slice(PipelineLayoutCreate) layouts;
|
||||
Slice(PipelineCreate) pipelines;
|
||||
};
|
||||
|
||||
STRUCT(PipelineIndex)
|
||||
{
|
||||
u32 value;
|
||||
};
|
||||
|
||||
STRUCT(PipelineLayoutIndex)
|
||||
{
|
||||
u32 value;
|
||||
};
|
||||
|
||||
STRUCT(DescriptorSetIndex)
|
||||
{
|
||||
u32 value;
|
||||
};
|
||||
|
||||
typedef enum BufferType : u8
|
||||
{
|
||||
BUFFER_TYPE_VERTEX,
|
||||
BUFFER_TYPE_INDEX,
|
||||
BUFFER_TYPE_STAGING,
|
||||
} BufferType;
|
||||
|
||||
STRUCT(HostBufferCopy)
|
||||
{
|
||||
String source;
|
||||
u64 destination_offset;
|
||||
};
|
||||
declare_slice(HostBufferCopy);
|
||||
|
||||
STRUCT(LocalBufferCopyRegion)
|
||||
{
|
||||
u64 source_offset;
|
||||
u64 destination_offset;
|
||||
u64 size;
|
||||
};
|
||||
declare_slice(LocalBufferCopyRegion);
|
||||
|
||||
EXPORT Renderer* renderer_initialize(Arena* arena);
|
||||
EXPORT RenderWindow* renderer_window_initialize(Renderer* renderer, OSWindow window);
|
||||
EXPORT PipelineIndex renderer_graphics_pipelines_create(Renderer* renderer, Arena* arena, GraphicsPipelinesCreate create_data);
|
||||
EXPORT PipelineLayoutIndex renderer_pipeline_get_layout(PipelineIndex pipeline);
|
||||
EXPORT void renderer_window_frame_begin(Renderer* renderer, RenderWindow* window);
|
||||
EXPORT void renderer_window_frame_end(Renderer* renderer, RenderWindow* window);
|
||||
EXPORT TextureIndex renderer_texture_create(Renderer* renderer, TextureMemory texture_memory);
|
||||
EXPORT uint2 renderer_font_compute_string_rect(Renderer* renderer, RenderFontType type, String string);
|
||||
EXPORT void window_command_begin(RenderWindow* window);
|
||||
EXPORT void window_command_end(RenderWindow* window);
|
||||
EXPORT void window_render_begin(RenderWindow* window);
|
||||
EXPORT void window_render_end(RenderWindow* window);
|
||||
|
||||
EXPORT void window_draw_indexed(RenderWindow* window, u32 index_count, u32 instance_count, u32 first_index, s32 vertex_offset, u32 first_instance);
|
||||
|
||||
EXPORT void window_rect_texture_update_begin(RenderWindow* window);
|
||||
EXPORT void renderer_queue_font_update(Renderer* renderer, RenderWindow* window, RenderFontType type, TextureAtlas atlas);
|
||||
EXPORT void window_queue_rect_texture_update(RenderWindow* window, RectTextureSlot slot, TextureIndex texture_index);
|
||||
EXPORT void window_rect_texture_update_end(Renderer* renderer, RenderWindow* window);
|
||||
|
||||
EXPORT u32 window_pipeline_add_vertices(RenderWindow* window, BBPipeline pipeline_index, String vertex_memory, u32 vertex_count);
|
||||
EXPORT void window_pipeline_add_indices(RenderWindow* window, BBPipeline pipeline_index, Slice(u32) indices);
|
||||
EXPORT void window_render_rect(RenderWindow* window, RectDraw draw);
|
||||
EXPORT void window_render_text(Renderer* renderer, RenderWindow* window, String string, float4 color, RenderFontType font_type, u32 x_offset, u32 y_offset);
|
@ -1,8 +0,0 @@
|
||||
#include <std/base.h>
|
||||
|
||||
EXPORT s32 string_first_ch(String string, u8 ch);
|
||||
EXPORT s64 string_last_ch(String string, u8 ch);
|
||||
EXPORT u8 string_starts_with(String string, String start);
|
||||
EXPORT u8 string_ends_with(String string, String end);
|
||||
EXPORT u64 string_first_ocurrence(String string, String substring);
|
||||
EXPORT u64 string_last_ocurrence(String string, String substring);
|
@ -1,192 +0,0 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/base.h>
|
||||
#include <std/os.h>
|
||||
#include <std/virtual_buffer.h>
|
||||
|
||||
typedef enum OSEventType
|
||||
{
|
||||
OS_EVENT_TYPE_MOUSE_BUTTON,
|
||||
OS_EVENT_TYPE_CURSOR_POSITION,
|
||||
OS_EVENT_TYPE_CURSOR_ENTER,
|
||||
OS_EVENT_TYPE_WINDOW_FOCUS,
|
||||
OS_EVENT_TYPE_WINDOW_POSITION,
|
||||
OS_EVENT_TYPE_WINDOW_CLOSE,
|
||||
} OSEventType;
|
||||
|
||||
STRUCT(OSEventDescriptor)
|
||||
{
|
||||
u32 index:24;
|
||||
OSEventType type:8;
|
||||
};
|
||||
static_assert(sizeof(OSEventDescriptor) == 4);
|
||||
decl_vb(OSEventDescriptor);
|
||||
|
||||
typedef enum OSEventMouseButtonKind : u8
|
||||
{
|
||||
OS_EVENT_MOUSE_BUTTON_1 = 0,
|
||||
OS_EVENT_MOUSE_BUTTON_2 = 1,
|
||||
OS_EVENT_MOUSE_BUTTON_3 = 2,
|
||||
OS_EVENT_MOUSE_BUTTON_4 = 3,
|
||||
OS_EVENT_MOUSE_BUTTON_5 = 4,
|
||||
OS_EVENT_MOUSE_BUTTON_6 = 5,
|
||||
OS_EVENT_MOUSE_BUTTON_7 = 6,
|
||||
OS_EVENT_MOUSE_BUTTON_8 = 7,
|
||||
OS_EVENT_MOUSE_BUTTON_COUNT = 8,
|
||||
OS_EVENT_MOUSE_LEFT = OS_EVENT_MOUSE_BUTTON_1,
|
||||
OS_EVENT_MOUSE_RIGHT = OS_EVENT_MOUSE_BUTTON_2,
|
||||
OS_EVENT_MOUSE_MIDDLE = OS_EVENT_MOUSE_BUTTON_3,
|
||||
} OSEventMouseButtonKind;
|
||||
|
||||
typedef enum OSEventMouseButtonAction : u8
|
||||
{
|
||||
OS_EVENT_MOUSE_RELAX = 0,
|
||||
OS_EVENT_MOUSE_RELEASE = 1,
|
||||
OS_EVENT_MOUSE_PRESS = 2,
|
||||
OS_EVENT_MOUSE_REPEAT = 3,
|
||||
} OSEventMouseButtonAction;
|
||||
|
||||
STRUCT(OSEventMouseButtonEvent)
|
||||
{
|
||||
OSEventMouseButtonAction action:2;
|
||||
u8 mod_shift:1;
|
||||
u8 mod_control:1;
|
||||
u8 mod_alt:1;
|
||||
u8 mod_super:1;
|
||||
u8 mod_caps_lock:1;
|
||||
u8 mod_num_lock:1;
|
||||
};
|
||||
|
||||
STRUCT(OSEventMouseButton)
|
||||
{
|
||||
OSEventMouseButtonKind button:3;
|
||||
u8 reserved:5;
|
||||
OSEventMouseButtonEvent event;
|
||||
};
|
||||
static_assert(sizeof(OSEventMouseButton) == sizeof(u16));
|
||||
decl_vb(OSEventMouseButton);
|
||||
|
||||
#define OS_EVENT_BITSET_SIZE (64)
|
||||
STRUCT(OSEventBitset)
|
||||
{
|
||||
u64 value;
|
||||
};
|
||||
decl_vb(OSEventBitset);
|
||||
|
||||
STRUCT(OSEventCursorPosition)
|
||||
{
|
||||
f64 x;
|
||||
f64 y;
|
||||
};
|
||||
decl_vb(OSEventCursorPosition);
|
||||
|
||||
STRUCT(OSEventWindowPosition)
|
||||
{
|
||||
u32 x;
|
||||
u32 y;
|
||||
};
|
||||
decl_vb(OSEventWindowPosition);
|
||||
|
||||
STRUCT(OSEventQueue)
|
||||
{
|
||||
VirtualBuffer(OSEventDescriptor) descriptors;
|
||||
VirtualBuffer(OSEventMouseButton) mouse_buttons;
|
||||
VirtualBuffer(OSEventBitset) window_focuses;
|
||||
u32 window_focuses_count;
|
||||
u32 cursor_enter_count;
|
||||
VirtualBuffer(OSEventBitset) cursor_enters;
|
||||
VirtualBuffer(OSEventCursorPosition) cursor_positions;
|
||||
VirtualBuffer(OSEventWindowPosition) window_positions;
|
||||
};
|
||||
|
||||
typedef void* OSWindow;
|
||||
|
||||
typedef void OSFramebufferResize(OSWindow window, void* context, u32 width, u32 height);
|
||||
typedef void OSWindowResize(OSWindow window, void* context, u32 width, u32 height);
|
||||
typedef void OSWindowRefresh(OSWindow window, void* context);
|
||||
typedef void OSWindowPosition(OSWindow window, void* context, u32 x, u32 y);
|
||||
typedef void OSWindowClose(OSWindow window, void* context);
|
||||
typedef void OSWindowFocus(OSWindow window, void* context, u8 focused);
|
||||
typedef void OSWindowIconify(OSWindow window, void* context, u8 iconified);
|
||||
typedef void OSWindowMaximize(OSWindow window, void* context, u8 maximized);
|
||||
typedef void OSWindowContentScale(OSWindow window, void* context, f32 x, f32 y);
|
||||
typedef void OSWindowKey(OSWindow window, void* context, s32 key, s32 scancode, s32 action, s32 mods);
|
||||
typedef void OSWindowCharacter(OSWindow window, void* context, u32 codepoint);
|
||||
typedef void OSWindowCharacterModifier(OSWindow window, void* context, u32 codepoint, s32 mods);
|
||||
typedef void OSWindowMouseButton(OSWindow window, void* context, s32 button, s32 action, s32 mods);
|
||||
typedef void OSWindowCursorPosition(OSWindow window, void* context, f64 x, f64 y);
|
||||
typedef void OSWindowCursorEnter(OSWindow window, void* context, u8 entered);
|
||||
typedef void OSWindowScroll(OSWindow window, void* context, f64 x, f64 y);
|
||||
typedef void OSWindowDrop(OSWindow window, void* context, CStringSlice paths);
|
||||
|
||||
STRUCT(OSWindowingCallbacks)
|
||||
{
|
||||
OSFramebufferResize* framebuffer_resize;
|
||||
OSWindowResize* window_resize;
|
||||
OSWindowRefresh* window_refresh;
|
||||
OSWindowPosition* window_position;
|
||||
OSWindowClose* window_close;
|
||||
OSWindowFocus* window_focus;
|
||||
OSWindowIconify* window_iconify;
|
||||
OSWindowMaximize* window_maximize;
|
||||
OSWindowContentScale* window_content_scale;
|
||||
OSWindowKey* window_key;
|
||||
OSWindowCharacter* window_character;
|
||||
OSWindowCharacterModifier* window_character_modifier;
|
||||
OSWindowMouseButton* window_mouse_button;
|
||||
OSWindowCursorPosition* window_cursor_position;
|
||||
OSWindowCursorEnter* window_cursor_enter;
|
||||
OSWindowScroll* window_scroll;
|
||||
OSWindowDrop* window_drop;
|
||||
};
|
||||
|
||||
STRUCT(OSWindowingInitializationOptions)
|
||||
{
|
||||
OSWindowingCallbacks callback;
|
||||
u8 should_use_x11;
|
||||
};
|
||||
|
||||
STRUCT(OSWindowSize)
|
||||
{
|
||||
u32 width;
|
||||
u32 height;
|
||||
};
|
||||
|
||||
STRUCT(OSWindowCreate)
|
||||
{
|
||||
String name;
|
||||
OSWindowSize size;
|
||||
void* context;
|
||||
OSWindowResize* resize_callback;
|
||||
OSWindowRefresh* refresh_callback;
|
||||
};
|
||||
|
||||
STRUCT(OSCursorPosition)
|
||||
{
|
||||
f64 x;
|
||||
f64 y;
|
||||
};
|
||||
|
||||
EXPORT void os_windowing_init(OSWindowingInitializationOptions options);
|
||||
EXPORT OSWindow os_window_create(OSWindowCreate create);
|
||||
EXPORT u8 os_window_should_close(OSWindow window);
|
||||
EXPORT void os_poll_events(OSEventQueue* event_queue);
|
||||
EXPORT OSCursorPosition os_window_cursor_position_get(OSWindow window);
|
||||
EXPORT OSWindowSize os_window_framebuffer_size_get(OSWindow window);
|
||||
|
||||
EXPORT u8 os_event_queue_get_window_focus(OSEventQueue* queue, u32 index);
|
||||
|
||||
#ifdef __linux__
|
||||
typedef unsigned long XID;
|
||||
typedef struct _XDisplay Display;
|
||||
typedef XID Window;
|
||||
|
||||
EXPORT Display* x11_display_get();
|
||||
EXPORT Window x11_window_get(OSWindow window);
|
||||
#endif
|
||||
|
||||
#ifdef _WIN32
|
||||
EXPORT HANDLE win32_window_get(OSWindow window);
|
||||
#endif
|
||||
|
||||
int window_create_surface(void* instance, OSWindow window, const void* allocator, void** surface);
|
@ -1,317 +1,160 @@
|
||||
#include <std/base.h>
|
||||
#include <std/os.h>
|
||||
#pragma once
|
||||
|
||||
u8 cast_u32_to_u8(u32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > UINT8_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u8)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u16 cast_u32_to_u16(u32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > UINT16_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u16)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
s16 cast_u32_to_s16(u32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > INT16_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (s16)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
s32 cast_u32_to_s32(u32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > INT32_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (s32)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u8 cast_u64_to_u8(u64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > UINT8_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u8)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u16 cast_u64_to_u16(u64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > UINT16_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u16)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u32 cast_u64_to_u32(u64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > UINT32_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u32)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
s32 cast_u64_to_s32(u64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > INT32_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (s32)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
s64 cast_u64_to_s64(u64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > INT64_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (s64)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u8 cast_s32_to_u8(s32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source < 0)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
if ((u32)source > UINT8_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u8)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u16 cast_s32_to_u16(s32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source < 0)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
if ((u32)source > UINT16_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u16)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u32 cast_s32_to_u32(s32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source < 0)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u32)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u64 cast_s32_to_u64(s32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source < 0)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u64)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
s16 cast_s32_to_s16(s32 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source > INT16_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
if (source < INT16_MIN)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (s16)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u16 cast_s64_to_u16(s64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source < 0)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
if (source > UINT16_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u16)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u32 cast_s64_to_u32(s64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source < 0)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u32)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u64 cast_s64_to_u64(s64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source < 0)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (u64)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
s32 cast_s64_to_s32(s64 source, const char* name, int line)
|
||||
{
|
||||
#if BB_DEBUG
|
||||
if (source < INT32_MIN)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
|
||||
if (source > INT32_MAX)
|
||||
{
|
||||
my_panic("Cast failed at {cstr}:{u32}\n", name, line);
|
||||
}
|
||||
#else
|
||||
unused(name);
|
||||
unused(line);
|
||||
#endif
|
||||
auto result = (s32)source;
|
||||
return result;
|
||||
}
|
||||
|
||||
u8 log2_alignment(u64 alignment)
|
||||
fn u8 log2_alignment(u64 alignment)
|
||||
{
|
||||
assert(alignment != 0);
|
||||
assert((alignment & (alignment - 1)) == 0);
|
||||
u64 left = (sizeof(alignment) * 8) - 1;
|
||||
auto right = cast_to(u64, s32, __builtin_clzll(alignment));
|
||||
auto result = cast_to(u8, u64, left - right);
|
||||
let_cast(u64, right, __builtin_clzll(alignment));
|
||||
let_cast(u8, result, left - right);
|
||||
return result;
|
||||
}
|
||||
|
||||
fn u128 u128_from_u64(u64 n)
|
||||
{
|
||||
#if defined(__TINYC__) || defined(_MSC_VER)
|
||||
u128 result = { .low = n };
|
||||
return result;
|
||||
#else
|
||||
return n;
|
||||
#endif
|
||||
}
|
||||
|
||||
fn u64 u64_from_u128(u128 n)
|
||||
{
|
||||
#if defined (__TINYC__) || defined(_MSC_VER)
|
||||
return n.low;
|
||||
#else
|
||||
return (u64)n;
|
||||
#endif
|
||||
}
|
||||
|
||||
fn u128 u128_shift_right(u128 value, u16 n)
|
||||
{
|
||||
#if defined (__TINYC__) || defined(_MSC_VER)
|
||||
u128 result = {0, 0};
|
||||
|
||||
if (n < 128)
|
||||
{
|
||||
if (n >= 64)
|
||||
{
|
||||
// If n >= 64, only the high part contributes to the low part
|
||||
result.low = value.high >> (n - 64);
|
||||
result.high = 0;
|
||||
}
|
||||
else
|
||||
{
|
||||
// Standard case: n < 64
|
||||
result.low = (value.low >> n) | (value.high << (64 - n));
|
||||
result.high = value.high >> n;
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
#else
|
||||
return value >> n;
|
||||
#endif
|
||||
}
|
||||
|
||||
fn u128 u128_shift_left(u128 value, u16 n)
|
||||
{
|
||||
#if defined(__TINYC__) || defined(_MSC_VER)
|
||||
u128 result = {0, 0};
|
||||
|
||||
if (n < 128)
|
||||
{
|
||||
if (n >= 64)
|
||||
{
|
||||
// If n >= 64, only the low part contributes to the high part
|
||||
result.high = value.low << (n - 64);
|
||||
result.low = 0;
|
||||
}
|
||||
else
|
||||
{
|
||||
// Standard case: n < 64
|
||||
result.high = (value.high << n) | (value.low >> (64 - n));
|
||||
result.low = value.low << n;
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
#else
|
||||
return value << n;
|
||||
#endif
|
||||
}
|
||||
|
||||
fn u128 u128_u64_or(u128 a, u64 b)
|
||||
{
|
||||
#if defined(__TINYC__) || defined(_MSC_VER)
|
||||
a.low |= b;
|
||||
return a;
|
||||
#else
|
||||
return a | b;
|
||||
#endif
|
||||
}
|
||||
|
||||
fn u128 u128_u64_add(u128 a, u64 b)
|
||||
{
|
||||
#if defined(__TINYC__) || defined(_MSC_VER)
|
||||
u128 result;
|
||||
|
||||
// Add the lower 64 bits and check for overflow
|
||||
result.low = a.low + b;
|
||||
u64 carry = (result.low < a.low) ? 1 : 0;
|
||||
|
||||
// Add the carry to the upper 64 bits
|
||||
result.high = a.high + carry;
|
||||
|
||||
return result;
|
||||
#else
|
||||
return a + b;
|
||||
#endif
|
||||
}
|
||||
|
||||
// Multiply two u128 values
|
||||
fn u128 u128_u64_mul(u128 a, u64 b)
|
||||
{
|
||||
#if defined(__TINYC__) || defined(_MSC_VER)
|
||||
u128 result = {0, 0};
|
||||
|
||||
// Compute low and high parts of the product
|
||||
u64 low_low = (a.low & 0xFFFFFFFF) * (b & 0xFFFFFFFF);
|
||||
u64 low_high = (a.low >> 32) * (b & 0xFFFFFFFF);
|
||||
u64 high_low = (a.low & 0xFFFFFFFF) * (b >> 32);
|
||||
u64 high_high = (a.low >> 32) * (b >> 32);
|
||||
|
||||
// Combine partial products for the lower 64 bits
|
||||
u64 carry = (low_low >> 32) + (low_high & 0xFFFFFFFF) + (high_low & 0xFFFFFFFF);
|
||||
result.low = (low_low & 0xFFFFFFFF) | (carry << 32);
|
||||
|
||||
// Add carry from lower to the high product
|
||||
result.high = a.high * b + (low_high >> 32) + (high_low >> 32) + (carry >> 32) + high_high;
|
||||
|
||||
return result;
|
||||
#else
|
||||
return a * b;
|
||||
#endif
|
||||
}
|
||||
|
||||
fn u64 u128_shift_right_by_64(u128 n)
|
||||
{
|
||||
#if defined(__TINYC__) || defined(_MSC_VER)
|
||||
return n.high;
|
||||
#else
|
||||
return n >> 64;
|
||||
#endif
|
||||
}
|
||||
|
||||
// Lehmer's generator
|
||||
// https://lemire.me/blog/2019/03/19/the-fastest-conventional-random-number-generator-that-can-pass-big-crush/
|
||||
may_be_unused global_variable u128 rn_state;
|
||||
may_be_unused fn u64 generate_random_number()
|
||||
{
|
||||
rn_state *= 0xda942042e4dd58b5;
|
||||
return rn_state >> 64;
|
||||
rn_state = u128_u64_mul(rn_state, 0xda942042e4dd58b5);
|
||||
return u128_shift_right_by_64(rn_state);
|
||||
}
|
||||
|
||||
u64 round_up_to_next_power_of_2(u64 n)
|
||||
fn u64 round_up_to_next_power_of_2(u64 n)
|
||||
{
|
||||
n -= 1;
|
||||
n |= n >> 1;
|
||||
@ -326,10 +169,10 @@ u64 round_up_to_next_power_of_2(u64 n)
|
||||
|
||||
may_be_unused fn u64 absolute_int(s64 n)
|
||||
{
|
||||
return n < 0 ? cast_to(u64, s64, -n) : cast_to(u64, s64, n);
|
||||
return n < 0 ? cast_to(u64, -n) : cast_to(u64, n);
|
||||
}
|
||||
|
||||
u64 parse_decimal(String string)
|
||||
fn u64 parse_decimal(String string)
|
||||
{
|
||||
u64 value = 0;
|
||||
for (u64 i = 0; i < string.length; i += 1)
|
||||
@ -342,7 +185,7 @@ u64 parse_decimal(String string)
|
||||
return value;
|
||||
}
|
||||
|
||||
u8 get_next_ch_safe(String string, u64 index)
|
||||
fn u8 get_next_ch_safe(String string, u64 index)
|
||||
{
|
||||
u64 next_index = index + 1;
|
||||
u64 is_in_range = next_index < string.length;
|
||||
@ -353,7 +196,7 @@ u8 get_next_ch_safe(String string, u64 index)
|
||||
return (u8)safe_result;
|
||||
}
|
||||
|
||||
u32 is_space(u8 ch, u8 next_ch)
|
||||
fn u32 is_space(u8 ch, u8 next_ch)
|
||||
{
|
||||
u32 is_comment = (ch == '/') & (next_ch == '/');
|
||||
u32 is_whitespace = ch == ' ';
|
||||
@ -375,12 +218,12 @@ fn u64 is_upper(u8 ch)
|
||||
return (ch >= 'A') & (ch <= 'Z');
|
||||
}
|
||||
|
||||
u64 is_alphabetic(u8 ch)
|
||||
fn u64 is_alphabetic(u8 ch)
|
||||
{
|
||||
return is_lower(ch) | is_upper(ch);
|
||||
}
|
||||
|
||||
u64 is_decimal_digit(u8 ch)
|
||||
fn u64 is_decimal_digit(u8 ch)
|
||||
{
|
||||
return (ch >= '0') & (ch <= '9');
|
||||
}
|
||||
@ -390,29 +233,28 @@ fn u64 is_hex_digit(u8 ch)
|
||||
return (is_decimal_digit(ch) | ((ch == 'a' | ch == 'A') | (ch == 'b' | ch == 'B'))) | (((ch == 'c' | ch == 'C') | (ch == 'd' | ch == 'D')) | ((ch == 'e' | ch == 'E') | (ch == 'f' | ch == 'F')));
|
||||
}
|
||||
|
||||
|
||||
u64 is_identifier_start(u8 ch)
|
||||
fn u64 is_identifier_start(u8 ch)
|
||||
{
|
||||
u64 alphabetic = is_alphabetic(ch);
|
||||
u64 is_underscore = ch == '_';
|
||||
return alphabetic | is_underscore;
|
||||
}
|
||||
|
||||
u64 is_identifier_ch(u8 ch)
|
||||
fn u64 is_identifier_ch(u8 ch)
|
||||
{
|
||||
u64 identifier_start = is_identifier_start(ch);
|
||||
u64 decimal = is_decimal_digit(ch);
|
||||
return identifier_start | decimal;
|
||||
}
|
||||
|
||||
Hash64 hash_byte(Hash64 source, u8 ch)
|
||||
fn Hash64 hash_byte(Hash64 source, u8 ch)
|
||||
{
|
||||
source ^= ch;
|
||||
source *= fnv_prime;
|
||||
return source;
|
||||
}
|
||||
|
||||
Hash64 hash_bytes(String bytes)
|
||||
fn Hash64 hash_bytes(String bytes)
|
||||
{
|
||||
u64 result = fnv_offset;
|
||||
for (u64 i = 0; i < bytes.length; i += 1)
|
||||
@ -423,7 +265,7 @@ Hash64 hash_bytes(String bytes)
|
||||
return result;
|
||||
}
|
||||
|
||||
Hash32 hash64_to_hash32(Hash64 hash64)
|
||||
fn Hash32 hash64_to_hash32(Hash64 hash64)
|
||||
{
|
||||
Hash32 low = hash64 & 0xffff;
|
||||
Hash32 high = (hash64 >> 32) & 0xffff;
|
||||
@ -431,148 +273,46 @@ Hash32 hash64_to_hash32(Hash64 hash64)
|
||||
return result;
|
||||
}
|
||||
|
||||
u64 align_forward(u64 value, u64 alignment)
|
||||
fn u64 align_forward(u64 value, u64 alignment)
|
||||
{
|
||||
u64 mask = alignment - 1;
|
||||
u64 result = (value + mask) & ~mask;
|
||||
return result;
|
||||
}
|
||||
|
||||
u64 align_backward(u64 value, u64 alignment)
|
||||
fn u64 align_backward(u64 value, u64 alignment)
|
||||
{
|
||||
u64 result = value & ~(alignment - 1);
|
||||
return result;
|
||||
}
|
||||
|
||||
u8 is_power_of_two(u64 value)
|
||||
fn u8 is_power_of_two(u64 value)
|
||||
{
|
||||
return (value & (value - 1)) == 0;
|
||||
}
|
||||
|
||||
u8 first_bit_set_32(u32 value)
|
||||
fn u8 first_bit_set_32(u32 value)
|
||||
{
|
||||
auto result = (u8)__builtin_ffs((s32)value);
|
||||
let(result, (u8)__builtin_ffs((s32)value));
|
||||
result -= result != 0;
|
||||
return result;
|
||||
}
|
||||
|
||||
u64 first_bit_set_64(u64 value)
|
||||
fn u64 first_bit_set_64(u64 value)
|
||||
{
|
||||
auto result = (u8) __builtin_ffs((s64)value);
|
||||
let(result, (u8) __builtin_ffs((s64)value));
|
||||
result -= result != 0;
|
||||
return result;
|
||||
}
|
||||
|
||||
Hash32 hash32_fib_end(Hash32 hash)
|
||||
fn Hash32 hash32_fib_end(Hash32 hash)
|
||||
{
|
||||
auto result = truncate_value(Hash32, ((hash + 1) * 11400714819323198485ull) >> 32);
|
||||
let(result, TRUNCATE(Hash32, ((hash + 1) * 11400714819323198485ull) >> 32));
|
||||
return result;
|
||||
}
|
||||
|
||||
Hash32 hash64_fib_end(Hash64 hash)
|
||||
fn Hash32 hash64_fib_end(Hash64 hash)
|
||||
{
|
||||
auto result = truncate_value(Hash32, ((hash + 1) * 11400714819323198485ull) >> 32);
|
||||
let(result, TRUNCATE(Hash32, ((hash + 1) * 11400714819323198485ull) >> 32));
|
||||
return result;
|
||||
}
|
||||
|
||||
#if LINK_LIBC == 0
|
||||
int strcmp(const char* s1, const char* s2)
|
||||
{
|
||||
auto diff = 0;
|
||||
while (1)
|
||||
{
|
||||
auto ch1 = *s1;
|
||||
auto ch2 = *s2;
|
||||
diff = ch1 - ch2;
|
||||
if (ch1 == 0 || ch2 == 0 || diff)
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
s1 += 1;
|
||||
s2 += 1;
|
||||
}
|
||||
|
||||
return diff;
|
||||
}
|
||||
void* memcpy(void* const restrict dst, const void* const restrict src, usize size)
|
||||
{
|
||||
auto* destination = (u8*)dst;
|
||||
auto* source = (u8*)src;
|
||||
|
||||
for (u64 i = 0; i < size; i += 1)
|
||||
{
|
||||
destination[i] = source[i];
|
||||
}
|
||||
|
||||
return dst;
|
||||
}
|
||||
|
||||
void* memmove(void* const dst, const void* const src, usize n)
|
||||
{
|
||||
// Implementation
|
||||
// https://opensource.apple.com/source/network_cmds/network_cmds-481.20.1/unbound/compat/memmove.c.auto.html
|
||||
uint8_t* from = (uint8_t*) src;
|
||||
uint8_t* to = (uint8_t*) dst;
|
||||
|
||||
if (from == to || n == 0)
|
||||
return dst;
|
||||
if (to > from && to-from < (s64)n) {
|
||||
/* to overlaps with from */
|
||||
/* <from......> */
|
||||
/* <to........> */
|
||||
/* copy in reverse, to avoid overwriting from */
|
||||
u64 i;
|
||||
for(i=n-1; i>=0; i--)
|
||||
to[i] = from[i];
|
||||
return dst;
|
||||
}
|
||||
if (from > to && from-to < (int)n) {
|
||||
/* to overlaps with from */
|
||||
/* <from......> */
|
||||
/* <to........> */
|
||||
/* copy forwards, to avoid overwriting from */
|
||||
u64 i;
|
||||
for(i=0; i<n; i++)
|
||||
to[i] = from[i];
|
||||
return dst;
|
||||
}
|
||||
memcpy(dst, src, n);
|
||||
return dst;
|
||||
}
|
||||
|
||||
void* memset(void* dst, int n, usize size)
|
||||
{
|
||||
u8 ch = cast_to(u8, s32, n);
|
||||
auto* destination = (u8*)dst;
|
||||
for (u64 i = 0; i < size; i += 1)
|
||||
{
|
||||
destination[i] = ch;
|
||||
}
|
||||
|
||||
return dst;
|
||||
}
|
||||
|
||||
int memcmp(const void* a, const void* b, usize n)
|
||||
{
|
||||
auto *s1 = (u8*)a;
|
||||
auto *s2 = (u8*)b;
|
||||
|
||||
while (n-- > 0)
|
||||
{
|
||||
if (*s1++ != *s2++)
|
||||
return s1[-1] < s2[-1] ? -1 : 1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
usize strlen(const char* c_string)
|
||||
{
|
||||
auto* it = c_string;
|
||||
while (*it)
|
||||
{
|
||||
it += 1;
|
||||
}
|
||||
return (u64)(it - c_string);
|
||||
}
|
||||
#endif
|
||||
|
479
bootstrap/std/base.h
Normal file
479
bootstrap/std/base.h
Normal file
@ -0,0 +1,479 @@
|
||||
#pragma once
|
||||
|
||||
#define LINK_LIBC 1
|
||||
|
||||
#ifdef NDEBUG
|
||||
#define BB_DEBUG 0
|
||||
#else
|
||||
#define BB_DEBUG 1
|
||||
#endif
|
||||
|
||||
#define BB_INCLUDE_INTRINSIC 0
|
||||
#if BB_DEBUG == 0
|
||||
#undef BB_INCLUDE_INTRINSIC
|
||||
#define BB_INCLUDE_INTRINSIC 1
|
||||
#endif
|
||||
#if BB_INCLUDE_INTRINSIC
|
||||
#if defined(__x86_64__)
|
||||
#include <immintrin.h>
|
||||
#endif
|
||||
#endif
|
||||
#include <stdint.h>
|
||||
#include <stddef.h>
|
||||
|
||||
#define BB_SAFETY BB_DEBUG
|
||||
|
||||
#define STRUCT_FORWARD_DECL(S) typedef struct S S
|
||||
#define STRUCT(S) STRUCT_FORWARD_DECL(S); struct S
|
||||
#define UNION_FORWARD_DECL(U) typedef union U U
|
||||
#define UNION(U) UNION_FORWARD_DECL(U); union U
|
||||
|
||||
#define MIN(a, b) (((a) < (b)) ? (a) : (b))
|
||||
#define MAX(a, b) (((a) > (b)) ? (a) : (b))
|
||||
#define CLAMP(a, x, b) (((a)>(x))?(a):((b)<(x))?(b):(x))
|
||||
|
||||
#if _MSC_VER
|
||||
#define ENUM_START(EnumName, T) typedef T EnumName; typedef enum EnumName ## Flags
|
||||
#define ENUM_END(EnumName) EnumName ## Flags
|
||||
#else
|
||||
#define ENUM_START(EnumName, T) typedef enum EnumName : T
|
||||
#define ENUM_END(EnumName) EnumName
|
||||
#endif
|
||||
|
||||
#define ENUM(EnumName, T, ...) \
|
||||
ENUM_START(EnumName, T)\
|
||||
{\
|
||||
__VA_ARGS__\
|
||||
} ENUM_END(EnumName)
|
||||
|
||||
typedef uint8_t u8;
|
||||
typedef uint16_t u16;
|
||||
typedef uint32_t u32;
|
||||
typedef uint64_t u64;
|
||||
#if defined (__TINYC__) || defined(_MSC_VER)
|
||||
UNION(u128)
|
||||
{
|
||||
struct
|
||||
{
|
||||
u64 low;
|
||||
u64 high;
|
||||
};
|
||||
u64 v[2];
|
||||
};
|
||||
#else
|
||||
typedef __uint128_t u128;
|
||||
#endif
|
||||
typedef unsigned int uint;
|
||||
|
||||
typedef int8_t s8;
|
||||
typedef int16_t s16;
|
||||
typedef int32_t s32;
|
||||
typedef int64_t s64;
|
||||
#if !defined(__TINYC__) && !defined(_MSC_VER)
|
||||
typedef __int128_t s128;
|
||||
#endif
|
||||
|
||||
typedef size_t usize;
|
||||
|
||||
#if !defined(__TINYC__) && !defined(_MSC_VER)
|
||||
typedef _Float16 f16;
|
||||
#endif
|
||||
typedef float f32;
|
||||
typedef double f64;
|
||||
|
||||
typedef u32 Hash32;
|
||||
typedef u64 Hash64;
|
||||
|
||||
#if BB_DEBUG
|
||||
#define assert(x) if (unlikely(!(x))) { my_panic("Assert failed: \"" # x "\" at {cstr}:{u32}\n", __FILE__, __LINE__); }
|
||||
#else
|
||||
#define assert(x) unused(likely(x))
|
||||
#endif
|
||||
|
||||
#ifndef __cplusplus
|
||||
#if _MSC_VER
|
||||
#define unreachable_raw() __assume(0)
|
||||
#else
|
||||
#define unreachable_raw() __builtin_unreachable()
|
||||
#endif
|
||||
// Undefine unreachable if needed to provide a more safe-guard implementation
|
||||
#ifdef unreachable
|
||||
#undef unreachable
|
||||
#endif
|
||||
#if BB_DEBUG
|
||||
#define unreachable() my_panic("Unreachable triggered\n", __FILE__, __LINE__)
|
||||
#else
|
||||
#define unreachable() unreachable_raw()
|
||||
#endif
|
||||
#ifdef __TINYC__
|
||||
#define fix_unreachable() unreachable_raw()
|
||||
#else
|
||||
#define fix_unreachable()
|
||||
#endif
|
||||
|
||||
|
||||
#define static_assert(x) _Static_assert((x), "Static assert failed!")
|
||||
#define alignof(x) _Alignof(x)
|
||||
#else
|
||||
#define restrict __restrict
|
||||
#endif
|
||||
#ifndef BB_INFINITY
|
||||
#define BB_INFINITY __builtin_inff()
|
||||
#endif
|
||||
#ifndef BB_NAN
|
||||
#define BB_NAN __builtin_nanf("")
|
||||
#endif
|
||||
#define fn static
|
||||
#define method __attribute__((visibility("internal")))
|
||||
#define global_variable static
|
||||
#define forceinline __attribute__((always_inline))
|
||||
#if _MSC_VER
|
||||
#define expect(x, b) (!!(x))
|
||||
#else
|
||||
#define expect(x, b) __builtin_expect(!!(x), b)
|
||||
#endif
|
||||
#define likely(x) expect(x, 1)
|
||||
#define unlikely(x) expect(x, 0)
|
||||
#define breakpoint() __builtin_debugtrap()
|
||||
#define failed_execution() my_panic("Failed execution at {cstr}:{u32}\n", __FILE__, __LINE__)
|
||||
#define todo() my_panic("TODO at {cstr}:{u32}\n", __FILE__, __LINE__)
|
||||
|
||||
fn void print(const char* format, ...);
|
||||
fn u8 os_is_being_debugged();
|
||||
fn void os_exit(u32 exit_code);
|
||||
|
||||
#define my_panic(...) do \
|
||||
{\
|
||||
print(__VA_ARGS__);\
|
||||
if (os_is_being_debugged())\
|
||||
{\
|
||||
trap();\
|
||||
fix_unreachable();\
|
||||
}\
|
||||
else\
|
||||
{\
|
||||
os_exit(1);\
|
||||
fix_unreachable();\
|
||||
}\
|
||||
} while (0)
|
||||
|
||||
#if _MSC_VER
|
||||
#define trap() __fastfail(1)
|
||||
#elif __has_builtin(__builtin_trap)
|
||||
#define trap() __builtin_trap()
|
||||
#else
|
||||
fn void trap()
|
||||
{
|
||||
asm volatile("ud2");
|
||||
}
|
||||
#endif
|
||||
|
||||
#define let_pointer_cast(PointerChildType, var_name, value) PointerChildType* var_name = (PointerChildType*)(value)
|
||||
#if defined(__TINYC__) || defined(_MSC_VER)
|
||||
#define let(name, value) typeof(value) name = (value)
|
||||
#else
|
||||
#define let(name, value) __auto_type name = (value)
|
||||
#endif
|
||||
#define let_cast_unchecked(name, T, value) T name = (T)(value)
|
||||
#define let_cast(T, name, value) T name = cast_to(T, value)
|
||||
#define let_va_arg(T, name, args) T name = va_arg(args, T)
|
||||
#define assign_cast(to, from) to = cast_to(typeof(to), from)
|
||||
#define transmute(D, source) *(D*)&source
|
||||
|
||||
UNION(SafeInteger)
|
||||
{
|
||||
s64 signed_value;
|
||||
u64 unsigned_value;
|
||||
};
|
||||
|
||||
fn SafeInteger safe_integer_cast(SafeInteger value, u64 to_size, u64 to_signedness, u64 from_size, u64 from_signedness)
|
||||
{
|
||||
SafeInteger result;
|
||||
let(shifter, to_size * 8 - to_signedness);
|
||||
let(to_max, (u64)(1 << shifter) - 1);
|
||||
// A fix for 64-bit wrapping
|
||||
to_max = to_max == 0 ? UINT64_MAX : to_max;
|
||||
let(to_signed_min, -((s64)1 << shifter));
|
||||
if (from_signedness == to_signedness)
|
||||
{
|
||||
if (to_size < from_size)
|
||||
{
|
||||
switch (to_signedness)
|
||||
{
|
||||
case 0:
|
||||
{
|
||||
if (value.unsigned_value > to_max)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
} break;
|
||||
case 1:
|
||||
{
|
||||
if (value.signed_value < to_signed_min)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
if (value.signed_value > (s64)to_max)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (from_signedness)
|
||||
{
|
||||
if (value.signed_value < 0)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
else if (value.unsigned_value > to_max)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (value.unsigned_value > to_max)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
result = value;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
#define type_is_signed(T) ((T)(-1) < 0)
|
||||
#if BB_SAFETY
|
||||
#define safe_integer_cast_function(To, value) (To) ((value) < 0 ? (safe_integer_cast((SafeInteger) { .signed_value = (value) }, sizeof(To), type_is_signed(To), sizeof(typeof(value)), type_is_signed(typeof(value)))).signed_value : (safe_integer_cast((SafeInteger) { .signed_value = (value) }, sizeof(To), type_is_signed(To), sizeof(typeof(value)), type_is_signed(typeof(value)))).unsigned_value)
|
||||
#endif
|
||||
|
||||
#if BB_SAFETY
|
||||
#define cast_to(T, value) safe_integer_cast_function(T, value)
|
||||
#else
|
||||
#define cast_to(T, value) (T)(value)
|
||||
#endif
|
||||
|
||||
|
||||
typedef enum Corner
|
||||
{
|
||||
CORNER_00,
|
||||
CORNER_01,
|
||||
CORNER_10,
|
||||
CORNER_11,
|
||||
CORNER_COUNT,
|
||||
} Corner;
|
||||
|
||||
typedef enum Axis2
|
||||
{
|
||||
AXIS2_X,
|
||||
AXIS2_Y,
|
||||
AXIS2_COUNT,
|
||||
} Axis2;
|
||||
|
||||
// #ifdef __cplusplus
|
||||
// #define EXPORT extern "C"
|
||||
// #else
|
||||
// #define EXPORT
|
||||
// #endif
|
||||
|
||||
#if defined(__cplusplus) && defined(__linux__)
|
||||
#define NO_EXCEPT __THROW
|
||||
#else
|
||||
#define NO_EXCEPT
|
||||
#endif
|
||||
|
||||
|
||||
#define Slice(T) Slice_ ## T
|
||||
#define SliceP(T) SliceP_ ## T
|
||||
#define declare_slice_ex(T, StructName) STRUCT(StructName) \
|
||||
{\
|
||||
T* pointer;\
|
||||
u64 length;\
|
||||
}
|
||||
|
||||
#define declare_slice(T) declare_slice_ex(T, Slice(T))
|
||||
#define declare_slice_p(T) declare_slice_ex(T*, SliceP(T))
|
||||
|
||||
declare_slice(u8);
|
||||
declare_slice(u16);
|
||||
declare_slice(u32);
|
||||
declare_slice(u64);
|
||||
declare_slice(s8);
|
||||
declare_slice(s16);
|
||||
declare_slice(s32);
|
||||
declare_slice(s64);
|
||||
|
||||
declare_slice_p(char);
|
||||
declare_slice_p(void);
|
||||
|
||||
typedef Slice(u8) String;
|
||||
declare_slice(String);
|
||||
|
||||
#define NamedEnumMemberEnum(e, enum_member) e ## _ ## enum_member
|
||||
#define NamedEnumMemberString(e, enum_member) strlit(#enum_member)
|
||||
|
||||
typedef SliceP(char) CStringSlice;
|
||||
|
||||
#ifdef _WIN32
|
||||
typedef void* FileDescriptor;
|
||||
#else
|
||||
typedef int FileDescriptor;
|
||||
#endif
|
||||
|
||||
#define FOR_N(it, start, end) \
|
||||
for (u32 it = (start), end__ = (end); it < end__; ++it)
|
||||
|
||||
#define FOR_REV_N(it, start, end) \
|
||||
for (u32 it = (end), start__ = (start); (it--) > start__;)
|
||||
|
||||
#define FOR_BIT(it, start, bits) \
|
||||
for (typeof(bits) _bits_ = (bits), it = (start); _bits_; _bits_ >>= 1, ++it) if (_bits_ & 1)
|
||||
|
||||
#define FOREACH_SET(it, set) \
|
||||
FOR_N(_i, 0, ((set)->arr.capacity + 63) / 64) FOR_BIT(it, _i*64, (set)->arr.pointer[_i])
|
||||
|
||||
|
||||
#ifdef __TINYC__
|
||||
#define declare_vector_type #error
|
||||
#else
|
||||
#ifdef __clang__
|
||||
#define declare_vector_type(T, count, name) typedef T name __attribute__((ext_vector_type(count)))
|
||||
#else
|
||||
#define declare_vector_type(T, count, name) typedef T name __attribute__((vector_size(count)))
|
||||
#endif
|
||||
#endif
|
||||
#define array_length(arr) sizeof(arr) / sizeof((arr)[0])
|
||||
#define KB(n) ((n) * 1024)
|
||||
#define MB(n) ((n) * 1024 * 1024)
|
||||
#define GB(n) ((u64)(n) * 1024 * 1024 * 1024)
|
||||
#define TB(n) ((u64)(n) * 1024 * 1024 * 1024 * 1024)
|
||||
#define unused(x) (void)(x)
|
||||
#ifdef __clang__
|
||||
#define may_be_unused __attribute__((unused))
|
||||
#else
|
||||
#define may_be_unused
|
||||
#endif
|
||||
#if _MSC_VER
|
||||
#define BB_NORETURN __declspec(noreturn)
|
||||
#define BB_COLD __declspec(noinline)
|
||||
#elif defined(__TINYC__)
|
||||
#define BB_NORETURN __attribute__((noreturn))
|
||||
#define BB_COLD __attribute__((cold))
|
||||
#else
|
||||
#define BB_NORETURN [[noreturn]]
|
||||
#define BB_COLD [[gnu::cold]]
|
||||
#endif
|
||||
#define TRUNCATE(Destination, source) (Destination)(source)
|
||||
#define size_until_end(T, field_name) (sizeof(T) - offsetof(T, field_name))
|
||||
#define SWAP(a, b) \
|
||||
do {\
|
||||
static_assert(typeof(a) == typeof(b));\
|
||||
let(temp, a);\
|
||||
a = b;\
|
||||
b = temp;\
|
||||
} while (0)
|
||||
|
||||
#define slice_from_pointer_range(T, start, end) (Slice(T)) { .pointer = start, .length = (u64)(end - start), }
|
||||
|
||||
#define strlit_len(s) (sizeof(s) - 1)
|
||||
#define strlit(s) (String){ .pointer = (u8*)(s), .length = strlit_len(s), }
|
||||
#define ch_to_str(ch) (String){ .pointer = &ch, .length = 1 }
|
||||
#define array_to_slice(arr) { .pointer = (arr), .length = array_length(arr) }
|
||||
#define array_to_bytes(arr) { .pointer = (u8*)(arr), .length = sizeof(arr) }
|
||||
#define pointer_to_bytes(p) (String) { .pointer = (u8*)(p), .length = sizeof(*p) }
|
||||
#define scalar_to_bytes(s) pointer_to_bytes(&(s))
|
||||
#define string_to_c(s) ((char*)((s).pointer))
|
||||
#define cstr(s) ((String) { .pointer = (u8*)(s), .length = strlen((char*)s), } )
|
||||
|
||||
#define case_to_name(prefix, e) case prefix ## e: return strlit(#e)
|
||||
|
||||
const may_be_unused global_variable u8 brace_open = '{';
|
||||
const may_be_unused global_variable u8 brace_close = '}';
|
||||
|
||||
const may_be_unused global_variable u8 parenthesis_open = '(';
|
||||
const may_be_unused global_variable u8 parenthesis_close = ')';
|
||||
|
||||
const may_be_unused global_variable u8 bracket_open = '[';
|
||||
const may_be_unused global_variable u8 bracket_close = ']';
|
||||
|
||||
#define s_get(s, i) (s).pointer[i]
|
||||
#define s_get_pointer(s, i) &((s).pointer[i])
|
||||
#define s_get_slice(T, s, start, end) (Slice(T)){ .pointer = ((s).pointer) + (start), .length = (end) - (start) }
|
||||
#define s_equal(a, b) ((a).length == (b).length && memcmp((a).pointer, (b).pointer, sizeof(*((a).pointer)) * (a).length) == 0)
|
||||
|
||||
|
||||
fn u64 align_forward(u64 value, u64 alignment);
|
||||
fn u64 align_backward(u64 value, u64 alignment);
|
||||
fn u8 log2_alignment(u64 alignment);
|
||||
fn u8 is_power_of_two(u64 value);
|
||||
fn u8 first_bit_set_32(u32 value);
|
||||
fn u64 first_bit_set_64(u64 value);
|
||||
|
||||
fn u8 cast_u32_to_u8(u32 source, const char* name, int line);
|
||||
fn u16 cast_u32_to_u16(u32 source, const char* name, int line);
|
||||
fn s16 cast_u32_to_s16(u32 source, const char* name, int line);
|
||||
fn s32 cast_u32_to_s32(u32 source, const char* name, int line);
|
||||
fn u8 cast_u64_to_u8(u64 source, const char* name, int line);
|
||||
fn u16 cast_u64_to_u16(u64 source, const char* name, int line);
|
||||
fn u32 cast_u64_to_u32(u64 source, const char* name, int line);
|
||||
fn s32 cast_u64_to_s32(u64 source, const char* name, int line);
|
||||
fn s64 cast_u64_to_s64(u64 source, const char* name, int line);
|
||||
fn u8 cast_s32_to_u8(s32 source, const char* name, int line);
|
||||
fn u16 cast_s32_to_u16(s32 source, const char* name, int line);
|
||||
fn u32 cast_s32_to_u32(s32 source, const char* name, int line);
|
||||
fn u64 cast_s32_to_u64(s32 source, const char* name, int line);
|
||||
fn s16 cast_s32_to_s16(s32 source, const char* name, int line);
|
||||
fn u16 cast_s64_to_u16(s64 source, const char* name, int line);
|
||||
fn u32 cast_s64_to_u32(s64 source, const char* name, int line);
|
||||
fn u64 cast_s64_to_u64(s64 source, const char* name, int line);
|
||||
fn s32 cast_s64_to_s32(s64 source, const char* name, int line);
|
||||
|
||||
fn u32 format_decimal(String buffer, u64 decimal);
|
||||
fn u32 format_hexadecimal(String buffer, u64 hexadecimal);
|
||||
fn u64 format_float(String buffer, f64 value_double);
|
||||
|
||||
fn u64 is_decimal_digit(u8 ch);
|
||||
fn u32 is_space(u8 ch, u8 next_ch);
|
||||
fn u8 get_next_ch_safe(String string, u64 index);
|
||||
fn u64 is_identifier_start(u8 ch);
|
||||
fn u64 is_identifier_ch(u8 ch);
|
||||
fn u64 is_alphabetic(u8 ch);
|
||||
|
||||
fn u64 parse_decimal(String string);
|
||||
|
||||
global_variable const Hash64 fnv_offset = 14695981039346656037ull;
|
||||
global_variable const u64 fnv_prime = 1099511628211ull;
|
||||
|
||||
fn Hash32 hash32_fib_end(Hash32 hash);
|
||||
fn Hash32 hash64_fib_end(Hash64 hash);
|
||||
|
||||
fn Hash64 hash_byte(Hash64 source, u8 ch);
|
||||
fn Hash64 hash_bytes(String bytes);
|
||||
fn Hash32 hash64_to_hash32(Hash64 hash64);
|
||||
|
||||
fn u64 round_up_to_next_power_of_2(u64 n);
|
||||
|
||||
STRUCT(TextureIndex)
|
||||
{
|
||||
u32 value;
|
||||
};
|
||||
|
||||
fn u64 safe_flag(u64 value, u64 flag)
|
||||
{
|
||||
u64 result = value & ((u64)0 - flag);
|
||||
return result;
|
||||
}
|
||||
|
||||
#define member_from_offset(pointer, type, memory_offset) (*(type*)((u8*)pointer + memory_offset))
|
||||
#if _MSC_VER
|
||||
#define offset_of(T, member) offsetof(T, member)
|
||||
#else
|
||||
#define offset_of(T, member) __builtin_offsetof(T, member)
|
||||
#endif
|
||||
|
65
bootstrap/std/cocoa_windowing.c
Normal file
65
bootstrap/std/cocoa_windowing.c
Normal file
@ -0,0 +1,65 @@
|
||||
#pragma once
|
||||
|
||||
@implementation AppleApplicationDelegate
|
||||
- (void)applicationDidFinishLaunching:(NSNotification*)aNotification
|
||||
{
|
||||
}
|
||||
|
||||
@end
|
||||
@implementation AppleWindow
|
||||
@end
|
||||
|
||||
global_variable WindowConnection window_connection;
|
||||
|
||||
fn u8 windowing_initialize()
|
||||
{
|
||||
u8 result = 1;
|
||||
window_connection.application = [NSApplication sharedApplication];
|
||||
AppleApplicationDelegate* application_delegate = [[AppleApplicationDelegate alloc] init];
|
||||
NSApp.delegate = application_delegate;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
fn WindowingInstance* windowing_instantiate(WindowingInstantiate instantiate)
|
||||
{
|
||||
NSRect rect = { { 0, 0 }, { 800, 600 } };
|
||||
AppleWindow* window = [[AppleWindow alloc] initWithContentRect:rect styleMask:(NSWindowStyleMaskTitled | NSWindowStyleMaskClosable | NSWindowStyleMaskResizable) backing:NSBackingStoreBuffered defer:NO];
|
||||
window.title = @"Hello Metal";
|
||||
[window_connection.application activate];
|
||||
[window orderFrontRegardless];
|
||||
return window;
|
||||
}
|
||||
|
||||
fn WindowingSize windowing_get_instance_framebuffer_size(WindowingInstance* instance)
|
||||
{
|
||||
WindowingSize size;
|
||||
@autoreleasepool {
|
||||
const NSRect contentRect = instance.contentView.frame;
|
||||
const NSRect fbRect = [instance.contentView convertRectToBacking:contentRect];
|
||||
|
||||
size = (WindowingSize) {
|
||||
.width = fbRect.size.width,
|
||||
.height = fbRect.size.height,
|
||||
};
|
||||
} // autoreleasepool
|
||||
return size;
|
||||
}
|
||||
|
||||
fn void windowing_poll_events()
|
||||
{
|
||||
@autoreleasepool {
|
||||
while (1)
|
||||
{
|
||||
NSEvent* event = [NSApp nextEventMatchingMask:NSEventMaskAny untilDate:[NSDate distantPast] inMode:NSDefaultRunLoopMode dequeue:YES];
|
||||
|
||||
if (event == nil)
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
[NSApp sendEvent:event];
|
||||
}
|
||||
|
||||
} // autoreleasepool
|
||||
}
|
18
bootstrap/std/cocoa_windowing.h
Normal file
18
bootstrap/std/cocoa_windowing.h
Normal file
@ -0,0 +1,18 @@
|
||||
#pragma once
|
||||
|
||||
#import <Cocoa/Cocoa.h>
|
||||
#import <QuartzCore/CAMetalLayer.h>
|
||||
|
||||
@interface AppleApplicationDelegate : NSObject<NSApplicationDelegate>
|
||||
@end
|
||||
@interface AppleWindow : NSWindow
|
||||
@end
|
||||
@interface AppleWindowDelegate : NSObject<NSWindowDelegate>
|
||||
@end
|
||||
|
||||
typedef AppleWindow WindowingInstance;
|
||||
|
||||
STRUCT(WindowConnection)
|
||||
{
|
||||
NSApplication* application;
|
||||
};
|
1
bootstrap/std/directx12_rendering.c
Normal file
1
bootstrap/std/directx12_rendering.c
Normal file
@ -0,0 +1 @@
|
||||
#pragma once
|
1
bootstrap/std/directx12_rendering.h
Normal file
1
bootstrap/std/directx12_rendering.h
Normal file
@ -0,0 +1 @@
|
||||
#pragma once
|
@ -2,7 +2,7 @@
|
||||
#include <std/os.h>
|
||||
|
||||
#if LINK_LIBC == 0
|
||||
[[gnu::naked]] [[noreturn]] void _start()
|
||||
[[gnu::naked]] BB_NORETURN void _start()
|
||||
{
|
||||
__asm__ __volatile__(
|
||||
"\nxor %ebp, %ebp"
|
||||
|
@ -3,7 +3,7 @@
|
||||
void entry_point(int argc, char* argv[], char* envp[]);
|
||||
|
||||
#if LINK_LIBC == 0
|
||||
[[gnu::naked]] [[noreturn]] void _start();
|
||||
[[gnu::naked]] BB_NORETURN void _start();
|
||||
#endif
|
||||
|
||||
#if LINK_LIBC == 0
|
@ -1 +0,0 @@
|
||||
|
@ -1,15 +1,56 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/font_provider.h>
|
||||
|
||||
#define STBTT_STATIC
|
||||
#define STB_TRUETYPE_IMPLEMENTATION
|
||||
#define stbtt_uint8 u8
|
||||
#define stbtt_uint16 u16
|
||||
#define stbtt_uint32 u32
|
||||
#define stbtt_int8 s8
|
||||
#define stbtt_int16 s16
|
||||
#define stbtt_int32 s32
|
||||
|
||||
extern float sqrtf(float x);
|
||||
extern float roundf(float x);
|
||||
extern float floorf(float x);
|
||||
|
||||
extern double sqrt(double x);
|
||||
extern double fabs(double x);
|
||||
extern double floor(double x);
|
||||
extern double ceil(double x);
|
||||
extern double fmod(double x, double y);
|
||||
extern double pow(double x, double y);
|
||||
extern double acos(double x);
|
||||
extern double cos(double x);
|
||||
|
||||
#define STBTT_ifloor(x) ((int) floor(x))
|
||||
#define STBTT_iceil(x) ((int) ceil(x))
|
||||
#define STBTT_sqrt(x) sqrt(x)
|
||||
#define STBTT_pow(x,y) pow(x,y)
|
||||
#define STBTT_fmod(x,y) fmod(x,y)
|
||||
#define STBTT_cos(x) cos(x)
|
||||
#define STBTT_acos(x) acos(x)
|
||||
#define STBTT_fabs(x) fabs(x)
|
||||
#define STBTT_malloc(x,u) ((void)(u),malloc(x))
|
||||
#define STBTT_free(x,u) ((void)(u),free(x))
|
||||
#define STBTT_assert(x) assert(x)
|
||||
#define STBTT_strlen(x) strlen(x)
|
||||
#define STBTT_memcpy memcpy
|
||||
#define STBTT_memset memset
|
||||
|
||||
#ifdef __clang__
|
||||
#pragma clang diagnostic push
|
||||
#pragma clang diagnostic ignored "-Wunused-function"
|
||||
#endif
|
||||
#include <stb_truetype.h>
|
||||
#ifdef __clang__
|
||||
#pragma clang diagnostic pop
|
||||
#endif
|
||||
|
||||
TextureAtlas font_texture_atlas_create(Arena* arena, Renderer* renderer, TextureAtlasCreate create)
|
||||
fn TextureAtlas font_texture_atlas_create(Arena* arena, Renderer* renderer, TextureAtlasCreate create)
|
||||
{
|
||||
auto font_file = file_read(arena, create.font_path);
|
||||
let(font_file, file_read(arena, create.font_path));
|
||||
stbtt_fontinfo font_info;
|
||||
if (!stbtt_InitFont(&font_info, font_file.pointer, stbtt_GetFontOffsetForIndex(font_file.pointer, 0)))
|
||||
{
|
||||
@ -23,7 +64,7 @@ TextureAtlas font_texture_atlas_create(Arena* arena, Renderer* renderer, Texture
|
||||
result.height = (u32)sqrtf((f32)(create.text_height * create.text_height * character_count));
|
||||
result.width = result.height;
|
||||
result.pointer = arena_allocate(arena, u32, result.width * result.height);
|
||||
auto scale_factor = stbtt_ScaleForPixelHeight(&font_info, create.text_height);
|
||||
let(scale_factor, stbtt_ScaleForPixelHeight(&font_info, create.text_height));
|
||||
|
||||
int ascent;
|
||||
int descent;
|
||||
@ -40,25 +81,25 @@ TextureAtlas font_texture_atlas_create(Arena* arena, Renderer* renderer, Texture
|
||||
u32 first_character = ' ';
|
||||
u32 last_character = '~';
|
||||
|
||||
for (auto i = first_character; i <= last_character; ++i)
|
||||
for (let(i, first_character); i <= last_character; ++i)
|
||||
{
|
||||
u32 width;
|
||||
u32 height;
|
||||
int advance;
|
||||
int left_bearing;
|
||||
|
||||
auto ch = (u8)i;
|
||||
auto* character = &result.characters[i];
|
||||
let(ch, (u8)i);
|
||||
let(character, &result.characters[i]);
|
||||
stbtt_GetCodepointHMetrics(&font_info, ch, &advance, &left_bearing);
|
||||
|
||||
character->advance = (u32)roundf(advance * scale_factor);
|
||||
character->left_bearing = (u32)roundf(left_bearing * scale_factor);
|
||||
|
||||
u8* bitmap = stbtt_GetCodepointBitmap(&font_info, 0.0f, scale_factor, ch, (int*)&width, (int*)&height, &character->x_offset, &character->y_offset);
|
||||
auto* kerning_table = result.kerning_tables + i * character_count;
|
||||
let(kerning_table, result.kerning_tables + i * character_count);
|
||||
for (u32 j = first_character; j <= last_character; j += 1)
|
||||
{
|
||||
auto kerning_advance = stbtt_GetCodepointKernAdvance(&font_info, i, j);
|
||||
let(kerning_advance, stbtt_GetCodepointKernAdvance(&font_info, i, j));
|
||||
kerning_table[j] = (s32)roundf(kerning_advance * scale_factor);
|
||||
}
|
||||
|
||||
@ -78,16 +119,16 @@ TextureAtlas font_texture_atlas_create(Arena* arena, Renderer* renderer, Texture
|
||||
character->width = width;
|
||||
character->height = height;
|
||||
|
||||
auto* source = bitmap;
|
||||
auto* destination = result.pointer;
|
||||
let(source, bitmap);
|
||||
let(destination, result.pointer);
|
||||
|
||||
for (u32 bitmap_y = 0; bitmap_y < height; bitmap_y += 1)
|
||||
{
|
||||
for (u32 bitmap_x = 0; bitmap_x < width; bitmap_x += 1)
|
||||
{
|
||||
auto source_index = bitmap_y * width + bitmap_x;
|
||||
auto destination_index = (bitmap_y + y) * result.width + (bitmap_x + x);
|
||||
auto value = source[source_index];
|
||||
let(source_index, bitmap_y * width + bitmap_x);
|
||||
let(destination_index, (bitmap_y + y) * result.width + (bitmap_x + x));
|
||||
let(value, source[source_index]);
|
||||
destination[destination_index] = ((u32)value << 24) | 0xffffff;
|
||||
}
|
||||
}
|
||||
@ -108,17 +149,17 @@ TextureAtlas font_texture_atlas_create(Arena* arena, Renderer* renderer, Texture
|
||||
return result;
|
||||
}
|
||||
|
||||
uint2 texture_atlas_compute_string_rect(String string, const TextureAtlas* atlas)
|
||||
fn uint2 texture_atlas_compute_string_rect(String string, const TextureAtlas* atlas)
|
||||
{
|
||||
auto height = atlas->ascent - atlas->descent;
|
||||
let(height, atlas->ascent - atlas->descent);
|
||||
u32 x_offset = 0;
|
||||
u32 y_offset = height;
|
||||
|
||||
for (u64 i = 0; i < string.length; i += 1)
|
||||
{
|
||||
auto ch = string.pointer[i];
|
||||
auto* character = &atlas->characters[ch];
|
||||
auto kerning = (atlas->kerning_tables + ch * 256)[string.pointer[i + 1]];
|
||||
let(ch, string.pointer[i]);
|
||||
let(character, &atlas->characters[ch]);
|
||||
let(kerning, (atlas->kerning_tables + ch * 256)[string.pointer[i + 1]]);
|
||||
x_offset += character->advance + kerning;
|
||||
}
|
||||
|
||||
|
@ -1,8 +1,5 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/base.h>
|
||||
#include <std/os.h>
|
||||
|
||||
STRUCT(FontCharacter)
|
||||
{
|
||||
u32 advance;
|
||||
@ -34,7 +31,5 @@ STRUCT(TextureAtlasCreate)
|
||||
u32 text_height;
|
||||
};
|
||||
|
||||
#include <std/render.h>
|
||||
|
||||
EXPORT TextureAtlas font_texture_atlas_create(Arena* arena, Renderer* renderer, TextureAtlasCreate create);
|
||||
EXPORT uint2 texture_atlas_compute_string_rect(String string, const TextureAtlas* atlas);
|
||||
fn TextureAtlas font_texture_atlas_create(Arena* arena, Renderer* renderer, TextureAtlasCreate create);
|
||||
fn uint2 texture_atlas_compute_string_rect(String string, const TextureAtlas* atlas);
|
@ -1,6 +1,6 @@
|
||||
#include <std/format.h>
|
||||
|
||||
u32 format_hexadecimal(String buffer, u64 hexadecimal)
|
||||
fn u32 format_hexadecimal(String buffer, u64 hexadecimal)
|
||||
{
|
||||
u64 value = hexadecimal;
|
||||
if (value)
|
||||
@ -35,7 +35,7 @@ u32 format_hexadecimal(String buffer, u64 hexadecimal)
|
||||
}
|
||||
}
|
||||
|
||||
u32 format_decimal(String buffer, u64 decimal)
|
||||
fn u32 format_decimal(String buffer, u64 decimal)
|
||||
{
|
||||
u64 value = decimal;
|
||||
if (value)
|
||||
@ -299,9 +299,10 @@ global_variable const u64 DOUBLE_POW5_INV_SPLIT[DOUBLE_POW5_INV_TABLE_SIZE][2] =
|
||||
// Best case: use 128-bit type.
|
||||
fn u64 mul_shift_64(const u64 m, const u64* const mul, const s32 j)
|
||||
{
|
||||
const u128 b0 = ((u128) m) * mul[0];
|
||||
const u128 b2 = ((u128) m) * mul[1];
|
||||
return (u64) (((b0 >> 64) + b2) >> (j - 64));
|
||||
const u128 b0 = u128_u64_mul(u128_from_u64(m), mul[0]);
|
||||
const u128 b2 = u128_u64_mul(u128_from_u64(m), mul[1]);
|
||||
return u64_from_u128(u128_shift_right(u128_u64_add(b2, u128_shift_right_by_64(b0)), j - 64));
|
||||
// return (u64) (((b0 >> 64) + b2) >> (j - 64));
|
||||
}
|
||||
|
||||
fn u64 mul_shift_all_64(const u64 m, const u64* const mul, const s32 j, u64* const vp, u64* const vm, const u32 mmShift)
|
||||
@ -526,27 +527,27 @@ fn u8 multiple_of_power_of_2(const u64 value, const u32 p) {
|
||||
return (value & ((1ull << p) - 1)) == 0;
|
||||
}
|
||||
|
||||
static inline uint64_t div5(const uint64_t x) {
|
||||
fn inline uint64_t div5(const uint64_t x) {
|
||||
return x / 5;
|
||||
}
|
||||
|
||||
static inline uint64_t div10(const uint64_t x) {
|
||||
fn inline uint64_t div10(const uint64_t x) {
|
||||
return x / 10;
|
||||
}
|
||||
|
||||
static inline uint64_t div100(const uint64_t x) {
|
||||
fn inline uint64_t div100(const uint64_t x) {
|
||||
return x / 100;
|
||||
}
|
||||
|
||||
static inline uint64_t div1e8(const uint64_t x) {
|
||||
fn inline uint64_t div1e8(const uint64_t x) {
|
||||
return x / 100000000;
|
||||
}
|
||||
|
||||
static inline uint64_t div1e9(const uint64_t x) {
|
||||
fn inline uint64_t div1e9(const uint64_t x) {
|
||||
return x / 1000000000;
|
||||
}
|
||||
|
||||
static inline uint32_t mod1e9(const uint64_t x) {
|
||||
fn inline uint32_t mod1e9(const uint64_t x) {
|
||||
return (uint32_t) (x - 1000000000 * div1e9(x));
|
||||
}
|
||||
|
||||
@ -572,7 +573,7 @@ may_be_unused fn Double double_transform(u64 ieee_mantissa, u32 ieee_exponent)
|
||||
}
|
||||
|
||||
u8 is_even = (m2 & 1) == 0;
|
||||
auto accept_bounds = is_even;
|
||||
let(accept_bounds, is_even);
|
||||
|
||||
u64 mv = 4 * m2;
|
||||
u32 mm_shift = (ieee_mantissa != 0) | (ieee_exponent <= 1);
|
||||
@ -742,8 +743,8 @@ may_be_unused fn Double double_transform(u64 ieee_mantissa, u32 ieee_exponent)
|
||||
may_be_unused fn SmallIntResult small_int(u64 ieee_mantissa, u32 ieee_exponent)
|
||||
{
|
||||
SmallIntResult result = {};
|
||||
auto m2 = ((u64)1 << double_mantissa_bits) | ieee_mantissa;
|
||||
auto e2 = (s32)ieee_exponent - double_bias - double_mantissa_bits;
|
||||
let(m2, ((u64)1 << double_mantissa_bits) | ieee_mantissa);
|
||||
let(e2, (s32)ieee_exponent - double_bias - double_mantissa_bits);
|
||||
|
||||
if (e2 > 0)
|
||||
{
|
||||
@ -805,7 +806,7 @@ STRUCT(floating_decimal_64)
|
||||
|
||||
fn u8* digits2(u64 value)
|
||||
{
|
||||
auto str = strlit("00010203040506070809101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899");
|
||||
String str = strlit("00010203040506070809101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899");
|
||||
return str.pointer + (value * 2);
|
||||
}
|
||||
|
||||
@ -815,9 +816,9 @@ fn void write_float_decimal(String buffer, u64* value, u64 count)
|
||||
|
||||
while (i + 2 < count)
|
||||
{
|
||||
auto c = cast_to(u8, u64, *value % 100);
|
||||
let_cast(u8, c, *value % 100);
|
||||
*value /= 100;
|
||||
auto ptr = digits2(c);
|
||||
let(ptr, digits2(c));
|
||||
buffer.pointer[count - i - 1] = ptr[1];
|
||||
buffer.pointer[count - i - 2] = ptr[0];
|
||||
i += 2;
|
||||
@ -825,7 +826,7 @@ fn void write_float_decimal(String buffer, u64* value, u64 count)
|
||||
|
||||
while (i < count)
|
||||
{
|
||||
auto c = cast_to(u8, u64, *value % 10);
|
||||
let(c, cast_to(u8, *value % 10));
|
||||
*value /= 10;
|
||||
buffer.pointer[count - i - 1] = '0' + c;
|
||||
|
||||
@ -833,22 +834,20 @@ fn void write_float_decimal(String buffer, u64* value, u64 count)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
u64 format_float(String buffer, f64 value_double)
|
||||
fn u64 format_float(String buffer, f64 value_double)
|
||||
{
|
||||
auto value_int = *(u64*)&value_double;
|
||||
let(value_int, *(u64*)&value_double);
|
||||
u64 buffer_i = 0;
|
||||
|
||||
const u8 ieee_sign = ((value_int >> (double_mantissa_bits + double_exponent_bits)) & 1) != 0;
|
||||
const auto ieee_mantissa = value_int & (((u64)1 << double_mantissa_bits) - 1);
|
||||
const auto ieee_exponent = (u32)((value_int >> double_mantissa_bits) & (((u32)1 << double_exponent_bits) - 1));
|
||||
let(ieee_mantissa, value_int & (((u64)1 << double_mantissa_bits) - 1));
|
||||
let(ieee_exponent, (u32)((value_int >> double_mantissa_bits) & (((u32)1 << double_exponent_bits) - 1)));
|
||||
|
||||
if (ieee_exponent == (((u32)1 << double_exponent_bits) - 1) || (ieee_exponent == 0 && ieee_mantissa == 0))
|
||||
{
|
||||
if (ieee_mantissa)
|
||||
{
|
||||
auto nan = strlit("NaN");
|
||||
String nan = strlit("NaN");
|
||||
memcpy(&buffer.pointer[buffer_i], nan.pointer, nan.length);
|
||||
buffer_i += nan.length;
|
||||
}
|
||||
@ -862,13 +861,13 @@ u64 format_float(String buffer, f64 value_double)
|
||||
|
||||
if (ieee_exponent)
|
||||
{
|
||||
auto inf = strlit("Infinity");
|
||||
String inf = strlit("Infinity");
|
||||
memcpy(&buffer.pointer[buffer_i], inf.pointer, inf.length);
|
||||
buffer_i += inf.length;
|
||||
}
|
||||
else
|
||||
{
|
||||
auto e0 = strlit("0E0");
|
||||
String e0 = strlit("0E0");
|
||||
memcpy(&buffer.pointer[buffer_i], e0.pointer, e0.length);
|
||||
buffer_i += e0.length;
|
||||
}
|
||||
@ -876,7 +875,7 @@ u64 format_float(String buffer, f64 value_double)
|
||||
}
|
||||
else
|
||||
{
|
||||
auto small_int_result = small_int(ieee_mantissa, ieee_exponent);
|
||||
let(small_int_result, small_int(ieee_mantissa, ieee_exponent));
|
||||
Double result;
|
||||
if (small_int_result.is_small_int)
|
||||
{
|
||||
@ -934,8 +933,8 @@ u64 format_float(String buffer, f64 value_double)
|
||||
const uint32_t d0 = (d % 100) << 1;
|
||||
const uint32_t d1 = (d / 100) << 1;
|
||||
|
||||
auto base_index = buffer_i + olength;
|
||||
auto base = buffer.pointer + base_index;
|
||||
let(base_index, buffer_i + olength);
|
||||
let(base, buffer.pointer + base_index);
|
||||
memcpy(base - 1, DIGIT_TABLE + c0, 2);
|
||||
memcpy(base - 3, DIGIT_TABLE + c1, 2);
|
||||
memcpy(base - 5, DIGIT_TABLE + d0, 2);
|
||||
@ -944,7 +943,7 @@ u64 format_float(String buffer, f64 value_double)
|
||||
i += 8;
|
||||
}
|
||||
|
||||
auto output2 = (u32) output;
|
||||
let(output2, (u32)output);
|
||||
|
||||
while (output2 >= 10000)
|
||||
{
|
||||
@ -956,7 +955,7 @@ u64 format_float(String buffer, f64 value_double)
|
||||
output2 /= 10000;
|
||||
const u32 c0 = (c % 100) << 1;
|
||||
const u32 c1 = (c / 100) << 1;
|
||||
auto base_index = buffer_i + olength - i;
|
||||
let(base_index, buffer_i + olength - i);
|
||||
memcpy(buffer.pointer + base_index - 1, DIGIT_TABLE + c0, 2);
|
||||
memcpy(buffer.pointer + base_index - 3, DIGIT_TABLE + c1, 2);
|
||||
|
||||
@ -1022,7 +1021,7 @@ u64 format_float(String buffer, f64 value_double)
|
||||
} break;
|
||||
case FLOAT_FORMAT_DECIMAL:
|
||||
{
|
||||
auto dp_offset = result.exponent + cast_to(s32, u32, olength);
|
||||
let(dp_offset, result.exponent + cast_to(s32, olength));
|
||||
|
||||
if (dp_offset <= 0)
|
||||
{
|
||||
@ -1030,9 +1029,9 @@ u64 format_float(String buffer, f64 value_double)
|
||||
buffer.pointer[buffer_i + 1] = '.';
|
||||
buffer_i += 2;
|
||||
|
||||
// auto dp_index = buffer_i;
|
||||
// let(dp_index, buffer_i);
|
||||
|
||||
auto dp_poffset = (u32)(-dp_offset);
|
||||
let(dp_poffset, (u32)(-dp_offset));
|
||||
memset(buffer.pointer + buffer_i, '0', dp_poffset);
|
||||
buffer_i += dp_poffset;
|
||||
write_float_decimal(s_get_slice(u8, buffer, buffer_i, buffer.length), &output, olength);
|
||||
@ -1040,13 +1039,13 @@ u64 format_float(String buffer, f64 value_double)
|
||||
}
|
||||
else
|
||||
{
|
||||
auto dp_uoffset = (u64)dp_offset;
|
||||
let(dp_uoffset, (u64)dp_offset);
|
||||
if (dp_uoffset >= olength)
|
||||
{
|
||||
write_float_decimal(s_get_slice(u8, buffer, buffer_i, buffer.length), &output, olength);
|
||||
buffer_i += olength;
|
||||
auto length = dp_uoffset - olength;
|
||||
auto memset_slice = s_get_slice(u8, buffer, buffer_i, buffer_i + length);
|
||||
let(length, dp_uoffset - olength);
|
||||
String memset_slice = s_get_slice(u8, buffer, buffer_i, buffer_i + length);
|
||||
memset(memset_slice.pointer, 0, length);
|
||||
buffer_i += length;
|
||||
}
|
||||
@ -1054,7 +1053,7 @@ u64 format_float(String buffer, f64 value_double)
|
||||
{
|
||||
write_float_decimal(s_get_slice(u8, buffer, buffer_i + dp_uoffset + 1, buffer.length), &output, olength - dp_uoffset);
|
||||
buffer.pointer[buffer_i + dp_uoffset] = '.';
|
||||
// auto dp_index = buffer_i + dp_uoffset + 1;
|
||||
// let(dp_index, buffer_i + dp_uoffset + 1);
|
||||
write_float_decimal(s_get_slice(u8, buffer, buffer_i, buffer.length), &output, dp_uoffset);
|
||||
buffer_i += olength + 1;
|
||||
}
|
||||
@ -1066,7 +1065,7 @@ u64 format_float(String buffer, f64 value_double)
|
||||
return buffer_i;
|
||||
}
|
||||
|
||||
String format_string_va(String buffer, const char* format, va_list args)
|
||||
fn String format_string_va(String buffer, const char* format, va_list args)
|
||||
{
|
||||
u8* it = (u8*)format;
|
||||
u64 buffer_i = 0;
|
||||
@ -1083,7 +1082,7 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
if (*it == brace_open)
|
||||
{
|
||||
it += 1;
|
||||
auto next_ch = *it;
|
||||
let(next_ch, *it);
|
||||
|
||||
if (next_ch == brace_open)
|
||||
{
|
||||
@ -1107,7 +1106,7 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
{
|
||||
it += 1;
|
||||
done = 1;
|
||||
auto* cstring = va_arg(args, const u8*);
|
||||
let_va_arg(const u8*, cstring, args);
|
||||
while (*cstring)
|
||||
{
|
||||
buffer.pointer[buffer_i] = *cstring;
|
||||
@ -1119,7 +1118,7 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
}
|
||||
else
|
||||
{
|
||||
auto character = cast_to(u8, u32, va_arg(args, u32));
|
||||
let_cast(u8, character, cast_to(u8, va_arg(args, u32)));
|
||||
buffer.pointer[buffer_i] = character;
|
||||
buffer_i += 1;
|
||||
done = 1;
|
||||
@ -1172,7 +1171,7 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
u8* bit_count_end = it;
|
||||
u64 bit_count = parse_decimal(slice_from_pointer_range(u8, (u8*)bit_count_start, (u8*)bit_count_end));
|
||||
|
||||
typedef enum IntegerFormat : u8
|
||||
typedef enum IntegerFormat
|
||||
{
|
||||
INTEGER_FORMAT_HEXADECIMAL,
|
||||
INTEGER_FORMAT_DECIMAL,
|
||||
@ -1221,13 +1220,13 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
unreachable();
|
||||
}
|
||||
|
||||
auto buffer_slice = s_get_slice(u8, buffer, buffer_i, buffer.length);
|
||||
String buffer_slice = s_get_slice(u8, buffer, buffer_i, buffer.length);
|
||||
|
||||
switch (format)
|
||||
{
|
||||
case INTEGER_FORMAT_HEXADECIMAL:
|
||||
{
|
||||
auto written_characters = format_hexadecimal(buffer_slice, original_value);
|
||||
let(written_characters, format_hexadecimal(buffer_slice, original_value));
|
||||
buffer_i += written_characters;
|
||||
} break;
|
||||
case INTEGER_FORMAT_DECIMAL:
|
||||
@ -1246,7 +1245,7 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
value = (u64)original_value;
|
||||
}
|
||||
|
||||
auto written_characters = format_decimal(buffer_slice, value);
|
||||
let(written_characters, format_decimal(buffer_slice, value));
|
||||
buffer_i += written_characters;
|
||||
} break;
|
||||
case INTEGER_FORMAT_OCTAL:
|
||||
@ -1280,7 +1279,7 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
u8* bit_count_end = it;
|
||||
u64 bit_count = parse_decimal(slice_from_pointer_range(u8, (u8*)bit_count_start, (u8*)bit_count_end));
|
||||
|
||||
typedef enum IntegerFormat : u8
|
||||
typedef enum IntegerFormat
|
||||
{
|
||||
INTEGER_FORMAT_HEXADECIMAL,
|
||||
INTEGER_FORMAT_DECIMAL,
|
||||
@ -1329,18 +1328,18 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
unreachable();
|
||||
}
|
||||
|
||||
auto buffer_slice = s_get_slice(u8, buffer, buffer_i, buffer.length);
|
||||
let(buffer_slice, s_get_slice(u8, buffer, buffer_i, buffer.length));
|
||||
|
||||
switch (format)
|
||||
{
|
||||
case INTEGER_FORMAT_HEXADECIMAL:
|
||||
{
|
||||
auto written_characters = format_hexadecimal(buffer_slice, original_value);
|
||||
let(written_characters, format_hexadecimal(buffer_slice, original_value));
|
||||
buffer_i += written_characters;
|
||||
} break;
|
||||
case INTEGER_FORMAT_DECIMAL:
|
||||
{
|
||||
auto written_characters = format_decimal(buffer_slice, original_value);
|
||||
let(written_characters, format_decimal(buffer_slice, original_value));
|
||||
buffer_i += written_characters;
|
||||
} break;
|
||||
case INTEGER_FORMAT_OCTAL:
|
||||
@ -1372,11 +1371,11 @@ String format_string_va(String buffer, const char* format, va_list args)
|
||||
return (String) { .pointer = buffer.pointer, .length = buffer_i };
|
||||
}
|
||||
|
||||
String format_string(String buffer, const char* format, ...)
|
||||
fn String format_string(String buffer, const char* format, ...)
|
||||
{
|
||||
va_list args;
|
||||
va_start(args, format);
|
||||
auto result = format_string_va(buffer, format, args);
|
||||
let(result, format_string_va(buffer, format, args));
|
||||
va_end(args);
|
||||
return result;
|
||||
}
|
||||
|
6
bootstrap/std/format.h
Normal file
6
bootstrap/std/format.h
Normal file
@ -0,0 +1,6 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/base.h>
|
||||
|
||||
fn String format_string(String buffer, const char* format, ...);
|
||||
fn String format_string_va(String buffer, const char* format, va_list args);
|
148
bootstrap/std/metal_rendering.c
Normal file
148
bootstrap/std/metal_rendering.c
Normal file
@ -0,0 +1,148 @@
|
||||
#pragma once
|
||||
|
||||
global_variable Renderer renderer_memory;
|
||||
|
||||
fn NSString* apple_string(String string)
|
||||
{
|
||||
NSString* result = [[NSString alloc] initWithBytes:string.pointer length:string.length encoding:NSUTF8StringEncoding];
|
||||
return result;
|
||||
}
|
||||
|
||||
fn Renderer* rendering_initialize(Arena* arena)
|
||||
{
|
||||
Renderer* renderer = &renderer_memory;
|
||||
@autoreleasepool {
|
||||
renderer->device = MTLCreateSystemDefaultDevice();
|
||||
String shader_source = file_read(arena, strlit("bootstrap/std/shaders/rect.metal"));
|
||||
NSString* apple_shader_source = apple_string(shader_source);
|
||||
NSError* error = nil;
|
||||
id<MTLLibrary> library = [renderer->device newLibraryWithSource: apple_shader_source options:nil error:&error];
|
||||
if (!library)
|
||||
{
|
||||
// Inspect the error
|
||||
NSLog(@"Error Domain: %@", error.domain);
|
||||
NSLog(@"Error Code: %ld", (long)error.code);
|
||||
NSLog(@"Localized Description: %@", error.localizedDescription);
|
||||
|
||||
NSDictionary *userInfo = error.userInfo;
|
||||
if (userInfo) {
|
||||
NSLog(@"Additional Info: %@", userInfo);
|
||||
}
|
||||
|
||||
// Take action based on the error
|
||||
if ([error.domain isEqualToString:MTLLibraryErrorDomain]) {
|
||||
NSLog(@"Metal Library Compilation Error. Check the shader source.");
|
||||
} else {
|
||||
NSLog(@"Unexpected error occurred.");
|
||||
}
|
||||
}
|
||||
|
||||
id<MTLFunction> vertex = [library newFunctionWithName:@"vertex_main"];
|
||||
id<MTLFunction> fragment = [library newFunctionWithName:@"fragment_main"];
|
||||
|
||||
MTLRenderPipelineDescriptor* pipeline_descriptor = [[MTLRenderPipelineDescriptor alloc] init];
|
||||
pipeline_descriptor.vertexFunction = vertex;
|
||||
pipeline_descriptor.fragmentFunction = fragment;
|
||||
pipeline_descriptor.colorAttachments[0].pixelFormat = MTLPixelFormatBGRA8Unorm_sRGB;
|
||||
|
||||
id<MTLRenderPipelineState> pipeline_state = [renderer->device newRenderPipelineStateWithDescriptor:pipeline_descriptor error:&error];
|
||||
|
||||
if (!pipeline_state)
|
||||
{
|
||||
// Inspect the error
|
||||
NSLog(@"Error Domain: %@", error.domain);
|
||||
NSLog(@"Error Code: %ld", (long)error.code);
|
||||
NSLog(@"Localized Description: %@", error.localizedDescription);
|
||||
|
||||
NSDictionary *userInfo = error.userInfo;
|
||||
if (userInfo) {
|
||||
NSLog(@"Additional Info: %@", userInfo);
|
||||
}
|
||||
}
|
||||
|
||||
id<MTLCommandQueue> command_queue = [renderer->device newCommandQueue];
|
||||
}
|
||||
|
||||
return renderer;
|
||||
}
|
||||
|
||||
global_variable RenderWindow render_window_memory;
|
||||
|
||||
fn RenderWindow* rendering_initialize_window(Renderer* renderer, WindowingInstance* window)
|
||||
{
|
||||
RenderWindow* render_window = &render_window_memory;
|
||||
|
||||
CAMetalLayer* layer = [CAMetalLayer layer];
|
||||
render_window->layer = layer;
|
||||
layer.device = renderer->device;
|
||||
layer.pixelFormat = MTLPixelFormatBGRA8Unorm_sRGB;
|
||||
layer.framebufferOnly = true;
|
||||
layer.frame = window.frame;
|
||||
window.contentView.layer = layer;
|
||||
window.opaque = true;
|
||||
window.backgroundColor = nil;
|
||||
|
||||
return render_window;
|
||||
}
|
||||
|
||||
fn void renderer_window_frame_begin(Renderer* renderer, RenderWindow* window)
|
||||
{
|
||||
@autoreleasepool {
|
||||
id<CAMetalDrawable> drawable = [window->layer nextDrawable];
|
||||
MTLRenderPassDescriptor* render_pass_descriptor = [MTLRenderPassDescriptor renderPassDescriptor];
|
||||
MTLRenderPassColorAttachmentDescriptor* color_attachment = render_pass_descriptor.colorAttachments[0];
|
||||
color_attachment.clearColor = MTLClearColorMake(1, 1, 1, 1);
|
||||
color_attachment.storeAction = MTLStoreActionStore;
|
||||
color_attachment.texture = drawable.texture;
|
||||
|
||||
id<MTLCommandBuffer> command_buffer = [renderer->command_queue commandBuffer];
|
||||
|
||||
id<MTLRenderCommandEncoder> render_command_encoder = [command_buffer renderCommandEncoderWithDescriptor:render_pass_descriptor];
|
||||
[render_command_encoder setRenderPipelineState: renderer->pipeline_state];
|
||||
[render_command_encoder drawPrimitives:MTLPrimitiveTypeTriangle vertexStart:0 vertexCount:3];
|
||||
[render_command_encoder endEncoding];
|
||||
[command_buffer presentDrawable:drawable];
|
||||
[command_buffer commit];
|
||||
}
|
||||
}
|
||||
|
||||
fn void renderer_window_frame_end(Renderer* renderer, RenderWindow* window)
|
||||
{
|
||||
|
||||
// todo();
|
||||
}
|
||||
|
||||
fn TextureIndex renderer_texture_create(Renderer* renderer, TextureMemory texture_memory)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
|
||||
fn void window_rect_texture_update_begin(RenderWindow* window)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
|
||||
fn void renderer_queue_font_update(Renderer* renderer, RenderWindow* window, RenderFontType type, TextureAtlas atlas)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
|
||||
fn void window_queue_rect_texture_update(RenderWindow* window, RectTextureSlot slot, TextureIndex texture_index)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
|
||||
fn void window_rect_texture_update_end(Renderer* renderer, RenderWindow* window)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
|
||||
fn void window_render_rect(RenderWindow* window, RectDraw draw)
|
||||
{
|
||||
// todo();
|
||||
}
|
||||
|
||||
fn void window_render_text(Renderer* renderer, RenderWindow* window, String string, float4 color, RenderFontType font_type, u32 x_offset, u32 y_offset)
|
||||
{
|
||||
// todo();
|
||||
}
|
16
bootstrap/std/metal_rendering.h
Normal file
16
bootstrap/std/metal_rendering.h
Normal file
@ -0,0 +1,16 @@
|
||||
#pragma once
|
||||
|
||||
#import <Metal/Metal.h>
|
||||
|
||||
STRUCT(Renderer)
|
||||
{
|
||||
id<MTLDevice> device;
|
||||
id<MTLCommandQueue> command_queue;
|
||||
id<MTLRenderPipelineState> pipeline_state;
|
||||
};
|
||||
|
||||
STRUCT(RenderWindow)
|
||||
{
|
||||
CAMetalLayer* layer;
|
||||
};
|
||||
|
@ -1,26 +1,10 @@
|
||||
#include <std/os.h>
|
||||
#pragma once
|
||||
|
||||
#include <std/string.h>
|
||||
#include <std/format.h>
|
||||
|
||||
#ifdef _WIN32
|
||||
#define WIN32_LEAN_AND_MEAN 1
|
||||
#include <Windows.h>
|
||||
#else
|
||||
#include <fcntl.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/mman.h>
|
||||
#include <sys/wait.h>
|
||||
#include <sys/time.h>
|
||||
#include <sys/ptrace.h>
|
||||
#endif
|
||||
|
||||
#if LINK_LIBC
|
||||
#include <errno.h>
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
#include <time.h>
|
||||
#endif
|
||||
#include <std/string.c>
|
||||
#include <std/format.c>
|
||||
|
||||
#if _WIN32
|
||||
global_variable u64 cpu_frequency;
|
||||
@ -32,31 +16,31 @@ global_variable u64 cpu_frequency;
|
||||
#endif
|
||||
#endif
|
||||
|
||||
Timestamp os_timestamp()
|
||||
fn Timestamp os_timestamp()
|
||||
{
|
||||
Timestamp result;
|
||||
|
||||
#if _WIN32
|
||||
LARGE_INTEGER li;
|
||||
QueryPerformanceCounter(&li);
|
||||
result.value = li.QuadPart;
|
||||
result.value = u128_from_u64(li.QuadPart);
|
||||
#else
|
||||
struct timespec ts;
|
||||
clock_gettime(CLOCK_MONOTONIC, &ts);
|
||||
result.value = ((u128)ts.tv_sec << 64) | ts.tv_nsec;
|
||||
result.value = u128_u64_or(u128_shift_left(u128_from_u64(ts.tv_sec), 64), ts.tv_nsec);
|
||||
#endif
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
f64 os_resolve_timestamps(Timestamp start, Timestamp end, TimeUnit time_unit)
|
||||
fn f64 os_resolve_timestamps(Timestamp start, Timestamp end, TimeUnit time_unit)
|
||||
{
|
||||
f64 result;
|
||||
#if _WIN32
|
||||
auto start_tick = (s64)start.value;
|
||||
auto end_tick = (s64)end.value;
|
||||
let(start_tick, (s64)u64_from_u128(start.value));
|
||||
let(end_tick, (s64)u64_from_u128(end.value));
|
||||
|
||||
auto seconds = (f64)(end_tick - start_tick) / cpu_frequency;
|
||||
let(seconds, (f64)(end_tick - start_tick) / cpu_frequency);
|
||||
|
||||
switch (time_unit)
|
||||
{
|
||||
@ -74,15 +58,16 @@ f64 os_resolve_timestamps(Timestamp start, Timestamp end, TimeUnit time_unit)
|
||||
break;
|
||||
}
|
||||
#else
|
||||
auto segmented_nanoseconds = (s64)end.value - (s64)start.value;
|
||||
auto segmented_seconds = (s64)(end.value >> 64) - (s64)(start.value >> 64);
|
||||
let(segmented_nanoseconds, (s64)u64_from_u128(end.value) - (s64)u64_from_u128(start.value));
|
||||
let(segmented_seconds, (s64)u128_shift_right_by_64(end.value) - (s64)u128_shift_right_by_64(start.value));
|
||||
|
||||
if (segmented_nanoseconds < 0)
|
||||
{
|
||||
segmented_seconds -= 1;
|
||||
segmented_nanoseconds += 1000000000;
|
||||
}
|
||||
|
||||
auto total_ns = segmented_seconds * 1000000000 + segmented_nanoseconds;
|
||||
let(total_ns, segmented_seconds * 1000000000 + segmented_nanoseconds);
|
||||
|
||||
switch (time_unit)
|
||||
{
|
||||
@ -104,10 +89,10 @@ f64 os_resolve_timestamps(Timestamp start, Timestamp end, TimeUnit time_unit)
|
||||
return result;
|
||||
}
|
||||
|
||||
FileDescriptor os_stdout_get()
|
||||
fn FileDescriptor os_stdout_get()
|
||||
{
|
||||
#if _WIN32
|
||||
auto handle = GetStdHandle(STD_OUTPUT_HANDLE);
|
||||
let(handle, GetStdHandle(STD_OUTPUT_HANDLE));
|
||||
assert(handle != INVALID_HANDLE_VALUE);
|
||||
return handle;
|
||||
#else
|
||||
@ -115,11 +100,11 @@ FileDescriptor os_stdout_get()
|
||||
#endif
|
||||
}
|
||||
|
||||
String path_dir(String string)
|
||||
fn String path_dir(String string)
|
||||
{
|
||||
String result = {};
|
||||
auto index = string_last_ch(string, '/');
|
||||
if (index != -1)
|
||||
let(index, string_last_ch(string, '/'));
|
||||
if (index != STRING_NO_MATCH)
|
||||
{
|
||||
result = s_get_slice(u8, string, 0, index);
|
||||
}
|
||||
@ -127,34 +112,34 @@ String path_dir(String string)
|
||||
return result;
|
||||
}
|
||||
|
||||
String path_base(String string)
|
||||
fn String path_base(String string)
|
||||
{
|
||||
String result = {};
|
||||
auto maybe_index = string_last_ch(string, '/');
|
||||
if (maybe_index != -1)
|
||||
let(index, string_last_ch(string, '/'));
|
||||
if (index != STRING_NO_MATCH)
|
||||
{
|
||||
auto index = cast_to(u64, s64, maybe_index);
|
||||
result = s_get_slice(u8, string, index + 1, string.length);
|
||||
}
|
||||
#if _WIN32
|
||||
if (!result.pointer)
|
||||
{
|
||||
auto maybe_index = string_last_ch(string, '\\');
|
||||
auto index = cast_to(u64, s64, maybe_index);
|
||||
let(index, string_last_ch(string, '\\'));
|
||||
if (index != STRING_NO_MATCH)
|
||||
{
|
||||
result = s_get_slice(u8, string, index + 1, string.length);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
String path_no_extension(String string)
|
||||
fn String path_no_extension(String string)
|
||||
{
|
||||
String result = {};
|
||||
auto maybe_index = string_last_ch(string, '.');
|
||||
if (maybe_index != -1)
|
||||
let(index, string_last_ch(string, '.'));
|
||||
if (index != STRING_NO_MATCH)
|
||||
{
|
||||
auto index = cast_to(u64, s64, maybe_index);
|
||||
result = s_get_slice(u8, string, 0, index);
|
||||
}
|
||||
|
||||
@ -603,7 +588,7 @@ may_be_unused fn void* posix_mmap(void* address, size_t length, int protection_f
|
||||
return mmap(address, length, protection_flags, map_flags, fd, offset);
|
||||
#else
|
||||
#ifdef __linux__
|
||||
return (void*) syscall6(syscall_x86_64_mmap, (s64)address, cast_to(s64, u64, length), protection_flags, map_flags, fd, offset);
|
||||
return (void*) syscall6(syscall_x86_64_mmap, (s64)address, cast_to(s64, length), protection_flags, map_flags, fd, offset);
|
||||
#else
|
||||
#error "Unsupported operating system for static linking"
|
||||
#endif
|
||||
@ -616,7 +601,7 @@ may_be_unused fn int syscall_mprotect(void *address, size_t length, int protecti
|
||||
return mprotect(address, length, protection_flags);
|
||||
#else
|
||||
#ifdef __linux__
|
||||
return cast_to(s32, s64, syscall3(syscall_x86_64_mprotect, (s64)address, cast_to(s64, u64, length), protection_flags));
|
||||
return cast_to(s32, syscall3(syscall_x86_64_mprotect, (s64)address, cast_to(s64, length), protection_flags));
|
||||
#else
|
||||
return mprotect(address, length, protection_flags);
|
||||
#endif
|
||||
@ -629,7 +614,7 @@ may_be_unused fn int syscall_open(const char *file_path, int flags, int mode)
|
||||
return open(file_path, flags, mode);
|
||||
#else
|
||||
#ifdef __linux__
|
||||
return cast_to(s32, s64, syscall3(syscall_x86_64_open, (s64)file_path, flags, mode));
|
||||
return cast_to(s32, syscall3(syscall_x86_64_open, (s64)file_path, flags, mode));
|
||||
#else
|
||||
return open(file_path, flags, mode);
|
||||
#endif
|
||||
@ -642,7 +627,7 @@ may_be_unused fn int syscall_close(int fd)
|
||||
return close(fd);
|
||||
#else
|
||||
#ifdef __linux__
|
||||
return cast_to(s32, s64, syscall1(syscall_x86_64_close, fd));
|
||||
return cast_to(s32, syscall1(syscall_x86_64_close, fd));
|
||||
#else
|
||||
return close(fd);
|
||||
#endif
|
||||
@ -655,7 +640,7 @@ fn int syscall_fstat(int fd, struct stat *buffer)
|
||||
return fstat(fd, buffer);
|
||||
#else
|
||||
#ifdef __linux__
|
||||
return cast_to(s32, s64, syscall2(syscall_x86_64_fstat, fd, (s64)buffer));
|
||||
return cast_to(s32, syscall2(syscall_x86_64_fstat, fd, (s64)buffer));
|
||||
#else
|
||||
return fstat(fd, buffer);
|
||||
#endif
|
||||
@ -694,7 +679,7 @@ may_be_unused fn int syscall_mkdir(String path, u32 mode)
|
||||
#if LINK_LIBC
|
||||
return mkdir((char*)path.pointer, mode);
|
||||
#else
|
||||
return cast_to(s32, s64, syscall2(syscall_x86_64_mkdir, (s64)path.pointer, (s64)mode));
|
||||
return cast_to(s32, syscall2(syscall_x86_64_mkdir, (s64)path.pointer, (s64)mode));
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -704,7 +689,7 @@ may_be_unused fn int syscall_rmdir(String path)
|
||||
#if LINK_LIBC
|
||||
return rmdir((char*)path.pointer);
|
||||
#else
|
||||
return cast_to(s32, s64, syscall1(syscall_x86_64_rmdir, (s64)path.pointer));
|
||||
return cast_to(s32, syscall1(syscall_x86_64_rmdir, (s64)path.pointer));
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -714,7 +699,7 @@ may_be_unused fn int syscall_unlink(String path)
|
||||
#if LINK_LIBC
|
||||
return unlink((char*)path.pointer);
|
||||
#else
|
||||
return cast_to(s32, s64, syscall1(syscall_x86_64_unlink, (s64)path.pointer));
|
||||
return cast_to(s32, syscall1(syscall_x86_64_unlink, (s64)path.pointer));
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -723,7 +708,7 @@ may_be_unused fn pid_t syscall_fork()
|
||||
#if LINK_LIBC
|
||||
return fork();
|
||||
#else
|
||||
return cast_to(s32, s64, syscall0(syscall_x86_64_fork));
|
||||
return cast_to(s32, syscall0(syscall_x86_64_fork));
|
||||
#endif
|
||||
|
||||
}
|
||||
@ -742,7 +727,7 @@ may_be_unused fn pid_t syscall_waitpid(pid_t pid, int* status, int options)
|
||||
#if LINK_LIBC
|
||||
return waitpid(pid, status, options);
|
||||
#else
|
||||
return cast_to(s32, s64, syscall4(syscall_x86_64_wait4, pid, (s64)status, options, 0));
|
||||
return cast_to(s32, syscall4(syscall_x86_64_wait4, pid, (s64)status, options, 0));
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -751,11 +736,11 @@ may_be_unused fn int syscall_gettimeofday(struct timeval* tv, struct timezone* t
|
||||
#if LINK_LIBC
|
||||
return gettimeofday(tv, tz);
|
||||
#else
|
||||
return cast_to(s32, s64, syscall2(syscall_x86_64_gettimeofday, (s64)tv, (s64)tz));
|
||||
return cast_to(s32, syscall2(syscall_x86_64_gettimeofday, (s64)tv, (s64)tz));
|
||||
#endif
|
||||
}
|
||||
|
||||
may_be_unused [[noreturn]] [[gnu::cold]] fn void syscall_exit(int status)
|
||||
may_be_unused BB_NORETURN BB_COLD fn void syscall_exit(int status)
|
||||
{
|
||||
#if LINK_LIBC
|
||||
_exit(status);
|
||||
@ -784,12 +769,12 @@ may_be_unused fn u64 os_timer_get()
|
||||
#else
|
||||
struct timeval tv;
|
||||
syscall_gettimeofday(&tv, 0);
|
||||
auto result = os_timer_freq() * cast_to(u64, s64, tv.tv_sec) + cast_to(u64, s64, tv.tv_usec);
|
||||
let(result, os_timer_freq() * cast_to(u64, tv.tv_sec) + cast_to(u64, tv.tv_usec));
|
||||
return result;
|
||||
#endif
|
||||
}
|
||||
|
||||
u8 os_file_descriptor_is_valid(FileDescriptor fd)
|
||||
fn u8 os_file_descriptor_is_valid(FileDescriptor fd)
|
||||
{
|
||||
#if _WIN32
|
||||
return fd != INVALID_HANDLE_VALUE;
|
||||
@ -798,7 +783,7 @@ u8 os_file_descriptor_is_valid(FileDescriptor fd)
|
||||
#endif
|
||||
}
|
||||
|
||||
FileDescriptor os_file_open(String path, OSFileOpenFlags flags, OSFilePermissions permissions)
|
||||
fn FileDescriptor os_file_open(String path, OSFileOpenFlags flags, OSFilePermissions permissions)
|
||||
{
|
||||
assert(path.pointer[path.length] == 0);
|
||||
#if _WIN32
|
||||
@ -816,7 +801,7 @@ FileDescriptor os_file_open(String path, OSFileOpenFlags flags, OSFilePermission
|
||||
dwFlagsAndAttributes |= flags.directory * FILE_FLAG_BACKUP_SEMANTICS;
|
||||
HANDLE hTemplateFile = 0;
|
||||
|
||||
auto handle = CreateFileA(string_to_c(path), dwDesiredAccess, dwShareMode, lpSecurityAttributes, dwCreationDisposition, dwFlagsAndAttributes, hTemplateFile);
|
||||
let(handle, CreateFileA(string_to_c(path), dwDesiredAccess, dwShareMode, lpSecurityAttributes, dwCreationDisposition, dwFlagsAndAttributes, hTemplateFile));
|
||||
return handle;
|
||||
#else
|
||||
int posix_flags = 0;
|
||||
@ -836,12 +821,12 @@ FileDescriptor os_file_open(String path, OSFileOpenFlags flags, OSFilePermission
|
||||
{
|
||||
posix_permissions = 0644;
|
||||
}
|
||||
auto result = syscall_open((char*)path.pointer, posix_flags, posix_permissions);
|
||||
let(result, syscall_open((char*)path.pointer, posix_flags, posix_permissions));
|
||||
return result;
|
||||
#endif
|
||||
}
|
||||
|
||||
u64 os_file_get_size(FileDescriptor fd)
|
||||
fn u64 os_file_get_size(FileDescriptor fd)
|
||||
{
|
||||
#if _WIN32
|
||||
LARGE_INTEGER file_size;
|
||||
@ -852,20 +837,20 @@ u64 os_file_get_size(FileDescriptor fd)
|
||||
struct stat stat_buffer;
|
||||
int stat_result = syscall_fstat(fd, &stat_buffer);
|
||||
assert(stat_result == 0);
|
||||
auto size = cast_to(u64, s64, stat_buffer.st_size);
|
||||
let_cast(u64, size, stat_buffer.st_size);
|
||||
return size;
|
||||
#endif
|
||||
}
|
||||
|
||||
void os_file_write(FileDescriptor fd, String content)
|
||||
fn void os_file_write(FileDescriptor fd, String content)
|
||||
{
|
||||
#if _WIN32
|
||||
DWORD bytes_written = 0;
|
||||
BOOL result = WriteFile(fd, content.pointer, cast_to(u32, u64, content.length), &bytes_written, 0);
|
||||
BOOL result = WriteFile(fd, content.pointer, cast_to(u32, content.length), &bytes_written, 0);
|
||||
assert(result != 0);
|
||||
#else
|
||||
auto result = syscall_write(fd, content.pointer, content.length);
|
||||
assert(cast_to(u64, s64, result) == content.length);
|
||||
let(result, syscall_write(fd, content.pointer, content.length));
|
||||
assert(cast_to(u64, result) == content.length);
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -878,15 +863,15 @@ may_be_unused fn u64 os_file_read(FileDescriptor fd, String buffer, u64 byte_cou
|
||||
{
|
||||
#if _WIN32
|
||||
DWORD read = 0;
|
||||
BOOL result = ReadFile(fd, buffer.pointer, cast_to(u32, u64, byte_count), &read, 0);
|
||||
BOOL result = ReadFile(fd, buffer.pointer, cast_to(u32, byte_count), &read, 0);
|
||||
assert(result != 0);
|
||||
bytes_read = read;
|
||||
#else
|
||||
auto result = syscall_read(fd, buffer.pointer, byte_count);
|
||||
let(result, syscall_read(fd, buffer.pointer, byte_count));
|
||||
assert(result > 0);
|
||||
if (result > 0)
|
||||
{
|
||||
bytes_read = cast_to(u64, s64, result);
|
||||
assign_cast(bytes_read, result);
|
||||
}
|
||||
#endif
|
||||
}
|
||||
@ -894,18 +879,18 @@ may_be_unused fn u64 os_file_read(FileDescriptor fd, String buffer, u64 byte_cou
|
||||
return bytes_read;
|
||||
}
|
||||
|
||||
void os_file_close(FileDescriptor fd)
|
||||
fn void os_file_close(FileDescriptor fd)
|
||||
{
|
||||
#if _WIN32
|
||||
BOOL result = CloseHandle(fd);
|
||||
assert(result != 0);
|
||||
#else
|
||||
auto result = syscall_close(fd);
|
||||
let(result, syscall_close(fd));
|
||||
assert(result == 0);
|
||||
#endif
|
||||
}
|
||||
|
||||
void calibrate_cpu_timer()
|
||||
fn void calibrate_cpu_timer()
|
||||
{
|
||||
#ifndef SILENT
|
||||
#if _WIN32
|
||||
@ -925,7 +910,7 @@ void calibrate_cpu_timer()
|
||||
|
||||
while (os_elapsed < os_wait_time)
|
||||
{
|
||||
auto os_end = os_timer_get();
|
||||
let(os_end, os_timer_get());
|
||||
os_elapsed = os_end - os_start;
|
||||
}
|
||||
|
||||
@ -937,7 +922,7 @@ void calibrate_cpu_timer()
|
||||
#endif
|
||||
}
|
||||
|
||||
u8* os_reserve(u64 base, u64 size, OSReserveProtectionFlags protection, OSReserveMapFlags map)
|
||||
fn u8* os_reserve(u64 base, u64 size, OSReserveProtectionFlags protection, OSReserveMapFlags map)
|
||||
{
|
||||
#if _WIN32
|
||||
DWORD map_flags = 0;
|
||||
@ -956,7 +941,7 @@ u8* os_reserve(u64 base, u64 size, OSReserveProtectionFlags protection, OSReserv
|
||||
#endif
|
||||
}
|
||||
|
||||
void os_commit(void* address, u64 size)
|
||||
fn void os_commit(void* address, u64 size)
|
||||
{
|
||||
#if _WIN32
|
||||
VirtualAlloc(address, size, MEM_COMMIT, PAGE_READWRITE);
|
||||
@ -966,7 +951,7 @@ void os_commit(void* address, u64 size)
|
||||
#endif
|
||||
}
|
||||
|
||||
void os_directory_make(String path)
|
||||
fn void os_directory_make(String path)
|
||||
{
|
||||
assert(path.pointer[path.length] == 0);
|
||||
#if _WIN32
|
||||
@ -976,7 +961,12 @@ void os_directory_make(String path)
|
||||
#endif
|
||||
}
|
||||
|
||||
void print(const char* format, ...)
|
||||
BB_NORETURN BB_COLD fn void os_exit(u32 exit_code)
|
||||
{
|
||||
exit(exit_code);
|
||||
}
|
||||
|
||||
fn void print(const char* format, ...)
|
||||
{
|
||||
#ifndef SILENT
|
||||
u8 stack_buffer[16*1024];
|
||||
@ -993,13 +983,13 @@ void print(const char* format, ...)
|
||||
static_assert(sizeof(Arena) == 64);
|
||||
const global_variable u64 minimum_position = sizeof(Arena);
|
||||
|
||||
Arena* arena_init(u64 reserved_size, u64 granularity, u64 initial_size)
|
||||
fn Arena* arena_initialize(u64 reserved_size, u64 granularity, u64 initial_size)
|
||||
{
|
||||
auto protection_flags = (OSReserveProtectionFlags) {
|
||||
OSReserveProtectionFlags protection_flags = {
|
||||
.read = 1,
|
||||
.write = 1,
|
||||
};
|
||||
auto map_flags = (OSReserveMapFlags) {
|
||||
OSReserveMapFlags map_flags = {
|
||||
.priv = 1,
|
||||
.anon = 1,
|
||||
.noreserve = 1,
|
||||
@ -1015,12 +1005,12 @@ Arena* arena_init(u64 reserved_size, u64 granularity, u64 initial_size)
|
||||
return arena;
|
||||
}
|
||||
|
||||
Arena* arena_init_default(u64 initial_size)
|
||||
fn Arena* arena_initialize_default(u64 initial_size)
|
||||
{
|
||||
return arena_init(default_size, minimum_granularity, initial_size);
|
||||
return arena_initialize(default_size, minimum_granularity, initial_size);
|
||||
}
|
||||
|
||||
u8* arena_allocate_bytes(Arena* arena, u64 size, u64 alignment)
|
||||
fn u8* arena_allocate_bytes(Arena* arena, u64 size, u64 alignment)
|
||||
{
|
||||
u64 aligned_offset = align_forward(arena->position, alignment);
|
||||
u64 aligned_size_after = aligned_offset + size;
|
||||
@ -1034,13 +1024,13 @@ u8* arena_allocate_bytes(Arena* arena, u64 size, u64 alignment)
|
||||
arena->os_position = committed_size;
|
||||
}
|
||||
|
||||
auto* result = (u8*)arena + aligned_offset;
|
||||
let(result, (u8*)arena + aligned_offset);
|
||||
arena->position = aligned_size_after;
|
||||
assert(arena->position <= arena->os_position);
|
||||
return result;
|
||||
}
|
||||
|
||||
String arena_join_string(Arena* arena, Slice(String) pieces)
|
||||
fn String arena_join_string(Arena* arena, Slice(String) pieces)
|
||||
{
|
||||
u64 size = 0;
|
||||
for (u64 i = 0; i < pieces.length; i += 1)
|
||||
@ -1050,7 +1040,7 @@ String arena_join_string(Arena* arena, Slice(String) pieces)
|
||||
}
|
||||
|
||||
u8* pointer = arena_allocate_bytes(arena, size + 1, 1);
|
||||
auto* it = pointer;
|
||||
let(it, pointer);
|
||||
for (u64 i = 0; i < pieces.length; i += 1)
|
||||
{
|
||||
String piece = pieces.pointer[i];
|
||||
@ -1063,19 +1053,16 @@ String arena_join_string(Arena* arena, Slice(String) pieces)
|
||||
return (String) { .pointer = pointer, .length = size };
|
||||
}
|
||||
|
||||
|
||||
void arena_reset(Arena* arena)
|
||||
fn void arena_reset(Arena* arena)
|
||||
{
|
||||
arena->position = minimum_position;
|
||||
memset(arena + 1, 0, arena->position - minimum_position);
|
||||
}
|
||||
|
||||
#define transmute(D, source) *(D*)&source
|
||||
|
||||
String file_read(Arena* arena, String path)
|
||||
fn String file_read(Arena* arena, String path)
|
||||
{
|
||||
String result = {};
|
||||
auto file_descriptor = os_file_open(path, (OSFileOpenFlags) {
|
||||
let(file_descriptor, os_file_open(path, (OSFileOpenFlags) {
|
||||
.truncate = 0,
|
||||
.executable = 0,
|
||||
.write = 0,
|
||||
@ -1083,11 +1070,11 @@ String file_read(Arena* arena, String path)
|
||||
.create = 0,
|
||||
}, (OSFilePermissions) {
|
||||
.readable = 1,
|
||||
});
|
||||
}));
|
||||
|
||||
if (os_file_descriptor_is_valid(file_descriptor))
|
||||
{
|
||||
auto file_size = os_file_get_size(file_descriptor);
|
||||
let(file_size, os_file_get_size(file_descriptor));
|
||||
if (file_size > 0)
|
||||
{
|
||||
result = (String){
|
||||
@ -1112,10 +1099,10 @@ String file_read(Arena* arena, String path)
|
||||
return result;
|
||||
}
|
||||
|
||||
void file_write(FileWriteOptions options)
|
||||
fn void file_write(FileWriteOptions options)
|
||||
{
|
||||
print("Writing file \"{s}\"...\n", options.path);
|
||||
auto fd = os_file_open(options.path, (OSFileOpenFlags) {
|
||||
let(fd, os_file_open(options.path, (OSFileOpenFlags) {
|
||||
.write = 1,
|
||||
.truncate = 1,
|
||||
.create = 1,
|
||||
@ -1124,36 +1111,40 @@ void file_write(FileWriteOptions options)
|
||||
.readable = 1,
|
||||
.writable = 1,
|
||||
.executable = options.executable,
|
||||
});
|
||||
}));
|
||||
assert(os_file_descriptor_is_valid(fd));
|
||||
|
||||
os_file_write(fd, options.content);
|
||||
os_file_close(fd);
|
||||
}
|
||||
|
||||
void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
fn void run_command(Arena* arena, CStringSlice arguments, char* envp[], RunCommandOptions run_options)
|
||||
{
|
||||
print("Running command:\n");
|
||||
assert(arguments.length > 0);
|
||||
assert(arguments.pointer[arguments.length - 1] == 0);
|
||||
|
||||
if (run_options.debug)
|
||||
{
|
||||
print("Running command:\n");
|
||||
for (u32 i = 0; i < arguments.length - 1; i += 1)
|
||||
{
|
||||
char* argument = arguments.pointer[i];
|
||||
print("{cstr} ", argument);
|
||||
}
|
||||
print("\n");
|
||||
}
|
||||
|
||||
#if _WIN32
|
||||
auto start_timestamp = os_timestamp();
|
||||
let(start_timestamp, os_timestamp());
|
||||
|
||||
u32 length = 0;
|
||||
for (u32 i = 0; i < arguments.length; i += 1)
|
||||
{
|
||||
auto argument = arguments.pointer[i];
|
||||
let(argument, arguments.pointer[i]);
|
||||
if (argument)
|
||||
{
|
||||
auto string_len = strlen(argument);
|
||||
length += cast_to(u32, u64, string_len + 1);
|
||||
let(string_len, strlen(argument));
|
||||
length += cast_to(u32, string_len + 1);
|
||||
}
|
||||
}
|
||||
|
||||
@ -1161,10 +1152,10 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
u32 byte_i = 0;
|
||||
for (u32 i = 0; i < arguments.length; i += 1)
|
||||
{
|
||||
auto argument = arguments.pointer[i];
|
||||
let(argument, arguments.pointer[i]);
|
||||
if (argument)
|
||||
{
|
||||
auto len = strlen(argument);
|
||||
let(len, strlen(argument));
|
||||
memcpy(&bytes[byte_i], argument, len);
|
||||
byte_i += len;
|
||||
bytes[byte_i] = ' ';
|
||||
@ -1172,7 +1163,7 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
}
|
||||
}
|
||||
bytes[byte_i - 1] = 0;
|
||||
auto end_timestamp = os_timestamp();
|
||||
let(end_timestamp, os_timestamp());
|
||||
|
||||
PROCESS_INFORMATION process_information = {};
|
||||
STARTUPINFOA startup_info = {};
|
||||
@ -1181,19 +1172,27 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
startup_info.hStdOutput = GetStdHandle(STD_OUTPUT_HANDLE);
|
||||
startup_info.hStdError = GetStdHandle(STD_ERROR_HANDLE);
|
||||
|
||||
auto handle_inheritance = 1;
|
||||
auto start = os_timestamp();
|
||||
let(handle_inheritance, 1);
|
||||
let(start, os_timestamp());
|
||||
if (CreateProcessA(0, bytes, 0, 0, handle_inheritance, 0, 0, 0, &startup_info, &process_information))
|
||||
{
|
||||
WaitForSingleObject(process_information.hProcess, INFINITE);
|
||||
auto end = os_timestamp();
|
||||
auto ms = os_resolve_timestamps(start, end, TIME_UNIT_MILLISECONDS);
|
||||
let(end, os_timestamp());
|
||||
let(ms, os_resolve_timestamps(start, end, TIME_UNIT_MILLISECONDS));
|
||||
|
||||
if (run_options.debug)
|
||||
{
|
||||
print("Process ran in {f64} ms\n", ms);
|
||||
}
|
||||
|
||||
DWORD exit_code;
|
||||
if (GetExitCodeProcess(process_information.hProcess, &exit_code))
|
||||
{
|
||||
if (run_options.debug)
|
||||
{
|
||||
print("Process ran with exit code: 0x{u32:x}\n", exit_code);
|
||||
}
|
||||
|
||||
if (exit_code != 0)
|
||||
{
|
||||
failed_execution();
|
||||
@ -1209,8 +1208,7 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
}
|
||||
else
|
||||
{
|
||||
print("Failure\n");
|
||||
auto err = GetLastError();
|
||||
let(err, GetLastError());
|
||||
LPSTR lpMsgBuf;
|
||||
DWORD bufSize = FormatMessageA(
|
||||
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||
@ -1221,9 +1219,10 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
0,
|
||||
NULL
|
||||
);
|
||||
unused(bufSize);
|
||||
print("CreateProcessA call failed: {cstr}\n", lpMsgBuf);
|
||||
todo();
|
||||
}
|
||||
|
||||
unused(start_timestamp);
|
||||
unused(end_timestamp);
|
||||
unused(envp);
|
||||
@ -1236,13 +1235,13 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
todo();
|
||||
}
|
||||
|
||||
auto start_timestamp = os_timestamp();
|
||||
let(start_timestamp, os_timestamp());
|
||||
|
||||
if (pid == 0)
|
||||
{
|
||||
// close(pipes[0]);
|
||||
// fcntl(pipes[1], F_SETFD, FD_CLOEXEC);
|
||||
auto result = syscall_execve(arguments.pointer[0], arguments.pointer, envp);
|
||||
let(result, syscall_execve(arguments.pointer[0], arguments.pointer, envp));
|
||||
#if LINK_LIBC
|
||||
my_panic("Execve failed! Error: {cstr}\n", strerror(errno));
|
||||
#else
|
||||
@ -1254,26 +1253,27 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
int status = 0;
|
||||
int options = 0;
|
||||
pid_t result = syscall_waitpid(pid, &status, options);
|
||||
auto end_timestamp = os_timestamp();
|
||||
let(end_timestamp, os_timestamp());
|
||||
int success = 0;
|
||||
if (result == pid)
|
||||
{
|
||||
if (run_options.debug)
|
||||
{
|
||||
print("{cstr} ", arguments.pointer[0]);
|
||||
|
||||
if (WIFEXITED(status))
|
||||
{
|
||||
auto exit_code = WEXITSTATUS(status);
|
||||
success = exit_code == 0;
|
||||
let(exit_code, WEXITSTATUS(status));
|
||||
print("exited with code {u32}\n", exit_code);
|
||||
}
|
||||
else if (WIFSIGNALED(status))
|
||||
{
|
||||
auto signal_code = WTERMSIG(status);
|
||||
let(signal_code, WTERMSIG(status));
|
||||
print("was signaled: {u32}\n", signal_code);
|
||||
}
|
||||
else if (WIFSTOPPED(status))
|
||||
{
|
||||
auto stopped_code = WSTOPSIG(status);
|
||||
let(stopped_code, WSTOPSIG(status));
|
||||
print("was stopped: {u32}\n", stopped_code);
|
||||
}
|
||||
else
|
||||
@ -1281,9 +1281,16 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
print("terminated unexpectedly with status {u32}\n", status);
|
||||
}
|
||||
}
|
||||
|
||||
if (WIFEXITED(status))
|
||||
{
|
||||
let(exit_code, WEXITSTATUS(status));
|
||||
success = exit_code == 0;
|
||||
}
|
||||
}
|
||||
else if (result == -1)
|
||||
{
|
||||
auto waitpid_error = errno;
|
||||
let(waitpid_error, errno);
|
||||
print("Error waiting for process termination: {u32}\n", waitpid_error);
|
||||
trap();
|
||||
}
|
||||
@ -1294,37 +1301,37 @@ void run_command(Arena* arena, CStringSlice arguments, char* envp[])
|
||||
|
||||
if (!success)
|
||||
{
|
||||
print("Program failed to run!\n");
|
||||
print("Program failed to run successfully!\n");
|
||||
failed_execution();
|
||||
}
|
||||
auto ms = os_resolve_timestamps(start_timestamp, end_timestamp, TIME_UNIT_MILLISECONDS);
|
||||
auto ticks =
|
||||
#if LINK_LIBC
|
||||
0
|
||||
#else
|
||||
cpu_frequency != 0
|
||||
|
||||
if (run_options.debug)
|
||||
{
|
||||
let(ms, os_resolve_timestamps(start_timestamp, end_timestamp, TIME_UNIT_MILLISECONDS));
|
||||
u32 ticks = 0;
|
||||
#if LINK_LIBC == 0
|
||||
ticks = cpu_frequency != 0;
|
||||
#endif
|
||||
;
|
||||
print("Command run successfully in {f64} {cstr}\n", ms, ticks ? "ticks" : "ms");
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
u8 os_is_being_debugged()
|
||||
fn u8 os_is_being_debugged()
|
||||
{
|
||||
u8 result = 0;
|
||||
#if _WIN32
|
||||
result = IsDebuggerPresent();
|
||||
#else
|
||||
auto request =
|
||||
#ifdef __APPLE__
|
||||
PT_TRACE_ME;
|
||||
let(request, PT_TRACE_ME);
|
||||
#else
|
||||
PTRACE_TRACEME;
|
||||
let(request, PTRACE_TRACEME);
|
||||
#endif
|
||||
if (ptrace(request, 0, 0, 0) == -1)
|
||||
{
|
||||
auto error = errno;
|
||||
let(error, errno);
|
||||
if (error == EPERM)
|
||||
{
|
||||
result = 1;
|
||||
@ -1335,7 +1342,7 @@ u8 os_is_being_debugged()
|
||||
return result;
|
||||
}
|
||||
|
||||
void print_string(String message)
|
||||
fn void print_string(String message)
|
||||
{
|
||||
#ifndef SILENT
|
||||
// TODO: check writes
|
||||
@ -1347,9 +1354,105 @@ void print_string(String message)
|
||||
#endif
|
||||
}
|
||||
|
||||
fn String os_get_environment_variable(const char* name)
|
||||
{
|
||||
String result = {};
|
||||
char* env = getenv(name);
|
||||
if (env)
|
||||
{
|
||||
result = cstr(env);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
fn u64 os_readlink(String path, String buffer)
|
||||
{
|
||||
u64 result = 0;
|
||||
assert(path.pointer[path.length] == 0);
|
||||
let(sys_result, readlink(string_to_c(path), string_to_c(buffer), buffer.length));
|
||||
if (sys_result > 0)
|
||||
{
|
||||
assign_cast(result, sys_result);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
fn String os_readlink_allocate(Arena* arena, String path)
|
||||
{
|
||||
String result = {};
|
||||
u8 buffer[4096];
|
||||
let(bytes, os_readlink(path, (String)array_to_slice(buffer)));
|
||||
|
||||
if (bytes > 0)
|
||||
{
|
||||
result.pointer = arena_allocate(arena, u8, bytes + 1);
|
||||
result.length = bytes;
|
||||
memcpy(result.pointer, buffer, bytes);
|
||||
result.pointer[bytes] = 0;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
#ifndef _WIN32
|
||||
fn String os_realpath(String path, String buffer)
|
||||
{
|
||||
String result = {};
|
||||
assert(path.pointer[path.length] == 0);
|
||||
char* system_result = realpath(string_to_c(path), string_to_c(buffer));
|
||||
if (system_result)
|
||||
{
|
||||
result = cstr(system_result);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
#endif
|
||||
|
||||
fn void os_free(void* pointer)
|
||||
{
|
||||
free(pointer);
|
||||
}
|
||||
|
||||
#if _WIN32
|
||||
HANDLE os_windows_get_module_handle()
|
||||
fn HANDLE os_windows_get_module_handle()
|
||||
{
|
||||
return GetModuleHandleW(0);
|
||||
}
|
||||
#endif
|
||||
|
||||
// TODO: structure this better
|
||||
#if _WIN32
|
||||
fn OSLibrary os_library_load(const char* library_name)
|
||||
{
|
||||
OSLibrary library = {};
|
||||
library.handle = LoadLibraryA(library_name);
|
||||
return library;
|
||||
}
|
||||
|
||||
fn OSSymbol os_symbol_load(OSLibrary library, const char* symbol_name)
|
||||
{
|
||||
OSSymbol symbol = GetProcAddress(library.handle, symbol_name);
|
||||
return symbol;
|
||||
}
|
||||
#else
|
||||
fn OSLibrary os_library_load(const char* library_name)
|
||||
{
|
||||
OSLibrary library = {};
|
||||
library.handle = dlopen(library_name, RTLD_NOW | RTLD_LOCAL);
|
||||
return library;
|
||||
}
|
||||
|
||||
fn OSSymbol os_symbol_load(OSLibrary library, const char* symbol_name)
|
||||
{
|
||||
OSSymbol symbol = dlsym(library.handle, symbol_name);
|
||||
return symbol;
|
||||
}
|
||||
#endif
|
||||
|
||||
fn u8 os_library_is_valid(OSLibrary library)
|
||||
{
|
||||
return library.handle != 0;
|
||||
}
|
||||
|
161
bootstrap/std/os.h
Normal file
161
bootstrap/std/os.h
Normal file
@ -0,0 +1,161 @@
|
||||
#pragma once
|
||||
|
||||
#include <stdarg.h>
|
||||
#include <stddef.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <time.h>
|
||||
#include <errno.h>
|
||||
|
||||
#ifdef _WIN32
|
||||
#define WIN32_LEAN_AND_MEAN 1
|
||||
#include <Windows.h>
|
||||
#else
|
||||
#include <dlfcn.h>
|
||||
#include <fcntl.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/mman.h>
|
||||
#include <sys/wait.h>
|
||||
#include <sys/time.h>
|
||||
#include <sys/ptrace.h>
|
||||
#endif
|
||||
|
||||
typedef enum TimeUnit
|
||||
{
|
||||
TIME_UNIT_NANOSECONDS,
|
||||
TIME_UNIT_MICROSECONDS,
|
||||
TIME_UNIT_MILLISECONDS,
|
||||
TIME_UNIT_SECONDS,
|
||||
} TimeUnit;
|
||||
|
||||
STRUCT(RunCommandOptions)
|
||||
{
|
||||
u64 debug:1;
|
||||
};
|
||||
|
||||
STRUCT(Timestamp)
|
||||
{
|
||||
u128 value;
|
||||
};
|
||||
|
||||
STRUCT(OSFileOpenFlags)
|
||||
{
|
||||
u32 truncate:1;
|
||||
u32 executable:1;
|
||||
u32 write:1;
|
||||
u32 read:1;
|
||||
u32 create:1;
|
||||
u32 directory:1;
|
||||
};
|
||||
|
||||
STRUCT(OSFilePermissions)
|
||||
{
|
||||
u8 readable:1;
|
||||
u8 writable:1;
|
||||
u8 executable:1;
|
||||
};
|
||||
|
||||
STRUCT(OSReserveProtectionFlags)
|
||||
{
|
||||
u32 read:1;
|
||||
u32 write:1;
|
||||
u32 execute:1;
|
||||
u32 reserved:29;
|
||||
};
|
||||
|
||||
STRUCT(OSReserveMapFlags)
|
||||
{
|
||||
u32 priv:1;
|
||||
u32 anon:1;
|
||||
u32 noreserve:1;
|
||||
u32 reserved:29;
|
||||
};
|
||||
|
||||
STRUCT(Arena)
|
||||
{
|
||||
u64 reserved_size;
|
||||
u64 position;
|
||||
u64 os_position;
|
||||
u64 granularity;
|
||||
u8 reserved[4 * 8];
|
||||
};
|
||||
|
||||
STRUCT(FileWriteOptions)
|
||||
{
|
||||
String path;
|
||||
String content;
|
||||
u8 executable;
|
||||
};
|
||||
|
||||
#if __APPLE__
|
||||
#define MY_PAGE_SIZE KB(16)
|
||||
#else
|
||||
#define MY_PAGE_SIZE KB(4)
|
||||
#endif
|
||||
const global_variable u64 page_size = MY_PAGE_SIZE;
|
||||
|
||||
global_variable u64 minimum_granularity = MY_PAGE_SIZE;
|
||||
// global_variable u64 middle_granularity = MB(2);
|
||||
global_variable u64 default_size = GB(4);
|
||||
|
||||
fn void print(const char* format, ...);
|
||||
fn void run_command(Arena* arena, CStringSlice arguments, char* envp[], RunCommandOptions options);
|
||||
fn String file_read(Arena* arena, String path);
|
||||
fn void file_write(FileWriteOptions options);
|
||||
|
||||
fn String path_dir(String string);
|
||||
fn String path_base(String string);
|
||||
fn String path_no_extension(String string);
|
||||
|
||||
|
||||
fn Arena* arena_initialize(u64 reserved_size, u64 granularity, u64 initial_size);
|
||||
fn Arena* arena_initialize_default(u64 initial_size);
|
||||
fn void arena_clear(Arena* arena);
|
||||
fn String arena_join_string(Arena* arena, Slice(String) pieces);
|
||||
fn u8* arena_allocate_bytes(Arena* arena, u64 size, u64 alignment);
|
||||
fn void arena_reset(Arena* arena);
|
||||
|
||||
#define arena_allocate(arena, T, count) (T*)(arena_allocate_bytes(arena, sizeof(T) * count, alignof(T)))
|
||||
#define arena_allocate_slice(arena, T, count) (Slice(T)){ .pointer = arena_allocate(arena, T, count), .length = count }
|
||||
|
||||
fn u8* os_reserve(u64 base, u64 size, OSReserveProtectionFlags protection, OSReserveMapFlags map);
|
||||
fn void os_commit(void* address, u64 size);
|
||||
|
||||
fn u8 os_file_descriptor_is_valid(FileDescriptor fd);
|
||||
fn FileDescriptor os_file_open(String path, OSFileOpenFlags flags, OSFilePermissions permissions);
|
||||
fn void os_file_close(FileDescriptor fd);
|
||||
fn u64 os_file_get_size(FileDescriptor fd);
|
||||
fn void os_file_write(FileDescriptor fd, String content);
|
||||
fn FileDescriptor os_stdout_get();
|
||||
fn void os_directory_make(String path);
|
||||
BB_NORETURN BB_COLD fn void os_exit(u32 exit_code);
|
||||
|
||||
fn void calibrate_cpu_timer();
|
||||
|
||||
fn void print_string(String string);
|
||||
|
||||
fn Timestamp os_timestamp();
|
||||
fn f64 os_resolve_timestamps(Timestamp start, Timestamp end, TimeUnit time_unit);
|
||||
fn u8 os_is_being_debugged();
|
||||
|
||||
#if _WIN32
|
||||
typedef void* HANDLE;
|
||||
fn HANDLE os_windows_get_module_handle();
|
||||
#endif
|
||||
|
||||
#if _WIN32
|
||||
#define EXECUTABLE_EXTENSION ".exe"
|
||||
#else
|
||||
#define EXECUTABLE_EXTENSION ""
|
||||
#endif
|
||||
|
||||
STRUCT(OSLibrary)
|
||||
{
|
||||
void* handle;
|
||||
};
|
||||
|
||||
typedef void* OSSymbol;
|
||||
|
||||
fn OSLibrary os_library_load(const char* library_name);
|
||||
fn OSSymbol os_symbol_load(OSLibrary library, const char* symbol_name);
|
60
bootstrap/std/project.h
Normal file
60
bootstrap/std/project.h
Normal file
@ -0,0 +1,60 @@
|
||||
#pragma once
|
||||
|
||||
#ifndef BUILD_DIR
|
||||
#define BUILD_DIR "cache"
|
||||
#endif
|
||||
|
||||
typedef enum RenderingBackend
|
||||
{
|
||||
RENDERING_BACKEND_NONE,
|
||||
RENDERING_BACKEND_METAL,
|
||||
RENDERING_BACKEND_VULKAN,
|
||||
RENDERING_BACKEND_DIRECTX12,
|
||||
RENDERING_BACKEND_COUNT,
|
||||
} RenderingBackend;
|
||||
|
||||
typedef enum WindowingBackend
|
||||
{
|
||||
WINDOWING_BACKEND_NONE,
|
||||
WINDOWING_BACKEND_WIN32,
|
||||
WINDOWING_BACKEND_X11,
|
||||
WINDOWING_BACKEND_WAYLAND,
|
||||
WINDOWING_BACKEND_COCOA,
|
||||
WINDOWING_BACKEND_COUNT,
|
||||
} WindowingBackend;
|
||||
|
||||
fn u8 rendering_backend_is_valid(RenderingBackend rendering_backend)
|
||||
{
|
||||
u8 valid = rendering_backend != RENDERING_BACKEND_COUNT;
|
||||
|
||||
if (valid && rendering_backend != RENDERING_BACKEND_NONE)
|
||||
{
|
||||
#ifdef __linux__
|
||||
valid = rendering_backend == RENDERING_BACKEND_VULKAN;
|
||||
#elif __APPLE__
|
||||
valid = rendering_backend == RENDERING_BACKEND_METAL || rendering_backend == RENDERING_BACKEND_VULKAN;
|
||||
#elif _WIN32
|
||||
valid = rendering_backend == RENDERING_BACKEND_DIRECTX12 || rendering_backend == RENDERING_BACKEND_VULKAN;
|
||||
#endif
|
||||
}
|
||||
|
||||
return valid;
|
||||
}
|
||||
|
||||
fn u8 windowing_backend_is_valid(WindowingBackend windowing_backend)
|
||||
{
|
||||
u8 valid = windowing_backend != WINDOWING_BACKEND_COUNT;
|
||||
|
||||
if (valid && windowing_backend != WINDOWING_BACKEND_NONE)
|
||||
{
|
||||
#ifdef __linux__
|
||||
valid = windowing_backend == WINDOWING_BACKEND_WAYLAND || windowing_backend == WINDOWING_BACKEND_X11;
|
||||
#elif _WIN32
|
||||
valid = windowing_backend == WINDOWING_BACKEND_WIN32;
|
||||
#elif __APPLE__
|
||||
valid = windowing_backend == WINDOWING_BACKEND_COCOA;
|
||||
#endif
|
||||
}
|
||||
|
||||
return valid;
|
||||
}
|
@ -1 +0,0 @@
|
||||
#include <Metal.hpp>
|
15
bootstrap/std/rendering.c
Normal file
15
bootstrap/std/rendering.c
Normal file
@ -0,0 +1,15 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/font_provider.c>
|
||||
|
||||
#if BB_RENDERING_BACKEND_VULKAN
|
||||
#include <std/vulkan_rendering.c>
|
||||
#endif
|
||||
|
||||
#if BB_RENDERING_BACKEND_METAL
|
||||
#include <std/metal_rendering.c>
|
||||
#endif
|
||||
|
||||
#if BB_RENDERING_BACKEND_DIRECTX12
|
||||
#include <std/directx12_rendering.c>
|
||||
#endif
|
334
bootstrap/std/rendering.h
Normal file
334
bootstrap/std/rendering.h
Normal file
@ -0,0 +1,334 @@
|
||||
#pragma once
|
||||
|
||||
#if BB_RENDERING_BACKEND_VULKAN
|
||||
#include <std/vulkan_rendering.h>
|
||||
#endif
|
||||
|
||||
#if BB_RENDERING_BACKEND_METAL
|
||||
#include <std/metal_rendering.h>
|
||||
#endif
|
||||
|
||||
#if BB_RENDERING_BACKEND_DIRECTX12
|
||||
#include <std/directx12_rendering.h>
|
||||
#endif
|
||||
|
||||
#ifdef __clang__
|
||||
#define BB_HAS_NATIVE_FLOAT2 1
|
||||
#define BB_HAS_NATIVE_FLOAT3 1
|
||||
#define BB_HAS_NATIVE_FLOAT4 1
|
||||
#define BB_HAS_NATIVE_UINT2 1
|
||||
#define BB_HAS_NATIVE_UINT3 1
|
||||
#define BB_HAS_NATIVE_UINT4 1
|
||||
#else
|
||||
#define BB_HAS_NATIVE_FLOAT2 0
|
||||
#define BB_HAS_NATIVE_FLOAT3 0
|
||||
#define BB_HAS_NATIVE_FLOAT4 0
|
||||
#define BB_HAS_NATIVE_UINT2 0
|
||||
#define BB_HAS_NATIVE_UINT3 0
|
||||
#define BB_HAS_NATIVE_UINT4 0
|
||||
#endif
|
||||
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
declare_vector_type(float, 2, float2);
|
||||
#else
|
||||
UNION(float2)
|
||||
{
|
||||
struct
|
||||
{
|
||||
float x, y;
|
||||
};
|
||||
float v[2];
|
||||
};
|
||||
#endif
|
||||
|
||||
#if BB_HAS_NATIVE_FLOAT3
|
||||
declare_vector_type(float, 3, float3);
|
||||
#else
|
||||
UNION(float3)
|
||||
{
|
||||
struct
|
||||
{
|
||||
float x, y, z;
|
||||
};
|
||||
float v[3];
|
||||
};
|
||||
#endif
|
||||
|
||||
#if BB_HAS_NATIVE_FLOAT4
|
||||
declare_vector_type(float, 4, float4);
|
||||
#else
|
||||
UNION(float4)
|
||||
{
|
||||
struct
|
||||
{
|
||||
float x, y, z, w;
|
||||
};
|
||||
float v[4];
|
||||
};
|
||||
#endif
|
||||
|
||||
#if BB_HAS_NATIVE_UINT2
|
||||
declare_vector_type(uint, 2, uint2);
|
||||
#else
|
||||
UNION(uint2)
|
||||
{
|
||||
struct
|
||||
{
|
||||
uint x, y;
|
||||
};
|
||||
uint v[2];
|
||||
};
|
||||
#endif
|
||||
|
||||
#if BB_HAS_NATIVE_UINT3
|
||||
declare_vector_type(uint, 3, uint3);
|
||||
#else
|
||||
UNION(uint3)
|
||||
{
|
||||
struct
|
||||
{
|
||||
uint x, y, z;
|
||||
};
|
||||
uint v[3];
|
||||
};
|
||||
#endif
|
||||
|
||||
#if BB_HAS_NATIVE_UINT4
|
||||
declare_vector_type(uint, 4, uint4);
|
||||
#else
|
||||
UNION(uint4)
|
||||
{
|
||||
struct
|
||||
{
|
||||
uint x, y, z, w;
|
||||
};
|
||||
uint v[4];
|
||||
};
|
||||
#endif
|
||||
|
||||
typedef float2 vec2;
|
||||
typedef float3 vec3;
|
||||
typedef float4 vec4;
|
||||
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
#define VEC2(_x, y) ((vec2){_x, _y})
|
||||
#else
|
||||
#define VEC2(_x, _y) ((vec2){ .x = _x, .y = _y})
|
||||
#endif
|
||||
|
||||
#if BB_HAS_NATIVE_FLOAT3
|
||||
#define VEC3(_x, _y, _z) ((vec3){_x, _y, _z})
|
||||
#else
|
||||
#define VEC3(_x, _y, _z) ((vec3){ .x = _x, .y = _y, .z = _z})
|
||||
#endif
|
||||
|
||||
#if BB_HAS_NATIVE_FLOAT4
|
||||
#define VEC4(_x, _y, _z, _w) ((vec4){_x, _y, _z, _w})
|
||||
#else
|
||||
#define VEC4(_x, _y, _z, _w) ((vec4){ .x = _x, .y = _y, .z = _z, .w = _w})
|
||||
#endif
|
||||
|
||||
fn float2 float2_add(float2 a, float2 b)
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
return a + b;
|
||||
#else
|
||||
float2 result;
|
||||
result.x = a.x + b.x;
|
||||
result.y = a.y + b.y;
|
||||
return result;
|
||||
#endif
|
||||
}
|
||||
|
||||
fn float2 float2_sub(float2 a, float2 b)
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
return a - b;
|
||||
#else
|
||||
float2 result;
|
||||
result.x = a.x - b.x;
|
||||
result.y = a.y - b.y;
|
||||
return result;
|
||||
#endif
|
||||
}
|
||||
|
||||
UNION(F32Interval2)
|
||||
{
|
||||
struct
|
||||
{
|
||||
vec2 min;
|
||||
vec2 max;
|
||||
};
|
||||
struct
|
||||
{
|
||||
float2 p0;
|
||||
float2 p1;
|
||||
};
|
||||
struct
|
||||
{
|
||||
f32 x0;
|
||||
f32 y0;
|
||||
f32 x1;
|
||||
f32 y1;
|
||||
};
|
||||
float2 v[2];
|
||||
};
|
||||
static_assert(sizeof(F32Interval2) == 4 * sizeof(f32));
|
||||
|
||||
typedef struct Renderer Renderer;
|
||||
typedef struct RenderWindow RenderWindow;
|
||||
typedef struct Pipeline Pipeline;
|
||||
|
||||
STRUCT(RectDraw)
|
||||
{
|
||||
F32Interval2 vertex;
|
||||
F32Interval2 texture;
|
||||
vec4 colors[4];
|
||||
u32 texture_index;
|
||||
};
|
||||
|
||||
#include "../std/shaders/rect.inc"
|
||||
typedef struct RectVertex RectVertex;
|
||||
decl_vb(RectVertex);
|
||||
|
||||
typedef enum BBPipeline
|
||||
{
|
||||
BB_PIPELINE_RECT,
|
||||
BB_PIPELINE_COUNT,
|
||||
} BBPipeline;
|
||||
|
||||
typedef enum RenderFontType
|
||||
{
|
||||
RENDER_FONT_TYPE_MONOSPACE,
|
||||
RENDER_FONT_TYPE_PROPORTIONAL,
|
||||
RENDER_FONT_TYPE_COUNT,
|
||||
} RenderFontType;
|
||||
|
||||
typedef enum RectTextureSlot
|
||||
{
|
||||
RECT_TEXTURE_SLOT_WHITE,
|
||||
RECT_TEXTURE_SLOT_MONOSPACE_FONT,
|
||||
RECT_TEXTURE_SLOT_PROPORTIONAL_FONT,
|
||||
RECT_TEXTURE_SLOT_COUNT
|
||||
} RectTextureSlot;
|
||||
|
||||
typedef enum TextureFormat
|
||||
{
|
||||
TEXTURE_FORMAT_R8_UNORM,
|
||||
TEXTURE_FORMAT_R8G8B8A8_SRGB,
|
||||
} TextureFormat;
|
||||
|
||||
STRUCT(TextureMemory)
|
||||
{
|
||||
void* pointer;
|
||||
u32 width;
|
||||
u32 height;
|
||||
u32 depth;
|
||||
TextureFormat format;
|
||||
};
|
||||
|
||||
ENUM(ShaderStage, u8,
|
||||
SHADER_STAGE_VERTEX,
|
||||
SHADER_STAGE_FRAGMENT,
|
||||
);
|
||||
|
||||
STRUCT(PipelineCreate)
|
||||
{
|
||||
Slice(u16) shader_source_indices;
|
||||
u16 layout_index;
|
||||
};
|
||||
declare_slice(PipelineCreate);
|
||||
|
||||
STRUCT(PushConstantRange)
|
||||
{
|
||||
u16 offset;
|
||||
u16 size;
|
||||
ShaderStage stage;
|
||||
};
|
||||
declare_slice(PushConstantRange);
|
||||
|
||||
ENUM(DescriptorType, u8,
|
||||
DESCRIPTOR_TYPE_IMAGE_PLUS_SAMPLER,
|
||||
DESCRIPTOR_TYPE_COUNT,
|
||||
);
|
||||
|
||||
STRUCT(DescriptorSetLayoutBinding)
|
||||
{
|
||||
u8 binding;
|
||||
DescriptorType type;
|
||||
ShaderStage stage;
|
||||
u8 count;
|
||||
};
|
||||
declare_slice(DescriptorSetLayoutBinding);
|
||||
|
||||
STRUCT(DescriptorSetLayoutCreate)
|
||||
{
|
||||
Slice(DescriptorSetLayoutBinding) bindings;
|
||||
};
|
||||
declare_slice(DescriptorSetLayoutCreate);
|
||||
|
||||
STRUCT(PipelineLayoutCreate)
|
||||
{
|
||||
Slice(PushConstantRange) push_constant_ranges;
|
||||
Slice(DescriptorSetLayoutCreate) descriptor_set_layouts;
|
||||
};
|
||||
declare_slice(PipelineLayoutCreate);
|
||||
|
||||
STRUCT(GraphicsPipelinesCreate)
|
||||
{
|
||||
Slice(String) shader_binaries;
|
||||
Slice(PipelineLayoutCreate) layouts;
|
||||
Slice(PipelineCreate) pipelines;
|
||||
};
|
||||
|
||||
STRUCT(PipelineIndex)
|
||||
{
|
||||
u32 value;
|
||||
};
|
||||
|
||||
STRUCT(PipelineLayoutIndex)
|
||||
{
|
||||
u32 value;
|
||||
};
|
||||
|
||||
STRUCT(DescriptorSetIndex)
|
||||
{
|
||||
u32 value;
|
||||
};
|
||||
|
||||
ENUM(BufferType, u8,
|
||||
BUFFER_TYPE_VERTEX,
|
||||
BUFFER_TYPE_INDEX,
|
||||
BUFFER_TYPE_STAGING,
|
||||
);
|
||||
|
||||
STRUCT(HostBufferCopy)
|
||||
{
|
||||
String source;
|
||||
u64 destination_offset;
|
||||
};
|
||||
declare_slice(HostBufferCopy);
|
||||
|
||||
STRUCT(LocalBufferCopyRegion)
|
||||
{
|
||||
u64 source_offset;
|
||||
u64 destination_offset;
|
||||
u64 size;
|
||||
};
|
||||
declare_slice(LocalBufferCopyRegion);
|
||||
|
||||
#include <std/font_provider.h>
|
||||
|
||||
fn Renderer* rendering_initialize(Arena* arena);
|
||||
fn RenderWindow* rendering_initialize_window(Renderer* renderer, WindowingInstance* window);
|
||||
fn void renderer_window_frame_begin(Renderer* renderer, RenderWindow* window);
|
||||
fn void renderer_window_frame_end(Renderer* renderer, RenderWindow* window);
|
||||
fn TextureIndex renderer_texture_create(Renderer* renderer, TextureMemory texture_memory);
|
||||
|
||||
fn void window_rect_texture_update_begin(RenderWindow* window);
|
||||
fn void renderer_queue_font_update(Renderer* renderer, RenderWindow* window, RenderFontType type, TextureAtlas atlas);
|
||||
fn void window_queue_rect_texture_update(RenderWindow* window, RectTextureSlot slot, TextureIndex texture_index);
|
||||
fn void window_rect_texture_update_end(Renderer* renderer, RenderWindow* window);
|
||||
|
||||
fn void window_render_rect(RenderWindow* window, RectDraw draw);
|
||||
fn void window_render_text(Renderer* renderer, RenderWindow* window, String string, float4 color, RenderFontType font_type, u32 x_offset, u32 y_offset);
|
122
bootstrap/std/shaders/rect.metal
Normal file
122
bootstrap/std/shaders/rect.metal
Normal file
@ -0,0 +1,122 @@
|
||||
#include <metal_stdlib>
|
||||
using namespace metal;
|
||||
|
||||
// Structs to match buffer data
|
||||
struct RectVertex
|
||||
{
|
||||
float2 p0; // Bottom-left corner position
|
||||
float2 extent; // Rectangle size (width and height)
|
||||
float2 uv0; // UV coordinates for bottom-left
|
||||
float corner_radius; // Corner radius
|
||||
float softness; // Edge softness
|
||||
float4 colors[4]; // Per-vertex colors
|
||||
uint texture_index; // Texture index
|
||||
};
|
||||
|
||||
struct RectFragmentShaderInput
|
||||
{
|
||||
float4 vertex_position [[position]];
|
||||
float4 color; // Vertex color
|
||||
float2 uv; // UV coordinates
|
||||
float2 position; // Position in world space
|
||||
float2 center; // Center of the rectangle
|
||||
float2 half_size; // Half dimensions of the rectangle
|
||||
float corner_radius; // Corner radius
|
||||
float softness; // Edge softness
|
||||
};
|
||||
|
||||
// Push constants (Metal uses constant buffers or argument buffers for this)
|
||||
struct PushConstants
|
||||
{
|
||||
device const RectVertex* vertex_buffer;
|
||||
float width; // Screen width
|
||||
float height; // Screen height
|
||||
};
|
||||
|
||||
// Vertex shader main function
|
||||
vertex RectFragmentShaderInput vertex_main(const device PushConstants& push_constants [[buffer(0)]], uint vertex_id [[vertex_id]])
|
||||
{
|
||||
// Constants for normalized quad corners
|
||||
constexpr float2 vertices[] = {
|
||||
float2(-1.0, -1.0),
|
||||
float2( 1.0, -1.0),
|
||||
float2(-1.0, 1.0),
|
||||
float2( 1.0, 1.0)
|
||||
};
|
||||
|
||||
// Fetch the vertex data from the buffer
|
||||
const RectVertex v = push_constants.vertex_buffer[vertex_id / 4];
|
||||
|
||||
// Rectangle calculations
|
||||
float2 p0 = v.p0;
|
||||
float2 p1 = p0 + v.extent;
|
||||
float2 position_center = (p1 + p0) * 0.5;
|
||||
float2 half_size = (p1 - p0) * 0.5;
|
||||
float2 position = vertices[vertex_id % 4] * half_size + position_center;
|
||||
|
||||
// Screen-space transformation
|
||||
float2 ndc_position = float2(2.0 * position.x / push_constants.width - 1.0,
|
||||
2.0 * position.y / push_constants.height - 1.0);
|
||||
|
||||
// Texture UV calculations
|
||||
float2 uv0 = v.uv0;
|
||||
float2 uv1 = uv0 + v.extent;
|
||||
float2 texture_center = (uv1 + uv0) * 0.5;
|
||||
float2 uv = vertices[vertex_id % 4] * half_size + texture_center;
|
||||
|
||||
// Output to fragment shader
|
||||
RectFragmentShaderInput out_data;
|
||||
out_data.color = v.colors[vertex_id % 4];
|
||||
out_data.uv = uv;
|
||||
out_data.position = position;
|
||||
out_data.vertex_position = float4(ndc_position.x, ndc_position.y, 0, 1);
|
||||
out_data.center = position_center;
|
||||
out_data.half_size = half_size;
|
||||
out_data.corner_radius = v.corner_radius;
|
||||
out_data.softness = v.softness;
|
||||
|
||||
return out_data;
|
||||
}
|
||||
|
||||
// Uniform buffer for textures
|
||||
struct FragmentUniforms
|
||||
{
|
||||
array<texture2d<float>, 100> textures; // Array of textures
|
||||
sampler texture_sampler; // Sampler for texture sampling
|
||||
};
|
||||
|
||||
// Calculate the signed distance field (SDF) for a rounded rectangle
|
||||
float rounded_rect_sdf(float2 position, float2 center, float2 half_size, float radius)
|
||||
{
|
||||
float2 r2 = float2(radius, radius);
|
||||
float2 d2_no_r2 = abs(center - position) - half_size;
|
||||
float2 d2 = d2_no_r2 + r2;
|
||||
float negative_euclidean_distance = min(max(d2.x, d2.y), 0.0);
|
||||
float positive_euclidean_distance = length(max(d2, 0.0));
|
||||
return negative_euclidean_distance + positive_euclidean_distance - radius;
|
||||
}
|
||||
|
||||
// Fragment shader function
|
||||
fragment float4 fragment_main(RectFragmentShaderInput inputs [[stage_in]], constant FragmentUniforms &uniforms [[buffer(0)]], uint texture_index)
|
||||
{
|
||||
// Texture size
|
||||
float2 texture_size = float2(uniforms.textures[texture_index].get_width(), uniforms.textures[texture_index].get_height());
|
||||
float2 uv = float2(inputs.uv.x / texture_size.x, inputs.uv.y / texture_size.y);
|
||||
|
||||
// Sample texture
|
||||
float4 sampled = uniforms.textures[texture_index].sample(uniforms.texture_sampler, uv);
|
||||
|
||||
// Compute softness padding
|
||||
float softness = inputs.softness;
|
||||
float softness_padding_scalar = max(0.0, softness * 2.0 - 1.0);
|
||||
float2 softness_padding = float2(softness_padding_scalar, softness_padding_scalar);
|
||||
|
||||
// Compute signed distance
|
||||
float distance = rounded_rect_sdf(inputs.position, inputs.center, inputs.half_size - softness_padding, inputs.corner_radius);
|
||||
|
||||
// Compute SDF factor
|
||||
float sdf_factor = 1.0 - smoothstep(0.0, 2.0 * softness, distance);
|
||||
|
||||
// Compute final color
|
||||
return inputs.color * sampled * sdf_factor;
|
||||
}
|
@ -1,8 +1,8 @@
|
||||
#include <std/string.h>
|
||||
|
||||
s32 string_first_ch(String string, u8 ch)
|
||||
u64 string_first_ch(String string, u8 ch)
|
||||
{
|
||||
s32 result = -1;
|
||||
u64 result = STRING_NO_MATCH;
|
||||
for (u64 i = 0; i < string.length; i += 1)
|
||||
{
|
||||
if (string.pointer[i] == ch)
|
||||
@ -15,16 +15,16 @@ s32 string_first_ch(String string, u8 ch)
|
||||
return result;
|
||||
}
|
||||
|
||||
s64 string_last_ch(String string, u8 ch)
|
||||
u64 string_last_ch(String string, u8 ch)
|
||||
{
|
||||
s64 result = -1;
|
||||
u64 result = STRING_NO_MATCH;
|
||||
u64 i = string.length;
|
||||
while (i > 0)
|
||||
{
|
||||
i -= 1;
|
||||
if (string.pointer[i] == ch)
|
||||
{
|
||||
result = cast_to(s64, u64, i);
|
||||
result = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
@ -47,8 +47,8 @@ u8 string_starts_with(String string, String start)
|
||||
u64 i;
|
||||
for (i = 0; i < start.length; i += 1)
|
||||
{
|
||||
auto start_ch = start.pointer[i];
|
||||
auto string_ch = string.pointer[i];
|
||||
let(start_ch, start.pointer[i]);
|
||||
let(string_ch, string.pointer[i]);
|
||||
if (unlikely(string_ch != start_ch))
|
||||
{
|
||||
break;
|
||||
@ -72,8 +72,8 @@ u8 string_ends_with(String string, String end)
|
||||
u64 offset = string.length - end.length;
|
||||
for (i = 0; i < end.length; i += 1)
|
||||
{
|
||||
auto start_ch = end.pointer[i];
|
||||
auto string_ch = string.pointer[i + offset];
|
||||
let(start_ch, end.pointer[i]);
|
||||
let(string_ch, string.pointer[i + offset]);
|
||||
if (unlikely(string_ch != start_ch))
|
||||
{
|
||||
break;
|
||||
@ -86,9 +86,9 @@ u8 string_ends_with(String string, String end)
|
||||
return result;
|
||||
}
|
||||
|
||||
u64 string_first_ocurrence(String string, String substring)
|
||||
u64 string_first_occurrence(String string, String substring)
|
||||
{
|
||||
s32 result = UINT64_MAX;
|
||||
u64 result = STRING_NO_MATCH;
|
||||
|
||||
if (substring.length < string.length)
|
||||
{
|
||||
@ -99,7 +99,7 @@ u64 string_first_ocurrence(String string, String substring)
|
||||
break;
|
||||
}
|
||||
|
||||
auto s = s_get_slice(u8, string, i, i + substring.length);
|
||||
String s = s_get_slice(u8, string, i, i + substring.length);
|
||||
if (s_equal(s, substring))
|
||||
{
|
||||
result = i;
|
||||
@ -122,7 +122,12 @@ u64 string_first_ocurrence(String string, String substring)
|
||||
return result;
|
||||
}
|
||||
|
||||
u64 string_last_ocurrence(String string, String substring)
|
||||
fn u64 string_last_occurrence(String string, String substring)
|
||||
{
|
||||
todo();
|
||||
}
|
||||
|
||||
fn u8 string_contains(String string, String substring)
|
||||
{
|
||||
return string_first_occurrence(string, substring) != STRING_NO_MATCH;
|
||||
}
|
||||
|
10
bootstrap/std/string.h
Normal file
10
bootstrap/std/string.h
Normal file
@ -0,0 +1,10 @@
|
||||
#include <std/base.h>
|
||||
|
||||
#define STRING_NO_MATCH UINT64_MAX
|
||||
|
||||
fn u64 string_first_ch(String string, u8 ch);
|
||||
fn u64 string_last_ch(String string, u8 ch);
|
||||
fn u8 string_starts_with(String string, String start);
|
||||
fn u8 string_ends_with(String string, String end);
|
||||
fn u64 string_first_occurrence(String string, String substring);
|
||||
fn u64 string_last_occurrence(String string, String substring);
|
@ -1,14 +1,13 @@
|
||||
#include <std/ui_builder.h>
|
||||
#include <std/render.h>
|
||||
#pragma once
|
||||
|
||||
UI_Signal ui_button(String string)
|
||||
fn UI_Signal ui_button(String string)
|
||||
{
|
||||
auto* widget = ui_widget_make((UI_WidgetFlags) {
|
||||
let(widget, ui_widget_make((UI_WidgetFlags) {
|
||||
.draw_text = 1,
|
||||
.draw_background = 1,
|
||||
.mouse_clickable = 1,
|
||||
.keyboard_pressable = 1,
|
||||
}, string);
|
||||
}, string));
|
||||
|
||||
UI_Signal signal = ui_signal_from_widget(widget);
|
||||
return signal;
|
||||
|
@ -3,4 +3,4 @@
|
||||
#include <std/base.h>
|
||||
#include <std/ui_core.h>
|
||||
|
||||
EXPORT UI_Signal ui_button(String string);
|
||||
fn UI_Signal ui_button(String string);
|
@ -13,27 +13,25 @@
|
||||
// https://www.rfleury.com/p/ui-bonus-1-simple-single-line-text
|
||||
// https://www.rfleury.com/p/codebase-walkthrough-multi-window
|
||||
|
||||
#include <std/ui_core.h>
|
||||
#include <std/format.h>
|
||||
#include <std/string.h>
|
||||
#pragma once
|
||||
|
||||
UI_State* ui_state = 0;
|
||||
global_variable UI_State* ui_state = 0;
|
||||
|
||||
fn void ui_autopop(UI_State* state)
|
||||
{
|
||||
auto* restrict stack_end = (u32*)((u8*)&state->stacks + sizeof(state->stacks));
|
||||
let(stack_end, (u32*)((u8*)&state->stacks + sizeof(state->stacks)));
|
||||
|
||||
auto* restrict bitset_pointer = (u64*)&state->stack_autopops;
|
||||
let(bitset_pointer, (u64*)&state->stack_autopops);
|
||||
u64 bitset_index = 0;
|
||||
for (auto* restrict stack_pointer = (u32*)&state->stacks; stack_pointer != stack_end; stack_pointer += sizeof(VirtualBuffer(u8)) / sizeof(u32))
|
||||
for (let(stack_pointer, (u32*)&state->stacks); stack_pointer != stack_end; stack_pointer += sizeof(VirtualBuffer(u8)) / sizeof(u32))
|
||||
{
|
||||
auto bitset = *bitset_pointer;
|
||||
auto shift_value = 1 << bitset_index;
|
||||
auto autopop = (bitset & shift_value) != 0;
|
||||
auto mask = ~shift_value;
|
||||
let(bitset, *bitset_pointer);
|
||||
let(shift_value, 1 << bitset_index);
|
||||
let(autopop, (bitset & shift_value) != 0);
|
||||
let(mask, ~shift_value);
|
||||
*bitset_pointer = bitset & mask;
|
||||
auto* restrict length_pointer = stack_pointer + (offsetof(VirtualBuffer(u8), length) / sizeof(u32));
|
||||
auto current_length = *length_pointer;
|
||||
let(length_pointer, stack_pointer + (offsetof(VirtualBuffer(u8), length) / sizeof(u32)));
|
||||
let(current_length, *length_pointer);
|
||||
assert(!autopop | current_length);
|
||||
*length_pointer -= autopop;
|
||||
|
||||
@ -43,19 +41,19 @@ fn void ui_autopop(UI_State* state)
|
||||
}
|
||||
}
|
||||
|
||||
void ui_state_select(UI_State* state)
|
||||
fn void ui_state_select(UI_State* state)
|
||||
{
|
||||
ui_state = state;
|
||||
}
|
||||
|
||||
UI_State* ui_state_get()
|
||||
fn UI_State* ui_state_get()
|
||||
{
|
||||
return ui_state;
|
||||
}
|
||||
|
||||
fn Arena* ui_build_arena()
|
||||
{
|
||||
auto* arena = ui_state->build_arenas[ui_state->build_count % array_length(ui_state->build_arenas)];
|
||||
let(arena, ui_state->build_arenas[ui_state->build_count % array_length(ui_state->build_arenas)]);
|
||||
return arena;
|
||||
}
|
||||
|
||||
@ -65,9 +63,9 @@ fn UI_Key ui_key_null()
|
||||
return key;
|
||||
}
|
||||
|
||||
UI_State* ui_state_allocate(Renderer* renderer, RenderWindow* window)
|
||||
fn UI_State* ui_state_allocate(Renderer* renderer, RenderWindow* window)
|
||||
{
|
||||
Arena* arena = arena_init(GB(8), MB(2), MB(2));
|
||||
Arena* arena = arena_initialize(GB(8), MB(2), MB(2));
|
||||
UI_State* state = arena_allocate(arena, UI_State, 1);
|
||||
state->renderer = renderer;
|
||||
state->render_window = window;
|
||||
@ -77,7 +75,7 @@ UI_State* ui_state_allocate(Renderer* renderer, RenderWindow* window)
|
||||
|
||||
for (u64 i = 0; i < array_length(state->build_arenas); i += 1)
|
||||
{
|
||||
state->build_arenas[i] = arena_init(GB(8), MB(2), MB(2));
|
||||
state->build_arenas[i] = arena_initialize(GB(8), MB(2), MB(2));
|
||||
}
|
||||
|
||||
state->stack_nulls = (UI_StateStackNulls){
|
||||
@ -87,11 +85,11 @@ UI_State* ui_state_allocate(Renderer* renderer, RenderWindow* window)
|
||||
.pref_height = {},
|
||||
};
|
||||
|
||||
auto* stack_end = (u32*)((u8*)&state->stacks + sizeof(state->stacks));
|
||||
let(stack_end, (u32*)((u8*)&state->stacks + sizeof(state->stacks)));
|
||||
|
||||
for (auto* stack_pointer = (u32*)&state->stacks; stack_pointer != stack_end; stack_pointer += sizeof(VirtualBuffer(u8)) / sizeof(u32))
|
||||
for (let(stack_pointer, (u32*)&state->stacks); stack_pointer != stack_end; stack_pointer += sizeof(VirtualBuffer(u8)) / sizeof(u32))
|
||||
{
|
||||
auto* length_pointer = stack_pointer + (offsetof(VirtualBuffer(u8), length) / sizeof(u32));
|
||||
let(length_pointer, stack_pointer + (offsetof(VirtualBuffer(u8), length) / sizeof(u32)));
|
||||
assert(*length_pointer == 0);
|
||||
}
|
||||
|
||||
@ -100,18 +98,17 @@ UI_State* ui_state_allocate(Renderer* renderer, RenderWindow* window)
|
||||
|
||||
fn u64 ui_widget_index_from_key(UI_Key key)
|
||||
{
|
||||
auto length = ui_state->widget_table.length;
|
||||
let(length, ui_state->widget_table.length);
|
||||
assert(is_power_of_two(length));
|
||||
return key.value & (length - 1);
|
||||
}
|
||||
|
||||
auto text_end_delimiter = strlit("##");
|
||||
auto hash_start_delimiter = strlit("###");
|
||||
|
||||
fn String ui_text_from_key_string(String string)
|
||||
{
|
||||
String result = string;
|
||||
auto index = string_first_ocurrence(string, text_end_delimiter);
|
||||
String text_end_delimiter = strlit("##");
|
||||
let(index, string_first_occurrence(string, text_end_delimiter));
|
||||
if (index < string.length)
|
||||
{
|
||||
result.length = index;
|
||||
@ -122,7 +119,8 @@ fn String ui_text_from_key_string(String string)
|
||||
fn String ui_hash_from_key_string(String string)
|
||||
{
|
||||
String result = string;
|
||||
auto index = string_first_ocurrence(string, hash_start_delimiter);
|
||||
String hash_start_delimiter = strlit("###");
|
||||
let(index, string_first_occurrence(string, hash_start_delimiter));
|
||||
if (index < string.length)
|
||||
{
|
||||
result = s_get_slice(u8, string, index, string.length);
|
||||
@ -153,9 +151,9 @@ fn UI_Key ui_key_from_string_format(UI_Key seed, char* format, ...)
|
||||
u8 buffer[256];
|
||||
va_list args;
|
||||
va_start(args, format);
|
||||
auto string = format_string_va((String)array_to_slice(buffer), format, args);
|
||||
let(string, format_string_va((String)array_to_slice(buffer), format, args));
|
||||
va_end(args);
|
||||
auto result = ui_key_from_string(seed, string);
|
||||
let(result, ui_key_from_string(seed, string));
|
||||
return result;
|
||||
}
|
||||
|
||||
@ -164,13 +162,13 @@ fn u8 ui_key_equal(UI_Key a, UI_Key b)
|
||||
return a.value == b.value;
|
||||
}
|
||||
|
||||
UI_Widget* ui_widget_from_key(UI_Key key)
|
||||
fn UI_Widget* ui_widget_from_key(UI_Key key)
|
||||
{
|
||||
UI_Widget* result = 0;
|
||||
|
||||
if (!ui_key_equal(key, ui_key_null()))
|
||||
{
|
||||
auto index = ui_widget_index_from_key(key);
|
||||
let(index, ui_widget_index_from_key(key));
|
||||
for (UI_Widget* widget = ui_state->widget_table.pointer[index].first; widget; widget = widget->hash_next)
|
||||
{
|
||||
if (ui_key_equal(widget->key, key))
|
||||
@ -184,10 +182,10 @@ UI_Widget* ui_widget_from_key(UI_Key key)
|
||||
return result;
|
||||
}
|
||||
|
||||
UI_Widget* ui_widget_make_from_key(UI_WidgetFlags flags, UI_Key key)
|
||||
fn UI_Widget* ui_widget_make_from_key(UI_WidgetFlags flags, UI_Key key)
|
||||
{
|
||||
auto* widget = ui_widget_from_key(key);
|
||||
static auto count = 0;
|
||||
let(widget, ui_widget_from_key(key));
|
||||
static let(count, 0);
|
||||
count += 1;
|
||||
|
||||
if (widget)
|
||||
@ -202,12 +200,12 @@ UI_Widget* ui_widget_make_from_key(UI_WidgetFlags flags, UI_Key key)
|
||||
u8 first_frame = 0;
|
||||
if (!widget)
|
||||
{
|
||||
auto index = ui_widget_index_from_key(key);
|
||||
let(index, ui_widget_index_from_key(key));
|
||||
first_frame = 1;
|
||||
|
||||
widget = arena_allocate(ui_state->arena, UI_Widget, 1);
|
||||
|
||||
auto* table_widget_slot = &ui_state->widget_table.pointer[index];
|
||||
let(table_widget_slot, &ui_state->widget_table.pointer[index]);
|
||||
if (!table_widget_slot->last)
|
||||
{
|
||||
table_widget_slot->first = widget;
|
||||
@ -221,7 +219,7 @@ UI_Widget* ui_widget_make_from_key(UI_WidgetFlags flags, UI_Key key)
|
||||
}
|
||||
}
|
||||
|
||||
auto* parent = ui_top(parent);
|
||||
let(parent, ui_top(parent));
|
||||
|
||||
if (parent)
|
||||
{
|
||||
@ -232,7 +230,7 @@ UI_Widget* ui_widget_make_from_key(UI_WidgetFlags flags, UI_Key key)
|
||||
}
|
||||
else
|
||||
{
|
||||
auto* previous_last = parent->last;
|
||||
let(previous_last, parent->last);
|
||||
previous_last->next = widget;
|
||||
widget->previous = previous_last;
|
||||
parent->last = widget;
|
||||
@ -266,15 +264,15 @@ UI_Widget* ui_widget_make_from_key(UI_WidgetFlags flags, UI_Key key)
|
||||
return widget;
|
||||
}
|
||||
|
||||
UI_Widget* ui_widget_make(UI_WidgetFlags flags, String string)
|
||||
fn UI_Widget* ui_widget_make(UI_WidgetFlags flags, String string)
|
||||
{
|
||||
// TODO:
|
||||
auto seed = ui_key_null();
|
||||
let(seed, ui_key_null());
|
||||
|
||||
auto hash_string = ui_hash_from_key_string(string);
|
||||
auto key = ui_key_from_string(seed, hash_string);
|
||||
let(hash_string, ui_hash_from_key_string(string));
|
||||
let(key, ui_key_from_string(seed, hash_string));
|
||||
|
||||
auto* widget = ui_widget_make_from_key(flags, key);
|
||||
let(widget, ui_widget_make_from_key(flags, key));
|
||||
|
||||
if (flags.draw_text)
|
||||
{
|
||||
@ -284,25 +282,25 @@ UI_Widget* ui_widget_make(UI_WidgetFlags flags, String string)
|
||||
return widget;
|
||||
}
|
||||
|
||||
UI_Widget* ui_widget_make_format(UI_WidgetFlags flags, const char* format, ...)
|
||||
fn UI_Widget* ui_widget_make_format(UI_WidgetFlags flags, const char* format, ...)
|
||||
{
|
||||
va_list args;
|
||||
u8 buffer[4096];
|
||||
va_start(args, format);
|
||||
auto string = format_string_va((String)array_to_slice(buffer), format, args);
|
||||
let(string, format_string_va((String)array_to_slice(buffer), format, args));
|
||||
va_end(args);
|
||||
|
||||
auto* result = ui_widget_make(flags, string);
|
||||
let(result, ui_widget_make(flags, string));
|
||||
return result;
|
||||
}
|
||||
|
||||
UI_Signal ui_signal_from_widget(UI_Widget* widget)
|
||||
fn UI_Signal ui_signal_from_widget(UI_Widget* widget)
|
||||
{
|
||||
auto rect = widget->rect;
|
||||
auto mouse_position = ui_state->mouse_position;
|
||||
let(rect, widget->rect);
|
||||
let(mouse_position, ui_state->mouse_position);
|
||||
UI_Signal signal = {
|
||||
.clicked_left =
|
||||
(widget->flags.mouse_clickable & (ui_state->mouse_button_events[OS_EVENT_MOUSE_LEFT].action == OS_EVENT_MOUSE_RELEASE)) &
|
||||
(widget->flags.mouse_clickable & (ui_state->mouse_button_events[WINDOWING_EVENT_MOUSE_LEFT].action == WINDOWING_EVENT_MOUSE_RELEASE)) &
|
||||
((mouse_position.x >= rect.x0) & (mouse_position.x <= rect.x1)) &
|
||||
((mouse_position.y >= rect.y0) & (mouse_position.y <= rect.y1)),
|
||||
};
|
||||
@ -311,16 +309,16 @@ UI_Signal ui_signal_from_widget(UI_Widget* widget)
|
||||
|
||||
fn void ui_stack_reset(UI_State* state)
|
||||
{
|
||||
auto* stack_end = (u32*)((u8*)&state->stacks + sizeof(state->stacks));
|
||||
let(stack_end, (u32*)((u8*)&state->stacks + sizeof(state->stacks)));
|
||||
|
||||
for (auto* stack_pointer = (u32*)&state->stacks; stack_pointer != stack_end; stack_pointer += sizeof(VirtualBuffer(u8)) / sizeof(u32))
|
||||
for (let(stack_pointer, (u32*)&state->stacks); stack_pointer != stack_end; stack_pointer += sizeof(VirtualBuffer(u8)) / sizeof(u32))
|
||||
{
|
||||
auto* length_pointer = stack_pointer + (offsetof(VirtualBuffer(u8), length) / sizeof(u32));
|
||||
let(length_pointer, stack_pointer + (offsetof(VirtualBuffer(u8), length) / sizeof(u32)));
|
||||
*length_pointer = 0;
|
||||
}
|
||||
}
|
||||
|
||||
UI_Size ui_pixels(u32 width, f32 strictness)
|
||||
fn UI_Size ui_pixels(u32 width, f32 strictness)
|
||||
{
|
||||
return (UI_Size) {
|
||||
.kind = UI_SIZE_PIXEL_COUNT,
|
||||
@ -329,7 +327,7 @@ UI_Size ui_pixels(u32 width, f32 strictness)
|
||||
};
|
||||
}
|
||||
|
||||
UI_Size ui_percentage(f32 percentage, f32 strictness)
|
||||
fn UI_Size ui_percentage(f32 percentage, f32 strictness)
|
||||
{
|
||||
return (UI_Size) {
|
||||
.kind = UI_SIZE_PERCENTAGE,
|
||||
@ -338,9 +336,9 @@ UI_Size ui_percentage(f32 percentage, f32 strictness)
|
||||
};
|
||||
}
|
||||
|
||||
UI_Size ui_em(f32 value, f32 strictness)
|
||||
fn UI_Size ui_em(f32 value, f32 strictness)
|
||||
{
|
||||
auto font_size = ui_top(font_size);
|
||||
let(font_size, ui_top(font_size));
|
||||
assert(font_size);
|
||||
return (UI_Size) {
|
||||
.kind = UI_SIZE_PIXEL_COUNT,
|
||||
@ -349,44 +347,44 @@ UI_Size ui_em(f32 value, f32 strictness)
|
||||
};
|
||||
}
|
||||
|
||||
u8 ui_build_begin(OSWindow os_window, f64 frame_time, OSEventQueue* event_queue)
|
||||
fn u8 ui_build_begin(WindowingInstance* window, f64 frame_time, WindowingEventQueue* event_queue)
|
||||
{
|
||||
ui_state->build_count += 1;
|
||||
auto* build_arena = ui_build_arena();
|
||||
let(build_arena, ui_build_arena());
|
||||
arena_reset(build_arena);
|
||||
ui_state->frame_time = frame_time;
|
||||
ui_state->os_window = os_window;
|
||||
ui_state->window = window;
|
||||
|
||||
ui_stack_reset(ui_state);
|
||||
|
||||
u8 open = 1;
|
||||
|
||||
auto mouse_button_count = 0;
|
||||
let(mouse_button_count, 0);
|
||||
for (u32 generic_event_index = 0; open & (generic_event_index < event_queue->descriptors.length); generic_event_index += 1)
|
||||
{
|
||||
auto event_descriptor = event_queue->descriptors.pointer[generic_event_index];
|
||||
let(event_descriptor, event_queue->descriptors.pointer[generic_event_index]);
|
||||
u32 event_index = event_descriptor.index;
|
||||
|
||||
switch (event_descriptor.type)
|
||||
{
|
||||
case OS_EVENT_TYPE_MOUSE_BUTTON:
|
||||
case WINDOWING_EVENT_TYPE_MOUSE_BUTTON:
|
||||
{
|
||||
auto button = event_queue->mouse_buttons.pointer[event_index];
|
||||
auto previous_button_event = ui_state->mouse_button_events[button.button];
|
||||
let(button, event_queue->mouse_buttons.pointer[event_index]);
|
||||
let(previous_button_event, ui_state->mouse_button_events[button.button]);
|
||||
switch (button.event.action)
|
||||
{
|
||||
case OS_EVENT_MOUSE_RELAX:
|
||||
case WINDOWING_EVENT_MOUSE_RELAX:
|
||||
unreachable();
|
||||
case OS_EVENT_MOUSE_RELEASE:
|
||||
case WINDOWING_EVENT_MOUSE_RELEASE:
|
||||
{
|
||||
assert(previous_button_event.action == OS_EVENT_MOUSE_PRESS);
|
||||
assert(previous_button_event.action == WINDOWING_EVENT_MOUSE_PRESS);
|
||||
} break;
|
||||
case OS_EVENT_MOUSE_PRESS:
|
||||
case WINDOWING_EVENT_MOUSE_PRESS:
|
||||
{
|
||||
// TODO: handle properly
|
||||
assert(previous_button_event.action == OS_EVENT_MOUSE_RELAX || mouse_button_count);
|
||||
assert(previous_button_event.action == WINDOWING_EVENT_MOUSE_RELAX || mouse_button_count);
|
||||
} break;
|
||||
case OS_EVENT_MOUSE_REPEAT:
|
||||
case WINDOWING_EVENT_MOUSE_REPEAT:
|
||||
{
|
||||
unreachable();
|
||||
} break;
|
||||
@ -395,27 +393,27 @@ u8 ui_build_begin(OSWindow os_window, f64 frame_time, OSEventQueue* event_queue)
|
||||
ui_state->mouse_button_events[button.button] = button.event;
|
||||
mouse_button_count += 1;
|
||||
} break;
|
||||
case OS_EVENT_TYPE_WINDOW_FOCUS:
|
||||
case WINDOWING_EVENT_TYPE_WINDOW_FOCUS:
|
||||
{
|
||||
} break;
|
||||
case OS_EVENT_TYPE_CURSOR_POSITION:
|
||||
case WINDOWING_EVENT_TYPE_CURSOR_POSITION:
|
||||
{
|
||||
auto mouse_position = event_queue->cursor_positions.pointer[event_index];
|
||||
let(mouse_position, event_queue->cursor_positions.pointer[event_index]);
|
||||
ui_state->mouse_position = (UI_MousePosition) {
|
||||
.x = mouse_position.x,
|
||||
.y = mouse_position.y,
|
||||
};
|
||||
} break;
|
||||
case OS_EVENT_TYPE_CURSOR_ENTER:
|
||||
case WINDOWING_EVENT_TYPE_CURSOR_ENTER:
|
||||
{
|
||||
todo();
|
||||
} break;
|
||||
case OS_EVENT_TYPE_WINDOW_POSITION:
|
||||
case WINDOWING_EVENT_TYPE_WINDOW_POSITION:
|
||||
{
|
||||
// event_queue->window_positions.pointer[event_index];
|
||||
// todo();
|
||||
} break;
|
||||
case OS_EVENT_TYPE_WINDOW_CLOSE:
|
||||
case WINDOWING_EVENT_TYPE_WINDOW_CLOSE:
|
||||
{
|
||||
open = 0;
|
||||
} break;
|
||||
@ -426,7 +424,7 @@ u8 ui_build_begin(OSWindow os_window, f64 frame_time, OSEventQueue* event_queue)
|
||||
{
|
||||
for (u64 i = 0; i < ui_state->widget_table.length; i += 1)
|
||||
{
|
||||
auto* widget_table_element = &ui_state->widget_table.pointer[i];
|
||||
let(widget_table_element, &ui_state->widget_table.pointer[i]);
|
||||
for (UI_Widget* widget = widget_table_element->first, *next = 0; widget; widget = next)
|
||||
{
|
||||
next = widget->hash_next;
|
||||
@ -457,19 +455,20 @@ u8 ui_build_begin(OSWindow os_window, f64 frame_time, OSEventQueue* event_queue)
|
||||
}
|
||||
}
|
||||
|
||||
auto framebuffer_size = os_window_framebuffer_size_get(os_window);
|
||||
let(framebuffer_size, windowing_get_instance_framebuffer_size(window));
|
||||
ui_push_next_only(pref_width, ui_pixels(framebuffer_size.width, 1.0f));
|
||||
ui_push_next_only(pref_height, ui_pixels(framebuffer_size.height, 1.0f));
|
||||
ui_push_next_only(child_layout_axis, AXIS2_Y);
|
||||
|
||||
auto* root = ui_widget_make_format((UI_WidgetFlags) {}, "window_root_{u64}", os_window);
|
||||
let(root, ui_widget_make_format((UI_WidgetFlags) {}, "window_root_{u64}", window));
|
||||
assert(!ui_state->stack_autopops.child_layout_axis);
|
||||
|
||||
ui_push(parent, root);
|
||||
|
||||
ui_push(font_size, 12);
|
||||
ui_push(text_color, ((float4) { 0.9, 0.9, 0.02, 1 }));
|
||||
ui_push(background_color, ((float4) { 0.1, 0.1, 0.1, 1 }));
|
||||
|
||||
ui_push(text_color, VEC4(0.9, 0.9, 0.02, 1));
|
||||
ui_push(background_color, VEC4(0.1, 0.1, 0.1, 1));
|
||||
ui_push(pref_width, ui_percentage(1.0, 0.0));
|
||||
ui_push(pref_height, ui_percentage(1.0, 0.0));
|
||||
// ui_push(pref_height, ui_em(1.8, 0.0));
|
||||
@ -482,13 +481,17 @@ fn void ui_compute_independent_sizes(UI_Widget* widget)
|
||||
{
|
||||
for (Axis2 axis = 0; axis < AXIS2_COUNT; axis += 1)
|
||||
{
|
||||
auto pref_size = widget->pref_size[axis];
|
||||
let(pref_size, widget->pref_size[axis]);
|
||||
switch (pref_size.kind)
|
||||
{
|
||||
default: break; case UI_SIZE_COUNT: unreachable();
|
||||
case UI_SIZE_PIXEL_COUNT:
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
widget->computed_size[axis] = floorf(widget->pref_size[axis].value);
|
||||
#else
|
||||
widget->computed_size.v[axis] = floorf(widget->pref_size[axis].value);
|
||||
#endif
|
||||
} break;
|
||||
}
|
||||
}
|
||||
@ -504,7 +507,7 @@ fn void ui_compute_upward_dependent_sizes(UI_Widget* widget)
|
||||
// TODO: optimize loop out if possible
|
||||
for (Axis2 axis = 0; axis < AXIS2_COUNT; axis += 1)
|
||||
{
|
||||
auto pref_size = widget->pref_size[axis];
|
||||
let(pref_size, widget->pref_size[axis]);
|
||||
switch (pref_size.kind)
|
||||
{
|
||||
default: break; case UI_SIZE_COUNT: unreachable();
|
||||
@ -514,7 +517,11 @@ fn void ui_compute_upward_dependent_sizes(UI_Widget* widget)
|
||||
{
|
||||
if (ancestor->pref_size[axis].kind != UI_SIZE_BY_CHILDREN)
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
widget->computed_size[axis] = floorf(ancestor->computed_size[axis] * widget->pref_size[axis].value);
|
||||
#else
|
||||
widget->computed_size.v[axis] = floorf(ancestor->computed_size.v[axis] * widget->pref_size[axis].value);
|
||||
#endif
|
||||
break;
|
||||
}
|
||||
}
|
||||
@ -537,7 +544,7 @@ fn void ui_compute_downward_dependent_sizes(UI_Widget* widget)
|
||||
|
||||
for (Axis2 axis = 0; axis < AXIS2_COUNT; axis += 1)
|
||||
{
|
||||
auto pref_size = widget->pref_size[axis];
|
||||
let(pref_size, widget->pref_size[axis]);
|
||||
switch (pref_size.kind)
|
||||
{
|
||||
default: break; case UI_SIZE_COUNT: unreachable();
|
||||
@ -553,7 +560,11 @@ fn void ui_resolve_conflicts(UI_Widget* widget)
|
||||
{
|
||||
for (Axis2 axis = 0; axis < AXIS2_COUNT; axis += 1)
|
||||
{
|
||||
auto available_space = widget->computed_size[axis];
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
let(available_space, widget->computed_size[axis]);
|
||||
#else
|
||||
let(available_space, widget->computed_size.v[axis]);
|
||||
#endif
|
||||
f32 taken_space = 0;
|
||||
f32 total_fixup_budget = 0;
|
||||
|
||||
@ -565,18 +576,30 @@ fn void ui_resolve_conflicts(UI_Widget* widget)
|
||||
{
|
||||
if (axis == widget->child_layout_axis)
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
taken_space += child_widget->computed_size[axis];
|
||||
#else
|
||||
taken_space += child_widget->computed_size.v[axis];
|
||||
#endif
|
||||
}
|
||||
else
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
taken_space = MAX(taken_space, child_widget->computed_size[axis]);
|
||||
#else
|
||||
taken_space = MAX(taken_space, child_widget->computed_size.v[axis]);
|
||||
#endif
|
||||
}
|
||||
auto fixup_budget_this_child = child_widget->computed_size[axis] * (1 - child_widget->pref_size[axis].strictness);
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
let(fixup_budget_this_child, child_widget->computed_size[axis] * (1 - child_widget->pref_size[axis].strictness));
|
||||
#else
|
||||
let(fixup_budget_this_child, child_widget->computed_size.v[axis] * (1 - child_widget->pref_size[axis].strictness));
|
||||
#endif
|
||||
total_fixup_budget += fixup_budget_this_child;
|
||||
}
|
||||
}
|
||||
|
||||
auto conflict = taken_space - available_space;
|
||||
let(conflict, taken_space - available_space);
|
||||
|
||||
if (conflict > 0 && total_fixup_budget > 0)
|
||||
{
|
||||
@ -584,7 +607,11 @@ fn void ui_resolve_conflicts(UI_Widget* widget)
|
||||
{
|
||||
if (!(child_widget->flags.v & (UI_WIDGET_FLAG_FLOATING_X << axis)))
|
||||
{
|
||||
auto fixup_budget_this_child = child_widget->computed_size[axis] * (1 - child_widget->pref_size[axis].strictness);
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
let(fixup_budget_this_child, child_widget->computed_size[axis] * (1 - child_widget->pref_size[axis].strictness));
|
||||
#else
|
||||
let(fixup_budget_this_child, child_widget->computed_size.v[axis] * (1 - child_widget->pref_size[axis].strictness));
|
||||
#endif
|
||||
f32 fixup_size_this_child = 0;
|
||||
|
||||
if (axis == widget->child_layout_axis)
|
||||
@ -593,11 +620,19 @@ fn void ui_resolve_conflicts(UI_Widget* widget)
|
||||
}
|
||||
else
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
fixup_size_this_child = child_widget->computed_size[axis] - available_space;
|
||||
#else
|
||||
fixup_size_this_child = child_widget->computed_size.v[axis] - available_space;
|
||||
#endif
|
||||
}
|
||||
|
||||
fixup_size_this_child = CLAMP(0, fixup_size_this_child, fixup_budget_this_child);
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
child_widget->computed_size[axis] = floorf(child_widget->computed_size[axis] - fixup_size_this_child);
|
||||
#else
|
||||
child_widget->computed_size.v[axis] = floorf(child_widget->computed_size.v[axis] - fixup_size_this_child);
|
||||
#endif
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -611,8 +646,13 @@ fn void ui_resolve_conflicts(UI_Widget* widget)
|
||||
{
|
||||
if (!(child_widget->flags.v & (UI_WIDGET_FLAG_FLOATING_X << axis)))
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
child_widget->computed_relative_position[axis] = p;
|
||||
p += child_widget->computed_size[axis];
|
||||
#else
|
||||
child_widget->computed_relative_position.v[axis] = p;
|
||||
p += child_widget->computed_size.v[axis];
|
||||
#endif
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -622,22 +662,32 @@ fn void ui_resolve_conflicts(UI_Widget* widget)
|
||||
{
|
||||
if (!(child_widget->flags.v & (UI_WIDGET_FLAG_FLOATING_X << axis)))
|
||||
{
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
child_widget->computed_relative_position[axis] = 0;
|
||||
#else
|
||||
child_widget->computed_relative_position.v[axis] = 0;
|
||||
#endif
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (UI_Widget* child_widget = widget->first; child_widget; child_widget = child_widget->next)
|
||||
{
|
||||
auto last_relative_rect = child_widget->relative_rect;
|
||||
let(last_relative_rect, child_widget->relative_rect);
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
child_widget->relative_rect.p0[axis] = child_widget->computed_relative_position[axis];
|
||||
child_widget->relative_rect.p1[axis] = child_widget->relative_rect.p0[axis] + child_widget->computed_size[axis];
|
||||
#else
|
||||
child_widget->relative_rect.p0.v[axis] = child_widget->computed_relative_position.v[axis];
|
||||
child_widget->relative_rect.p1.v[axis] = child_widget->relative_rect.p0.v[axis] + child_widget->computed_size.v[axis];
|
||||
#endif
|
||||
|
||||
float2 last_corner_01 = { last_relative_rect.x0, last_relative_rect.y1 };
|
||||
float2 last_corner_10 = { last_relative_rect.x1, last_relative_rect.y0 };
|
||||
float2 this_corner_01 = { child_widget->relative_rect.x0, child_widget->relative_rect.y1 };
|
||||
float2 this_corner_10 = { child_widget->relative_rect.x1, child_widget->relative_rect.y0 };
|
||||
|
||||
#if BB_HAS_NATIVE_FLOAT2
|
||||
child_widget->relative_corner_delta[CORNER_00][axis] = child_widget->relative_rect.p0[axis] - last_relative_rect.p0[axis];
|
||||
child_widget->relative_corner_delta[CORNER_01][axis] = this_corner_01[axis] - last_corner_01[axis];
|
||||
child_widget->relative_corner_delta[CORNER_10][axis] = this_corner_10[axis] - last_corner_10[axis];
|
||||
@ -651,6 +701,21 @@ fn void ui_resolve_conflicts(UI_Widget* widget)
|
||||
child_widget->rect.p0[axis] = floorf(child_widget->rect.p0[axis]);
|
||||
child_widget->rect.p1[axis] = floorf(child_widget->rect.p1[axis]);
|
||||
}
|
||||
#else
|
||||
child_widget->relative_corner_delta[CORNER_00].v[axis] = child_widget->relative_rect.p0.v[axis] - last_relative_rect.p0.v[axis];
|
||||
child_widget->relative_corner_delta[CORNER_01].v[axis] = this_corner_01.v[axis] - last_corner_01.v[axis];
|
||||
child_widget->relative_corner_delta[CORNER_10].v[axis] = this_corner_10.v[axis] - last_corner_10.v[axis];
|
||||
child_widget->relative_corner_delta[CORNER_11].v[axis] = child_widget->relative_rect.p1.v[axis] - last_relative_rect.p1.v[axis];
|
||||
|
||||
child_widget->rect.p0.v[axis] = widget->rect.p0.v[axis] + child_widget->relative_rect.p0.v[axis] - widget->view_offset.v[axis];
|
||||
child_widget->rect.p1.v[axis] = child_widget->rect.p0.v[axis] + child_widget->computed_size.v[axis];
|
||||
|
||||
if (!(child_widget->flags.v & (UI_WIDGET_FLAG_FLOATING_X << axis)))
|
||||
{
|
||||
child_widget->rect.p0.v[axis] = floorf(child_widget->rect.p0.v[axis]);
|
||||
child_widget->rect.p1.v[axis] = floorf(child_widget->rect.p1.v[axis]);
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
for (UI_Widget* child_widget = widget->first; child_widget; child_widget = child_widget->next)
|
||||
@ -660,15 +725,15 @@ fn void ui_resolve_conflicts(UI_Widget* widget)
|
||||
}
|
||||
}
|
||||
|
||||
void ui_build_end()
|
||||
fn void ui_build_end()
|
||||
{
|
||||
// Clear release button presses
|
||||
for (u32 i = 0; i < array_length(ui_state->mouse_button_events); i += 1)
|
||||
{
|
||||
auto* event = &ui_state->mouse_button_events[i];
|
||||
if (event->action == OS_EVENT_MOUSE_RELEASE)
|
||||
let(event, &ui_state->mouse_button_events[i]);
|
||||
if (event->action == WINDOWING_EVENT_MOUSE_RELEASE)
|
||||
{
|
||||
event->action = OS_EVENT_MOUSE_RELAX;
|
||||
event->action = WINDOWING_EVENT_MOUSE_RELAX;
|
||||
}
|
||||
}
|
||||
|
||||
@ -690,10 +755,10 @@ STRUCT(WidgetIterator)
|
||||
#define ui_widget_recurse_depth_first_preorder(widget) ui_widget_recurse_depth_first((widget), offset_of(UI_Widget, next), offset_of(UI_Widget, first))
|
||||
#define ui_widget_recurse_depth_first_postorder(widget) ui_widget_recurse_depth_first((widget), offset_of(UI_Widget, previous), offset_of(UI_Widget, last))
|
||||
|
||||
WidgetIterator ui_widget_recurse_depth_first(UI_Widget* widget, u64 sibling_offset, u64 child_offset)
|
||||
fn WidgetIterator ui_widget_recurse_depth_first(UI_Widget* widget, u64 sibling_offset, u64 child_offset)
|
||||
{
|
||||
WidgetIterator it = {};
|
||||
auto* child = member_from_offset(widget, UI_Widget*, child_offset);
|
||||
let(child, member_from_offset(widget, UI_Widget*, child_offset));
|
||||
if (child)
|
||||
{
|
||||
it.next = child;
|
||||
@ -703,7 +768,7 @@ WidgetIterator ui_widget_recurse_depth_first(UI_Widget* widget, u64 sibling_offs
|
||||
{
|
||||
for (UI_Widget* w = widget; w; w = w->parent)
|
||||
{
|
||||
auto* sibling = member_from_offset(w, UI_Widget*, sibling_offset);
|
||||
let(sibling, member_from_offset(w, UI_Widget*, sibling_offset));
|
||||
if (sibling)
|
||||
{
|
||||
it.next = sibling;
|
||||
@ -717,7 +782,7 @@ WidgetIterator ui_widget_recurse_depth_first(UI_Widget* widget, u64 sibling_offs
|
||||
return it;
|
||||
}
|
||||
|
||||
void ui_draw()
|
||||
fn void ui_draw()
|
||||
{
|
||||
UI_Widget* root = ui_state->root;
|
||||
|
||||
@ -743,4 +808,3 @@ void ui_draw()
|
||||
widget = ui_widget_recurse_depth_first_postorder(widget).next;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1,17 +1,11 @@
|
||||
#pragma once
|
||||
|
||||
#include <std/base.h>
|
||||
#include <std/window.h>
|
||||
#include <std/os.h>
|
||||
#include <std/render.h>
|
||||
|
||||
typedef enum UI_SizeKind : u8
|
||||
{
|
||||
ENUM(UI_SizeKind, u8,
|
||||
UI_SIZE_PIXEL_COUNT,
|
||||
UI_SIZE_PERCENTAGE,
|
||||
UI_SIZE_BY_CHILDREN,
|
||||
UI_SIZE_COUNT,
|
||||
} UI_SizeKind;
|
||||
);
|
||||
|
||||
STRUCT(UI_Size)
|
||||
{
|
||||
@ -33,8 +27,7 @@ STRUCT(UI_MousePosition)
|
||||
f64 y;
|
||||
};
|
||||
|
||||
typedef enum UI_WidgetFlagEnum : u64
|
||||
{
|
||||
ENUM(UI_WidgetFlagEnum, u64,
|
||||
UI_WIDGET_FLAG_DISABLED = 1 << 0,
|
||||
UI_WIDGET_FLAG_MOUSE_CLICKABLE = 1 << 1,
|
||||
UI_WIDGET_FLAG_KEYBOARD_PRESSABLE = 1 << 2,
|
||||
@ -44,7 +37,7 @@ typedef enum UI_WidgetFlagEnum : u64
|
||||
UI_WIDGET_FLAG_OVERFLOW_Y = 1 << 6,
|
||||
UI_WIDGET_FLAG_FLOATING_X = 1 << 7,
|
||||
UI_WIDGET_FLAG_FLOATING_Y = 1 << 8,
|
||||
} UI_WidgetFlagEnum;
|
||||
);
|
||||
|
||||
UNION(UI_WidgetFlags)
|
||||
{
|
||||
@ -153,7 +146,7 @@ STRUCT(UI_State)
|
||||
Arena* build_arenas[2];
|
||||
Renderer* renderer;
|
||||
RenderWindow* render_window;
|
||||
OSWindow os_window;
|
||||
WindowingInstance* window;
|
||||
u64 build_count;
|
||||
f64 frame_time;
|
||||
UI_Widget* root;
|
||||
@ -161,7 +154,7 @@ STRUCT(UI_State)
|
||||
Slice(UI_WidgetSlot) widget_table;
|
||||
UI_Widget* free_widget_list;
|
||||
u64 free_widget_count;
|
||||
OSEventMouseButtonEvent mouse_button_events[OS_EVENT_MOUSE_BUTTON_COUNT];
|
||||
WindowingEventMouseButtonEvent mouse_button_events[WINDOWING_EVENT_MOUSE_BUTTON_COUNT];
|
||||
u8 focused:1;
|
||||
|
||||
UI_StateStacks stacks;
|
||||
@ -190,8 +183,6 @@ STRUCT(UI_Signal)
|
||||
};
|
||||
};
|
||||
|
||||
extern UI_State* ui_state;
|
||||
|
||||
#define ui_stack_autopop_set(field_name, value) ui_state->stack_autopops.field_name = (value)
|
||||
#define ui_stack_push_impl(field_name, value, auto_pop_value) do \
|
||||
{\
|
||||
@ -199,34 +190,34 @@ extern UI_State* ui_state;
|
||||
ui_stack_autopop_set(field_name, auto_pop_value);\
|
||||
} while (0)
|
||||
|
||||
fn u8* ui_pop_generic(VirtualBuffer(u8)* stack, u32 element_size)
|
||||
{
|
||||
auto length = stack->length;
|
||||
|
||||
assert(length > 0);
|
||||
auto next_length = length - 1;
|
||||
auto index = next_length;
|
||||
auto* result = &stack->pointer[index * element_size];
|
||||
stack->length = next_length;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
#define ui_push(field_name, value) ui_stack_push_impl(field_name, value, 0)
|
||||
#define ui_push_next_only(field_name, value) ui_stack_push_impl(field_name, value, 1)
|
||||
#define ui_pop(field_name) (typeof(ui_state->stacks.field_name.pointer)) ui_pop_generic((VirtualBuffer(u8)*)&ui_state->stacks.field_name, sizeof(*ui_state->stacks.field_name.pointer))
|
||||
#define ui_top(field_name) (ui_state->stacks.field_name.length ? ui_state->stacks.field_name.pointer[ui_state->stacks.field_name.length - 1] : ui_state->stack_nulls.field_name)
|
||||
|
||||
EXPORT UI_State* ui_state_allocate(Renderer* renderer, RenderWindow* window);
|
||||
EXPORT void ui_state_select(UI_State* state);
|
||||
EXPORT u8 ui_build_begin(OSWindow window, f64 frame_time, OSEventQueue* event_queue);
|
||||
EXPORT void ui_build_end();
|
||||
EXPORT void ui_draw();
|
||||
EXPORT UI_Signal ui_signal_from_widget(UI_Widget* widget);
|
||||
EXPORT UI_State* ui_state_get();
|
||||
fn u8* ui_pop_generic(VirtualBuffer(u8)* stack, u32 element_size)
|
||||
{
|
||||
let(length, stack->length);
|
||||
|
||||
EXPORT UI_Widget* ui_widget_make(UI_WidgetFlags flags, String string);
|
||||
EXPORT UI_Widget* ui_widget_make_format(UI_WidgetFlags flags, const char* format, ...);
|
||||
EXPORT UI_Size ui_pixels(u32 width, f32 strictness);
|
||||
EXPORT UI_Size ui_percentage(f32 percentage, f32 strictness);
|
||||
EXPORT UI_Size ui_em(f32 value, f32 strictness);
|
||||
assert(length > 0);
|
||||
let(next_length, length - 1);
|
||||
let(index, next_length);
|
||||
let(result, &stack->pointer[index * element_size]);
|
||||
stack->length = next_length;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
fn UI_State* ui_state_allocate(Renderer* renderer, RenderWindow* window);
|
||||
fn void ui_state_select(UI_State* state);
|
||||
fn u8 ui_build_begin(WindowingInstance* window, f64 frame_time, WindowingEventQueue* event_queue);
|
||||
fn void ui_build_end();
|
||||
fn void ui_draw();
|
||||
fn UI_Signal ui_signal_from_widget(UI_Widget* widget);
|
||||
fn UI_State* ui_state_get();
|
||||
|
||||
fn UI_Widget* ui_widget_make(UI_WidgetFlags flags, String string);
|
||||
fn UI_Widget* ui_widget_make_format(UI_WidgetFlags flags, const char* format, ...);
|
||||
fn UI_Size ui_pixels(u32 width, f32 strictness);
|
||||
fn UI_Size ui_percentage(f32 percentage, f32 strictness);
|
||||
fn UI_Size ui_em(f32 value, f32 strictness);
|
@ -1,7 +1,7 @@
|
||||
#include <std/virtual_buffer.h>
|
||||
#include <std/os.h>
|
||||
|
||||
void vb_generic_ensure_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count)
|
||||
fn void vb_generic_ensure_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count)
|
||||
{
|
||||
u32 old_capacity = vb->capacity;
|
||||
u32 wanted_capacity = vb->length + item_count;
|
||||
@ -13,20 +13,20 @@ void vb_generic_ensure_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_c
|
||||
vb->pointer = os_reserve(0, item_size * UINT32_MAX, (OSReserveProtectionFlags) {}, (OSReserveMapFlags) { .priv = 1, .anon = 1, .noreserve = 1 });
|
||||
}
|
||||
|
||||
u32 old_page_capacity = cast_to(u32, u64, align_forward(old_capacity * item_size, minimum_granularity));
|
||||
u32 new_page_capacity = cast_to(u32, u64, align_forward(wanted_capacity * item_size, minimum_granularity));
|
||||
let_cast(u32, old_page_capacity, align_forward(old_capacity * item_size, minimum_granularity));
|
||||
let_cast(u32, new_page_capacity, align_forward(wanted_capacity * item_size, minimum_granularity));
|
||||
|
||||
u32 commit_size = new_page_capacity - old_page_capacity;
|
||||
let(commit_size, new_page_capacity - old_page_capacity);
|
||||
void* commit_pointer = vb->pointer + old_page_capacity;
|
||||
|
||||
os_commit(commit_pointer, commit_size);
|
||||
|
||||
u32 new_capacity = new_page_capacity / item_size;
|
||||
let(new_capacity, new_page_capacity / item_size);
|
||||
vb->capacity = new_capacity;
|
||||
}
|
||||
}
|
||||
|
||||
u8* vb_generic_add_assume_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count)
|
||||
fn u8* vb_generic_add_assume_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count)
|
||||
{
|
||||
u32 index = vb->length;
|
||||
assert(vb->capacity >= index + item_count);
|
||||
@ -34,29 +34,29 @@ u8* vb_generic_add_assume_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 ite
|
||||
return vb->pointer + (index * item_size);
|
||||
}
|
||||
|
||||
u8* vb_generic_add(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count)
|
||||
fn u8* vb_generic_add(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count)
|
||||
{
|
||||
vb_generic_ensure_capacity(vb, item_size, item_count);
|
||||
return vb_generic_add_assume_capacity(vb, item_size, item_count);
|
||||
}
|
||||
|
||||
u8* vb_append_bytes(VirtualBuffer(u8*) vb, Slice(u8) bytes)
|
||||
fn u8* vb_append_bytes(VirtualBuffer(u8*) vb, Slice(u8) bytes)
|
||||
{
|
||||
auto len = cast_to(u32, u64, bytes.length);
|
||||
let_cast(u32, len, bytes.length);
|
||||
vb_generic_ensure_capacity(vb, sizeof(u8), len);
|
||||
auto* pointer = vb_generic_add_assume_capacity(vb, sizeof(u8), len);
|
||||
let(pointer, vb_generic_add_assume_capacity(vb, sizeof(u8), len));
|
||||
memcpy(pointer, bytes.pointer, len);
|
||||
return pointer;
|
||||
}
|
||||
|
||||
void vb_copy_string(VirtualBuffer(u8)* buffer, String string)
|
||||
fn void vb_copy_string(VirtualBuffer(u8)* buffer, String string)
|
||||
{
|
||||
auto length = cast_to(u32, u64, string.length);
|
||||
auto* pointer = vb_add(buffer, length);
|
||||
let_cast(u32, length, string.length);
|
||||
let(pointer, vb_add(buffer, length));
|
||||
memcpy(pointer, string.pointer, length);
|
||||
}
|
||||
|
||||
u64 vb_copy_string_zero_terminated(VirtualBuffer(u8)* buffer, String string)
|
||||
fn u64 vb_copy_string_zero_terminated(VirtualBuffer(u8)* buffer, String string)
|
||||
{
|
||||
assert(string.pointer[string.length] == 0);
|
||||
string.length += 1;
|
||||
|
@ -45,9 +45,9 @@ decl_vb(f64);
|
||||
#define vb_copy_any_array(vb, arr) memcpy(vb_generic_add(vb, vb_size_of_element(vb), sizeof(arr)), (arr), sizeof(arr))
|
||||
#define vb_copy_any_slice(vb, slice) memcpy(vb_generic_add(vb, vb_size_of_element(vb), sizeof(*((slice).pointer)) * (slice).length), (slice).pointer, sizeof(*((slice).pointer)) * (slice).length)
|
||||
|
||||
void vb_generic_ensure_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count);
|
||||
u8* vb_generic_add_assume_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count);
|
||||
u8* vb_generic_add(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count);
|
||||
u8* vb_append_bytes(VirtualBuffer(u8*) vb, Slice(u8) bytes);
|
||||
void vb_copy_string(VirtualBuffer(u8)* buffer, String string);
|
||||
u64 vb_copy_string_zero_terminated(VirtualBuffer(u8)* buffer, String string);
|
||||
fn void vb_generic_ensure_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count);
|
||||
fn u8* vb_generic_add_assume_capacity(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count);
|
||||
fn u8* vb_generic_add(VirtualBuffer(u8)* vb, u32 item_size, u32 item_count);
|
||||
fn u8* vb_append_bytes(VirtualBuffer(u8*) vb, Slice(u8) bytes);
|
||||
fn void vb_copy_string(VirtualBuffer(u8)* buffer, String string);
|
||||
fn u64 vb_copy_string_zero_terminated(VirtualBuffer(u8)* buffer, String string);
|
@ -1,9 +1,10 @@
|
||||
#include <std/render.h>
|
||||
#include <std/string.h>
|
||||
#include <std/image_loader.h>
|
||||
#include <std/virtual_buffer.h>
|
||||
#pragma once
|
||||
|
||||
#include <volk.h>
|
||||
#define vulkan_load_function_generic(fn, load_fn, context) fn = (PFN_ ## fn) load_fn(context, #fn)
|
||||
#define vulkan_load_instance_function(instance, fn) vulkan_load_function_generic(fn, vkGetInstanceProcAddr, instance)
|
||||
#define vulkan_load_device_function(device, fn) vulkan_load_function_generic(fn, vkGetDeviceProcAddr, device)
|
||||
|
||||
global_variable OSLibrary vulkan_library;
|
||||
|
||||
#define MAX_SWAPCHAIN_IMAGE_COUNT (16)
|
||||
#define MAX_FRAME_COUNT (2)
|
||||
@ -115,7 +116,6 @@ STRUCT(IndexBuffer)
|
||||
|
||||
STRUCT(Renderer)
|
||||
{
|
||||
VkInstance instance;
|
||||
VkAllocationCallbacks* allocator;
|
||||
VkPhysicalDevice physical_device;
|
||||
VkDevice device;
|
||||
@ -192,11 +192,11 @@ STRUCT(VulkanCopyImageArgs)
|
||||
VulkanCopyImage destination;
|
||||
};
|
||||
|
||||
|
||||
global_variable Renderer renderer_memory;
|
||||
global_variable RenderWindow renderer_window_memory;
|
||||
global_variable VulkanTexture textures[MAX_TEXTURE_COUNT];
|
||||
global_variable u32 texture_count;
|
||||
global_variable VkInstance instance;
|
||||
|
||||
fn String vulkan_result_to_string(VkResult result)
|
||||
{
|
||||
@ -255,7 +255,7 @@ fn String vulkan_result_to_string(VkResult result)
|
||||
}
|
||||
}
|
||||
|
||||
[[noreturn]] [[gnu::cold]] fn void wrong_vulkan_result(VkResult result, String call_string, String file, int line)
|
||||
BB_NORETURN BB_COLD fn void wrong_vulkan_result(VkResult result, String call_string, String file, int line)
|
||||
{
|
||||
unused(result);
|
||||
unused(call_string);
|
||||
@ -270,15 +270,15 @@ fn void buffer_copy_to_local_command(VkCommandBuffer command_buffer, Slice(Local
|
||||
{
|
||||
for (u64 i = 0; i < copies.length; i += 1)
|
||||
{
|
||||
auto copy = copies.pointer[i];
|
||||
auto* source_buffer = ©.source;
|
||||
auto* destination_buffer = ©.destination;
|
||||
let(copy, copies.pointer[i]);
|
||||
let(source_buffer, ©.source);
|
||||
let(destination_buffer, ©.destination);
|
||||
|
||||
VkBufferCopy2 buffer_copies[MAX_LOCAL_BUFFER_COPY_COUNT];
|
||||
|
||||
for (u64 i = 0; i < copy.regions.length; i += 1)
|
||||
{
|
||||
auto copy_region = copy.regions.pointer[i];
|
||||
let(copy_region, copy.regions.pointer[i]);
|
||||
buffer_copies[i] = (VkBufferCopy2) {
|
||||
.sType = VK_STRUCTURE_TYPE_BUFFER_COPY_2,
|
||||
.pNext = 0,
|
||||
@ -301,16 +301,16 @@ fn void buffer_copy_to_local_command(VkCommandBuffer command_buffer, Slice(Local
|
||||
}
|
||||
}
|
||||
|
||||
void buffer_copy_to_host(VulkanBuffer buffer, Slice(HostBufferCopy) regions)
|
||||
fn void buffer_copy_to_host(VulkanBuffer buffer, Slice(HostBufferCopy) regions)
|
||||
{
|
||||
assert(buffer.type == BUFFER_TYPE_STAGING);
|
||||
|
||||
auto* buffer_pointer = (u8*)buffer.address;
|
||||
let(buffer_pointer, (u8*)buffer.address);
|
||||
|
||||
for (u64 i = 0; i < regions.length; i += 1)
|
||||
{
|
||||
auto region = regions.pointer[i];
|
||||
auto* destination = buffer_pointer + region.destination_offset;
|
||||
let(region, regions.pointer[i]);
|
||||
let(destination, buffer_pointer + region.destination_offset);
|
||||
assert(destination + region.source.length <= (u8*)buffer.address + buffer.size);
|
||||
#define USE_MEMCPY 1
|
||||
#if USE_MEMCPY
|
||||
@ -396,7 +396,7 @@ fn GPUMemory vk_allocate_memory(VkDevice device, const VkAllocationCallbacks* al
|
||||
u32 memory_type_index;
|
||||
for (memory_type_index = 0; memory_type_index < memory_properties.memoryTypeCount; memory_type_index += 1)
|
||||
{
|
||||
auto memory_type = memory_properties.memoryTypes[memory_type_index];
|
||||
let(memory_type, memory_properties.memoryTypes[memory_type_index]);
|
||||
|
||||
if ((memory_requirements.memoryTypeBits & (1 << memory_type_index)) != 0 && (memory_type.propertyFlags & flags) == flags)
|
||||
{
|
||||
@ -498,7 +498,7 @@ fn VulkanBuffer buffer_create(Renderer* renderer, u64 size, BufferType type)
|
||||
VkMemoryPropertyFlags memory_flags =
|
||||
(VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT * is_dst) |
|
||||
((VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT) * is_src);
|
||||
auto result = vk_buffer_create(renderer->device, renderer->allocator, renderer->memory_properties, size, usage, memory_flags);
|
||||
let(result, vk_buffer_create(renderer->device, renderer->allocator, renderer->memory_properties, size, usage, memory_flags));
|
||||
result.type = type;
|
||||
return result;
|
||||
}
|
||||
@ -524,7 +524,7 @@ fn u8 vk_layer_is_supported(String layer_name)
|
||||
{
|
||||
VkLayerProperties* properties = &layers[i];
|
||||
|
||||
auto candidate_layer_name = cstr(properties->layerName);
|
||||
let(candidate_layer_name, cstr(properties->layerName));
|
||||
if (s_equal(candidate_layer_name, layer_name))
|
||||
{
|
||||
supported = 1;
|
||||
@ -549,7 +549,7 @@ fn String message_severity_to_string(VkDebugUtilsMessageSeverityFlagBitsEXT mess
|
||||
return strlit("WARNING");
|
||||
case VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT:
|
||||
return strlit("ERROR");
|
||||
case VK_DEBUG_UTILS_MESSAGE_SEVERITY_FLAG_BITS_MAX_ENUM_EXT:
|
||||
default:
|
||||
unreachable();
|
||||
}
|
||||
}
|
||||
@ -566,9 +566,10 @@ fn String message_type_to_string(VkDebugUtilsMessageTypeFlagBitsEXT message_type
|
||||
return strlit("PERFORMANCE");
|
||||
case VK_DEBUG_UTILS_MESSAGE_TYPE_DEVICE_ADDRESS_BINDING_BIT_EXT:
|
||||
return strlit("DEVICE_ADDRESS_BINDING");
|
||||
case VK_DEBUG_UTILS_MESSAGE_TYPE_FLAG_BITS_MAX_ENUM_EXT:
|
||||
default:
|
||||
unreachable();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
fn VkBool32 VKAPI_CALL debug_callback(VkDebugUtilsMessageSeverityFlagBitsEXT message_severity, VkDebugUtilsMessageTypeFlagsEXT message_type, const VkDebugUtilsMessengerCallbackDataEXT* callback_data, void* user_data)
|
||||
@ -638,7 +639,7 @@ fn void immediate_end(ImmediateContext context)
|
||||
|
||||
vkok(vkQueueSubmit2(context.queue, array_length(submit_info), submit_info, context.fence));
|
||||
VkBool32 wait_all = 1;
|
||||
auto timeout = ~(u64)0;
|
||||
let(timeout, ~(u64)0);
|
||||
vkok(vkWaitForFences(context.device, array_length(fences), fences, wait_all, timeout));
|
||||
}
|
||||
|
||||
@ -775,21 +776,82 @@ fn void vk_image_copy(VkCommandBuffer command_buffer, VulkanCopyImageArgs args)
|
||||
vkCmdBlitImage2(command_buffer, &blit_info);
|
||||
}
|
||||
|
||||
|
||||
Renderer* renderer_initialize(Arena* arena)
|
||||
fn Renderer* rendering_initialize(Arena* arena)
|
||||
{
|
||||
Renderer* renderer = &renderer_memory;
|
||||
vkok(volkInitialize());
|
||||
|
||||
auto api_version = volkGetInstanceVersion();
|
||||
#ifdef __linux__
|
||||
vulkan_library = os_library_load("libvulkan.so.1");
|
||||
#elif _WIN32
|
||||
vulkan_library = os_library_load("vulkan-1.dll");
|
||||
#elif __APPLE__
|
||||
vulkan_library = os_library_load("libvulkan.dylib");
|
||||
|
||||
if (!os_library_is_valid(vulkan_library))
|
||||
{
|
||||
vulkan_library = os_library_load("libvulkan.1.dylib");
|
||||
}
|
||||
|
||||
if (!os_library_is_valid(vulkan_library))
|
||||
{
|
||||
vulkan_library = os_library_load("libMoltenVK.dylib");
|
||||
}
|
||||
|
||||
if (!os_library_is_valid(vulkan_library))
|
||||
{
|
||||
vulkan_library = os_library_load("vulkan.framework/vulkan");
|
||||
}
|
||||
|
||||
if (!os_library_is_valid(vulkan_library))
|
||||
{
|
||||
vulkan_library = os_library_load("MoltenVK.framework/MoltenVK");
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!os_library_is_valid(vulkan_library))
|
||||
{
|
||||
failed_execution();
|
||||
}
|
||||
|
||||
vkGetInstanceProcAddr = os_symbol_load(vulkan_library, "vkGetInstanceProcAddr");
|
||||
if (!vkGetInstanceProcAddr)
|
||||
{
|
||||
failed_execution();
|
||||
}
|
||||
|
||||
vulkan_load_instance_function(0, vkEnumerateInstanceVersion);
|
||||
vulkan_load_instance_function(0, vkCreateInstance);
|
||||
|
||||
u32 api_version = 0;
|
||||
if (vkEnumerateInstanceVersion)
|
||||
{
|
||||
vkok(vkEnumerateInstanceVersion(&api_version));
|
||||
}
|
||||
|
||||
if (!api_version)
|
||||
{
|
||||
if (vkCreateInstance)
|
||||
{
|
||||
api_version = VK_API_VERSION_1_0;
|
||||
}
|
||||
}
|
||||
|
||||
if (!api_version)
|
||||
{
|
||||
failed_execution();
|
||||
}
|
||||
|
||||
if (api_version < VK_API_VERSION_1_3)
|
||||
{
|
||||
failed_execution();
|
||||
}
|
||||
|
||||
// Proceed to load Vulkan instance-level functions
|
||||
vulkan_load_instance_function(0, vkEnumerateInstanceLayerProperties);
|
||||
|
||||
{
|
||||
#if BB_DEBUG
|
||||
auto debug_layer = strlit("VK_LAYER_KHRONOS_validation");
|
||||
let(debug_layer, strlit("VK_LAYER_KHRONOS_validation"));
|
||||
if (!vk_layer_is_supported(debug_layer))
|
||||
{
|
||||
failed_execution();
|
||||
@ -798,7 +860,7 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
{
|
||||
string_to_c(debug_layer),
|
||||
};
|
||||
auto layer_count = array_length(layers);
|
||||
let(layer_count, array_length(layers));
|
||||
#else
|
||||
const char** layers = 0;
|
||||
u32 layer_count = 0;
|
||||
@ -852,11 +914,10 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
};
|
||||
|
||||
#endif
|
||||
auto* pNext =
|
||||
#if BB_DEBUG
|
||||
enable_shader_debug_printf ? (void*)&validation_features : (void*)&msg_ci;
|
||||
let(pNext, enable_shader_debug_printf ? (void*)&validation_features : (void*)&msg_ci);
|
||||
#else
|
||||
(void*)0;
|
||||
void* pNext = 0;
|
||||
#endif
|
||||
VkApplicationInfo app_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_APPLICATION_INFO,
|
||||
@ -877,19 +938,40 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
.enabledExtensionCount = array_length(extensions),
|
||||
};
|
||||
|
||||
vkok(vkCreateInstance(&ci, renderer->allocator, &renderer->instance));
|
||||
volkLoadInstance(renderer->instance);
|
||||
vkok(vkCreateInstance(&ci, renderer->allocator, &instance));
|
||||
|
||||
#if BB_DEBUG
|
||||
vulkan_load_instance_function(instance, vkCreateDebugUtilsMessengerEXT);
|
||||
VkDebugUtilsMessengerEXT messenger;
|
||||
vkok(vkCreateDebugUtilsMessengerEXT(renderer->instance, &msg_ci, renderer->allocator, &messenger));
|
||||
vkok(vkCreateDebugUtilsMessengerEXT(instance, &msg_ci, renderer->allocator, &messenger));
|
||||
#endif
|
||||
}
|
||||
|
||||
vulkan_load_instance_function(instance, vkGetDeviceProcAddr);
|
||||
vulkan_load_instance_function(instance, vkEnumeratePhysicalDevices);
|
||||
vulkan_load_instance_function(instance, vkGetPhysicalDeviceMemoryProperties);
|
||||
vulkan_load_instance_function(instance, vkGetPhysicalDeviceProperties);
|
||||
vulkan_load_instance_function(instance, vkGetPhysicalDeviceQueueFamilyProperties);
|
||||
vulkan_load_instance_function(instance, vkCreateDevice);
|
||||
vulkan_load_instance_function(instance, vkGetPhysicalDeviceSurfaceCapabilitiesKHR);
|
||||
vulkan_load_instance_function(instance, vkGetPhysicalDeviceSurfacePresentModesKHR);
|
||||
#ifdef VK_USE_PLATFORM_XCB_KHR
|
||||
vulkan_load_instance_function(instance, vkCreateXcbSurfaceKHR);
|
||||
#endif
|
||||
#ifdef VK_USE_PLATFORM_WIN32_KHR
|
||||
vulkan_load_instance_function(instance, vkCreateWin32SurfaceKHR);
|
||||
#endif
|
||||
#ifdef VK_USE_PLATFORM_WIN32_KHR
|
||||
vulkan_load_instance_function(instance, vkCreateWin32SurfaceKHR);
|
||||
#endif
|
||||
#ifdef VK_USE_PLATFORM_METAL_EXT
|
||||
vulkan_load_instance_function(instance, vkCreateMetalSurfaceEXT);
|
||||
#endif
|
||||
|
||||
{
|
||||
u32 physical_device_count;
|
||||
VkPhysicalDevice physical_devices[256];
|
||||
vkok(vkEnumeratePhysicalDevices(renderer->instance, &physical_device_count, 0));
|
||||
vkok(vkEnumeratePhysicalDevices(instance, &physical_device_count, 0));
|
||||
|
||||
if (physical_device_count == 0)
|
||||
{
|
||||
@ -901,7 +983,7 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
failed_execution();
|
||||
}
|
||||
|
||||
vkok(vkEnumeratePhysicalDevices(renderer->instance, &physical_device_count, physical_devices));
|
||||
vkok(vkEnumeratePhysicalDevices(instance, &physical_device_count, physical_devices));
|
||||
|
||||
renderer->physical_device = physical_devices[0];
|
||||
}
|
||||
@ -1026,6 +1108,61 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
vkok(vkCreateDevice(renderer->physical_device, &ci, renderer->allocator, &renderer->device));
|
||||
}
|
||||
|
||||
// Load device table
|
||||
vulkan_load_device_function(renderer->device, vkCreateSwapchainKHR);
|
||||
vulkan_load_device_function(renderer->device, vkDestroySwapchainKHR);
|
||||
vulkan_load_device_function(renderer->device, vkGetSwapchainImagesKHR);
|
||||
vulkan_load_device_function(renderer->device, vkGetImageMemoryRequirements);
|
||||
vulkan_load_device_function(renderer->device, vkGetBufferMemoryRequirements);
|
||||
vulkan_load_device_function(renderer->device, vkMapMemory);
|
||||
vulkan_load_device_function(renderer->device, vkUnmapMemory);
|
||||
vulkan_load_device_function(renderer->device, vkAllocateMemory);
|
||||
vulkan_load_device_function(renderer->device, vkBindImageMemory);
|
||||
vulkan_load_device_function(renderer->device, vkBindBufferMemory);
|
||||
vulkan_load_device_function(renderer->device, vkBindBufferMemory);
|
||||
vulkan_load_device_function(renderer->device, vkGetDeviceQueue);
|
||||
vulkan_load_device_function(renderer->device, vkCreateCommandPool);
|
||||
vulkan_load_device_function(renderer->device, vkAllocateCommandBuffers);
|
||||
vulkan_load_device_function(renderer->device, vkCreateFence);
|
||||
vulkan_load_device_function(renderer->device, vkCreateSemaphore);
|
||||
vulkan_load_device_function(renderer->device, vkCreateSampler);
|
||||
vulkan_load_device_function(renderer->device, vkCreateShaderModule);
|
||||
vulkan_load_device_function(renderer->device, vkCreateDescriptorSetLayout);
|
||||
vulkan_load_device_function(renderer->device, vkCreatePipelineLayout);
|
||||
vulkan_load_device_function(renderer->device, vkCreateGraphicsPipelines);
|
||||
vulkan_load_device_function(renderer->device, vkCreateImage);
|
||||
vulkan_load_device_function(renderer->device, vkCreateImageView);
|
||||
vulkan_load_device_function(renderer->device, vkCreateBuffer);
|
||||
vulkan_load_device_function(renderer->device, vkCreateDescriptorPool);
|
||||
vulkan_load_device_function(renderer->device, vkAllocateDescriptorSets);
|
||||
vulkan_load_device_function(renderer->device, vkResetFences);
|
||||
vulkan_load_device_function(renderer->device, vkResetCommandBuffer);
|
||||
vulkan_load_device_function(renderer->device, vkBeginCommandBuffer);
|
||||
vulkan_load_device_function(renderer->device, vkCmdPipelineBarrier2);
|
||||
vulkan_load_device_function(renderer->device, vkCmdCopyBufferToImage);
|
||||
vulkan_load_device_function(renderer->device, vkEndCommandBuffer);
|
||||
vulkan_load_device_function(renderer->device, vkQueueSubmit2);
|
||||
vulkan_load_device_function(renderer->device, vkWaitForFences);
|
||||
vulkan_load_device_function(renderer->device, vkUpdateDescriptorSets);
|
||||
vulkan_load_device_function(renderer->device, vkAcquireNextImageKHR);
|
||||
vulkan_load_device_function(renderer->device, vkGetBufferDeviceAddress);
|
||||
vulkan_load_device_function(renderer->device, vkCmdCopyBuffer2);
|
||||
vulkan_load_device_function(renderer->device, vkCmdSetViewport);
|
||||
vulkan_load_device_function(renderer->device, vkCmdSetScissor);
|
||||
vulkan_load_device_function(renderer->device, vkCmdBeginRendering);
|
||||
vulkan_load_device_function(renderer->device, vkCmdEndRendering);
|
||||
vulkan_load_device_function(renderer->device, vkCmdBindPipeline);
|
||||
vulkan_load_device_function(renderer->device, vkCmdBindDescriptorSets);
|
||||
vulkan_load_device_function(renderer->device, vkCmdBindIndexBuffer);
|
||||
vulkan_load_device_function(renderer->device, vkCmdPushConstants);
|
||||
vulkan_load_device_function(renderer->device, vkCmdDrawIndexed);
|
||||
vulkan_load_device_function(renderer->device, vkCmdBlitImage2);
|
||||
vulkan_load_device_function(renderer->device, vkQueuePresentKHR);
|
||||
vulkan_load_device_function(renderer->device, vkDeviceWaitIdle);
|
||||
vulkan_load_device_function(renderer->device, vkDestroyImageView);
|
||||
vulkan_load_device_function(renderer->device, vkDestroyImage);
|
||||
vulkan_load_device_function(renderer->device, vkFreeMemory);
|
||||
|
||||
vkGetDeviceQueue(renderer->device, graphics_queue_family_index, 0, &renderer->graphics_queue);
|
||||
|
||||
renderer->immediate.device = renderer->device;
|
||||
@ -1079,8 +1216,8 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
}
|
||||
|
||||
String shader_binaries[] = {
|
||||
strlit(BUILD_DIR "/rect.vert.spv"),
|
||||
strlit(BUILD_DIR "/rect.frag.spv"),
|
||||
strlit(BUILD_DIR "/" "rect.vert.spv"),
|
||||
strlit(BUILD_DIR "/" "rect.frag.spv"),
|
||||
};
|
||||
|
||||
PipelineLayoutCreate pipeline_layouts[] = {
|
||||
@ -1112,17 +1249,17 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
.layout_index = 0,
|
||||
},
|
||||
};
|
||||
auto create_data = (GraphicsPipelinesCreate) {
|
||||
GraphicsPipelinesCreate create_data = {
|
||||
.layouts = array_to_slice(pipeline_layouts),
|
||||
.pipelines = array_to_slice(pipeline_create),
|
||||
.shader_binaries = array_to_slice(shader_binaries),
|
||||
};
|
||||
auto graphics_pipeline_count = create_data.pipelines.length;
|
||||
let(graphics_pipeline_count, create_data.pipelines.length);
|
||||
assert(graphics_pipeline_count);
|
||||
auto pipeline_layout_count = create_data.layouts.length;
|
||||
let(pipeline_layout_count, create_data.layouts.length);
|
||||
assert(pipeline_layout_count);
|
||||
assert(pipeline_layout_count <= graphics_pipeline_count);
|
||||
auto shader_count = create_data.shader_binaries.length;
|
||||
let(shader_count, create_data.shader_binaries.length);
|
||||
|
||||
VkPipeline pipeline_handles[BB_PIPELINE_COUNT];
|
||||
VkPipelineShaderStageCreateInfo shader_create_infos[MAX_SHADER_MODULE_COUNT_PER_PIPELINE];
|
||||
@ -1246,13 +1383,13 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
.stencilAttachmentFormat = 0,
|
||||
};
|
||||
|
||||
auto* shader_modules = arena_allocate(arena, VkShaderModule, shader_count);
|
||||
let(shader_modules, arena_allocate(arena, VkShaderModule, shader_count));
|
||||
|
||||
for (u64 i = 0; i < shader_count; i += 1)
|
||||
{
|
||||
String shader_binary_path = create_data.shader_binaries.pointer[i];
|
||||
|
||||
auto binary = file_read(arena, shader_binary_path);
|
||||
let(binary, file_read(arena, shader_binary_path));
|
||||
|
||||
VkShaderModuleCreateInfo create_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO,
|
||||
@ -1265,10 +1402,10 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
|
||||
for (u64 pipeline_index = 0; pipeline_index < pipeline_layout_count; pipeline_index += 1)
|
||||
{
|
||||
auto create = create_data.layouts.pointer[pipeline_index];
|
||||
auto descriptor_set_layout_count = create.descriptor_set_layouts.length;
|
||||
auto push_constant_range_count = create.push_constant_ranges.length;
|
||||
auto* pipeline = &renderer->pipelines[pipeline_index];
|
||||
let(create, create_data.layouts.pointer[pipeline_index]);
|
||||
let(descriptor_set_layout_count, create.descriptor_set_layouts.length);
|
||||
let(push_constant_range_count, create.push_constant_ranges.length);
|
||||
let(pipeline, &renderer->pipelines[pipeline_index]);
|
||||
pipeline->descriptor_set_count = descriptor_set_layout_count;
|
||||
pipeline->push_constant_range_count = push_constant_range_count;
|
||||
|
||||
@ -1281,14 +1418,14 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
|
||||
for (u64 descriptor_set_layout_index = 0; descriptor_set_layout_index < descriptor_set_layout_count; descriptor_set_layout_index += 1)
|
||||
{
|
||||
auto set_layout_create = create.descriptor_set_layouts.pointer[descriptor_set_layout_index];
|
||||
auto binding_count = set_layout_create.bindings.length;
|
||||
auto* descriptor_set_layout_bindings = &pipeline->descriptor_set_layout_bindings[descriptor_set_layout_index];
|
||||
let(set_layout_create, create.descriptor_set_layouts.pointer[descriptor_set_layout_index]);
|
||||
let(binding_count, set_layout_create.bindings.length);
|
||||
let(descriptor_set_layout_bindings, &pipeline->descriptor_set_layout_bindings[descriptor_set_layout_index]);
|
||||
descriptor_set_layout_bindings->count = binding_count;
|
||||
|
||||
for (u64 binding_index = 0; binding_index < binding_count; binding_index += 1)
|
||||
{
|
||||
auto binding_descriptor = set_layout_create.bindings.pointer[binding_index];
|
||||
let(binding_descriptor, set_layout_create.bindings.pointer[binding_index]);
|
||||
|
||||
VkDescriptorType descriptor_type = vulkan_descriptor_type(binding_descriptor.type);
|
||||
|
||||
@ -1321,7 +1458,7 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
|
||||
for (u64 push_constant_index = 0; push_constant_index < push_constant_range_count; push_constant_index += 1)
|
||||
{
|
||||
auto push_constant_descriptor = create.push_constant_ranges.pointer[push_constant_index];
|
||||
let(push_constant_descriptor, create.push_constant_ranges.pointer[push_constant_index]);
|
||||
pipeline->push_constant_ranges[push_constant_index] = (VkPushConstantRange) {
|
||||
.stageFlags = vulkan_shader_stage(push_constant_descriptor.stage),
|
||||
.offset = push_constant_descriptor.offset,
|
||||
@ -1344,8 +1481,8 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
|
||||
for (u64 i = 0; i < graphics_pipeline_count; i += 1)
|
||||
{
|
||||
auto create = create_data.pipelines.pointer[i];
|
||||
auto pipeline_shader_count = create.shader_source_indices.length;
|
||||
let(create, create_data.pipelines.pointer[i]);
|
||||
let(pipeline_shader_count, create.shader_source_indices.length);
|
||||
if (pipeline_shader_count > MAX_SHADER_MODULE_COUNT_PER_PIPELINE)
|
||||
{
|
||||
failed_execution();
|
||||
@ -1353,8 +1490,8 @@ Renderer* renderer_initialize(Arena* arena)
|
||||
|
||||
for (u64 i = 0; i < pipeline_shader_count; i += 1)
|
||||
{
|
||||
auto shader_index = create.shader_source_indices.pointer[i];
|
||||
auto shader_source_path = create_data.shader_binaries.pointer[shader_index];
|
||||
let(shader_index, create.shader_source_indices.pointer[i]);
|
||||
let(shader_source_path, create_data.shader_binaries.pointer[shader_index]);
|
||||
|
||||
shader_create_infos[i] = (VkPipelineShaderStageCreateInfo) {
|
||||
.sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO,
|
||||
@ -1543,43 +1680,82 @@ fn void swapchain_recreate(Renderer* renderer, RenderWindow* window)
|
||||
});
|
||||
}
|
||||
|
||||
// typedef void GLFWwindow;
|
||||
// extern VkResult glfwCreateWindowSurface(VkInstance instance, GLFWwindow* window, const VkAllocationCallbacks* allocator, VkSurfaceKHR* surface);
|
||||
|
||||
RenderWindow* renderer_window_initialize(Renderer* renderer, OSWindow window)
|
||||
fn RenderWindow* rendering_initialize_window(Renderer* renderer, WindowingInstance* window)
|
||||
{
|
||||
RenderWindow* result = &renderer_window_memory;
|
||||
vkok((VkResult)window_create_surface(renderer->instance, window, renderer->allocator, (void**)&result->surface));
|
||||
// TODO: truly allocate
|
||||
RenderWindow* render_window = &renderer_window_memory;
|
||||
|
||||
swapchain_recreate(renderer, result);
|
||||
#if BB_WINDOWING_BACKEND_X11
|
||||
VkXcbSurfaceCreateInfoKHR create_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_XCB_SURFACE_CREATE_INFO_KHR,
|
||||
.pNext = 0,
|
||||
.flags = 0,
|
||||
.connection = xcb_connection_get(),
|
||||
.window = xcb_window_from_windowing_instance(window),
|
||||
};
|
||||
vkok(vkCreateXcbSurfaceKHR(instance, &create_info, renderer->allocator, &render_window->surface));
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_WIN32
|
||||
VkWin32SurfaceCreateInfoKHR create_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR,
|
||||
.pNext = 0,
|
||||
.flags = 0,
|
||||
.hinstance = windowing_connection.instance,
|
||||
.hwnd = window->handle,
|
||||
};
|
||||
vkok(vkCreateWin32SurfaceKHR(instance, &create_info, renderer->allocator, &render_window->surface));
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_COCOA
|
||||
CAMetalLayer* layer = [CAMetalLayer layer];
|
||||
layer.pixelFormat = MTLPixelFormatBGRA8Unorm_sRGB;
|
||||
layer.framebufferOnly = true;
|
||||
layer.frame = window.frame;
|
||||
window.contentView.layer = layer;
|
||||
window.opaque = true;
|
||||
window.backgroundColor = nil;
|
||||
|
||||
VkMetalSurfaceCreateInfoEXT create_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_METAL_SURFACE_CREATE_INFO_EXT,
|
||||
.pNext = 0,
|
||||
.flags = 0,
|
||||
.pLayer = [CAMetalLayer layer],
|
||||
};
|
||||
vkCreateMetalSurfaceEXT(instance, &create_info, renderer->allocator, &render_window->surface);
|
||||
#endif
|
||||
|
||||
WindowingSize window_size = windowing_get_instance_framebuffer_size(window);
|
||||
|
||||
swapchain_recreate(renderer, render_window);
|
||||
|
||||
for (u64 frame_index = 0; frame_index < MAX_FRAME_COUNT; frame_index += 1)
|
||||
{
|
||||
for (u64 pipeline_index = 0; pipeline_index < BB_PIPELINE_COUNT; pipeline_index += 1)
|
||||
{
|
||||
result->frames[frame_index].pipeline_instantiations[pipeline_index].vertex_buffer.gpu.type = BUFFER_TYPE_VERTEX;
|
||||
result->frames[frame_index].pipeline_instantiations[pipeline_index].index_buffer.gpu.type = BUFFER_TYPE_INDEX;
|
||||
result->frames[frame_index].pipeline_instantiations[pipeline_index].transient_buffer.type = BUFFER_TYPE_STAGING;
|
||||
render_window->frames[frame_index].pipeline_instantiations[pipeline_index].vertex_buffer.gpu.type = BUFFER_TYPE_VERTEX;
|
||||
render_window->frames[frame_index].pipeline_instantiations[pipeline_index].index_buffer.gpu.type = BUFFER_TYPE_INDEX;
|
||||
render_window->frames[frame_index].pipeline_instantiations[pipeline_index].transient_buffer.type = BUFFER_TYPE_STAGING;
|
||||
}
|
||||
}
|
||||
|
||||
for (u64 pipeline_index = 0; pipeline_index < BB_PIPELINE_COUNT; pipeline_index += 1)
|
||||
{
|
||||
auto* pipeline_descriptor = &renderer->pipelines[pipeline_index];
|
||||
auto* pipeline_instantiation = &result->pipeline_instantiations[pipeline_index];
|
||||
let(pipeline_descriptor, &renderer->pipelines[pipeline_index]);
|
||||
let(pipeline_instantiation, &render_window->pipeline_instantiations[pipeline_index]);
|
||||
|
||||
u16 descriptor_type_counter[DESCRIPTOR_TYPE_COUNT] = {};
|
||||
|
||||
for (u64 descriptor_index = 0; descriptor_index < pipeline_descriptor->descriptor_set_count; descriptor_index += 1)
|
||||
{
|
||||
auto* descriptor_set_layout_bindings = &pipeline_descriptor->descriptor_set_layout_bindings[descriptor_index];
|
||||
let(descriptor_set_layout_bindings, &pipeline_descriptor->descriptor_set_layout_bindings[descriptor_index]);
|
||||
|
||||
for (u64 binding_index = 0; binding_index < descriptor_set_layout_bindings->count; binding_index += 1)
|
||||
{
|
||||
auto* binding_descriptor = &descriptor_set_layout_bindings->buffer[binding_index];
|
||||
auto descriptor_type = descriptor_type_from_vulkan(binding_descriptor->descriptorType);
|
||||
auto* counter_ptr = &descriptor_type_counter[descriptor_type];
|
||||
auto old_counter = *counter_ptr;
|
||||
let(binding_descriptor, &descriptor_set_layout_bindings->buffer[binding_index]);
|
||||
let(descriptor_type, descriptor_type_from_vulkan(binding_descriptor->descriptorType));
|
||||
let(counter_ptr, &descriptor_type_counter[descriptor_type]);
|
||||
let(old_counter, *counter_ptr);
|
||||
*counter_ptr = old_counter + binding_descriptor->descriptorCount;
|
||||
}
|
||||
}
|
||||
@ -1589,10 +1765,10 @@ RenderWindow* renderer_window_initialize(Renderer* renderer, OSWindow window)
|
||||
|
||||
for (DescriptorType i = 0; i < DESCRIPTOR_TYPE_COUNT; i += 1)
|
||||
{
|
||||
auto count = descriptor_type_counter[i];
|
||||
let(count, descriptor_type_counter[i]);
|
||||
if (count)
|
||||
{
|
||||
auto* pool_size = &pool_sizes[pool_size_count];
|
||||
let(pool_size, &pool_sizes[pool_size_count]);
|
||||
pool_size_count += 1;
|
||||
|
||||
*pool_size = (VkDescriptorPoolSize) {
|
||||
@ -1643,7 +1819,7 @@ RenderWindow* renderer_window_initialize(Renderer* renderer, OSWindow window)
|
||||
.flags = 0,
|
||||
};
|
||||
|
||||
WindowFrame* frame = &result->frames[i];
|
||||
WindowFrame* frame = &render_window->frames[i];
|
||||
vkok(vkCreateCommandPool(renderer->device, &command_pool_create_info, renderer->allocator, &frame->command_pool));
|
||||
|
||||
VkCommandBufferAllocateInfo command_buffer_allocate_info = {
|
||||
@ -1660,7 +1836,7 @@ RenderWindow* renderer_window_initialize(Renderer* renderer, OSWindow window)
|
||||
frame->bound_pipeline = BB_PIPELINE_COUNT;
|
||||
}
|
||||
|
||||
return result;
|
||||
return render_window;
|
||||
}
|
||||
|
||||
fn WindowFrame* window_frame(RenderWindow* window)
|
||||
@ -1668,10 +1844,10 @@ fn WindowFrame* window_frame(RenderWindow* window)
|
||||
return &window->frames[window->frame_index % MAX_FRAME_COUNT];
|
||||
}
|
||||
|
||||
void renderer_window_frame_begin(Renderer* renderer, RenderWindow* window)
|
||||
fn void renderer_window_frame_begin(Renderer* renderer, RenderWindow* window)
|
||||
{
|
||||
auto* frame = window_frame(window);
|
||||
auto timeout = ~(u64)0;
|
||||
let(frame, window_frame(window));
|
||||
let(timeout, ~(u64)0);
|
||||
|
||||
u32 fence_count = 1;
|
||||
VkBool32 wait_all = 1;
|
||||
@ -1695,14 +1871,14 @@ void renderer_window_frame_begin(Renderer* renderer, RenderWindow* window)
|
||||
// Reset frame data
|
||||
for (u32 i = 0; i < array_length(window->pipeline_instantiations); i += 1)
|
||||
{
|
||||
auto* pipeline_instantiation = &frame->pipeline_instantiations[i];
|
||||
let(pipeline_instantiation, &frame->pipeline_instantiations[i]);
|
||||
pipeline_instantiation->vertex_buffer.cpu.length = 0;
|
||||
pipeline_instantiation->vertex_buffer.count = 0;
|
||||
pipeline_instantiation->index_buffer.cpu.length = 0;
|
||||
}
|
||||
}
|
||||
|
||||
void buffer_destroy(Renderer* renderer, VulkanBuffer buffer)
|
||||
fn void buffer_destroy(Renderer* renderer, VulkanBuffer buffer)
|
||||
{
|
||||
if (buffer.handle)
|
||||
{
|
||||
@ -1729,9 +1905,9 @@ fn void buffer_ensure_capacity(Renderer* renderer, VulkanBuffer* buffer, u64 nee
|
||||
}
|
||||
}
|
||||
|
||||
void renderer_window_frame_end(Renderer* renderer, RenderWindow* window)
|
||||
fn void renderer_window_frame_end(Renderer* renderer, RenderWindow* window)
|
||||
{
|
||||
auto* frame = window_frame(window);
|
||||
let(frame, window_frame(window));
|
||||
|
||||
VkCommandBufferBeginInfo command_buffer_begin_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO,
|
||||
@ -1741,13 +1917,13 @@ void renderer_window_frame_end(Renderer* renderer, RenderWindow* window)
|
||||
|
||||
for (u32 i = 0; i < BB_PIPELINE_COUNT; i += 1)
|
||||
{
|
||||
auto* frame_pipeline_instantiation = &frame->pipeline_instantiations[i];
|
||||
let(frame_pipeline_instantiation, &frame->pipeline_instantiations[i]);
|
||||
|
||||
if (likely(frame_pipeline_instantiation->vertex_buffer.cpu.length))
|
||||
{
|
||||
auto new_vertex_buffer_size = frame_pipeline_instantiation->vertex_buffer.cpu.length * sizeof(*frame_pipeline_instantiation->vertex_buffer.cpu.pointer);
|
||||
auto new_index_buffer_size = frame_pipeline_instantiation->index_buffer.cpu.length * sizeof(*frame_pipeline_instantiation->index_buffer.cpu.pointer);
|
||||
auto new_transient_buffer_size = new_vertex_buffer_size + new_index_buffer_size;
|
||||
let(new_vertex_buffer_size, frame_pipeline_instantiation->vertex_buffer.cpu.length * sizeof(*frame_pipeline_instantiation->vertex_buffer.cpu.pointer));
|
||||
let(new_index_buffer_size, frame_pipeline_instantiation->index_buffer.cpu.length * sizeof(*frame_pipeline_instantiation->index_buffer.cpu.pointer));
|
||||
let(new_transient_buffer_size, new_vertex_buffer_size + new_index_buffer_size);
|
||||
|
||||
buffer_ensure_capacity(renderer, &frame_pipeline_instantiation->transient_buffer, new_transient_buffer_size);
|
||||
buffer_ensure_capacity(renderer, &frame_pipeline_instantiation->vertex_buffer.gpu, new_vertex_buffer_size);
|
||||
@ -1857,9 +2033,9 @@ void renderer_window_frame_end(Renderer* renderer, RenderWindow* window)
|
||||
|
||||
for (u32 i = 0; i < BB_PIPELINE_COUNT; i += 1)
|
||||
{
|
||||
auto* pipeline = &renderer->pipelines[i];
|
||||
auto* pipeline_instantiation = &window->pipeline_instantiations[i];
|
||||
auto* frame_pipeline_instantiation = &frame->pipeline_instantiations[i];
|
||||
let(pipeline, &renderer->pipelines[i]);
|
||||
let(pipeline_instantiation, &window->pipeline_instantiations[i]);
|
||||
let(frame_pipeline_instantiation, &frame->pipeline_instantiations[i]);
|
||||
|
||||
if (likely(frame_pipeline_instantiation->vertex_buffer.cpu.length))
|
||||
{
|
||||
@ -1883,15 +2059,14 @@ void renderer_window_frame_end(Renderer* renderer, RenderWindow* window)
|
||||
}
|
||||
|
||||
// Send vertex buffer and screen dimensions to the shader
|
||||
auto push_constants = (GPUDrawPushConstants)
|
||||
{
|
||||
GPUDrawPushConstants push_constants = {
|
||||
.vertex_buffer = frame_pipeline_instantiation->vertex_buffer.gpu.address,
|
||||
.width = window->width,
|
||||
.height = window->height,
|
||||
};
|
||||
|
||||
{
|
||||
auto push_constant_range = pipeline->push_constant_ranges[0];
|
||||
let(push_constant_range, pipeline->push_constant_ranges[0]);
|
||||
vkCmdPushConstants(frame->command_buffer, pipeline->layout, push_constant_range.stageFlags, push_constant_range.offset, push_constant_range.size, &push_constants);
|
||||
frame->push_constants = push_constants;
|
||||
}
|
||||
@ -2032,15 +2207,17 @@ fn u32 format_channel_count(TextureFormat format)
|
||||
case TEXTURE_FORMAT_R8G8B8A8_SRGB:
|
||||
return 4;
|
||||
}
|
||||
|
||||
unreachable();
|
||||
}
|
||||
|
||||
TextureIndex renderer_texture_create(Renderer* renderer, TextureMemory texture_memory)
|
||||
fn TextureIndex renderer_texture_create(Renderer* renderer, TextureMemory texture_memory)
|
||||
{
|
||||
assert(texture_memory.depth == 1);
|
||||
|
||||
auto texture_index = texture_count;
|
||||
let(texture_index, texture_count);
|
||||
texture_count += 1;
|
||||
auto* texture = &textures[texture_index];
|
||||
let(texture, &textures[texture_index]);
|
||||
texture->image = vk_image_create(renderer->device, renderer->allocator, renderer->memory_properties, (VulkanImageCreate) {
|
||||
.width = texture_memory.width,
|
||||
.height = texture_memory.height,
|
||||
@ -2050,10 +2227,10 @@ TextureIndex renderer_texture_create(Renderer* renderer, TextureMemory texture_m
|
||||
});
|
||||
texture->sampler = renderer->sampler;
|
||||
|
||||
auto image_size = (u64)texture_memory.depth * texture_memory.width * texture_memory.height * format_channel_count(texture_memory.format);
|
||||
let(image_size, (u64)texture_memory.depth * texture_memory.width * texture_memory.height * format_channel_count(texture_memory.format));
|
||||
VkBufferUsageFlags buffer_usage_flags = VK_BUFFER_USAGE_TRANSFER_SRC_BIT;
|
||||
VkMemoryPropertyFlags buffer_memory_flags = VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT;
|
||||
auto transfer_buffer = vk_buffer_create(renderer->device, renderer->allocator, renderer->memory_properties, image_size, buffer_usage_flags, buffer_memory_flags);
|
||||
let(transfer_buffer, vk_buffer_create(renderer->device, renderer->allocator, renderer->memory_properties, image_size, buffer_usage_flags, buffer_memory_flags));
|
||||
memcpy((void*)transfer_buffer.address, texture_memory.pointer, image_size);
|
||||
|
||||
immediate_start(renderer->immediate);
|
||||
@ -2093,15 +2270,15 @@ TextureIndex renderer_texture_create(Renderer* renderer, TextureMemory texture_m
|
||||
return (TextureIndex) { .value = texture_index };
|
||||
}
|
||||
|
||||
void window_draw_indexed(RenderWindow* window, u32 index_count, u32 instance_count, u32 first_index, s32 vertex_offset, u32 first_instance)
|
||||
fn void window_draw_indexed(RenderWindow* window, u32 index_count, u32 instance_count, u32 first_index, s32 vertex_offset, u32 first_instance)
|
||||
{
|
||||
auto* frame = window_frame(window);
|
||||
let(frame, window_frame(window));
|
||||
vkCmdDrawIndexed(frame->command_buffer, index_count, instance_count, first_index, vertex_offset, first_instance);
|
||||
}
|
||||
|
||||
fn void window_texture_update_begin(RenderWindow* window, BBPipeline pipeline_index, u32 descriptor_count)
|
||||
{
|
||||
auto* pipeline_instantiation = &window->pipeline_instantiations[pipeline_index];
|
||||
let(pipeline_instantiation, &window->pipeline_instantiations[pipeline_index]);
|
||||
assert(descriptor_count <= array_length(pipeline_instantiation->texture_descriptors));
|
||||
|
||||
pipeline_instantiation->descriptor_set_update = (VkWriteDescriptorSet) {
|
||||
@ -2118,14 +2295,14 @@ fn void window_texture_update_begin(RenderWindow* window, BBPipeline pipeline_in
|
||||
};
|
||||
}
|
||||
|
||||
void window_rect_texture_update_begin(RenderWindow* window)
|
||||
fn void window_rect_texture_update_begin(RenderWindow* window)
|
||||
{
|
||||
window_texture_update_begin(window, BB_PIPELINE_RECT, RECT_TEXTURE_SLOT_COUNT);
|
||||
}
|
||||
|
||||
fn void window_queue_pipeline_texture_update(RenderWindow* window, BBPipeline pipeline_index, u32 resource_slot, TextureIndex texture_index)
|
||||
{
|
||||
auto* pipeline_instantiation = &window->pipeline_instantiations[pipeline_index];
|
||||
let(pipeline_instantiation, &window->pipeline_instantiations[pipeline_index]);
|
||||
VkDescriptorImageInfo* descriptor_image = &pipeline_instantiation->texture_descriptors[resource_slot];
|
||||
VulkanTexture* texture = &textures[texture_index.value];
|
||||
*descriptor_image = (VkDescriptorImageInfo) {
|
||||
@ -2135,22 +2312,22 @@ fn void window_queue_pipeline_texture_update(RenderWindow* window, BBPipeline pi
|
||||
};
|
||||
}
|
||||
|
||||
void window_queue_rect_texture_update(RenderWindow* window, RectTextureSlot slot, TextureIndex texture_index)
|
||||
fn void window_queue_rect_texture_update(RenderWindow* window, RectTextureSlot slot, TextureIndex texture_index)
|
||||
{
|
||||
window_queue_pipeline_texture_update(window, BB_PIPELINE_RECT, slot, texture_index);
|
||||
}
|
||||
|
||||
void renderer_queue_font_update(Renderer* renderer, RenderWindow* window, RenderFontType type, TextureAtlas atlas)
|
||||
fn void renderer_queue_font_update(Renderer* renderer, RenderWindow* window, RenderFontType type, TextureAtlas atlas)
|
||||
{
|
||||
static_assert(RECT_TEXTURE_SLOT_MONOSPACE_FONT < RECT_TEXTURE_SLOT_PROPORTIONAL_FONT);
|
||||
auto slot = RECT_TEXTURE_SLOT_MONOSPACE_FONT + type;
|
||||
let(slot, RECT_TEXTURE_SLOT_MONOSPACE_FONT + type);
|
||||
window_queue_rect_texture_update(window, slot, atlas.texture);
|
||||
renderer->fonts[type] = atlas;
|
||||
}
|
||||
|
||||
fn void window_texture_update_end(Renderer* renderer, RenderWindow* window, BBPipeline pipeline_index)
|
||||
{
|
||||
auto* pipeline_instantiation = &window->pipeline_instantiations[pipeline_index];
|
||||
let(pipeline_instantiation, &window->pipeline_instantiations[pipeline_index]);
|
||||
u32 descriptor_copy_count = 0;
|
||||
VkCopyDescriptorSet* descriptor_copies = 0;
|
||||
VkWriteDescriptorSet descriptor_set_writes[] = {
|
||||
@ -2159,41 +2336,41 @@ fn void window_texture_update_end(Renderer* renderer, RenderWindow* window, BBPi
|
||||
vkUpdateDescriptorSets(renderer->device, array_length(descriptor_set_writes), descriptor_set_writes, descriptor_copy_count, descriptor_copies);
|
||||
}
|
||||
|
||||
void window_rect_texture_update_end(Renderer* renderer, RenderWindow* window)
|
||||
fn void window_rect_texture_update_end(Renderer* renderer, RenderWindow* window)
|
||||
{
|
||||
window_texture_update_end(renderer, window, BB_PIPELINE_RECT);
|
||||
}
|
||||
|
||||
u32 window_pipeline_add_vertices(RenderWindow* window, BBPipeline pipeline_index, String vertex_memory, u32 vertex_count)
|
||||
fn u32 window_pipeline_add_vertices(RenderWindow* window, BBPipeline pipeline_index, String vertex_memory, u32 vertex_count)
|
||||
{
|
||||
auto* frame = window_frame(window);
|
||||
auto* vertex_buffer = &frame->pipeline_instantiations[pipeline_index].vertex_buffer;
|
||||
let(frame, window_frame(window));
|
||||
let(vertex_buffer, &frame->pipeline_instantiations[pipeline_index].vertex_buffer);
|
||||
vb_copy_string(&vertex_buffer->cpu, vertex_memory);
|
||||
auto vertex_offset = vertex_buffer->count;
|
||||
let(vertex_offset, vertex_buffer->count);
|
||||
vertex_buffer->count = vertex_offset + vertex_count;
|
||||
return vertex_offset;
|
||||
}
|
||||
|
||||
void window_pipeline_add_indices(RenderWindow* window, BBPipeline pipeline_index, Slice(u32) indices)
|
||||
fn void window_pipeline_add_indices(RenderWindow* window, BBPipeline pipeline_index, Slice(u32) indices)
|
||||
{
|
||||
auto* frame = window_frame(window);
|
||||
auto* index_pointer = vb_add(&frame->pipeline_instantiations[pipeline_index].index_buffer.cpu, indices.length);
|
||||
let(frame, window_frame(window));
|
||||
let(index_pointer, vb_add(&frame->pipeline_instantiations[pipeline_index].index_buffer.cpu, indices.length));
|
||||
memcpy(index_pointer, indices.pointer, indices.length * sizeof(*indices.pointer));
|
||||
}
|
||||
|
||||
void window_render_rect(RenderWindow* window, RectDraw draw)
|
||||
fn void window_render_rect(RenderWindow* window, RectDraw draw)
|
||||
{
|
||||
auto p0 = draw.vertex.p0;
|
||||
auto uv0 = draw.texture.p0;
|
||||
let(p0, draw.vertex.p0);
|
||||
let(uv0, draw.texture.p0);
|
||||
if (draw.texture.p1.x != 0)
|
||||
{
|
||||
assert(draw.texture.p1.x - draw.texture.p0.x == draw.vertex.p1.x - draw.vertex.p0.x);
|
||||
assert(draw.texture.p1.y - draw.texture.p0.y == draw.vertex.p1.y - draw.vertex.p0.y);
|
||||
}
|
||||
|
||||
auto corner_radius = 5.0f;
|
||||
let(corner_radius, 5.0f);
|
||||
|
||||
auto extent = draw.vertex.p1 - p0;
|
||||
let(extent, float2_sub(draw.vertex.p1, p0));
|
||||
RectVertex vertices[] = {
|
||||
(RectVertex) {
|
||||
.p0 = p0,
|
||||
@ -2233,7 +2410,7 @@ void window_render_rect(RenderWindow* window, RectDraw draw)
|
||||
},
|
||||
};
|
||||
|
||||
auto vertex_offset = window_pipeline_add_vertices(window, BB_PIPELINE_RECT, (String)array_to_bytes(vertices), array_length(vertices));
|
||||
let(vertex_offset, window_pipeline_add_vertices(window, BB_PIPELINE_RECT, (String)array_to_bytes(vertices), array_length(vertices)));
|
||||
|
||||
u32 indices[] = {
|
||||
vertex_offset + 0,
|
||||
@ -2248,25 +2425,25 @@ void window_render_rect(RenderWindow* window, RectDraw draw)
|
||||
}
|
||||
|
||||
// TODO: support gradient
|
||||
void window_render_text(Renderer* renderer, RenderWindow* window, String string, float4 color, RenderFontType font_type, u32 x_offset, u32 y_offset)
|
||||
fn void window_render_text(Renderer* renderer, RenderWindow* window, String string, float4 color, RenderFontType font_type, u32 x_offset, u32 y_offset)
|
||||
{
|
||||
auto* texture_atlas = &renderer->fonts[font_type];
|
||||
auto height = texture_atlas->ascent - texture_atlas->descent;
|
||||
auto texture_index = texture_atlas->texture.value;
|
||||
let(texture_atlas, &renderer->fonts[font_type]);
|
||||
let(height, texture_atlas->ascent - texture_atlas->descent);
|
||||
let(texture_index, texture_atlas->texture.value);
|
||||
|
||||
for (u64 i = 0; i < string.length; i += 1)
|
||||
{
|
||||
auto ch = string.pointer[i];
|
||||
auto* character = &texture_atlas->characters[ch];
|
||||
let(ch, string.pointer[i]);
|
||||
let(character, &texture_atlas->characters[ch]);
|
||||
|
||||
auto uv_x = character->x;
|
||||
auto uv_y = character->y;
|
||||
let(uv_x, character->x);
|
||||
let(uv_y, character->y);
|
||||
|
||||
auto char_width = character->width;
|
||||
auto char_height = character->height;
|
||||
let(char_width, character->width);
|
||||
let(char_height, character->height);
|
||||
|
||||
auto pos_x = x_offset;
|
||||
auto pos_y = y_offset + character->y_offset + height + texture_atlas->descent; // Offset of the height to render the character from the bottom (y + height) up (y)
|
||||
let(pos_x, x_offset);
|
||||
let(pos_y, y_offset + character->y_offset + height + texture_atlas->descent); // Offset of the height to render the character from the bottom (y + height) up (y)
|
||||
vec2 p0 = { pos_x, pos_y };
|
||||
vec2 uv0 = { uv_x, uv_y };
|
||||
vec2 extent = { char_width, char_height };
|
||||
@ -2307,7 +2484,7 @@ void window_render_text(Renderer* renderer, RenderWindow* window, String string,
|
||||
},
|
||||
};
|
||||
|
||||
auto vertex_offset = window_pipeline_add_vertices(window, BB_PIPELINE_RECT, (String)array_to_bytes(vertices), array_length(vertices));
|
||||
let(vertex_offset, window_pipeline_add_vertices(window, BB_PIPELINE_RECT, (String)array_to_bytes(vertices), array_length(vertices)));
|
||||
|
||||
u32 indices[] = {
|
||||
vertex_offset + 0,
|
||||
@ -2320,14 +2497,7 @@ void window_render_text(Renderer* renderer, RenderWindow* window, String string,
|
||||
|
||||
window_pipeline_add_indices(window, BB_PIPELINE_RECT, (Slice(u32))array_to_slice(indices));
|
||||
|
||||
auto kerning = (texture_atlas->kerning_tables + ch * 256)[string.pointer[i + 1]];
|
||||
u32 kerning = (texture_atlas->kerning_tables + ch * 256)[string.pointer[i + 1]];
|
||||
x_offset += character->advance + kerning;
|
||||
}
|
||||
}
|
||||
|
||||
uint2 renderer_font_compute_string_rect(Renderer* renderer, RenderFontType type, String string)
|
||||
{
|
||||
auto* texture_atlas = &renderer->fonts[type];
|
||||
auto result = texture_atlas_compute_string_rect(string, texture_atlas);
|
||||
return result;
|
||||
}
|
130
bootstrap/std/vulkan_rendering.h
Normal file
130
bootstrap/std/vulkan_rendering.h
Normal file
@ -0,0 +1,130 @@
|
||||
#pragma once
|
||||
|
||||
#if BB_WINDOWING_BACKEND_X11
|
||||
#define VK_USE_PLATFORM_XCB_KHR
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_COCOA
|
||||
#define VK_USE_PLATFORM_METAL_EXT
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_WIN32
|
||||
#define VK_USE_PLATFORM_WIN32_KHR
|
||||
#endif
|
||||
|
||||
// Code from Volk
|
||||
#ifndef VK_NO_PROTOTYPES
|
||||
#define VK_NO_PROTOTYPES
|
||||
#endif
|
||||
|
||||
#ifndef VULKAN_H_
|
||||
#if defined(VK_USE_PLATFORM_WIN32_KHR)
|
||||
#include <vulkan/vk_platform.h>
|
||||
#include <vulkan/vulkan_core.h>
|
||||
|
||||
/* When VK_USE_PLATFORM_WIN32_KHR is defined, instead of including vulkan.h directly, we include individual parts of the SDK
|
||||
* This is necessary to avoid including <windows.h> which is very heavy - it takes 200ms to parse without WIN32_LEAN_AND_MEAN
|
||||
* and 100ms to parse with it. vulkan_win32.h only needs a few symbols that are easy to redefine ourselves.
|
||||
*/
|
||||
typedef unsigned long DWORD;
|
||||
typedef const wchar_t* LPCWSTR;
|
||||
typedef void* HANDLE;
|
||||
typedef struct HINSTANCE__* HINSTANCE;
|
||||
typedef struct HWND__* HWND;
|
||||
typedef struct HMONITOR__* HMONITOR;
|
||||
typedef struct _SECURITY_ATTRIBUTES SECURITY_ATTRIBUTES;
|
||||
|
||||
#include <vulkan/vulkan_win32.h>
|
||||
|
||||
#ifdef VK_ENABLE_BETA_EXTENSIONS
|
||||
#include <vulkan/vulkan_beta.h>
|
||||
#endif
|
||||
#else
|
||||
#include <vulkan/vulkan.h>
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#define vulkan_function_pointer(n) PFN_ ## n n
|
||||
#define vulkan_global_function_pointer(n) global_variable vulkan_function_pointer(n)
|
||||
// INSTANCE FUNCTIONS START
|
||||
// These functions require no instance
|
||||
vulkan_global_function_pointer(vkGetInstanceProcAddr);
|
||||
vulkan_global_function_pointer(vkEnumerateInstanceVersion);
|
||||
vulkan_global_function_pointer(vkEnumerateInstanceLayerProperties);
|
||||
vulkan_global_function_pointer(vkCreateInstance);
|
||||
|
||||
// These functions require an instance as a parameter
|
||||
vulkan_global_function_pointer(vkGetDeviceProcAddr);
|
||||
vulkan_global_function_pointer(vkCreateDebugUtilsMessengerEXT);
|
||||
vulkan_global_function_pointer(vkEnumeratePhysicalDevices);
|
||||
vulkan_global_function_pointer(vkGetPhysicalDeviceMemoryProperties);
|
||||
vulkan_global_function_pointer(vkGetPhysicalDeviceProperties);
|
||||
vulkan_global_function_pointer(vkGetPhysicalDeviceQueueFamilyProperties);
|
||||
vulkan_global_function_pointer(vkGetPhysicalDeviceSurfaceCapabilitiesKHR);
|
||||
vulkan_global_function_pointer(vkGetPhysicalDeviceSurfacePresentModesKHR);
|
||||
vulkan_global_function_pointer(vkCreateDevice);
|
||||
|
||||
#if defined(VK_KHR_xcb_surface)
|
||||
vulkan_global_function_pointer(vkCreateXcbSurfaceKHR);
|
||||
#endif
|
||||
#if defined(VK_KHR_win32_surface)
|
||||
vulkan_global_function_pointer(vkCreateWin32SurfaceKHR);
|
||||
#endif
|
||||
#if defined(VK_EXT_metal_surface)
|
||||
vulkan_global_function_pointer(vkCreateMetalSurfaceEXT);
|
||||
#endif
|
||||
// INSTANCE FUNCTIONS END
|
||||
|
||||
vulkan_global_function_pointer(vkCreateSwapchainKHR);
|
||||
vulkan_global_function_pointer(vkCmdCopyBuffer2);
|
||||
vulkan_global_function_pointer(vkAllocateMemory);
|
||||
vulkan_global_function_pointer(vkCreateBuffer);
|
||||
vulkan_global_function_pointer(vkGetBufferMemoryRequirements);
|
||||
vulkan_global_function_pointer(vkBindBufferMemory);
|
||||
vulkan_global_function_pointer(vkMapMemory);
|
||||
vulkan_global_function_pointer(vkGetBufferDeviceAddress);
|
||||
vulkan_global_function_pointer(vkResetFences);
|
||||
vulkan_global_function_pointer(vkResetCommandBuffer);
|
||||
vulkan_global_function_pointer(vkBeginCommandBuffer);
|
||||
vulkan_global_function_pointer(vkEndCommandBuffer);
|
||||
vulkan_global_function_pointer(vkQueueSubmit2);
|
||||
vulkan_global_function_pointer(vkWaitForFences);
|
||||
vulkan_global_function_pointer(vkCreateImage);
|
||||
vulkan_global_function_pointer(vkGetImageMemoryRequirements);
|
||||
vulkan_global_function_pointer(vkBindImageMemory);
|
||||
vulkan_global_function_pointer(vkCreateImageView);
|
||||
vulkan_global_function_pointer(vkCmdPipelineBarrier2);
|
||||
vulkan_global_function_pointer(vkCmdBlitImage2);
|
||||
vulkan_global_function_pointer(vkGetDeviceQueue);
|
||||
vulkan_global_function_pointer(vkCreateCommandPool);
|
||||
vulkan_global_function_pointer(vkAllocateCommandBuffers);
|
||||
vulkan_global_function_pointer(vkCreateFence);
|
||||
vulkan_global_function_pointer(vkCreateSampler);
|
||||
vulkan_global_function_pointer(vkCreateShaderModule);
|
||||
vulkan_global_function_pointer(vkCreateDescriptorSetLayout);
|
||||
vulkan_global_function_pointer(vkCreatePipelineLayout);
|
||||
vulkan_global_function_pointer(vkCreateGraphicsPipelines);
|
||||
vulkan_global_function_pointer(vkDestroyImageView);
|
||||
vulkan_global_function_pointer(vkDestroyImage);
|
||||
vulkan_global_function_pointer(vkFreeMemory);
|
||||
vulkan_global_function_pointer(vkDeviceWaitIdle);
|
||||
vulkan_global_function_pointer(vkDestroySwapchainKHR);
|
||||
vulkan_global_function_pointer(vkGetSwapchainImagesKHR);
|
||||
vulkan_global_function_pointer(vkCreateDescriptorPool);
|
||||
vulkan_global_function_pointer(vkAllocateDescriptorSets);
|
||||
vulkan_global_function_pointer(vkCreateSemaphore);
|
||||
vulkan_global_function_pointer(vkAcquireNextImageKHR);
|
||||
vulkan_global_function_pointer(vkDestroyBuffer);
|
||||
vulkan_global_function_pointer(vkUnmapMemory);
|
||||
vulkan_global_function_pointer(vkCmdSetViewport);
|
||||
vulkan_global_function_pointer(vkCmdSetScissor);
|
||||
vulkan_global_function_pointer(vkCmdBeginRendering);
|
||||
vulkan_global_function_pointer(vkCmdBindPipeline);
|
||||
vulkan_global_function_pointer(vkCmdBindDescriptorSets);
|
||||
vulkan_global_function_pointer(vkCmdBindIndexBuffer);
|
||||
vulkan_global_function_pointer(vkCmdPushConstants);
|
||||
vulkan_global_function_pointer(vkCmdDrawIndexed);
|
||||
vulkan_global_function_pointer(vkCmdEndRendering);
|
||||
vulkan_global_function_pointer(vkQueuePresentKHR);
|
||||
vulkan_global_function_pointer(vkCmdCopyBufferToImage);
|
||||
vulkan_global_function_pointer(vkUpdateDescriptorSets);
|
1
bootstrap/std/wayland_windowing.c
Normal file
1
bootstrap/std/wayland_windowing.c
Normal file
@ -0,0 +1 @@
|
||||
#pragma once
|
1
bootstrap/std/wayland_windowing.h
Normal file
1
bootstrap/std/wayland_windowing.h
Normal file
@ -0,0 +1 @@
|
||||
#pragma once
|
59
bootstrap/std/win32_windowing.c
Normal file
59
bootstrap/std/win32_windowing.c
Normal file
@ -0,0 +1,59 @@
|
||||
#pragma once
|
||||
|
||||
#pragma comment(lib, "user32")
|
||||
|
||||
fn LRESULT window_callback(HWND window, UINT message, WPARAM w_parameter, LPARAM l_parameter)
|
||||
{
|
||||
return DefWindowProcW(window, message, w_parameter, l_parameter);
|
||||
}
|
||||
|
||||
fn u8 windowing_initialize()
|
||||
{
|
||||
HINSTANCE instance = GetModuleHandleW(0);
|
||||
windowing_connection.instance = instance;
|
||||
WNDCLASSEXW window_class = {
|
||||
.cbSize = sizeof(window_class),
|
||||
.lpfnWndProc = window_callback,
|
||||
.hInstance = instance,
|
||||
.lpszClassName = L"window",
|
||||
.hCursor = LoadCursorA(0, IDC_ARROW),
|
||||
.hIcon = LoadIcon(instance, MAKEINTRESOURCE(1)),
|
||||
.style = CS_VREDRAW|CS_HREDRAW,
|
||||
};
|
||||
RegisterClassExW(&window_class);
|
||||
windowing_connection.window_class = window_class;
|
||||
return 1;
|
||||
}
|
||||
|
||||
fn WindowingInstance* windowing_instantiate(WindowingInstantiate instantiate)
|
||||
{
|
||||
// TODO:
|
||||
WindowingInstance* window = &windowing_instances[0];
|
||||
window->handle = CreateWindowExW(WS_EX_APPWINDOW, L"window", L"Bloat Buster", WS_OVERLAPPEDWINDOW | WS_SIZEBOX, CW_USEDEFAULT, CW_USEDEFAULT, instantiate.size.width, instantiate.size.height, 0, 0, windowing_connection.instance, 0);
|
||||
ShowWindow(window->handle, SW_SHOW);
|
||||
return window;
|
||||
}
|
||||
|
||||
fn WindowingSize windowing_get_instance_framebuffer_size(WindowingInstance* instance)
|
||||
{
|
||||
RECT area;
|
||||
GetClientRect(instance->handle, &area);
|
||||
WindowingSize size = {
|
||||
.width = area.right,
|
||||
.height = area.bottom,
|
||||
};
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
fn void windowing_poll_events()
|
||||
{
|
||||
MSG msg;
|
||||
HWND handle;
|
||||
|
||||
while (PeekMessageW(&msg, NULL, 0, 0, PM_REMOVE))
|
||||
{
|
||||
TranslateMessage(&msg);
|
||||
DispatchMessageW(&msg);
|
||||
}
|
||||
}
|
13
bootstrap/std/win32_windowing.h
Normal file
13
bootstrap/std/win32_windowing.h
Normal file
@ -0,0 +1,13 @@
|
||||
#pragma once
|
||||
|
||||
STRUCT(WindowingConnection)
|
||||
{
|
||||
HINSTANCE instance;
|
||||
WNDCLASSEXW window_class;
|
||||
};
|
||||
|
||||
STRUCT(WindowingInstance)
|
||||
{
|
||||
HWND handle;
|
||||
};
|
||||
|
@ -1,434 +0,0 @@
|
||||
#include <std/window.h>
|
||||
|
||||
#ifndef __APPLE__
|
||||
#include <volk.h>
|
||||
#endif
|
||||
#define GLFW_INCLUDE_NONE
|
||||
#include <GLFW/glfw3.h>
|
||||
#include <GLFW/glfw3native.h>
|
||||
|
||||
global_variable OSWindowingCallbacks callbacks;
|
||||
|
||||
// TODO: thread local
|
||||
global_variable OSEventQueue* event_queue = 0;
|
||||
fn void monitor_callback(GLFWmonitor* monitor, int event)
|
||||
{
|
||||
unused(monitor);
|
||||
unused(event);
|
||||
todo();
|
||||
}
|
||||
|
||||
fn void joystick_callback(int joystick_id, int event)
|
||||
{
|
||||
unused(joystick_id);
|
||||
unused(event);
|
||||
todo();
|
||||
}
|
||||
|
||||
fn void bitset_list_add(VirtualBuffer(OSEventBitset)* list, u32* counter, u64 value)
|
||||
{
|
||||
auto event_index = *counter;
|
||||
if (unlikely(event_index % OS_EVENT_BITSET_SIZE == 0))
|
||||
{
|
||||
*vb_add(list, 1) = (OSEventBitset) {
|
||||
.value = 0,
|
||||
};
|
||||
}
|
||||
|
||||
auto bitset_index = event_index / OS_EVENT_BITSET_SIZE;
|
||||
u64 bit_index = event_index % OS_EVENT_BITSET_SIZE;
|
||||
list->pointer[bitset_index].value |= (value << bit_index);
|
||||
}
|
||||
|
||||
void os_windowing_init(OSWindowingInitializationOptions options)
|
||||
{
|
||||
#ifdef __linux__
|
||||
int platform_hint = options.should_use_x11 ? GLFW_PLATFORM_X11 : GLFW_PLATFORM_WAYLAND;
|
||||
glfwInitHint(GLFW_PLATFORM, platform_hint);
|
||||
#endif
|
||||
|
||||
if (glfwInit() != GLFW_TRUE)
|
||||
{
|
||||
failed_execution();
|
||||
}
|
||||
|
||||
glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API);
|
||||
// From GLFW documentation:
|
||||
// This is called when a monitor is connected to or disconnected from the system.
|
||||
glfwSetMonitorCallback(&monitor_callback);
|
||||
glfwSetJoystickCallback(&joystick_callback);
|
||||
|
||||
callbacks = options.callback;
|
||||
}
|
||||
|
||||
fn void glfw_window_drop_callback(GLFWwindow* window, int path_count, const char* paths[])
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
auto* drop_callback = callbacks.window_drop;
|
||||
print("DROP\n");
|
||||
if (drop_callback)
|
||||
{
|
||||
drop_callback(window, context, (CStringSlice) { .pointer = (char**)paths, .length = path_count });
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_scroll_callback(GLFWwindow* window, double x, double y)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
auto* scroll_callback = callbacks.window_scroll;
|
||||
print("SCROLL\n");
|
||||
if (scroll_callback)
|
||||
{
|
||||
scroll_callback(window, context, x, y);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_cursor_enter_callback(GLFWwindow* window, int entered)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
auto* cursor_enter_callback = callbacks.window_cursor_enter;
|
||||
print("CURSOR_ENTER: {u32}\n", entered);
|
||||
|
||||
auto event_index = event_queue->cursor_enter_count;
|
||||
*vb_add(&event_queue->descriptors, 1) = (OSEventDescriptor) {
|
||||
.type = OS_EVENT_TYPE_WINDOW_FOCUS,
|
||||
.index = event_index,
|
||||
};
|
||||
|
||||
bitset_list_add(&event_queue->cursor_enters, &event_queue->cursor_enter_count, entered);
|
||||
|
||||
if (cursor_enter_callback)
|
||||
{
|
||||
cursor_enter_callback(window, context, entered);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_cursor_position_callback(GLFWwindow* window, double x, double y)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
auto* cursor_position_callback = callbacks.window_cursor_position;
|
||||
*vb_add(&event_queue->descriptors, 1) = (OSEventDescriptor) {
|
||||
.index = event_queue->cursor_positions.length,
|
||||
.type = OS_EVENT_TYPE_CURSOR_POSITION,
|
||||
};
|
||||
*vb_add(&event_queue->cursor_positions, 1) = (OSEventCursorPosition) {
|
||||
.x = x,
|
||||
.y = y,
|
||||
};
|
||||
|
||||
if (cursor_position_callback)
|
||||
{
|
||||
cursor_position_callback(window, context, x, y);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_mouse_button_callback(GLFWwindow* window, int button, int action, int mods)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
auto* mouse_button_callback = callbacks.window_mouse_button;
|
||||
*vb_add(&event_queue->descriptors, 1) = (OSEventDescriptor) {
|
||||
.index = event_queue->mouse_buttons.length,
|
||||
.type = OS_EVENT_TYPE_MOUSE_BUTTON,
|
||||
};
|
||||
print("Button: {u32:x}. Action: {u32:x}. Mods: {u32:x}\n", button, action, mods);
|
||||
|
||||
OSEventMouseButtonAction os_action;
|
||||
switch (action)
|
||||
{
|
||||
case GLFW_RELEASE: os_action = OS_EVENT_MOUSE_RELEASE; break;
|
||||
case GLFW_PRESS: os_action = OS_EVENT_MOUSE_PRESS; break;
|
||||
case GLFW_REPEAT: os_action = OS_EVENT_MOUSE_REPEAT; break;
|
||||
default: unreachable();
|
||||
}
|
||||
|
||||
OSEventMouseButtonKind mouse_button = 0;
|
||||
switch (button)
|
||||
{
|
||||
case GLFW_MOUSE_BUTTON_1:
|
||||
mouse_button = OS_EVENT_MOUSE_BUTTON_1;
|
||||
break;
|
||||
case GLFW_MOUSE_BUTTON_2:
|
||||
mouse_button = OS_EVENT_MOUSE_BUTTON_2;
|
||||
break;
|
||||
case GLFW_MOUSE_BUTTON_3:
|
||||
mouse_button = OS_EVENT_MOUSE_BUTTON_3;
|
||||
break;
|
||||
case GLFW_MOUSE_BUTTON_4:
|
||||
mouse_button = OS_EVENT_MOUSE_BUTTON_4;
|
||||
break;
|
||||
case GLFW_MOUSE_BUTTON_5:
|
||||
mouse_button = OS_EVENT_MOUSE_BUTTON_5;
|
||||
break;
|
||||
case GLFW_MOUSE_BUTTON_6:
|
||||
mouse_button = OS_EVENT_MOUSE_BUTTON_6;
|
||||
break;
|
||||
case GLFW_MOUSE_BUTTON_7:
|
||||
mouse_button = OS_EVENT_MOUSE_BUTTON_7;
|
||||
break;
|
||||
case GLFW_MOUSE_BUTTON_8:
|
||||
mouse_button = OS_EVENT_MOUSE_BUTTON_8;
|
||||
break;
|
||||
}
|
||||
|
||||
*vb_add(&event_queue->mouse_buttons, 1) = (OSEventMouseButton) {
|
||||
.button = mouse_button,
|
||||
.event = (OSEventMouseButtonEvent){
|
||||
.action = os_action,
|
||||
.mod_shift = !!(mods & GLFW_MOD_SHIFT),
|
||||
.mod_control = !!(mods & GLFW_MOD_CONTROL),
|
||||
.mod_alt = !!(mods & GLFW_MOD_ALT),
|
||||
.mod_super = !!(mods & GLFW_MOD_SUPER),
|
||||
.mod_caps_lock = !!(mods & GLFW_MOD_CAPS_LOCK),
|
||||
.mod_num_lock = !!(mods & GLFW_MOD_NUM_LOCK),
|
||||
},
|
||||
};
|
||||
|
||||
if (mouse_button_callback)
|
||||
{
|
||||
mouse_button_callback(window, context, button, action, mods);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_character_modifier_callback(GLFWwindow* window, unsigned int codepoint, int mods)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
auto* character_modifier_callback = callbacks.window_character_modifier;
|
||||
// print("CHAR_MODIFIER. Codepoint: {u32}. Mods: {u32}\n", codepoint, mods);
|
||||
if (character_modifier_callback)
|
||||
{
|
||||
character_modifier_callback(window, context, codepoint, mods);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_character_callback(GLFWwindow* window, unsigned int codepoint)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
auto* character_callback = callbacks.window_character;
|
||||
// print("CHAR. Codepoint: {u32}\n", codepoint);
|
||||
if (character_callback)
|
||||
{
|
||||
character_callback(window, context, codepoint);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_key_callback(GLFWwindow* window, int key, int scancode, int action, int mods)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
auto* key_callback = callbacks.window_key;
|
||||
print("Key: {u32}. Scancode: {u32}. Action: {u32}. Mods: {u32}\n", key, scancode, action, mods);
|
||||
if (key_callback)
|
||||
{
|
||||
key_callback(window, context, key, scancode, action, mods);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_content_scale_callback(GLFWwindow* window, float x, float y)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
print("CONTENT_SCALE\n");
|
||||
auto* content_scale_callback = callbacks.window_content_scale;
|
||||
if (content_scale_callback)
|
||||
{
|
||||
content_scale_callback(window, context, x, y);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_maximize_callback(GLFWwindow* window, int maximized)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
print("MAXIMIZE\n");
|
||||
auto* maximize_callback = callbacks.window_maximize;
|
||||
if (maximize_callback)
|
||||
{
|
||||
maximize_callback(window, context, maximized);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_iconify_callback(GLFWwindow* window, int iconified)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
print("ICONIFY\n");
|
||||
auto* iconify_callback = callbacks.window_iconify;
|
||||
if (iconify_callback)
|
||||
{
|
||||
iconify_callback(window, context, iconified);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_focus_callback(GLFWwindow* window, int focused)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
print("FOCUS\n");
|
||||
auto* focus_callback = callbacks.window_focus;
|
||||
auto event_index = event_queue->window_focuses_count;
|
||||
*vb_add(&event_queue->descriptors, 1) = (OSEventDescriptor) {
|
||||
.type = OS_EVENT_TYPE_WINDOW_FOCUS,
|
||||
.index = event_index,
|
||||
};
|
||||
|
||||
bitset_list_add(&event_queue->window_focuses, &event_queue->window_focuses_count, focused);
|
||||
|
||||
if (focus_callback)
|
||||
{
|
||||
focus_callback(window, context, focused);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_close_callback(GLFWwindow* window)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
*vb_add(&event_queue->descriptors, 1) = (OSEventDescriptor) {
|
||||
.type = OS_EVENT_TYPE_WINDOW_CLOSE,
|
||||
};
|
||||
|
||||
auto* close_callback = callbacks.window_close;
|
||||
if (close_callback)
|
||||
{
|
||||
close_callback(window, context);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_position_callback(GLFWwindow* window, int x, int y)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
print("WINDOW_POSITION: {u32}x{u32}\n", x, y);
|
||||
auto* position_callback = callbacks.window_position;
|
||||
|
||||
*vb_add(&event_queue->descriptors, 1) = (OSEventDescriptor) {
|
||||
.index = event_queue->window_positions.length,
|
||||
.type = OS_EVENT_TYPE_WINDOW_POSITION,
|
||||
};
|
||||
|
||||
*vb_add(&event_queue->window_positions, 1) = (OSEventWindowPosition) {
|
||||
.x = x,
|
||||
.y = y,
|
||||
};
|
||||
|
||||
if (position_callback)
|
||||
{
|
||||
position_callback(window, context, x, y);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_size_callback(GLFWwindow* window, int width, int height)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
print("WINDOW_SIZE\n");
|
||||
auto* window_resize_callback = callbacks.window_resize;
|
||||
if (window_resize_callback)
|
||||
{
|
||||
window_resize_callback(window, context, width, height);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_framebuffer_size_callback(GLFWwindow* window, int width, int height)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(window);
|
||||
print("FRAMEBUFFER_SIZE\n");
|
||||
auto* framebuffer_resize_callback = callbacks.framebuffer_resize;
|
||||
if (framebuffer_resize_callback)
|
||||
{
|
||||
framebuffer_resize_callback(window, context, width, height);
|
||||
}
|
||||
}
|
||||
|
||||
fn void glfw_window_refresh_callback(GLFWwindow* w)
|
||||
{
|
||||
void* context = glfwGetWindowUserPointer(w);
|
||||
print("REFRESH\n");
|
||||
auto refresh_callback = callbacks.window_refresh;
|
||||
if (refresh_callback)
|
||||
{
|
||||
refresh_callback(w, context);
|
||||
}
|
||||
}
|
||||
|
||||
OSWindow os_window_create(OSWindowCreate create)
|
||||
{
|
||||
// glfwWindowHint(GLFW_MAXIMIZED, GLFW_TRUE);
|
||||
|
||||
GLFWmonitor* monitor = 0;
|
||||
GLFWwindow* share = 0;
|
||||
GLFWwindow* window = glfwCreateWindow(create.size.width, create.size.height, string_to_c(create.name), monitor, share);
|
||||
|
||||
glfwSetWindowUserPointer(window, create.context);
|
||||
|
||||
glfwSetWindowPosCallback(window, &glfw_window_position_callback);
|
||||
glfwSetWindowSizeCallback(window, &glfw_window_size_callback);
|
||||
glfwSetWindowCloseCallback(window, &glfw_window_close_callback);
|
||||
glfwSetWindowFocusCallback(window, &glfw_window_focus_callback);
|
||||
glfwSetWindowIconifyCallback(window, &glfw_window_iconify_callback); // Minimize callback
|
||||
glfwSetWindowMaximizeCallback(window, &glfw_window_maximize_callback);
|
||||
glfwSetFramebufferSizeCallback(window, &glfw_framebuffer_size_callback);
|
||||
glfwSetWindowRefreshCallback(window, &glfw_window_refresh_callback);
|
||||
glfwSetWindowContentScaleCallback(window, &glfw_window_content_scale_callback);
|
||||
glfwSetKeyCallback(window, &glfw_window_key_callback);
|
||||
glfwSetCharCallback(window, &glfw_window_character_callback);
|
||||
glfwSetCharModsCallback(window, &glfw_window_character_modifier_callback);
|
||||
glfwSetMouseButtonCallback(window, &glfw_window_mouse_button_callback);
|
||||
glfwSetCursorPosCallback(window, &glfw_window_cursor_position_callback);
|
||||
glfwSetCursorEnterCallback(window, &glfw_window_cursor_enter_callback);
|
||||
glfwSetScrollCallback(window, &glfw_window_scroll_callback);
|
||||
glfwSetDropCallback(window, &glfw_window_drop_callback);
|
||||
|
||||
return window;
|
||||
}
|
||||
|
||||
u8 os_window_should_close(OSWindow window)
|
||||
{
|
||||
return glfwWindowShouldClose(window);
|
||||
}
|
||||
|
||||
fn void os_event_queue_reset(OSEventQueue* queue)
|
||||
{
|
||||
queue->descriptors.length = 0;
|
||||
queue->mouse_buttons.length = 0;
|
||||
queue->cursor_positions.length = 0;
|
||||
}
|
||||
|
||||
fn u8 os_event_bitset_list(VirtualBuffer(OSEventBitset) bitset, u32 index)
|
||||
{
|
||||
auto bitset_index = index / bitset.length;
|
||||
auto bit_index = index % bitset.length;
|
||||
return !!(bitset.pointer[bitset_index].value & bit_index);
|
||||
}
|
||||
|
||||
u8 os_event_queue_get_window_focus(OSEventQueue* queue, u32 index)
|
||||
{
|
||||
assert(index < queue->window_focuses.length);
|
||||
auto result = os_event_bitset_list(queue->window_focuses, index);
|
||||
return result;
|
||||
}
|
||||
|
||||
void os_poll_events(OSEventQueue* queue)
|
||||
{
|
||||
os_event_queue_reset(queue);
|
||||
event_queue = queue;
|
||||
assert(queue->descriptors.length == 0);
|
||||
glfwPollEvents();
|
||||
}
|
||||
|
||||
OSWindowSize os_window_framebuffer_size_get(OSWindow window)
|
||||
{
|
||||
int width;
|
||||
int height;
|
||||
glfwGetFramebufferSize(window, &width, &height);
|
||||
return (OSWindowSize)
|
||||
{
|
||||
.width = width,
|
||||
.height = height,
|
||||
};
|
||||
}
|
||||
|
||||
OSCursorPosition os_window_cursor_position_get(OSWindow window)
|
||||
{
|
||||
OSCursorPosition result;
|
||||
glfwGetCursorPos(window, &result.x, &result.y);
|
||||
return result;
|
||||
}
|
||||
|
||||
int window_create_surface(void* instance, OSWindow window, const void* allocator, void** surface)
|
||||
{
|
||||
return glfwCreateWindowSurface(instance, window, allocator, (VkSurfaceKHR*)surface);
|
||||
}
|
||||
|
15
bootstrap/std/windowing.c
Normal file
15
bootstrap/std/windowing.c
Normal file
@ -0,0 +1,15 @@
|
||||
#if BB_WINDOWING_BACKEND_X11
|
||||
#include <std/x11_windowing.c>
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_WAYLAND
|
||||
#include <std/wayland_windowing.c>
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_COCOA
|
||||
#include <std/cocoa_windowing.c>
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_WIN32
|
||||
#include <std/win32_windowing.c>
|
||||
#endif
|
206
bootstrap/std/windowing.h
Normal file
206
bootstrap/std/windowing.h
Normal file
@ -0,0 +1,206 @@
|
||||
#pragma once
|
||||
|
||||
#if BB_WINDOWING_BACKEND_X11
|
||||
#include <std/x11_windowing.h>
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_WAYLAND
|
||||
#include <std/wayland_windowing.h>
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_COCOA
|
||||
#include <std/cocoa_windowing.h>
|
||||
#endif
|
||||
|
||||
#if BB_WINDOWING_BACKEND_WIN32
|
||||
#include <std/win32_windowing.h>
|
||||
#endif
|
||||
|
||||
typedef enum WindowingEventType
|
||||
{
|
||||
WINDOWING_EVENT_TYPE_MOUSE_BUTTON,
|
||||
WINDOWING_EVENT_TYPE_CURSOR_POSITION,
|
||||
WINDOWING_EVENT_TYPE_CURSOR_ENTER,
|
||||
WINDOWING_EVENT_TYPE_WINDOW_FOCUS,
|
||||
WINDOWING_EVENT_TYPE_WINDOW_POSITION,
|
||||
WINDOWING_EVENT_TYPE_WINDOW_CLOSE,
|
||||
} WindowingEventType;
|
||||
|
||||
STRUCT(WindowingEventDescriptor)
|
||||
{
|
||||
u32 index:24;
|
||||
WindowingEventType type:8;
|
||||
};
|
||||
static_assert(sizeof(WindowingEventDescriptor) == 4);
|
||||
decl_vb(WindowingEventDescriptor);
|
||||
|
||||
ENUM_START(WindowingEventMouseButtonKind, u8)
|
||||
{
|
||||
WINDOWING_EVENT_MOUSE_BUTTON_1 = 0,
|
||||
WINDOWING_EVENT_MOUSE_BUTTON_2 = 1,
|
||||
WINDOWING_EVENT_MOUSE_BUTTON_3 = 2,
|
||||
WINDOWING_EVENT_MOUSE_BUTTON_4 = 3,
|
||||
WINDOWING_EVENT_MOUSE_BUTTON_5 = 4,
|
||||
WINDOWING_EVENT_MOUSE_BUTTON_6 = 5,
|
||||
WINDOWING_EVENT_MOUSE_BUTTON_7 = 6,
|
||||
WINDOWING_EVENT_MOUSE_BUTTON_8 = 7,
|
||||
WINDOWING_EVENT_MOUSE_LEFT = WINDOWING_EVENT_MOUSE_BUTTON_1,
|
||||
WINDOWING_EVENT_MOUSE_RIGHT = WINDOWING_EVENT_MOUSE_BUTTON_2,
|
||||
WINDOWING_EVENT_MOUSE_MIDDLE = WINDOWING_EVENT_MOUSE_BUTTON_3,
|
||||
}
|
||||
ENUM_END(WindowingEventMouseButtonKind);
|
||||
#define WINDOWING_EVENT_MOUSE_BUTTON_COUNT (WINDOWING_EVENT_MOUSE_BUTTON_8 + 1)
|
||||
|
||||
ENUM_START(WindowingEventMouseButtonAction, u8)
|
||||
{
|
||||
WINDOWING_EVENT_MOUSE_RELAX = 0,
|
||||
WINDOWING_EVENT_MOUSE_RELEASE = 1,
|
||||
WINDOWING_EVENT_MOUSE_PRESS = 2,
|
||||
WINDOWING_EVENT_MOUSE_REPEAT = 3,
|
||||
} ENUM_END(WindowingEventMouseButtonAction);
|
||||
|
||||
STRUCT(WindowingEventMouseButtonEvent)
|
||||
{
|
||||
WindowingEventMouseButtonAction action:2;
|
||||
u8 mod_shift:1;
|
||||
u8 mod_control:1;
|
||||
u8 mod_alt:1;
|
||||
u8 mod_super:1;
|
||||
u8 mod_caps_lock:1;
|
||||
u8 mod_num_lock:1;
|
||||
};
|
||||
|
||||
STRUCT(WindowingEventMouseButton)
|
||||
{
|
||||
WindowingEventMouseButtonKind button:3;
|
||||
u8 reserved:5;
|
||||
WindowingEventMouseButtonEvent event;
|
||||
};
|
||||
static_assert(sizeof(WindowingEventMouseButton) == sizeof(u16));
|
||||
decl_vb(WindowingEventMouseButton);
|
||||
|
||||
#define WINDOWING_EVENT_BITSET_SIZE (64)
|
||||
STRUCT(WindowingEventBitset)
|
||||
{
|
||||
u64 value;
|
||||
};
|
||||
decl_vb(WindowingEventBitset);
|
||||
|
||||
STRUCT(WindowingEventCursorPosition)
|
||||
{
|
||||
f64 x;
|
||||
f64 y;
|
||||
};
|
||||
decl_vb(WindowingEventCursorPosition);
|
||||
|
||||
STRUCT(WindowingEventWindowPosition)
|
||||
{
|
||||
u32 x;
|
||||
u32 y;
|
||||
};
|
||||
decl_vb(WindowingEventWindowPosition);
|
||||
|
||||
STRUCT(WindowingEventQueue)
|
||||
{
|
||||
VirtualBuffer(WindowingEventDescriptor) descriptors;
|
||||
VirtualBuffer(WindowingEventMouseButton) mouse_buttons;
|
||||
VirtualBuffer(WindowingEventBitset) window_focuses;
|
||||
u32 window_focuses_count;
|
||||
u32 cursor_enter_count;
|
||||
VirtualBuffer(WindowingEventBitset) cursor_enters;
|
||||
VirtualBuffer(WindowingEventCursorPosition) cursor_positions;
|
||||
VirtualBuffer(WindowingEventWindowPosition) window_positions;
|
||||
};
|
||||
|
||||
// typedef void OSFramebufferResize(OSWindow window, void* context, u32 width, u32 height);
|
||||
// typedef void OSWindowResize(OSWindow window, void* context, u32 width, u32 height);
|
||||
// typedef void OSWindowRefresh(OSWindow window, void* context);
|
||||
// typedef void OSWindowPosition(OSWindow window, void* context, u32 x, u32 y);
|
||||
// typedef void OSWindowClose(OSWindow window, void* context);
|
||||
// typedef void OSWindowFocus(OSWindow window, void* context, u8 focused);
|
||||
// typedef void OSWindowIconify(OSWindow window, void* context, u8 iconified);
|
||||
// typedef void OSWindowMaximize(OSWindow window, void* context, u8 maximized);
|
||||
// typedef void OSWindowContentScale(OSWindow window, void* context, f32 x, f32 y);
|
||||
// typedef void OSWindowKey(OSWindow window, void* context, s32 key, s32 scancode, s32 action, s32 mods);
|
||||
// typedef void OSWindowCharacter(OSWindow window, void* context, u32 codepoint);
|
||||
// typedef void OSWindowCharacterModifier(OSWindow window, void* context, u32 codepoint, s32 mods);
|
||||
// typedef void OSWindowMouseButton(OSWindow window, void* context, s32 button, s32 action, s32 mods);
|
||||
// typedef void OSWindowCursorPosition(OSWindow window, void* context, f64 x, f64 y);
|
||||
// typedef void OSWindowCursorEnter(OSWindow window, void* context, u8 entered);
|
||||
// typedef void OSWindowScroll(OSWindow window, void* context, f64 x, f64 y);
|
||||
// typedef void OSWindowDrop(OSWindow window, void* context, CStringSlice paths);
|
||||
|
||||
// STRUCT(OSWindowingCallbacks)
|
||||
// {
|
||||
// // OSFramebufferResize* framebuffer_resize;
|
||||
// // OSWindowResize* window_resize;
|
||||
// // OSWindowRefresh* window_refresh;
|
||||
// // OSWindowPosition* window_position;
|
||||
// // OSWindowClose* window_close;
|
||||
// // OSWindowFocus* window_focus;
|
||||
// // OSWindowIconify* window_iconify;
|
||||
// // OSWindowMaximize* window_maximize;
|
||||
// // OSWindowContentScale* window_content_scale;
|
||||
// // OSWindowKey* window_key;
|
||||
// // OSWindowCharacter* window_character;
|
||||
// // OSWindowCharacterModifier* window_character_modifier;
|
||||
// // OSWindowMouseButton* window_mouse_button;
|
||||
// // OSWindowCursorPosition* window_cursor_position;
|
||||
// // OSWindowCursorEnter* window_cursor_enter;
|
||||
// // OSWindowScroll* window_scroll;
|
||||
// // OSWindowDrop* window_drop;
|
||||
// };
|
||||
|
||||
// STRUCT(OSWindowCreate)
|
||||
// {
|
||||
// String name;
|
||||
// OSWindowSize size;
|
||||
// void* context;
|
||||
// // OSWindowResize* resize_callback;
|
||||
// // OSWindowRefresh* refresh_callback;
|
||||
// };
|
||||
|
||||
STRUCT(WindowingCursorPosition)
|
||||
{
|
||||
f64 x;
|
||||
f64 y;
|
||||
};
|
||||
|
||||
// NEW API START
|
||||
STRUCT(WindowingOffset)
|
||||
{
|
||||
u32 x;
|
||||
u32 y;
|
||||
};
|
||||
|
||||
STRUCT(WindowingSize)
|
||||
{
|
||||
u32 width;
|
||||
u32 height;
|
||||
};
|
||||
|
||||
STRUCT(WindowingInstantiate)
|
||||
{
|
||||
String name;
|
||||
WindowingOffset offset;
|
||||
WindowingSize size;
|
||||
void* context;
|
||||
};
|
||||
|
||||
fn u8 windowing_initialize();
|
||||
fn WindowingInstance* windowing_instantiate(WindowingInstantiate instantiate);
|
||||
fn void windowing_poll_events();
|
||||
fn WindowingSize windowing_get_instance_framebuffer_size(WindowingInstance* window);
|
||||
// NEW API END
|
||||
|
||||
|
||||
// fn OSWindow os_window_create(OSWindowCreate create);
|
||||
// fn u8 os_window_should_close(OSWindow window);
|
||||
// fn OSCursorPosition os_window_cursor_position_get(OSWindow window);
|
||||
//
|
||||
// fn u8 os_event_queue_get_window_focus(OSEventQueue* queue, u32 index);
|
||||
|
||||
#ifndef __APPLE__
|
||||
global_variable WindowingConnection windowing_connection;
|
||||
global_variable WindowingInstance windowing_instances[256];
|
||||
#endif
|
197
bootstrap/std/x11_windowing.c
Normal file
197
bootstrap/std/x11_windowing.c
Normal file
@ -0,0 +1,197 @@
|
||||
#pragma once
|
||||
|
||||
global_variable xcb_window_t windowing_instance_handles[256];
|
||||
|
||||
typedef enum WindowingEvent : u32
|
||||
{
|
||||
WINDOWING_EVENT_CLOSE,
|
||||
WINDOWING_EVENT_COUNT,
|
||||
} WindowingEvent;
|
||||
|
||||
fn xcb_connection_t* xcb_connection_get()
|
||||
{
|
||||
return windowing_connection.handle;
|
||||
}
|
||||
|
||||
fn xcb_window_t xcb_window_from_windowing_instance(WindowingInstance* instance)
|
||||
{
|
||||
return instance->handle;
|
||||
}
|
||||
|
||||
fn void x11_intern_atoms(u32 atom_count, String* names, xcb_intern_atom_cookie_t* cookies, xcb_intern_atom_reply_t** replies)
|
||||
{
|
||||
xcb_connection_t* connection = windowing_connection.handle;
|
||||
|
||||
for (u64 i = 0; i < atom_count; i += 1)
|
||||
{
|
||||
String atom_name = names[i];
|
||||
|
||||
cookies[i] = xcb_intern_atom(connection, 0, atom_name.length, string_to_c(atom_name));
|
||||
}
|
||||
|
||||
for (u64 i = 0; i < atom_count; i += 1)
|
||||
{
|
||||
replies[i] = xcb_intern_atom_reply(connection, cookies[i], 0);
|
||||
}
|
||||
}
|
||||
|
||||
typedef enum X11Atom
|
||||
{
|
||||
X11_ATOM_WM_PROTOCOLS,
|
||||
X11_ATOM_WM_DELETE_WINDOW,
|
||||
X11_ATOM_COUNT,
|
||||
} X11Atom;
|
||||
|
||||
global_variable String atom_names[X11_ATOM_COUNT] = {
|
||||
strlit("WM_PROTOCOLS"),
|
||||
strlit("WM_DELETE_WINDOW"),
|
||||
};
|
||||
global_variable xcb_intern_atom_reply_t* atom_replies[array_length(atom_names)];
|
||||
global_variable xcb_intern_atom_cookie_t atom_cookies[array_length(atom_names)];
|
||||
|
||||
fn u8 windowing_initialize()
|
||||
{
|
||||
u8 result = 0;
|
||||
|
||||
windowing_connection.handle = xcb_connect(0, 0);
|
||||
if (windowing_connection.handle)
|
||||
{
|
||||
if (!xcb_connection_has_error(windowing_connection.handle))
|
||||
{
|
||||
windowing_connection.setup = xcb_get_setup(windowing_connection.handle);
|
||||
|
||||
if (windowing_connection.setup)
|
||||
{
|
||||
x11_intern_atoms(array_length(atom_names), atom_names, atom_cookies, atom_replies);
|
||||
|
||||
if (atom_replies[X11_ATOM_WM_PROTOCOLS])
|
||||
{
|
||||
if (atom_replies[X11_ATOM_WM_DELETE_WINDOW])
|
||||
{
|
||||
result = 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
fn WindowingInstance* windowing_instantiate(WindowingInstantiate create)
|
||||
{
|
||||
xcb_connection_t* connection = windowing_connection.handle;
|
||||
xcb_screen_iterator_t iter = xcb_setup_roots_iterator(windowing_connection.setup);
|
||||
xcb_screen_t *screen = iter.data;
|
||||
|
||||
/* Create a window */
|
||||
xcb_window_t window_handle = xcb_generate_id(connection);
|
||||
|
||||
u32 i;
|
||||
for (i = 0; i < array_length(windowing_instance_handles); i += 1)
|
||||
{
|
||||
xcb_window_t* window_handle_pointer = &windowing_instance_handles[i];
|
||||
if (!*window_handle_pointer)
|
||||
{
|
||||
*window_handle_pointer = window_handle;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
WindowingInstance* window = &windowing_instances[i];
|
||||
window->handle = window_handle;
|
||||
|
||||
u32 value_mask = XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK;
|
||||
u32 value_list[] = {
|
||||
screen->black_pixel,
|
||||
XCB_EVENT_MASK_EXPOSURE | XCB_EVENT_MASK_KEY_PRESS | XCB_EVENT_MASK_STRUCTURE_NOTIFY
|
||||
};
|
||||
|
||||
xcb_create_window(
|
||||
connection, /* Connection */
|
||||
XCB_COPY_FROM_PARENT, /* Depth (same as parent) */
|
||||
window_handle, /* Window ID */
|
||||
screen->root, /* Parent window (root) */
|
||||
create.offset.x, create.offset.y, /* X, Y */
|
||||
create.size.width, create.size.height,
|
||||
10, /* Border width */
|
||||
XCB_WINDOW_CLASS_INPUT_OUTPUT, /* Class */
|
||||
screen->root_visual, /* Visual */
|
||||
value_mask, /* Value mask */
|
||||
value_list /* Value list */
|
||||
);
|
||||
|
||||
xcb_change_property(connection, XCB_PROP_MODE_REPLACE, window_handle, atom_replies[X11_ATOM_WM_PROTOCOLS]->atom, XCB_ATOM_ATOM, 32, 1, &atom_replies[X11_ATOM_WM_DELETE_WINDOW]->atom);
|
||||
|
||||
xcb_map_window(connection, window_handle);
|
||||
|
||||
/* Flush requests to the X server */
|
||||
xcb_flush(connection);
|
||||
|
||||
return window;
|
||||
}
|
||||
|
||||
fn void windowing_poll_events()
|
||||
{
|
||||
xcb_generic_event_t *event;
|
||||
xcb_connection_t* connection = windowing_connection.handle;
|
||||
|
||||
while ((event = xcb_poll_for_event(connection)))
|
||||
{
|
||||
switch (event->response_type & ~0x80) {
|
||||
case XCB_EXPOSE:
|
||||
break;
|
||||
case XCB_KEY_PRESS:
|
||||
break;
|
||||
case XCB_CLIENT_MESSAGE:
|
||||
{
|
||||
let_pointer_cast(xcb_client_message_event_t, client_message_event, event);
|
||||
if (client_message_event->data.data32[0] == atom_replies[X11_ATOM_WM_DELETE_WINDOW]->atom)
|
||||
{
|
||||
xcb_window_t window_handle = client_message_event->window;
|
||||
u32 i;
|
||||
u32 window_handle_count = array_length(windowing_instance_handles);
|
||||
for (i = 0; i < window_handle_count; i += 1)
|
||||
{
|
||||
xcb_window_t* window_handle_pointer = &windowing_instance_handles[i];
|
||||
if (window_handle == *window_handle_pointer)
|
||||
{
|
||||
windowing_instances[i].handle = 0;
|
||||
*window_handle_pointer = 0;
|
||||
// TODO: For now do this
|
||||
os_exit(0);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (i == window_handle_count)
|
||||
{
|
||||
os_exit(1);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
trap();
|
||||
}
|
||||
} break;
|
||||
case XCB_DESTROY_NOTIFY:
|
||||
trap();
|
||||
default:
|
||||
break;
|
||||
}
|
||||
os_free(event);
|
||||
}
|
||||
}
|
||||
|
||||
fn WindowingSize windowing_get_instance_framebuffer_size(WindowingInstance* instance)
|
||||
{
|
||||
WindowingSize result = {};
|
||||
xcb_connection_t* connection = windowing_connection.handle;
|
||||
xcb_window_t window = instance->handle;
|
||||
xcb_get_geometry_cookie_t cookie = xcb_get_geometry(connection, window);
|
||||
xcb_get_geometry_reply_t* reply = xcb_get_geometry_reply(connection, cookie, 0);
|
||||
result.width = reply->width;
|
||||
result.height = reply->height;
|
||||
os_free(reply);
|
||||
return result;
|
||||
}
|
17
bootstrap/std/x11_windowing.h
Normal file
17
bootstrap/std/x11_windowing.h
Normal file
@ -0,0 +1,17 @@
|
||||
#pragma once
|
||||
|
||||
#include <xcb/xcb.h>
|
||||
|
||||
STRUCT(WindowingConnection)
|
||||
{
|
||||
xcb_connection_t* handle;
|
||||
const xcb_setup_t* setup;
|
||||
};
|
||||
|
||||
STRUCT(WindowingInstance)
|
||||
{
|
||||
xcb_window_t handle;
|
||||
};
|
||||
|
||||
fn xcb_connection_t* xcb_connection_get();
|
||||
fn xcb_window_t xcb_window_from_windowing_instance(WindowingInstance* instance);
|
23
build.bat
Normal file
23
build.bat
Normal file
@ -0,0 +1,23 @@
|
||||
@echo off
|
||||
setlocal enableextensions
|
||||
|
||||
if not defined BB_CI (
|
||||
set BB_CI=0
|
||||
)
|
||||
|
||||
if not defined BB_BUILD_TYPE (
|
||||
set BB_BUILD_TYPE=debug
|
||||
)
|
||||
|
||||
set BUILD_DIR=cache
|
||||
mkdir %BUILD_DIR% > NUL 2>&1
|
||||
set BUILD_OUT=cache\build.exe
|
||||
|
||||
if "%BB_CI%" == "0" (
|
||||
%VK_SDK_PATH%\Bin\glslangValidator.exe -V bootstrap\std\shaders\rect.vert -o cache\rect.vert.spv --quiet || exit /b 1
|
||||
%VK_SDK_PATH%\Bin\glslangValidator.exe -V bootstrap\std\shaders\rect.frag -o cache\rect.frag.spv --quiet || exit /b 1
|
||||
)
|
||||
|
||||
|
||||
cl /Zi /Y- /Gm- /std:clatest /diagnostics:caret -FC /nologo build.c /Fd%BUILD_DIR%\ /Fo%BUILD_DIR%\ /Fe%BUILD_OUT% -Ibootstrap -DBB_TIMETRACE=0 -DBB_BUILD_TYPE=\"%BB_BUILD_TYPE%\" -DBB_CI=%BB_CI% /link /INCREMENTAL:NO || exit /b 1
|
||||
%BUILD_OUT%
|
725
build.c
Normal file
725
build.c
Normal file
@ -0,0 +1,725 @@
|
||||
#include <std/base.h>
|
||||
#include <std/os.h>
|
||||
#include <std/project.h>
|
||||
|
||||
#include <std/base.c>
|
||||
#include <std/os.c>
|
||||
|
||||
typedef enum C_Compiler
|
||||
{
|
||||
C_COMPILER_GCC,
|
||||
C_COMPILER_CLANG,
|
||||
C_COMPILER_MSVC,
|
||||
C_COMPILER_TCC,
|
||||
C_COMPILER_COUNT,
|
||||
} C_Compiler;
|
||||
|
||||
global_variable char* c_compiler_names[] = {
|
||||
"gcc",
|
||||
"clang",
|
||||
"cl",
|
||||
"tcc",
|
||||
};
|
||||
|
||||
typedef enum CompilerArgumentStyle
|
||||
{
|
||||
COMPILER_ARGUMENT_STYLE_GNU,
|
||||
COMPILER_ARGUMENT_STYLE_MSVC,
|
||||
COMPILER_ARGUMENT_STYLE_COUNT,
|
||||
} CompilerArgumentStyle;
|
||||
|
||||
global_variable C_Compiler preferred_c_compiler = C_COMPILER_COUNT;
|
||||
global_variable char** environment_pointer;
|
||||
|
||||
typedef enum BuildType
|
||||
{
|
||||
BUILD_TYPE_DEBUG,
|
||||
BUILD_TYPE_RELEASE_SAFE,
|
||||
BUILD_TYPE_RELEASE_FAST,
|
||||
BUILD_TYPE_RELEASE_SMALL,
|
||||
BUILD_TYPE_COUNT,
|
||||
} BuildType;
|
||||
|
||||
const char* build_type_strings[BUILD_TYPE_COUNT] = {
|
||||
"debug",
|
||||
"release_safe",
|
||||
"release_fast",
|
||||
"release_small",
|
||||
};
|
||||
|
||||
char* optimization_switches[COMPILER_ARGUMENT_STYLE_COUNT][BUILD_TYPE_COUNT] = {
|
||||
[COMPILER_ARGUMENT_STYLE_GNU] = {
|
||||
[BUILD_TYPE_DEBUG] = "-O0",
|
||||
[BUILD_TYPE_RELEASE_SAFE] = "-O2",
|
||||
[BUILD_TYPE_RELEASE_FAST] = "-O3",
|
||||
[BUILD_TYPE_RELEASE_SMALL] = "-Oz",
|
||||
},
|
||||
[COMPILER_ARGUMENT_STYLE_MSVC] = {
|
||||
[BUILD_TYPE_DEBUG] = "/Od",
|
||||
[BUILD_TYPE_RELEASE_SAFE] = "/Ox",
|
||||
[BUILD_TYPE_RELEASE_FAST] = "/O2",
|
||||
[BUILD_TYPE_RELEASE_SMALL] = "/O1",
|
||||
},
|
||||
};
|
||||
|
||||
STRUCT(CompileFlags)
|
||||
{
|
||||
u64 colored_output:1;
|
||||
u64 debug:1;
|
||||
u64 error_limit:1;
|
||||
u64 time_trace:1;
|
||||
};
|
||||
|
||||
STRUCT(CompileOptions)
|
||||
{
|
||||
String source_path;
|
||||
String output_path;
|
||||
String compiler_path;
|
||||
RenderingBackend rendering_backend;
|
||||
WindowingBackend windowing_backend;
|
||||
BuildType build_type;
|
||||
CompileFlags flags;
|
||||
};
|
||||
|
||||
typedef enum CompilerSwitch
|
||||
{
|
||||
COMPILER_SWITCH_DEBUG_INFO,
|
||||
COMPILER_SWITCH_COUNT,
|
||||
} CompilerSwitch;
|
||||
|
||||
|
||||
global_variable char* compiler_switches[COMPILER_ARGUMENT_STYLE_COUNT][COMPILER_SWITCH_COUNT] = {
|
||||
[COMPILER_ARGUMENT_STYLE_GNU] = {
|
||||
[COMPILER_SWITCH_DEBUG_INFO] = "-g",
|
||||
},
|
||||
[COMPILER_ARGUMENT_STYLE_MSVC] = {
|
||||
[COMPILER_SWITCH_DEBUG_INFO] = "/Zi",
|
||||
},
|
||||
};
|
||||
|
||||
fn String file_find_in_path(Arena* arena, String file, String path_env)
|
||||
{
|
||||
String result = {};
|
||||
assert(path_env.pointer);
|
||||
|
||||
String path_it = path_env;
|
||||
u8 buffer[4096];
|
||||
|
||||
#if _WIN32
|
||||
u8 env_path_separator = ';';
|
||||
u8 path_separator = '\\';
|
||||
#else
|
||||
u8 env_path_separator = ':';
|
||||
u8 path_separator = '/';
|
||||
#endif
|
||||
|
||||
while (path_it.length)
|
||||
{
|
||||
let(index, string_first_ch(path_it, env_path_separator));
|
||||
index = unlikely(index == STRING_NO_MATCH) ? path_it.length : index;
|
||||
let(path_chunk, s_get_slice(u8, path_it, 0, index));
|
||||
|
||||
u64 i = 0;
|
||||
|
||||
memcpy(&buffer[i], path_chunk.pointer, path_chunk.length);
|
||||
i += path_chunk.length;
|
||||
|
||||
buffer[i] = path_separator;
|
||||
i += 1;
|
||||
|
||||
memcpy(&buffer[i], file.pointer, file.length);
|
||||
i += file.length;
|
||||
|
||||
#if _WIN32
|
||||
String exe_extension = strlit(".exe");
|
||||
memcpy(&buffer[i], exe_extension.pointer, exe_extension.length);
|
||||
i += exe_extension.length;
|
||||
#endif
|
||||
|
||||
buffer[i] = 0;
|
||||
i += 1;
|
||||
|
||||
let(total_length, i - 1);
|
||||
OSFileOpenFlags flags = {
|
||||
.read = 1,
|
||||
};
|
||||
OSFilePermissions permissions = {
|
||||
.readable = 1,
|
||||
.writable = 1,
|
||||
};
|
||||
|
||||
String path = { .pointer = buffer, .length = total_length };
|
||||
|
||||
FileDescriptor fd = os_file_open(path, flags, permissions);
|
||||
|
||||
if (os_file_descriptor_is_valid(fd))
|
||||
{
|
||||
os_file_close(fd);
|
||||
result.pointer = arena_allocate(arena, u8, total_length + 1);
|
||||
memcpy(result.pointer, buffer, total_length + 1);
|
||||
result.length = total_length;
|
||||
break;
|
||||
}
|
||||
|
||||
String new_path = s_get_slice(u8, path_it, index + (index != path_it.length), path_it.length);
|
||||
assert(new_path.length < path_env.length);
|
||||
path_it = new_path;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
fn C_Compiler c_compiler_from_path(String path)
|
||||
{
|
||||
C_Compiler result = C_COMPILER_COUNT;
|
||||
let(last_ch_slash, string_last_ch(path, '/'));
|
||||
let(start, last_ch_slash);
|
||||
#if _WIN32
|
||||
let(last_ch_backslash, string_last_ch(path, '\\'));
|
||||
start = MIN(last_ch_slash, last_ch_backslash);
|
||||
#endif
|
||||
assert(start != STRING_NO_MATCH); // This ensures us the path is not just the executable name
|
||||
let(compiler_name, s_get_slice(u8, path, start + 1, path.length));
|
||||
|
||||
for (C_Compiler i = 0; i < C_COMPILER_COUNT; i += 1)
|
||||
{
|
||||
let(candidate_compiler_name, cstr(c_compiler_names[i]));
|
||||
if (string_contains(compiler_name, candidate_compiler_name))
|
||||
{
|
||||
result = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
fn u8 c_compiler_is_supported_by_os(C_Compiler compiler)
|
||||
{
|
||||
#ifdef __linux__
|
||||
switch (compiler)
|
||||
{
|
||||
case C_COMPILER_TCC: case C_COMPILER_GCC: case C_COMPILER_CLANG: return 1;
|
||||
case C_COMPILER_MSVC: return 0;
|
||||
case C_COMPILER_COUNT: unreachable();
|
||||
}
|
||||
#elif __APPLE__
|
||||
switch (compiler)
|
||||
{
|
||||
case C_COMPILER_TCC: case C_COMPILER_CLANG: return 1;
|
||||
case C_COMPILER_MSVC: case C_COMPILER_GCC: return 0;
|
||||
case C_COMPILER_COUNT: unreachable();
|
||||
}
|
||||
#elif _WIN32
|
||||
switch (compiler)
|
||||
{
|
||||
case C_COMPILER_MSVC: case C_COMPILER_TCC: case C_COMPILER_CLANG: return 1;
|
||||
case C_COMPILER_GCC: return 0;
|
||||
}
|
||||
#endif
|
||||
unreachable();
|
||||
}
|
||||
|
||||
fn String c_compiler_to_string(C_Compiler c_compiler)
|
||||
{
|
||||
switch (c_compiler)
|
||||
{
|
||||
case C_COMPILER_GCC: return strlit("gcc");
|
||||
case C_COMPILER_MSVC: return strlit("MSVC");
|
||||
case C_COMPILER_CLANG: return strlit("clang");
|
||||
case C_COMPILER_TCC: return strlit("tcc");
|
||||
default: unreachable();
|
||||
}
|
||||
}
|
||||
|
||||
// Returns the absolute path of a C compiler
|
||||
fn String get_c_compiler_path(Arena* arena)
|
||||
{
|
||||
String cc_path = {};
|
||||
String cc_env = os_get_environment_variable("CC");
|
||||
String path_env = os_get_environment_variable("PATH");
|
||||
if (cc_env.pointer)
|
||||
{
|
||||
cc_path = cc_env;
|
||||
}
|
||||
#ifndef _WIN32
|
||||
else
|
||||
{
|
||||
cc_path = file_find_in_path(arena, strlit("cc"), path_env);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!cc_path.pointer)
|
||||
{
|
||||
#if _WIN32
|
||||
cc_path = strlit("cl.exe");
|
||||
#elif defined(__APPLE__)
|
||||
cc_path = strlit("clang");
|
||||
#elif defined(__linux__)
|
||||
cc_path = strlit("clang");
|
||||
#else
|
||||
#error "Operating system not supported"
|
||||
#endif
|
||||
}
|
||||
|
||||
let(no_path_sep, string_first_ch(cc_path, '/') == STRING_NO_MATCH);
|
||||
#ifdef _WIN32
|
||||
no_path_sep = no_path_sep && string_first_ch(cc_path, '\\') == STRING_NO_MATCH;
|
||||
#endif
|
||||
if (no_path_sep)
|
||||
{
|
||||
cc_path = file_find_in_path(arena, cc_path, path_env);
|
||||
}
|
||||
|
||||
#ifndef _WIN32
|
||||
if (cc_path.pointer)
|
||||
{
|
||||
u8 buffer[4096];
|
||||
let(realpath, os_realpath(cc_path, (String)array_to_slice(buffer)));
|
||||
if (!s_equal(realpath, cc_path))
|
||||
{
|
||||
cc_path.pointer = arena_allocate(arena, u8, realpath.length + 1);
|
||||
cc_path.length = realpath.length;
|
||||
memcpy(cc_path.pointer, realpath.pointer, realpath.length);
|
||||
cc_path.pointer[cc_path.length] = 0;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#if __APPLE__
|
||||
if (s_equal(cc_path, strlit("/usr/bin/cc")))
|
||||
{
|
||||
cc_path = strlit("/usr/bin/clang");
|
||||
}
|
||||
#endif
|
||||
|
||||
if (preferred_c_compiler != C_COMPILER_COUNT && c_compiler_is_supported_by_os(preferred_c_compiler))
|
||||
{
|
||||
String find_result = file_find_in_path(arena, c_compiler_to_string(preferred_c_compiler), path_env);
|
||||
if (find_result.pointer)
|
||||
{
|
||||
cc_path = find_result;
|
||||
}
|
||||
}
|
||||
|
||||
return cc_path;
|
||||
}
|
||||
|
||||
fn u8 c_compiler_supports_colored_output(C_Compiler compiler)
|
||||
{
|
||||
// TODO: fix
|
||||
switch (compiler)
|
||||
{
|
||||
case C_COMPILER_GCC: case C_COMPILER_CLANG: return 1;
|
||||
case C_COMPILER_TCC: case C_COMPILER_MSVC: return 0;
|
||||
default: unreachable();
|
||||
}
|
||||
}
|
||||
|
||||
fn char* c_compiler_get_error_limit_switch(C_Compiler compiler)
|
||||
{
|
||||
// TODO: fix
|
||||
switch (compiler)
|
||||
{
|
||||
case C_COMPILER_CLANG: return "-ferror-limit=1";
|
||||
case C_COMPILER_GCC: return "-fmax-errors=1";
|
||||
case C_COMPILER_MSVC: case C_COMPILER_TCC: return 0;
|
||||
default: unreachable();
|
||||
}
|
||||
}
|
||||
|
||||
fn char* c_compiler_get_highest_c_standard_flag(C_Compiler compiler)
|
||||
{
|
||||
switch (compiler)
|
||||
{
|
||||
case C_COMPILER_CLANG: case C_COMPILER_GCC: return "-std=gnu2x";
|
||||
case C_COMPILER_MSVC: return "/std:clatest";
|
||||
case C_COMPILER_TCC: return "-std=gnu2x"; // TODO: does it do anything in TCC?
|
||||
default: unreachable();
|
||||
}
|
||||
}
|
||||
|
||||
fn RenderingBackend rendering_backend_parse_env(String env)
|
||||
{
|
||||
unused(env);
|
||||
todo();
|
||||
}
|
||||
|
||||
fn RenderingBackend rendering_backend_pick()
|
||||
{
|
||||
RenderingBackend rendering_backend = RENDERING_BACKEND_COUNT;
|
||||
#if BB_CI
|
||||
rendering_backend = RENDERING_BACKEND_NONE;
|
||||
#else
|
||||
char* env = getenv("BB_RENDERING_BACKEND");
|
||||
if (env)
|
||||
{
|
||||
rendering_backend = rendering_backend_parse_env(cstr(env));
|
||||
}
|
||||
|
||||
if (!rendering_backend_is_valid(rendering_backend))
|
||||
{
|
||||
#ifdef __linux__
|
||||
rendering_backend = RENDERING_BACKEND_VULKAN;
|
||||
#elif defined(__APPLE__)
|
||||
rendering_backend = RENDERING_BACKEND_METAL;
|
||||
#elif _WIN32
|
||||
rendering_backend = RENDERING_BACKEND_VULKAN;
|
||||
#endif
|
||||
}
|
||||
#endif
|
||||
|
||||
return rendering_backend;
|
||||
}
|
||||
|
||||
fn WindowingBackend windowing_backend_parse_env(String env)
|
||||
{
|
||||
unused(env);
|
||||
todo();
|
||||
}
|
||||
|
||||
fn WindowingBackend windowing_backend_pick()
|
||||
{
|
||||
WindowingBackend windowing_backend = WINDOWING_BACKEND_COUNT;
|
||||
#if BB_CI
|
||||
windowing_backend = WINDOWING_BACKEND_NONE;
|
||||
#else
|
||||
// Only done for Linux because it is the only operating system in which two windowing backends officially coexist
|
||||
#ifdef __linux__
|
||||
char* env = getenv("BB_WINDOWING_BACKEND");
|
||||
if (env)
|
||||
{
|
||||
windowing_backend = windowing_backend_parse_env(cstr(env));
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!windowing_backend_is_valid(windowing_backend))
|
||||
{
|
||||
#ifdef __linux__
|
||||
// Prefer X11 over Wayland because:
|
||||
// 1) It works both on Wayland and on X11 desktops
|
||||
// 2) It works with debugging tools like RenderDoc
|
||||
windowing_backend = WINDOWING_BACKEND_X11;
|
||||
#elif _WIN32
|
||||
windowing_backend = WINDOWING_BACKEND_WIN32;
|
||||
#elif __APPLE__
|
||||
windowing_backend = WINDOWING_BACKEND_COCOA;
|
||||
#endif
|
||||
}
|
||||
#endif
|
||||
|
||||
return windowing_backend;
|
||||
}
|
||||
|
||||
fn u8 c_compiler_supports_time_trace(C_Compiler compiler)
|
||||
{
|
||||
switch (compiler)
|
||||
{
|
||||
case C_COMPILER_CLANG: return 1;
|
||||
default: return 0;
|
||||
case C_COMPILER_COUNT: unreachable();
|
||||
}
|
||||
}
|
||||
|
||||
fn BuildType build_type_pick()
|
||||
{
|
||||
String build_type_string = strlit(BB_BUILD_TYPE);
|
||||
BuildType build_type;
|
||||
|
||||
for (build_type = 0; build_type < BUILD_TYPE_COUNT; build_type += 1)
|
||||
{
|
||||
if (s_equal(build_type_string, cstr(build_type_strings[build_type])))
|
||||
{
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return build_type;
|
||||
}
|
||||
|
||||
fn void compile_program(Arena* arena, CompileOptions options)
|
||||
{
|
||||
if (!options.compiler_path.pointer)
|
||||
{
|
||||
char* cc_env = getenv("CC");
|
||||
if (options.flags.debug)
|
||||
{
|
||||
print("Could not find a valid compiler for CC: \"{cstr}\"\n", cc_env ? cc_env : "");
|
||||
print("PATH: {cstr}\n", getenv("PATH"));
|
||||
}
|
||||
failed_execution();
|
||||
}
|
||||
|
||||
if (options.flags.debug)
|
||||
{
|
||||
print("C compiler path: {s}\n", options.compiler_path);
|
||||
}
|
||||
|
||||
C_Compiler c_compiler = c_compiler_from_path(options.compiler_path);
|
||||
if (c_compiler != C_COMPILER_COUNT)
|
||||
{
|
||||
String compiler_name = c_compiler_to_string(c_compiler);
|
||||
if (options.flags.debug)
|
||||
{
|
||||
print("Identified compiler as {s}\n", compiler_name);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
print("Unrecognized C compiler: {s}\n", options.compiler_path);
|
||||
os_exit(1);
|
||||
}
|
||||
char* args[4096];
|
||||
u64 arg_i = 0;
|
||||
#define add_arg(arg) args[arg_i++] = (arg)
|
||||
add_arg(string_to_c(options.compiler_path));
|
||||
if (c_compiler == C_COMPILER_MSVC)
|
||||
{
|
||||
add_arg("/nologo");
|
||||
}
|
||||
|
||||
#if __APPLE__
|
||||
add_arg("-x");
|
||||
add_arg("objective-c");
|
||||
#endif
|
||||
|
||||
add_arg(string_to_c(options.source_path));
|
||||
|
||||
if (c_compiler == C_COMPILER_MSVC)
|
||||
{
|
||||
String strings[] = {
|
||||
strlit("/Fe"),
|
||||
options.output_path,
|
||||
};
|
||||
String arg = arena_join_string(arena, (Slice(String))array_to_slice(strings));
|
||||
add_arg(string_to_c(arg));
|
||||
|
||||
add_arg("/Fo" BUILD_DIR "\\");
|
||||
add_arg("/Fd" BUILD_DIR "\\");
|
||||
}
|
||||
else
|
||||
{
|
||||
add_arg("-o");
|
||||
add_arg(string_to_c(options.output_path));
|
||||
}
|
||||
|
||||
add_arg("-Ibootstrap");
|
||||
add_arg("-Idependencies/stb");
|
||||
|
||||
char* c_include_path = getenv("C_INCLUDE_PATH");
|
||||
if (c_include_path)
|
||||
{
|
||||
String c_include_path_string = cstr(c_include_path);
|
||||
|
||||
u64 previous_i = 0;
|
||||
for (u64 i = 0; i < c_include_path_string.length; i += 1)
|
||||
{
|
||||
u8 ch = c_include_path_string.pointer[i];
|
||||
if (ch == ':')
|
||||
{
|
||||
todo();
|
||||
}
|
||||
}
|
||||
|
||||
String strings[] = {
|
||||
strlit("-I"),
|
||||
s_get_slice(u8, c_include_path_string, previous_i, c_include_path_string.length),
|
||||
};
|
||||
String arg = arena_join_string(arena, (Slice(String))array_to_slice(strings));
|
||||
add_arg(string_to_c(arg));
|
||||
}
|
||||
|
||||
let(debug_info, options.build_type != BUILD_TYPE_RELEASE_SMALL);
|
||||
if (debug_info)
|
||||
{
|
||||
add_arg(compiler_switches[c_compiler == C_COMPILER_MSVC][COMPILER_SWITCH_DEBUG_INFO]);
|
||||
}
|
||||
|
||||
if (c_compiler != C_COMPILER_TCC)
|
||||
{
|
||||
add_arg(optimization_switches[c_compiler == C_COMPILER_MSVC][options.build_type]);
|
||||
}
|
||||
|
||||
if (options.flags.colored_output && c_compiler_supports_colored_output(c_compiler))
|
||||
{
|
||||
add_arg("-fdiagnostics-color=auto");
|
||||
}
|
||||
|
||||
if (options.flags.error_limit)
|
||||
{
|
||||
char* error_limit = c_compiler_get_error_limit_switch(c_compiler);
|
||||
if (error_limit)
|
||||
{
|
||||
add_arg(error_limit);
|
||||
}
|
||||
}
|
||||
|
||||
if (options.flags.time_trace && c_compiler_supports_time_trace(c_compiler))
|
||||
{
|
||||
add_arg("-ftime-trace");
|
||||
}
|
||||
|
||||
if (c_compiler == C_COMPILER_MSVC)
|
||||
{
|
||||
add_arg("/diagnostics:caret");
|
||||
}
|
||||
else
|
||||
{
|
||||
add_arg("-fdiagnostics-show-option");
|
||||
}
|
||||
|
||||
add_arg(c_compiler_get_highest_c_standard_flag(c_compiler));
|
||||
|
||||
switch (options.windowing_backend)
|
||||
{
|
||||
case WINDOWING_BACKEND_NONE:
|
||||
{
|
||||
add_arg("-DBB_WINDOWING_BACKEND_NONE=1");
|
||||
} break;
|
||||
case WINDOWING_BACKEND_WIN32:
|
||||
{
|
||||
add_arg("-DBB_WINDOWING_BACKEND_WIN32=1");
|
||||
} break;
|
||||
case WINDOWING_BACKEND_COCOA:
|
||||
{
|
||||
add_arg("-DBB_WINDOWING_BACKEND_COCOA=1");
|
||||
} break;
|
||||
case WINDOWING_BACKEND_X11:
|
||||
{
|
||||
add_arg("-DBB_WINDOWING_BACKEND_X11=1");
|
||||
} break;
|
||||
case WINDOWING_BACKEND_WAYLAND:
|
||||
{
|
||||
add_arg("-DBB_WINDOWING_BACKEND_WAYLAND=1");
|
||||
} break;
|
||||
case WINDOWING_BACKEND_COUNT: unreachable();
|
||||
}
|
||||
|
||||
switch (options.rendering_backend)
|
||||
{
|
||||
case RENDERING_BACKEND_NONE:
|
||||
{
|
||||
add_arg("-DBB_RENDERING_BACKEND_NONE=1");
|
||||
} break;
|
||||
case RENDERING_BACKEND_METAL:
|
||||
{
|
||||
add_arg("-DBB_RENDERING_BACKEND_METAL=1");
|
||||
} break;
|
||||
case RENDERING_BACKEND_DIRECTX12:
|
||||
{
|
||||
add_arg("-DBB_RENDERING_BACKEND_DIRECTX12=1");
|
||||
} break;
|
||||
case RENDERING_BACKEND_VULKAN:
|
||||
{
|
||||
add_arg("-DBB_RENDERING_BACKEND_VULKAN=1");
|
||||
#if _WIN32
|
||||
char* vk_sdk_path = getenv("VK_SDK_PATH");
|
||||
if (vk_sdk_path)
|
||||
{
|
||||
if (c_compiler == C_COMPILER_MSVC)
|
||||
{
|
||||
String strings[] = {
|
||||
strlit("-I"),
|
||||
cstr(vk_sdk_path),
|
||||
strlit("\\Include"),
|
||||
};
|
||||
String arg = arena_join_string(arena, (Slice(String))array_to_slice(strings));
|
||||
add_arg(string_to_c(arg));
|
||||
}
|
||||
else
|
||||
{
|
||||
todo();
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
print("VK_SDK_PATH environment variable not found\n");
|
||||
}
|
||||
#endif
|
||||
} break;
|
||||
case RENDERING_BACKEND_COUNT: unreachable();
|
||||
}
|
||||
|
||||
#ifndef _WIN32
|
||||
add_arg("-lm");
|
||||
#endif
|
||||
|
||||
switch (options.windowing_backend)
|
||||
{
|
||||
case WINDOWING_BACKEND_NONE:
|
||||
{
|
||||
} break;
|
||||
case WINDOWING_BACKEND_WIN32:
|
||||
{
|
||||
} break;
|
||||
case WINDOWING_BACKEND_COCOA:
|
||||
{
|
||||
add_arg("-framework");
|
||||
add_arg("AppKit");
|
||||
} break;
|
||||
case WINDOWING_BACKEND_X11:
|
||||
{
|
||||
add_arg("-lxcb");
|
||||
} break;
|
||||
case WINDOWING_BACKEND_WAYLAND:
|
||||
{
|
||||
} break;
|
||||
case WINDOWING_BACKEND_COUNT: unreachable();
|
||||
}
|
||||
|
||||
switch (options.rendering_backend)
|
||||
{
|
||||
case RENDERING_BACKEND_NONE:
|
||||
{
|
||||
} break;
|
||||
case RENDERING_BACKEND_METAL:
|
||||
{
|
||||
add_arg("-framework");
|
||||
add_arg("Metal");
|
||||
add_arg("-framework");
|
||||
add_arg("QuartzCore");
|
||||
} break;
|
||||
case RENDERING_BACKEND_DIRECTX12:
|
||||
{
|
||||
} break;
|
||||
case RENDERING_BACKEND_VULKAN:
|
||||
{
|
||||
#if __APPLE__
|
||||
add_arg("-framework");
|
||||
add_arg("QuartzCore");
|
||||
#endif
|
||||
} break;
|
||||
case RENDERING_BACKEND_COUNT: unreachable();
|
||||
}
|
||||
|
||||
add_arg(0);
|
||||
CStringSlice arguments = { .pointer = args, .length = arg_i };
|
||||
RunCommandOptions run_options = {
|
||||
.debug = options.flags.debug,
|
||||
};
|
||||
run_command(arena, arguments, environment_pointer, run_options);
|
||||
}
|
||||
|
||||
int main(int argc, char* argv[], char** envp)
|
||||
{
|
||||
environment_pointer = envp;
|
||||
Arena* arena = arena_initialize_default(KB(64));
|
||||
|
||||
CompileOptions compile_options = {
|
||||
.compiler_path = get_c_compiler_path(arena),
|
||||
.source_path = strlit("bootstrap/bloat-buster/bb.c"),
|
||||
.output_path = strlit("cache/bb" EXECUTABLE_EXTENSION),
|
||||
.windowing_backend = windowing_backend_pick(),
|
||||
.rendering_backend = rendering_backend_pick(),
|
||||
.build_type = build_type_pick(),
|
||||
.flags = {
|
||||
.colored_output = 1,
|
||||
.error_limit = 1,
|
||||
.debug = 1,
|
||||
.time_trace = BB_TIMETRACE,
|
||||
},
|
||||
};
|
||||
compile_program(arena, compile_options);
|
||||
return 0;
|
||||
}
|
42
build.sh
Executable file
42
build.sh
Executable file
@ -0,0 +1,42 @@
|
||||
#!/usr/bin/env bash
|
||||
set -eu
|
||||
|
||||
if [[ -z "${BB_CI-}" ]]; then
|
||||
BB_CI=0
|
||||
fi
|
||||
|
||||
if [[ -z "${BB_BUILD_TYPE-}" ]]; then
|
||||
BB_BUILD_TYPE=debug
|
||||
fi
|
||||
|
||||
BUILD_DIR=cache
|
||||
mkdir -p $BUILD_DIR
|
||||
|
||||
if [[ "${BB_CI}" == "0" ]]; then
|
||||
glslangValidator -V bootstrap/std/shaders/rect.vert -o $BUILD_DIR/rect.vert.spv --quiet
|
||||
glslangValidator -V bootstrap/std/shaders/rect.frag -o $BUILD_DIR/rect.frag.spv --quiet
|
||||
fi
|
||||
|
||||
BUILD_OUT=$BUILD_DIR/build
|
||||
C_COMPILER=cc
|
||||
TIME_TRACE=1
|
||||
BB_TIMETRACE=0
|
||||
GCC_ARGS=
|
||||
CLANG_ARGS=
|
||||
TIME_TRACE_ARG=
|
||||
|
||||
if [[ $C_COMPILER == "clang"* ]]; then
|
||||
CLANG_ARGS=-ferror-limit=1
|
||||
if [[ "$TIME_TRACE" == "1" ]]; then
|
||||
CLANG_ARGS="$CLANG_ARGS -ftime-trace"
|
||||
BB_TIMETRACE=1
|
||||
else
|
||||
CLANG_ARGS="$CLANG_ARGS -ftime-trace"
|
||||
fi
|
||||
elif [[ $C_COMPILER == "gcc"* ]]; then
|
||||
GCC_ARGS=-fmax-errors=1
|
||||
fi
|
||||
|
||||
$C_COMPILER build.c -g -o $BUILD_OUT -Ibootstrap -std=gnu2x $CLANG_ARGS $GCC_ARGS -DBB_TIMETRACE=$BB_TIMETRACE -DBB_CI=$BB_CI -DBB_BUILD_TYPE=\"$BB_BUILD_TYPE\"
|
||||
$BUILD_OUT $@
|
||||
exit 0
|
@ -1,48 +0,0 @@
|
||||
# Usage:
|
||||
# cmake -P GenerateMappings.cmake <path/to/mappings.h.in> <path/to/mappings.h>
|
||||
|
||||
set(source_url "https://raw.githubusercontent.com/gabomdq/SDL_GameControllerDB/master/gamecontrollerdb.txt")
|
||||
set(source_path "${CMAKE_CURRENT_BINARY_DIR}/gamecontrollerdb.txt")
|
||||
set(template_path "${CMAKE_ARGV3}")
|
||||
set(target_path "${CMAKE_ARGV4}")
|
||||
|
||||
if (NOT EXISTS "${template_path}")
|
||||
message(FATAL_ERROR "Failed to find template file ${template_path}")
|
||||
endif()
|
||||
|
||||
file(DOWNLOAD "${source_url}" "${source_path}"
|
||||
STATUS download_status
|
||||
TLS_VERIFY on)
|
||||
|
||||
list(GET download_status 0 status_code)
|
||||
list(GET download_status 1 status_message)
|
||||
|
||||
if (status_code)
|
||||
message(FATAL_ERROR "Failed to download ${source_url}: ${status_message}")
|
||||
endif()
|
||||
|
||||
file(STRINGS "${source_path}" lines)
|
||||
foreach(line ${lines})
|
||||
if (line MATCHES "^[0-9a-fA-F]")
|
||||
if (line MATCHES "platform:Windows")
|
||||
if (GLFW_WIN32_MAPPINGS)
|
||||
string(APPEND GLFW_WIN32_MAPPINGS "\n")
|
||||
endif()
|
||||
string(APPEND GLFW_WIN32_MAPPINGS "\"${line}\",")
|
||||
elseif (line MATCHES "platform:Mac OS X")
|
||||
if (GLFW_COCOA_MAPPINGS)
|
||||
string(APPEND GLFW_COCOA_MAPPINGS "\n")
|
||||
endif()
|
||||
string(APPEND GLFW_COCOA_MAPPINGS "\"${line}\",")
|
||||
elseif (line MATCHES "platform:Linux")
|
||||
if (GLFW_LINUX_MAPPINGS)
|
||||
string(APPEND GLFW_LINUX_MAPPINGS "\n")
|
||||
endif()
|
||||
string(APPEND GLFW_LINUX_MAPPINGS "\"${line}\",")
|
||||
endif()
|
||||
endif()
|
||||
endforeach()
|
||||
|
||||
configure_file("${template_path}" "${target_path}" @ONLY NEWLINE_STYLE UNIX)
|
||||
file(REMOVE "${source_path}")
|
||||
|
38
dependencies/glfw-3.4/CMake/Info.plist.in
vendored
38
dependencies/glfw-3.4/CMake/Info.plist.in
vendored
@ -1,38 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple Computer//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>English</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>${MACOSX_BUNDLE_EXECUTABLE_NAME}</string>
|
||||
<key>CFBundleGetInfoString</key>
|
||||
<string>${MACOSX_BUNDLE_INFO_STRING}</string>
|
||||
<key>CFBundleIconFile</key>
|
||||
<string>${MACOSX_BUNDLE_ICON_FILE}</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
<string>${MACOSX_BUNDLE_GUI_IDENTIFIER}</string>
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleLongVersionString</key>
|
||||
<string>${MACOSX_BUNDLE_LONG_VERSION_STRING}</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>${MACOSX_BUNDLE_BUNDLE_NAME}</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>APPL</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
<string>${MACOSX_BUNDLE_SHORT_VERSION_STRING}</string>
|
||||
<key>CFBundleSignature</key>
|
||||
<string>????</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>${MACOSX_BUNDLE_BUNDLE_VERSION}</string>
|
||||
<key>CSResourcesFileMapped</key>
|
||||
<true/>
|
||||
<key>LSRequiresCarbon</key>
|
||||
<true/>
|
||||
<key>NSHumanReadableCopyright</key>
|
||||
<string>${MACOSX_BUNDLE_COPYRIGHT}</string>
|
||||
<key>NSHighResolutionCapable</key>
|
||||
<true/>
|
||||
</dict>
|
||||
</plist>
|
@ -1,29 +0,0 @@
|
||||
|
||||
if (NOT EXISTS "@GLFW_BINARY_DIR@/install_manifest.txt")
|
||||
message(FATAL_ERROR "Cannot find install manifest: \"@GLFW_BINARY_DIR@/install_manifest.txt\"")
|
||||
endif()
|
||||
|
||||
file(READ "@GLFW_BINARY_DIR@/install_manifest.txt" files)
|
||||
string(REGEX REPLACE "\n" ";" files "${files}")
|
||||
|
||||
foreach (file ${files})
|
||||
message(STATUS "Uninstalling \"$ENV{DESTDIR}${file}\"")
|
||||
if (EXISTS "$ENV{DESTDIR}${file}")
|
||||
exec_program("@CMAKE_COMMAND@" ARGS "-E remove \"$ENV{DESTDIR}${file}\""
|
||||
OUTPUT_VARIABLE rm_out
|
||||
RETURN_VALUE rm_retval)
|
||||
if (NOT "${rm_retval}" STREQUAL 0)
|
||||
MESSAGE(FATAL_ERROR "Problem when removing \"$ENV{DESTDIR}${file}\"")
|
||||
endif()
|
||||
elseif (IS_SYMLINK "$ENV{DESTDIR}${file}")
|
||||
EXEC_PROGRAM("@CMAKE_COMMAND@" ARGS "-E remove \"$ENV{DESTDIR}${file}\""
|
||||
OUTPUT_VARIABLE rm_out
|
||||
RETURN_VALUE rm_retval)
|
||||
if (NOT "${rm_retval}" STREQUAL 0)
|
||||
message(FATAL_ERROR "Problem when removing symlink \"$ENV{DESTDIR}${file}\"")
|
||||
endif()
|
||||
else()
|
||||
message(STATUS "File \"$ENV{DESTDIR}${file}\" does not exist.")
|
||||
endif()
|
||||
endforeach()
|
||||
|
13
dependencies/glfw-3.4/CMake/glfw3.pc.in
vendored
13
dependencies/glfw-3.4/CMake/glfw3.pc.in
vendored
@ -1,13 +0,0 @@
|
||||
prefix=@CMAKE_INSTALL_PREFIX@
|
||||
exec_prefix=${prefix}
|
||||
includedir=@CMAKE_INSTALL_FULL_INCLUDEDIR@
|
||||
libdir=@CMAKE_INSTALL_FULL_LIBDIR@
|
||||
|
||||
Name: GLFW
|
||||
Description: A multi-platform library for OpenGL, window and input
|
||||
Version: @GLFW_VERSION@
|
||||
URL: https://www.glfw.org/
|
||||
Requires.private: @GLFW_PKG_CONFIG_REQUIRES_PRIVATE@
|
||||
Libs: -L${libdir} -l@GLFW_LIB_NAME@@GLFW_LIB_NAME_SUFFIX@
|
||||
Libs.private: @GLFW_PKG_CONFIG_LIBS_PRIVATE@
|
||||
Cflags: -I${includedir}
|
@ -1,3 +0,0 @@
|
||||
include(CMakeFindDependencyMacro)
|
||||
find_dependency(Threads)
|
||||
include("${CMAKE_CURRENT_LIST_DIR}/glfw3Targets.cmake")
|
@ -1,13 +0,0 @@
|
||||
# Define the environment for cross-compiling with 32-bit MinGW-w64 Clang
|
||||
SET(CMAKE_SYSTEM_NAME Windows) # Target system name
|
||||
SET(CMAKE_SYSTEM_VERSION 1)
|
||||
SET(CMAKE_C_COMPILER "i686-w64-mingw32-clang")
|
||||
SET(CMAKE_CXX_COMPILER "i686-w64-mingw32-clang++")
|
||||
SET(CMAKE_RC_COMPILER "i686-w64-mingw32-windres")
|
||||
SET(CMAKE_RANLIB "i686-w64-mingw32-ranlib")
|
||||
|
||||
# Configure the behaviour of the find commands
|
||||
SET(CMAKE_FIND_ROOT_PATH "/usr/i686-w64-mingw32")
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER)
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY)
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY)
|
@ -1,13 +0,0 @@
|
||||
# Define the environment for cross-compiling with 32-bit MinGW-w64 GCC
|
||||
SET(CMAKE_SYSTEM_NAME Windows) # Target system name
|
||||
SET(CMAKE_SYSTEM_VERSION 1)
|
||||
SET(CMAKE_C_COMPILER "i686-w64-mingw32-gcc")
|
||||
SET(CMAKE_CXX_COMPILER "i686-w64-mingw32-g++")
|
||||
SET(CMAKE_RC_COMPILER "i686-w64-mingw32-windres")
|
||||
SET(CMAKE_RANLIB "i686-w64-mingw32-ranlib")
|
||||
|
||||
# Configure the behaviour of the find commands
|
||||
SET(CMAKE_FIND_ROOT_PATH "/usr/i686-w64-mingw32")
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER)
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY)
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY)
|
@ -1,17 +0,0 @@
|
||||
# Find EpollShim
|
||||
# Once done, this will define
|
||||
#
|
||||
# EPOLLSHIM_FOUND - System has EpollShim
|
||||
# EPOLLSHIM_INCLUDE_DIRS - The EpollShim include directories
|
||||
# EPOLLSHIM_LIBRARIES - The libraries needed to use EpollShim
|
||||
|
||||
find_path(EPOLLSHIM_INCLUDE_DIRS NAMES sys/epoll.h sys/timerfd.h HINTS /usr/local/include/libepoll-shim)
|
||||
find_library(EPOLLSHIM_LIBRARIES NAMES epoll-shim libepoll-shim HINTS /usr/local/lib)
|
||||
|
||||
if (EPOLLSHIM_INCLUDE_DIRS AND EPOLLSHIM_LIBRARIES)
|
||||
set(EPOLLSHIM_FOUND TRUE)
|
||||
endif (EPOLLSHIM_INCLUDE_DIRS AND EPOLLSHIM_LIBRARIES)
|
||||
|
||||
include(FindPackageHandleStandardArgs)
|
||||
find_package_handle_standard_args(EpollShim DEFAULT_MSG EPOLLSHIM_LIBRARIES EPOLLSHIM_INCLUDE_DIRS)
|
||||
mark_as_advanced(EPOLLSHIM_INCLUDE_DIRS EPOLLSHIM_LIBRARIES)
|
@ -1,18 +0,0 @@
|
||||
# Try to find OSMesa on a Unix system
|
||||
#
|
||||
# This will define:
|
||||
#
|
||||
# OSMESA_LIBRARIES - Link these to use OSMesa
|
||||
# OSMESA_INCLUDE_DIR - Include directory for OSMesa
|
||||
#
|
||||
# Copyright (c) 2014 Brandon Schaefer <brandon.schaefer@canonical.com>
|
||||
|
||||
if (NOT WIN32)
|
||||
|
||||
find_package (PkgConfig)
|
||||
pkg_check_modules (PKG_OSMESA QUIET osmesa)
|
||||
|
||||
set (OSMESA_INCLUDE_DIR ${PKG_OSMESA_INCLUDE_DIRS})
|
||||
set (OSMESA_LIBRARIES ${PKG_OSMESA_LIBRARIES})
|
||||
|
||||
endif ()
|
@ -1,13 +0,0 @@
|
||||
# Define the environment for cross-compiling with 64-bit MinGW-w64 Clang
|
||||
SET(CMAKE_SYSTEM_NAME Windows) # Target system name
|
||||
SET(CMAKE_SYSTEM_VERSION 1)
|
||||
SET(CMAKE_C_COMPILER "x86_64-w64-mingw32-clang")
|
||||
SET(CMAKE_CXX_COMPILER "x86_64-w64-mingw32-clang++")
|
||||
SET(CMAKE_RC_COMPILER "x86_64-w64-mingw32-windres")
|
||||
SET(CMAKE_RANLIB "x86_64-w64-mingw32-ranlib")
|
||||
|
||||
# Configure the behaviour of the find commands
|
||||
SET(CMAKE_FIND_ROOT_PATH "/usr/x86_64-w64-mingw32")
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER)
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY)
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY)
|
@ -1,13 +0,0 @@
|
||||
# Define the environment for cross-compiling with 64-bit MinGW-w64 GCC
|
||||
SET(CMAKE_SYSTEM_NAME Windows) # Target system name
|
||||
SET(CMAKE_SYSTEM_VERSION 1)
|
||||
SET(CMAKE_C_COMPILER "x86_64-w64-mingw32-gcc")
|
||||
SET(CMAKE_CXX_COMPILER "x86_64-w64-mingw32-g++")
|
||||
SET(CMAKE_RC_COMPILER "x86_64-w64-mingw32-windres")
|
||||
SET(CMAKE_RANLIB "x86_64-w64-mingw32-ranlib")
|
||||
|
||||
# Configure the behaviour of the find commands
|
||||
SET(CMAKE_FIND_ROOT_PATH "/usr/x86_64-w64-mingw32")
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER)
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY)
|
||||
SET(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY)
|
165
dependencies/glfw-3.4/CMakeLists.txt
vendored
165
dependencies/glfw-3.4/CMakeLists.txt
vendored
@ -1,165 +0,0 @@
|
||||
cmake_minimum_required(VERSION 3.4...3.28 FATAL_ERROR)
|
||||
|
||||
project(GLFW VERSION 3.4.0 LANGUAGES C)
|
||||
|
||||
if (POLICY CMP0069)
|
||||
cmake_policy(SET CMP0069 NEW)
|
||||
endif()
|
||||
|
||||
if (POLICY CMP0077)
|
||||
cmake_policy(SET CMP0077 NEW)
|
||||
endif()
|
||||
|
||||
set_property(GLOBAL PROPERTY USE_FOLDERS ON)
|
||||
|
||||
string(COMPARE EQUAL "${CMAKE_CURRENT_SOURCE_DIR}" "${CMAKE_SOURCE_DIR}" GLFW_STANDALONE)
|
||||
|
||||
option(BUILD_SHARED_LIBS "Build shared libraries" OFF)
|
||||
option(GLFW_BUILD_EXAMPLES "Build the GLFW example programs" ${GLFW_STANDALONE})
|
||||
option(GLFW_BUILD_TESTS "Build the GLFW test programs" ${GLFW_STANDALONE})
|
||||
option(GLFW_BUILD_DOCS "Build the GLFW documentation" ON)
|
||||
option(GLFW_INSTALL "Generate installation target" ON)
|
||||
|
||||
include(GNUInstallDirs)
|
||||
include(CMakeDependentOption)
|
||||
|
||||
if (GLFW_USE_OSMESA)
|
||||
message(FATAL_ERROR "GLFW_USE_OSMESA has been removed; set the GLFW_PLATFORM init hint")
|
||||
endif()
|
||||
|
||||
if (DEFINED GLFW_USE_WAYLAND AND UNIX AND NOT APPLE)
|
||||
message(FATAL_ERROR
|
||||
"GLFW_USE_WAYLAND has been removed; delete the CMake cache and set GLFW_BUILD_WAYLAND and GLFW_BUILD_X11 instead")
|
||||
endif()
|
||||
|
||||
cmake_dependent_option(GLFW_BUILD_WIN32 "Build support for Win32" ON "WIN32" OFF)
|
||||
cmake_dependent_option(GLFW_BUILD_COCOA "Build support for Cocoa" ON "APPLE" OFF)
|
||||
cmake_dependent_option(GLFW_BUILD_X11 "Build support for X11" ON "UNIX;NOT APPLE" OFF)
|
||||
cmake_dependent_option(GLFW_BUILD_WAYLAND "Build support for Wayland" ON "UNIX;NOT APPLE" OFF)
|
||||
|
||||
cmake_dependent_option(GLFW_USE_HYBRID_HPG "Force use of high-performance GPU on hybrid systems" OFF
|
||||
"WIN32" OFF)
|
||||
cmake_dependent_option(USE_MSVC_RUNTIME_LIBRARY_DLL "Use MSVC runtime library DLL" ON
|
||||
"MSVC" OFF)
|
||||
|
||||
set(GLFW_LIBRARY_TYPE "${GLFW_LIBRARY_TYPE}" CACHE STRING
|
||||
"Library type override for GLFW (SHARED, STATIC, OBJECT, or empty to follow BUILD_SHARED_LIBS)")
|
||||
|
||||
if (GLFW_LIBRARY_TYPE)
|
||||
if (GLFW_LIBRARY_TYPE STREQUAL "SHARED")
|
||||
set(GLFW_BUILD_SHARED_LIBRARY TRUE)
|
||||
else()
|
||||
set(GLFW_BUILD_SHARED_LIBRARY FALSE)
|
||||
endif()
|
||||
else()
|
||||
set(GLFW_BUILD_SHARED_LIBRARY ${BUILD_SHARED_LIBS})
|
||||
endif()
|
||||
|
||||
list(APPEND CMAKE_MODULE_PATH "${GLFW_SOURCE_DIR}/CMake/modules")
|
||||
|
||||
find_package(Threads REQUIRED)
|
||||
|
||||
#--------------------------------------------------------------------
|
||||
# Report backend selection
|
||||
#--------------------------------------------------------------------
|
||||
if (GLFW_BUILD_WIN32)
|
||||
message(STATUS "Including Win32 support")
|
||||
endif()
|
||||
if (GLFW_BUILD_COCOA)
|
||||
message(STATUS "Including Cocoa support")
|
||||
endif()
|
||||
if (GLFW_BUILD_WAYLAND)
|
||||
message(STATUS "Including Wayland support")
|
||||
endif()
|
||||
if (GLFW_BUILD_X11)
|
||||
message(STATUS "Including X11 support")
|
||||
endif()
|
||||
|
||||
#--------------------------------------------------------------------
|
||||
# Apply Microsoft C runtime library option
|
||||
# This is here because it also applies to tests and examples
|
||||
#--------------------------------------------------------------------
|
||||
if (MSVC AND NOT USE_MSVC_RUNTIME_LIBRARY_DLL)
|
||||
if (CMAKE_VERSION VERSION_LESS 3.15)
|
||||
foreach (flag CMAKE_C_FLAGS
|
||||
CMAKE_C_FLAGS_DEBUG
|
||||
CMAKE_C_FLAGS_RELEASE
|
||||
CMAKE_C_FLAGS_MINSIZEREL
|
||||
CMAKE_C_FLAGS_RELWITHDEBINFO)
|
||||
|
||||
if (flag MATCHES "/MD")
|
||||
string(REGEX REPLACE "/MD" "/MT" ${flag} "${${flag}}")
|
||||
endif()
|
||||
if (flag MATCHES "/MDd")
|
||||
string(REGEX REPLACE "/MDd" "/MTd" ${flag} "${${flag}}")
|
||||
endif()
|
||||
|
||||
endforeach()
|
||||
else()
|
||||
set(CMAKE_MSVC_RUNTIME_LIBRARY "MultiThreaded$<$<CONFIG:Debug>:Debug>")
|
||||
endif()
|
||||
endif()
|
||||
|
||||
#--------------------------------------------------------------------
|
||||
# Create generated files
|
||||
#--------------------------------------------------------------------
|
||||
include(CMakePackageConfigHelpers)
|
||||
|
||||
set(GLFW_CONFIG_PATH "${CMAKE_INSTALL_LIBDIR}/cmake/glfw3")
|
||||
|
||||
configure_package_config_file(CMake/glfw3Config.cmake.in
|
||||
src/glfw3Config.cmake
|
||||
INSTALL_DESTINATION "${GLFW_CONFIG_PATH}"
|
||||
NO_CHECK_REQUIRED_COMPONENTS_MACRO)
|
||||
|
||||
write_basic_package_version_file(src/glfw3ConfigVersion.cmake
|
||||
VERSION ${GLFW_VERSION}
|
||||
COMPATIBILITY SameMajorVersion)
|
||||
|
||||
#--------------------------------------------------------------------
|
||||
# Add subdirectories
|
||||
#--------------------------------------------------------------------
|
||||
add_subdirectory(src)
|
||||
|
||||
if (GLFW_BUILD_EXAMPLES)
|
||||
add_subdirectory(examples)
|
||||
endif()
|
||||
|
||||
if (GLFW_BUILD_TESTS)
|
||||
add_subdirectory(tests)
|
||||
endif()
|
||||
|
||||
if (GLFW_BUILD_DOCS)
|
||||
add_subdirectory(docs)
|
||||
endif()
|
||||
|
||||
#--------------------------------------------------------------------
|
||||
# Install files other than the library
|
||||
# The library is installed by src/CMakeLists.txt
|
||||
#--------------------------------------------------------------------
|
||||
if (GLFW_INSTALL)
|
||||
install(DIRECTORY include/GLFW DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}
|
||||
FILES_MATCHING PATTERN glfw3.h PATTERN glfw3native.h)
|
||||
|
||||
install(FILES "${GLFW_BINARY_DIR}/src/glfw3Config.cmake"
|
||||
"${GLFW_BINARY_DIR}/src/glfw3ConfigVersion.cmake"
|
||||
DESTINATION "${GLFW_CONFIG_PATH}")
|
||||
|
||||
install(EXPORT glfwTargets FILE glfw3Targets.cmake
|
||||
EXPORT_LINK_INTERFACE_LIBRARIES
|
||||
DESTINATION "${GLFW_CONFIG_PATH}")
|
||||
install(FILES "${GLFW_BINARY_DIR}/src/glfw3.pc"
|
||||
DESTINATION "${CMAKE_INSTALL_LIBDIR}/pkgconfig")
|
||||
|
||||
# Only generate this target if no higher-level project already has
|
||||
if (NOT TARGET uninstall)
|
||||
configure_file(CMake/cmake_uninstall.cmake.in
|
||||
cmake_uninstall.cmake IMMEDIATE @ONLY)
|
||||
|
||||
add_custom_target(uninstall
|
||||
"${CMAKE_COMMAND}" -P
|
||||
"${GLFW_BINARY_DIR}/cmake_uninstall.cmake")
|
||||
set_target_properties(uninstall PROPERTIES FOLDER "GLFW3")
|
||||
endif()
|
||||
endif()
|
||||
|
297
dependencies/glfw-3.4/CONTRIBUTORS.md
vendored
297
dependencies/glfw-3.4/CONTRIBUTORS.md
vendored
@ -1,297 +0,0 @@
|
||||
# Acknowledgements
|
||||
|
||||
GLFW exists because people around the world donated their time and lent their
|
||||
skills. This list only includes contributions to the main repository and
|
||||
excludes other invaluable contributions like language bindings and text and
|
||||
video tutorials.
|
||||
|
||||
- Bobyshev Alexander
|
||||
- Laurent Aphecetche
|
||||
- Matt Arsenault
|
||||
- Takuro Ashie
|
||||
- ashishgamedev
|
||||
- David Avedissian
|
||||
- Luca Bacci
|
||||
- Keith Bauer
|
||||
- John Bartholomew
|
||||
- Coşku Baş
|
||||
- Bayemite
|
||||
- Niklas Behrens
|
||||
- Andrew Belt
|
||||
- Nevyn Bengtsson
|
||||
- Niklas Bergström
|
||||
- Denis Bernard
|
||||
- BiBi
|
||||
- Doug Binks
|
||||
- blanco
|
||||
- Waris Boonyasiriwat
|
||||
- Kyle Brenneman
|
||||
- Rok Breulj
|
||||
- TheBrokenRail
|
||||
- Kai Burjack
|
||||
- Martin Capitanio
|
||||
- Nicolas Caramelli
|
||||
- David Carlier
|
||||
- Arturo Castro
|
||||
- Chi-kwan Chan
|
||||
- Victor Chernyakin
|
||||
- TheChocolateOre
|
||||
- Ali Chraghi
|
||||
- Joseph Chua
|
||||
- Ian Clarkson
|
||||
- Michał Cichoń
|
||||
- Lambert Clara
|
||||
- Anna Clarke
|
||||
- Josh Codd
|
||||
- Yaron Cohen-Tal
|
||||
- Omar Cornut
|
||||
- Andrew Corrigan
|
||||
- Bailey Cosier
|
||||
- Noel Cower
|
||||
- CuriouserThing
|
||||
- Bill Currie
|
||||
- Jason Daly
|
||||
- danhambleton
|
||||
- Jarrod Davis
|
||||
- Olivier Delannoy
|
||||
- Paul R. Deppe
|
||||
- Michael Dickens
|
||||
- Роман Донченко
|
||||
- Mario Dorn
|
||||
- Wolfgang Draxinger
|
||||
- Jonathan Dummer
|
||||
- Ralph Eastwood
|
||||
- Fredrik Ehnbom
|
||||
- Robin Eklind
|
||||
- Jan Ekström
|
||||
- Siavash Eliasi
|
||||
- Ahmad Fatoum
|
||||
- Nikita Fediuchin
|
||||
- Felipe Ferreira
|
||||
- Michael Fogleman
|
||||
- forworldm
|
||||
- Jason Francis
|
||||
- Gerald Franz
|
||||
- Mário Freitas
|
||||
- GeO4d
|
||||
- Marcus Geelnard
|
||||
- Gegy
|
||||
- ghuser404
|
||||
- Charles Giessen
|
||||
- Ryan C. Gordon
|
||||
- Stephen Gowen
|
||||
- Kovid Goyal
|
||||
- Kevin Grandemange
|
||||
- Eloi Marín Gratacós
|
||||
- Grzesiek11
|
||||
- Stefan Gustavson
|
||||
- Andrew Gutekanst
|
||||
- Stephen Gutekanst
|
||||
- Jonathan Hale
|
||||
- Daniel Hauser
|
||||
- hdf89shfdfs
|
||||
- Moritz Heinemann
|
||||
- Sylvain Hellegouarch
|
||||
- Björn Hempel
|
||||
- Matthew Henry
|
||||
- heromyth
|
||||
- Lucas Hinderberger
|
||||
- Paul Holden
|
||||
- Hajime Hoshi
|
||||
- Warren Hu
|
||||
- Charles Huber
|
||||
- Brent Huisman
|
||||
- Florian Hülsmann
|
||||
- illustris
|
||||
- InKryption
|
||||
- IntellectualKitty
|
||||
- Aaron Jacobs
|
||||
- JannikGM
|
||||
- Erik S. V. Jansson
|
||||
- jjYBdx4IL
|
||||
- Peter Johnson
|
||||
- Toni Jovanoski
|
||||
- Arseny Kapoulkine
|
||||
- Cem Karan
|
||||
- Osman Keskin
|
||||
- Koray Kilinc
|
||||
- Josh Kilmer
|
||||
- Byunghoon Kim
|
||||
- Cameron King
|
||||
- Peter Knut
|
||||
- Christoph Kubisch
|
||||
- Yuri Kunde Schlesner
|
||||
- Rokas Kupstys
|
||||
- Konstantin Käfer
|
||||
- Eric Larson
|
||||
- Guillaume Lebrun
|
||||
- Francis Lecavalier
|
||||
- Jong Won Lee
|
||||
- Robin Leffmann
|
||||
- Glenn Lewis
|
||||
- Shane Liesegang
|
||||
- Anders Lindqvist
|
||||
- Leon Linhart
|
||||
- Marco Lizza
|
||||
- lo-v-ol
|
||||
- Eyal Lotem
|
||||
- Aaron Loucks
|
||||
- Ned Loynd
|
||||
- Luflosi
|
||||
- lukect
|
||||
- Tristam MacDonald
|
||||
- Jean-Luc Mackail
|
||||
- Hans Mackowiak
|
||||
- Ramiro Magno
|
||||
- Дмитри Малышев
|
||||
- Zbigniew Mandziejewicz
|
||||
- Adam Marcus
|
||||
- Célestin Marot
|
||||
- Kyle McDonald
|
||||
- David V. McKay
|
||||
- David Medlock
|
||||
- Bryce Mehring
|
||||
- Jonathan Mercier
|
||||
- Marcel Metz
|
||||
- Liam Middlebrook
|
||||
- mightgoyardstill
|
||||
- Ave Milia
|
||||
- Icyllis Milica
|
||||
- Jonathan Miller
|
||||
- Kenneth Miller
|
||||
- Bruce Mitchener
|
||||
- Jack Moffitt
|
||||
- Ravi Mohan
|
||||
- Jeff Molofee
|
||||
- Alexander Monakov
|
||||
- Pierre Morel
|
||||
- Jon Morton
|
||||
- Pierre Moulon
|
||||
- Martins Mozeiko
|
||||
- Pascal Muetschard
|
||||
- James Murphy
|
||||
- Julian Møller
|
||||
- Julius Häger
|
||||
- Nat!
|
||||
- NateIsStalling
|
||||
- ndogxj
|
||||
- F. Nedelec
|
||||
- n3rdopolis
|
||||
- Kristian Nielsen
|
||||
- Joel Niemelä
|
||||
- Victor Nova
|
||||
- Kamil Nowakowski
|
||||
- onox
|
||||
- Denis Ovod
|
||||
- Ozzy
|
||||
- Andri Pálsson
|
||||
- luz paz
|
||||
- Peoro
|
||||
- Braden Pellett
|
||||
- Christopher Pelloux
|
||||
- Michael Pennington
|
||||
- Arturo J. Pérez
|
||||
- Vladimir Perminov
|
||||
- Olivier Perret
|
||||
- Anthony Pesch
|
||||
- Orson Peters
|
||||
- Emmanuel Gil Peyrot
|
||||
- Cyril Pichard
|
||||
- Pilzschaf
|
||||
- Keith Pitt
|
||||
- Stanislav Podgorskiy
|
||||
- Konstantin Podsvirov
|
||||
- Nathan Poirier
|
||||
- Pokechu22
|
||||
- Alexandre Pretyman
|
||||
- Pablo Prietz
|
||||
- przemekmirek
|
||||
- pthom
|
||||
- Martin Pulec
|
||||
- Guillaume Racicot
|
||||
- Juan Ramos
|
||||
- Christian Rauch
|
||||
- Philip Rideout
|
||||
- Eddie Ringle
|
||||
- Max Risuhin
|
||||
- Joe Roback
|
||||
- Jorge Rodriguez
|
||||
- Jari Ronkainen
|
||||
- Luca Rood
|
||||
- Ed Ropple
|
||||
- Aleksey Rybalkin
|
||||
- Mikko Rytkönen
|
||||
- Riku Salminen
|
||||
- Yoshinori Sano
|
||||
- Brandon Schaefer
|
||||
- Sebastian Schuberth
|
||||
- Scr3amer
|
||||
- Jan Schuerkamp
|
||||
- Christian Sdunek
|
||||
- Matt Sealey
|
||||
- Steve Sexton
|
||||
- Arkady Shapkin
|
||||
- Mingjie Shen
|
||||
- Ali Sherief
|
||||
- Yoshiki Shibukawa
|
||||
- Dmitri Shuralyov
|
||||
- Joao da Silva
|
||||
- Daniel Sieger
|
||||
- Daljit Singh
|
||||
- Michael Skec
|
||||
- Daniel Skorupski
|
||||
- Slemmie
|
||||
- Anthony Smith
|
||||
- Bradley Smith
|
||||
- Cliff Smolinsky
|
||||
- Patrick Snape
|
||||
- Erlend Sogge Heggen
|
||||
- Olivier Sohn
|
||||
- Julian Squires
|
||||
- Johannes Stein
|
||||
- Pontus Stenetorp
|
||||
- Michael Stocker
|
||||
- Justin Stoecker
|
||||
- Elviss Strazdins
|
||||
- Paul Sultana
|
||||
- Nathan Sweet
|
||||
- TTK-Bandit
|
||||
- Nuno Teixeira
|
||||
- Jared Tiala
|
||||
- Sergey Tikhomirov
|
||||
- Arthur Tombs
|
||||
- TronicLabs
|
||||
- Ioannis Tsakpinis
|
||||
- Samuli Tuomola
|
||||
- Matthew Turner
|
||||
- urraka
|
||||
- Elias Vanderstuyft
|
||||
- Stef Velzel
|
||||
- Jari Vetoniemi
|
||||
- Ricardo Vieira
|
||||
- Nicholas Vitovitch
|
||||
- Vladimír Vondruš
|
||||
- Simon Voordouw
|
||||
- Corentin Wallez
|
||||
- Torsten Walluhn
|
||||
- Patrick Walton
|
||||
- Jim Wang
|
||||
- Xo Wang
|
||||
- Andre Weissflog
|
||||
- Jay Weisskopf
|
||||
- Frank Wille
|
||||
- Andy Williams
|
||||
- Joel Winarske
|
||||
- Richard A. Wilkes
|
||||
- Tatsuya Yatagawa
|
||||
- Ryogo Yoshimura
|
||||
- Lukas Zanner
|
||||
- Andrey Zholos
|
||||
- Aihui Zhu
|
||||
- Santi Zupancic
|
||||
- Jonas Ådahl
|
||||
- Lasse Öörni
|
||||
- Leonard König
|
||||
- All the unmentioned and anonymous contributors in the GLFW community, for bug
|
||||
reports, patches, feedback, testing and encouragement
|
||||
|
23
dependencies/glfw-3.4/LICENSE.md
vendored
23
dependencies/glfw-3.4/LICENSE.md
vendored
@ -1,23 +0,0 @@
|
||||
Copyright (c) 2002-2006 Marcus Geelnard
|
||||
|
||||
Copyright (c) 2006-2019 Camilla Löwy
|
||||
|
||||
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.
|
||||
|
458
dependencies/glfw-3.4/README.md
vendored
458
dependencies/glfw-3.4/README.md
vendored
@ -1,458 +0,0 @@
|
||||
# GLFW
|
||||
|
||||
[](https://github.com/glfw/glfw/actions)
|
||||
[](https://ci.appveyor.com/project/elmindreda/glfw)
|
||||
|
||||
## Introduction
|
||||
|
||||
GLFW is an Open Source, multi-platform library for OpenGL, OpenGL ES and Vulkan
|
||||
application development. It provides a simple, platform-independent API for
|
||||
creating windows, contexts and surfaces, reading input, handling events, etc.
|
||||
|
||||
GLFW natively supports Windows, macOS and Linux and other Unix-like systems. On
|
||||
Linux both Wayland and X11 are supported.
|
||||
|
||||
GLFW is licensed under the [zlib/libpng
|
||||
license](https://www.glfw.org/license.html).
|
||||
|
||||
You can [download](https://www.glfw.org/download.html) the latest stable release
|
||||
as source or Windows binaries. Each release starting with 3.0 also has
|
||||
a corresponding [annotated tag](https://github.com/glfw/glfw/releases) with
|
||||
source and binary archives.
|
||||
|
||||
The [documentation](https://www.glfw.org/docs/latest/) is available online and is
|
||||
included in all source and binary archives. See the [release
|
||||
notes](https://www.glfw.org/docs/latest/news.html) for new features, caveats and
|
||||
deprecations in the latest release. For more details see the [version
|
||||
history](https://www.glfw.org/changelog.html).
|
||||
|
||||
The `master` branch is the stable integration branch and _should_ always compile
|
||||
and run on all supported platforms, although details of newly added features may
|
||||
change until they have been included in a release. New features and many bug
|
||||
fixes live in [other branches](https://github.com/glfw/glfw/branches/all) until
|
||||
they are stable enough to merge.
|
||||
|
||||
If you are new to GLFW, you may find the
|
||||
[tutorial](https://www.glfw.org/docs/latest/quick.html) for GLFW 3 useful. If
|
||||
you have used GLFW 2 in the past, there is a [transition
|
||||
guide](https://www.glfw.org/docs/latest/moving.html) for moving to the GLFW
|
||||
3 API.
|
||||
|
||||
GLFW exists because of the contributions of [many people](CONTRIBUTORS.md)
|
||||
around the world, whether by reporting bugs, providing community support, adding
|
||||
features, reviewing or testing code, debugging, proofreading docs, suggesting
|
||||
features or fixing bugs.
|
||||
|
||||
|
||||
## Compiling GLFW
|
||||
|
||||
GLFW is written primarily in C99, with parts of macOS support being written in
|
||||
Objective-C. GLFW itself requires only the headers and libraries for your OS
|
||||
and window system. It does not need any additional headers for context creation
|
||||
APIs (WGL, GLX, EGL, NSGL, OSMesa) or rendering APIs (OpenGL, OpenGL ES, Vulkan)
|
||||
to enable support for them.
|
||||
|
||||
GLFW supports compilation on Windows with Visual C++ 2013 and later, MinGW and
|
||||
MinGW-w64, on macOS with Clang and on Linux and other Unix-like systems with GCC
|
||||
and Clang. It will likely compile in other environments as well, but this is
|
||||
not regularly tested.
|
||||
|
||||
There are [pre-compiled binaries](https://www.glfw.org/download.html) available
|
||||
for all supported compilers on Windows and macOS.
|
||||
|
||||
See the [compilation guide](https://www.glfw.org/docs/latest/compile.html) for
|
||||
more information about how to compile GLFW yourself.
|
||||
|
||||
|
||||
## Using GLFW
|
||||
|
||||
See the [documentation](https://www.glfw.org/docs/latest/) for tutorials, guides
|
||||
and the API reference.
|
||||
|
||||
|
||||
## Contributing to GLFW
|
||||
|
||||
See the [contribution
|
||||
guide](https://github.com/glfw/glfw/blob/master/docs/CONTRIBUTING.md) for
|
||||
more information.
|
||||
|
||||
|
||||
## System requirements
|
||||
|
||||
GLFW supports Windows XP and later and macOS 10.8 and later. Linux and other
|
||||
Unix-like systems running the X Window System are supported even without
|
||||
a desktop environment or modern extensions, although some features require
|
||||
a running window or clipboard manager. The OSMesa backend requires Mesa 6.3.
|
||||
|
||||
See the [compatibility guide](https://www.glfw.org/docs/latest/compat.html)
|
||||
in the documentation for more information.
|
||||
|
||||
|
||||
## Dependencies
|
||||
|
||||
GLFW itself needs only CMake 3.1 or later and the headers and libraries for your
|
||||
OS and window system.
|
||||
|
||||
The examples and test programs depend on a number of tiny libraries. These are
|
||||
located in the `deps/` directory.
|
||||
|
||||
- [getopt\_port](https://github.com/kimgr/getopt_port/) for examples
|
||||
with command-line options
|
||||
- [TinyCThread](https://github.com/tinycthread/tinycthread) for threaded
|
||||
examples
|
||||
- [glad2](https://github.com/Dav1dde/glad) for loading OpenGL and Vulkan
|
||||
functions
|
||||
- [linmath.h](https://github.com/datenwolf/linmath.h) for linear algebra in
|
||||
examples
|
||||
- [Nuklear](https://github.com/Immediate-Mode-UI/Nuklear) for test and example UI
|
||||
- [stb\_image\_write](https://github.com/nothings/stb) for writing images to disk
|
||||
|
||||
The documentation is generated with [Doxygen](https://doxygen.org/) if CMake can
|
||||
find that tool.
|
||||
|
||||
|
||||
## Reporting bugs
|
||||
|
||||
Bugs are reported to our [issue tracker](https://github.com/glfw/glfw/issues).
|
||||
Please check the [contribution
|
||||
guide](https://github.com/glfw/glfw/blob/master/docs/CONTRIBUTING.md) for
|
||||
information on what to include when reporting a bug.
|
||||
|
||||
|
||||
## Changelog since 3.3.10
|
||||
|
||||
- Added `GLFW_PLATFORM` init hint for runtime platform selection (#1958)
|
||||
- Added `GLFW_ANY_PLATFORM`, `GLFW_PLATFORM_WIN32`, `GLFW_PLATFORM_COCOA`,
|
||||
`GLFW_PLATFORM_WAYLAND`, `GLFW_PLATFORM_X11` and `GLFW_PLATFORM_NULL` symbols to
|
||||
specify the desired platform (#1958)
|
||||
- Added `glfwGetPlatform` function to query what platform was selected (#1655,#1958)
|
||||
- Added `glfwPlatformSupported` function to query if a platform is supported
|
||||
(#1655,#1958)
|
||||
- Added `glfwInitAllocator` for setting a custom memory allocator (#544,#1628,#1947)
|
||||
- Added `GLFWallocator` struct and `GLFWallocatefun`, `GLFWreallocatefun` and
|
||||
`GLFWdeallocatefun` types (#544,#1628,#1947)
|
||||
- Added `glfwGetWindowTitle` function for querying window title (#1448,#1909,#2482)
|
||||
- Added `glfwInitVulkanLoader` for using a non-default Vulkan loader (#1374,#1890)
|
||||
- Added `GLFW_RESIZE_NWSE_CURSOR`, `GLFW_RESIZE_NESW_CURSOR`,
|
||||
`GLFW_RESIZE_ALL_CURSOR` and `GLFW_NOT_ALLOWED_CURSOR` cursor shapes (#427)
|
||||
- Added `GLFW_RESIZE_EW_CURSOR` alias for `GLFW_HRESIZE_CURSOR` (#427)
|
||||
- Added `GLFW_RESIZE_NS_CURSOR` alias for `GLFW_VRESIZE_CURSOR` (#427)
|
||||
- Added `GLFW_POINTING_HAND_CURSOR` alias for `GLFW_HAND_CURSOR` (#427)
|
||||
- Added `GLFW_MOUSE_PASSTHROUGH` window hint for letting mouse input pass
|
||||
through the window (#1236,#1568)
|
||||
- Added `GLFW_CURSOR_CAPTURED` cursor mode to confine the cursor to the window
|
||||
content area (#58)
|
||||
- Added `GLFW_POSITION_X` and `GLFW_POSITION_Y` window hints for initial position
|
||||
(#1603,#1747)
|
||||
- Added `GLFW_SCALE_FRAMEBUFFER` window hint for Wayland and macOS scaling (#2457)
|
||||
- Added `GLFW_ANY_POSITION` hint value for letting the window manager choose (#1603,#1747)
|
||||
- Added `GLFW_PLATFORM_UNAVAILABLE` error for platform detection failures (#1958)
|
||||
- Added `GLFW_FEATURE_UNAVAILABLE` error for platform limitations (#1692)
|
||||
- Added `GLFW_FEATURE_UNIMPLEMENTED` error for incomplete backends (#1692)
|
||||
- Added `GLFW_WAYLAND_APP_ID` window hint string for Wayland app\_id selection
|
||||
(#2121,#2122)
|
||||
- Added `GLFW_ANGLE_PLATFORM_TYPE` init hint and `GLFW_ANGLE_PLATFORM_TYPE_*`
|
||||
values to select ANGLE backend (#1380)
|
||||
- Added `GLFW_X11_XCB_VULKAN_SURFACE` init hint for selecting X11 Vulkan
|
||||
surface extension (#1793)
|
||||
- Added `GLFW_WIN32_KEYBOARD_MENU` window hint for enabling access to the window menu
|
||||
- Added `GLFW_WIN32_SHOWDEFAULT` window hint for applying the parent process
|
||||
show command (#2359)
|
||||
- Added `GLFW_NATIVE_INCLUDE_NONE` for disabling inclusion of native headers (#1348)
|
||||
- Added `GLFW_BUILD_WIN32` CMake option for enabling Win32 support (#1958)
|
||||
- Added `GLFW_BUILD_COCOA` CMake option for enabling Cocoa support (#1958)
|
||||
- Added `GLFW_BUILD_X11` CMake option for enabling X11 support (#1958)
|
||||
- Added `GLFW_LIBRARY_TYPE` CMake variable for overriding the library type
|
||||
(#279,#1307,#1497,#1574,#1928)
|
||||
- Added support for `XDG_SESSION_TYPE` environment variable
|
||||
- Added `GLFW_PKG_CONFIG_REQUIRES_PRIVATE` and `GLFW_PKG_CONFIG_LIBS_PRIVATE` CMake
|
||||
variables exposing pkg-config dependencies (#1307)
|
||||
- Made joystick subsystem initialize at first use (#1284,#1646)
|
||||
- Made `GLFW_DOUBLEBUFFER` a read-only window attribute
|
||||
- Made Wayland the preferred platform over X11 if both are available (#2035)
|
||||
- Updated the minimum required CMake version to 3.4
|
||||
- Updated gamepad mappings from upstream
|
||||
- Renamed `GLFW_USE_WAYLAND` CMake option to `GLFW_BUILD_WAYLAND` (#1958)
|
||||
- Disabled tests and examples by default when built as a CMake subdirectory
|
||||
- Removed `GLFW_USE_OSMESA` CMake option enabling the Null platform (#1958)
|
||||
- Removed CMake generated configuration header
|
||||
- Bugfix: `glfwGetVideoMode` returned an invalid mode on error (#1292)
|
||||
- [Win32] Added a version info resource to the GLFW DLL
|
||||
- [Win32] Made hidden helper window use its own window class
|
||||
- [Win32] Bugfix: The foreground lock timeout was overridden, ignoring the user
|
||||
- [Cocoa] Added `glfwGetCocoaView` native access function (#2235)
|
||||
- [Cocoa] Moved main menu creation to GLFW initialization time (#1649)
|
||||
- [Cocoa] Bugfix: Touching event queue from secondary thread before main thread
|
||||
would abort (#1649)
|
||||
- [Wayland] Added support for `glfwRequestWindowAttention` (#2287)
|
||||
- [Wayland] Added support for `glfwFocusWindow`
|
||||
- [Wayland] Added support for `GLFW_RESIZABLE` (#2203)
|
||||
- [Wayland] Added support for fractional scaling of window contents
|
||||
- [Wayland] Added dynamic loading of all Wayland libraries
|
||||
- [Wayland] Bugfix: `CLOCK_MONOTONIC` was not correctly enabled
|
||||
- [Wayland] Bugfix: `GLFW_HOVERED` was true when the cursor was over any
|
||||
fallback window decoration
|
||||
- [Wayland] Bugfix: Fallback decorations allowed resizing to invalid size
|
||||
(#2204)
|
||||
- [X11] Bugfix: Termination would segfault if the IM had been destroyed
|
||||
- [X11] Bugfix: Any IM started after initialization would not be detected
|
||||
- [Linux] Bugfix: Joystick evdev fds remained open in forks (#2446)
|
||||
- [POSIX] Removed use of deprecated function `gettimeofday`
|
||||
- [POSIX] Bugfix: `CLOCK_MONOTONIC` was not correctly tested for or enabled
|
||||
- [WGL] Disabled the DWM swap interval hack for Windows 8 and later (#1072)
|
||||
- [NSGL] Removed enforcement of forward-compatible flag for core contexts
|
||||
- [NSGL] Bugfix: A core profile OpenGL context was returned if 3.2+
|
||||
compatibility profile was requested
|
||||
- [EGL] Added platform selection via the `EGL_EXT_platform_base` extension
|
||||
(#442)
|
||||
- [EGL] Added ANGLE backend selection via `EGL_ANGLE_platform_angle` extension
|
||||
(#1380)
|
||||
|
||||
|
||||
## Changelog since 3.3
|
||||
|
||||
- Added `GLFW_WAYLAND_LIBDECOR` init hint for disabling libdecor support (#1639,#1693)
|
||||
- Bugfix: The CMake config-file package used an absolute path and was not
|
||||
relocatable (#1470)
|
||||
- Bugfix: Video modes with a duplicate screen area were discarded (#1555,#1556)
|
||||
- Bugfix: Compiling with -Wextra-semi caused warnings (#1440)
|
||||
- Bugfix: Built-in mappings failed because some OEMs re-used VID/PID (#1583)
|
||||
- Bugfix: Some extension loader headers did not prevent default OpenGL header
|
||||
inclusion (#1695)
|
||||
- Bugfix: Buffers were swapped at creation on single-buffered windows (#1873)
|
||||
- Bugfix: Gamepad mapping updates could spam `GLFW_INVALID_VALUE` due to
|
||||
incompatible controllers sharing hardware ID (#1763)
|
||||
- Bugfix: Native access functions for context handles did not check that the API matched
|
||||
- Bugfix: `glfwMakeContextCurrent` would access TLS slot before initialization
|
||||
- Bugfix: `glfwSetGammaRamp` could emit `GLFW_INVALID_VALUE` before initialization
|
||||
- Bugfix: `glfwGetJoystickUserPointer` returned `NULL` during disconnection (#2092)
|
||||
- Bugfix: `glfwGetKeyScancode` returned `0` on error when initialized instead of `-1`
|
||||
- Bugfix: Failure to make a newly created context current could cause segfault (#2327)
|
||||
- [Win32] Disabled framebuffer transparency on Windows 7 when DWM windows are
|
||||
opaque (#1512)
|
||||
- [Win32] Bugfix: `GLFW_INCLUDE_VULKAN` plus `VK_USE_PLATFORM_WIN32_KHR` caused
|
||||
symbol redefinition (#1524)
|
||||
- [Win32] Bugfix: The cursor position event was emitted before its cursor enter
|
||||
event (#1490)
|
||||
- [Win32] Bugfix: The window hint `GLFW_MAXIMIZED` did not move or resize the
|
||||
window (#1499)
|
||||
- [Win32] Bugfix: Disabled cursor mode interfered with some non-client actions
|
||||
- [Win32] Bugfix: Super key was not released after Win+V hotkey (#1622)
|
||||
- [Win32] Bugfix: `glfwGetKeyName` could access out of bounds and return an
|
||||
invalid pointer
|
||||
- [Win32] Bugfix: Some synthetic key events were reported as `GLFW_KEY_UNKNOWN`
|
||||
(#1623)
|
||||
- [Win32] Bugfix: Non-BMP Unicode codepoint input was reported as UTF-16
|
||||
- [Win32] Bugfix: Monitor functions could return invalid values after
|
||||
configuration change (#1761)
|
||||
- [Win32] Bugfix: Initialization would segfault on Windows 8 (not 8.1) (#1775)
|
||||
- [Win32] Bugfix: Duplicate size events were not filtered (#1610)
|
||||
- [Win32] Bugfix: Full screen windows were incorrectly resized by DPI changes
|
||||
(#1582)
|
||||
- [Win32] Bugfix: `GLFW_SCALE_TO_MONITOR` had no effect on systems older than
|
||||
Windows 10 version 1703 (#1511)
|
||||
- [Win32] Bugfix: `USE_MSVC_RUNTIME_LIBRARY_DLL` had no effect on CMake 3.15 or
|
||||
later (#1783,#1796)
|
||||
- [Win32] Bugfix: Compilation with LLVM for Windows failed (#1807,#1824,#1874)
|
||||
- [Win32] Bugfix: Content scale queries could fail silently (#1615)
|
||||
- [Win32] Bugfix: Content scales could have garbage values if monitor was recently
|
||||
disconnected (#1615)
|
||||
- [Win32] Bugfix: A window created maximized and undecorated would cover the whole
|
||||
monitor (#1806)
|
||||
- [Win32] Bugfix: The default restored window position was lost when creating a maximized
|
||||
window
|
||||
- [Win32] Bugfix: `glfwMaximizeWindow` would make a hidden window visible
|
||||
- [Win32] Bugfix: `Alt+PrtSc` would emit `GLFW_KEY_UNKNOWN` and a different
|
||||
scancode than `PrtSc` (#1993)
|
||||
- [Win32] Bugfix: `GLFW_KEY_PAUSE` scancode from `glfwGetKeyScancode` did not
|
||||
match event scancode (#1993)
|
||||
- [Win32] Bugfix: Instance-local operations used executable instance (#469,#1296,#1395)
|
||||
- [Win32] Bugfix: The OSMesa library was not unloaded on termination
|
||||
- [Win32] Bugfix: Right shift emitted `GLFW_KEY_UNKNOWN` when using a CJK IME (#2050)
|
||||
- [Win32] Bugfix: `glfwWaitEventsTimeout` did not return for some sent messages (#2408)
|
||||
- [Win32] Bugfix: Fix pkg-config for dynamic library on Windows (#2386, #2420)
|
||||
- [Win32] Bugfix: XInput could reportedly provide invalid DPad bit masks (#2291)
|
||||
- [Win32] Bugfix: Rapid clipboard calls could fail due to Clipboard History
|
||||
- [Win32] Bugfix: Disabled cursor mode doesn't work right when connected over RDP (#1276,#1279,#2431)
|
||||
- [Cocoa] Added support for `VK_EXT_metal_surface` (#1619)
|
||||
- [Cocoa] Added locating the Vulkan loader at runtime in an application bundle
|
||||
- [Cocoa] Changed `EGLNativeWindowType` from `NSView` to `CALayer` (#1169)
|
||||
- [Cocoa] Changed F13 key to report Print Screen for cross-platform consistency
|
||||
(#1786)
|
||||
- [Cocoa] Disabled macOS fullscreen when `GLFW_RESIZABLE` is false
|
||||
- [Cocoa] Removed dependency on the CoreVideo framework
|
||||
- [Cocoa] Bugfix: `glfwSetWindowSize` used a bottom-left anchor point (#1553)
|
||||
- [Cocoa] Bugfix: Window remained on screen after destruction until event poll
|
||||
(#1412)
|
||||
- [Cocoa] Bugfix: Event processing before window creation would assert (#1543)
|
||||
- [Cocoa] Bugfix: Undecorated windows could not be iconified on recent macOS
|
||||
- [Cocoa] Bugfix: Non-BMP Unicode codepoint input was reported as UTF-16
|
||||
(#1635)
|
||||
- [Cocoa] Bugfix: Failing to retrieve the refresh rate of built-in displays
|
||||
could leak memory
|
||||
- [Cocoa] Bugfix: Objective-C files were compiled as C with CMake 3.19 (#1787)
|
||||
- [Cocoa] Bugfix: Duplicate video modes were not filtered out (#1830)
|
||||
- [Cocoa] Bugfix: Menu bar was not clickable on macOS 10.15+ until it lost and
|
||||
regained focus (#1648,#1802)
|
||||
- [Cocoa] Bugfix: Monitor name query could segfault on macOS 11 (#1809,#1833)
|
||||
- [Cocoa] Bugfix: The install name of the installed dylib was relative (#1504)
|
||||
- [Cocoa] Bugfix: The MoltenVK layer contents scale was updated only after
|
||||
related events were emitted
|
||||
- [Cocoa] Bugfix: Moving the cursor programmatically would freeze it for
|
||||
a fraction of a second (#1962)
|
||||
- [Cocoa] Bugfix: `kIOMasterPortDefault` was deprecated in macOS 12.0 (#1980)
|
||||
- [Cocoa] Bugfix: `kUTTypeURL` was deprecated in macOS 12.0 (#2003)
|
||||
- [Cocoa] Bugfix: A connected Apple AirPlay would emit a useless error (#1791)
|
||||
- [Cocoa] Bugfix: The EGL and OSMesa libraries were not unloaded on termination
|
||||
- [Cocoa] Bugfix: `GLFW_MAXIMIZED` was always true when `GLFW_RESIZABLE` was false
|
||||
- [Cocoa] Bugfix: Changing `GLFW_DECORATED` in macOS fullscreen would abort
|
||||
application (#1886)
|
||||
- [Cocoa] Bugfix: Setting a monitor from macOS fullscreen would abort
|
||||
application (#2110)
|
||||
- [Cocoa] Bugfix: The Vulkan loader was not loaded from the `Frameworks` bundle
|
||||
subdirectory (#2113,#2120)
|
||||
- [Cocoa] Bugfix: Compilation failed on OS X 10.8 due to unconditional use of 10.9+
|
||||
symbols (#2161)
|
||||
- [Cocoa] Bugfix: Querying joystick elements could reportedly segfault on macOS
|
||||
13 Ventura (#2320)
|
||||
- [X11] Bugfix: The CMake files did not check for the XInput headers (#1480)
|
||||
- [X11] Bugfix: Key names were not updated when the keyboard layout changed
|
||||
(#1462,#1528)
|
||||
- [X11] Bugfix: Decorations could not be enabled after window creation (#1566)
|
||||
- [X11] Bugfix: Content scale fallback value could be inconsistent (#1578)
|
||||
- [X11] Bugfix: `glfwMaximizeWindow` had no effect on hidden windows
|
||||
- [X11] Bugfix: Clearing `GLFW_FLOATING` on a hidden window caused invalid read
|
||||
- [X11] Bugfix: Changing `GLFW_FLOATING` on a hidden window could silently fail
|
||||
- [X11] Bugfix: Disabled cursor mode was interrupted by indicator windows
|
||||
- [X11] Bugfix: Monitor physical dimensions could be reported as zero mm
|
||||
- [X11] Bugfix: Window position events were not emitted during resizing (#1613)
|
||||
- [X11] Bugfix: `glfwFocusWindow` could terminate on older WMs or without a WM
|
||||
- [X11] Bugfix: Querying a disconnected monitor could segfault (#1602)
|
||||
- [X11] Bugfix: IME input of CJK was broken for "C" locale (#1587,#1636)
|
||||
- [X11] Bugfix: Xlib errors caused by other parts of the application could be
|
||||
reported as GLFW errors
|
||||
- [X11] Bugfix: A handle race condition could cause a `BadWindow` error (#1633)
|
||||
- [X11] Bugfix: XKB path used keysyms instead of physical locations for
|
||||
non-printable keys (#1598)
|
||||
- [X11] Bugfix: Function keys were mapped to `GLFW_KEY_UNKNOWN` for some layout
|
||||
combinations (#1598)
|
||||
- [X11] Bugfix: Keys pressed simultaneously with others were not always
|
||||
reported (#1112,#1415,#1472,#1616)
|
||||
- [X11] Bugfix: Some window attributes were not applied on leaving fullscreen
|
||||
(#1863)
|
||||
- [X11] Bugfix: Changing `GLFW_FLOATING` could leak memory
|
||||
- [X11] Bugfix: Icon pixel format conversion worked only by accident, relying on
|
||||
undefined behavior (#1986)
|
||||
- [X11] Bugfix: Dynamic loading on OpenBSD failed due to soname differences
|
||||
- [X11] Bugfix: Waiting for events would fail if file descriptor was too large
|
||||
(#2024)
|
||||
- [X11] Bugfix: Joystick events could lead to busy-waiting (#1872)
|
||||
- [X11] Bugfix: `glfwWaitEvents*` did not continue for joystick events
|
||||
- [X11] Bugfix: `glfwPostEmptyEvent` could be ignored due to race condition
|
||||
(#379,#1281,#1285,#2033)
|
||||
- [X11] Bugfix: Dynamic loading on NetBSD failed due to soname differences
|
||||
- [X11] Bugfix: Left shift of int constant relied on undefined behavior (#1951)
|
||||
- [X11] Bugfix: The OSMesa libray was not unloaded on termination
|
||||
- [X11] Bugfix: A malformed response during selection transfer could cause a segfault
|
||||
- [X11] Bugfix: Some calls would reset Xlib to the default error handler (#2108)
|
||||
- [Wayland] Added improved fallback window decorations via libdecor (#1639,#1693)
|
||||
- [Wayland] Added support for key names via xkbcommon
|
||||
- [Wayland] Added support for file path drop events (#2040)
|
||||
- [Wayland] Added support for more human-readable monitor names where available
|
||||
- [Wayland] Disabled alpha channel for opaque windows on systems lacking
|
||||
`EGL_EXT_present_opaque` (#1895)
|
||||
- [Wayland] Removed support for `wl_shell` (#1443)
|
||||
- [Wayland] Bugfix: The `GLFW_HAND_CURSOR` shape used the wrong image (#1432)
|
||||
- [Wayland] Bugfix: Repeated keys could be reported with `NULL` window (#1704)
|
||||
- [Wayland] Bugfix: Retrieving partial framebuffer size would segfault
|
||||
- [Wayland] Bugfix: Scrolling offsets were inverted compared to other platforms
|
||||
(#1463)
|
||||
- [Wayland] Bugfix: Client-Side Decorations were destroyed in the wrong order
|
||||
(#1798)
|
||||
- [Wayland] Bugfix: Monitors physical size could report zero (#1784,#1792)
|
||||
- [Wayland] Bugfix: Some keys were not repeating in Wayland (#1908)
|
||||
- [Wayland] Bugfix: Non-arrow cursors are offset from the hotspot (#1706,#1899)
|
||||
- [Wayland] Bugfix: The `O_CLOEXEC` flag was not defined on FreeBSD
|
||||
- [Wayland] Bugfix: Key repeat could lead to a race condition (#1710)
|
||||
- [Wayland] Bugfix: Activating a window would emit two input focus events
|
||||
- [Wayland] Bugfix: Disable key repeat mechanism when window loses input focus
|
||||
- [Wayland] Bugfix: Window hiding and showing did not work (#1492,#1731)
|
||||
- [Wayland] Bugfix: A key being repeated was not released when window lost focus
|
||||
- [Wayland] Bugfix: Showing a hidden window did not emit a window refresh event
|
||||
- [Wayland] Bugfix: Full screen window creation did not ignore `GLFW_VISIBLE`
|
||||
- [Wayland] Bugfix: Some keys were reported as wrong key or `GLFW_KEY_UNKNOWN`
|
||||
- [Wayland] Bugfix: Text input did not repeat along with key repeat
|
||||
- [Wayland] Bugfix: `glfwPostEmptyEvent` sometimes had no effect (#1520,#1521)
|
||||
- [Wayland] Bugfix: `glfwSetClipboardString` would fail if set to result of
|
||||
`glfwGetClipboardString`
|
||||
- [Wayland] Bugfix: Data source creation error would cause double free at termination
|
||||
- [Wayland] Bugfix: Partial writes of clipboard string would cause beginning to repeat
|
||||
- [Wayland] Bugfix: Some errors would cause clipboard string transfer to hang
|
||||
- [Wayland] Bugfix: Drag and drop data was misinterpreted as clipboard string
|
||||
- [Wayland] Bugfix: MIME type matching was not performed for clipboard string
|
||||
- [Wayland] Bugfix: The OSMesa library was not unloaded on termination
|
||||
- [Wayland] Bugfix: `glfwCreateWindow` could emit `GLFW_FEATURE_UNAVAILABLE`
|
||||
- [Wayland] Bugfix: Lock key modifier bits were only set when lock keys were pressed
|
||||
- [Wayland] Bugfix: A window leaving full screen mode would be iconified (#1995)
|
||||
- [Wayland] Bugfix: A window leaving full screen mode ignored its desired size
|
||||
- [Wayland] Bugfix: `glfwSetWindowMonitor` did not update windowed mode size
|
||||
- [Wayland] Bugfix: `glfwRestoreWindow` would make a full screen window windowed
|
||||
- [Wayland] Bugfix: A window maximized or restored by the user would enter an
|
||||
inconsistent state
|
||||
- [Wayland] Bugfix: Window maximization events were not emitted
|
||||
- [Wayland] Bugfix: `glfwRestoreWindow` assumed it was always in windowed mode
|
||||
- [Wayland] Bugfix: `glfwSetWindowSize` would resize a full screen window
|
||||
- [Wayland] Bugfix: A window content scale event would be emitted every time
|
||||
the window resized
|
||||
- [Wayland] Bugfix: If `glfwInit` failed it would close stdin
|
||||
- [Wayland] Bugfix: Manual resizing with fallback decorations behaved erratically
|
||||
(#1991,#2115,#2127)
|
||||
- [Wayland] Bugfix: Size limits included frame size for fallback decorations
|
||||
- [Wayland] Bugfix: Updating `GLFW_DECORATED` had no effect on server-side
|
||||
decorations
|
||||
- [Wayland] Bugfix: A monitor would be reported as connected again if its scale
|
||||
changed
|
||||
- [Wayland] Bugfix: `glfwTerminate` would segfault if any monitor had changed
|
||||
scale
|
||||
- [Wayland] Bugfix: Window content scale events were not emitted when monitor
|
||||
scale changed
|
||||
- [Wayland] Bugfix: `glfwSetWindowAspectRatio` reported an error instead of
|
||||
applying the specified ratio
|
||||
- [Wayland] Bugfix: `GLFW_MAXIMIZED` window hint had no effect
|
||||
- [Wayland] Bugfix: `glfwRestoreWindow` had no effect before first show
|
||||
- [Wayland] Bugfix: Hiding and then showing a window caused program abort on
|
||||
wlroots compositors (#1268)
|
||||
- [Wayland] Bugfix: `GLFW_DECORATED` was ignored when showing a window with XDG
|
||||
decorations
|
||||
- [Wayland] Bugfix: Connecting a mouse after `glfwInit` would segfault (#1450)
|
||||
- [Wayland] Bugfix: Joysticks connected after `glfwInit` were not detected (#2198)
|
||||
- [Wayland] Bugfix: Fallback decorations emitted `GLFW_CURSOR_UNAVAILABLE` errors
|
||||
- [Linux] Bugfix: Joysticks without buttons were ignored (#2042,#2043)
|
||||
- [Linux] Bugfix: A small amount of memory could leak if initialization failed (#2229)
|
||||
- [NSGL] Bugfix: `GLFW_COCOA_RETINA_FRAMEBUFFER` had no effect on newer
|
||||
macOS versions (#1442)
|
||||
- [NSGL] Bugfix: Workaround for swap interval on 10.14 broke on 10.12 (#1483)
|
||||
- [NSGL] Bugfix: Defining `GL_SILENCE_DEPRECATION` externally caused
|
||||
a duplicate definition warning (#1840)
|
||||
- [EGL] Added loading of glvnd `libOpenGL.so.0` where available for OpenGL
|
||||
- [EGL] Bugfix: The `GLFW_DOUBLEBUFFER` context attribute was ignored (#1843)
|
||||
- [EGL] Bugfix: Setting `GLFW_CONTEXT_DEBUG` caused creation to fail (#2348)
|
||||
- [GLX] Added loading of glvnd `libGLX.so.0` where available
|
||||
- [GLX] Bugfix: Context creation failed if GLX 1.4 was not exported by GLX library
|
||||
|
||||
|
||||
## Contact
|
||||
|
||||
On [glfw.org](https://www.glfw.org/) you can find the latest version of GLFW, as
|
||||
well as news, documentation and other information about the project.
|
||||
|
||||
If you have questions related to the use of GLFW, we have a
|
||||
[forum](https://discourse.glfw.org/).
|
||||
|
||||
If you have a bug to report, a patch to submit or a feature you'd like to
|
||||
request, please file it in the
|
||||
[issue tracker](https://github.com/glfw/glfw/issues) on GitHub.
|
||||
|
||||
Finally, if you're interested in helping out with the development of GLFW or
|
||||
porting it to your favorite platform, join us on the forum or GitHub.
|
||||
|
230
dependencies/glfw-3.4/deps/getopt.c
vendored
230
dependencies/glfw-3.4/deps/getopt.c
vendored
@ -1,230 +0,0 @@
|
||||
/* Copyright (c) 2012, Kim Gräsman
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions are met:
|
||||
* * Redistributions of source code must retain the above copyright notice,
|
||||
* this list of conditions and the following disclaimer.
|
||||
* * Redistributions in binary form must reproduce the above copyright notice,
|
||||
* this list of conditions and the following disclaimer in the documentation
|
||||
* and/or other materials provided with the distribution.
|
||||
* * Neither the name of Kim Gräsman nor the names of contributors may be used
|
||||
* to endorse or promote products derived from this software without specific
|
||||
* prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||
* ARE DISCLAIMED. IN NO EVENT SHALL KIM GRÄSMAN BE LIABLE FOR ANY DIRECT,
|
||||
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include "getopt.h"
|
||||
|
||||
#include <stddef.h>
|
||||
#include <string.h>
|
||||
|
||||
const int no_argument = 0;
|
||||
const int required_argument = 1;
|
||||
const int optional_argument = 2;
|
||||
|
||||
char* optarg;
|
||||
int optopt;
|
||||
/* The variable optind [...] shall be initialized to 1 by the system. */
|
||||
int optind = 1;
|
||||
int opterr;
|
||||
|
||||
static char* optcursor = NULL;
|
||||
|
||||
/* Implemented based on [1] and [2] for optional arguments.
|
||||
optopt is handled FreeBSD-style, per [3].
|
||||
Other GNU and FreeBSD extensions are purely accidental.
|
||||
|
||||
[1] http://pubs.opengroup.org/onlinepubs/000095399/functions/getopt.html
|
||||
[2] http://www.kernel.org/doc/man-pages/online/pages/man3/getopt.3.html
|
||||
[3] http://www.freebsd.org/cgi/man.cgi?query=getopt&sektion=3&manpath=FreeBSD+9.0-RELEASE
|
||||
*/
|
||||
int getopt(int argc, char* const argv[], const char* optstring) {
|
||||
int optchar = -1;
|
||||
const char* optdecl = NULL;
|
||||
|
||||
optarg = NULL;
|
||||
opterr = 0;
|
||||
optopt = 0;
|
||||
|
||||
/* Unspecified, but we need it to avoid overrunning the argv bounds. */
|
||||
if (optind >= argc)
|
||||
goto no_more_optchars;
|
||||
|
||||
/* If, when getopt() is called argv[optind] is a null pointer, getopt()
|
||||
shall return -1 without changing optind. */
|
||||
if (argv[optind] == NULL)
|
||||
goto no_more_optchars;
|
||||
|
||||
/* If, when getopt() is called *argv[optind] is not the character '-',
|
||||
getopt() shall return -1 without changing optind. */
|
||||
if (*argv[optind] != '-')
|
||||
goto no_more_optchars;
|
||||
|
||||
/* If, when getopt() is called argv[optind] points to the string "-",
|
||||
getopt() shall return -1 without changing optind. */
|
||||
if (strcmp(argv[optind], "-") == 0)
|
||||
goto no_more_optchars;
|
||||
|
||||
/* If, when getopt() is called argv[optind] points to the string "--",
|
||||
getopt() shall return -1 after incrementing optind. */
|
||||
if (strcmp(argv[optind], "--") == 0) {
|
||||
++optind;
|
||||
goto no_more_optchars;
|
||||
}
|
||||
|
||||
if (optcursor == NULL || *optcursor == '\0')
|
||||
optcursor = argv[optind] + 1;
|
||||
|
||||
optchar = *optcursor;
|
||||
|
||||
/* FreeBSD: The variable optopt saves the last known option character
|
||||
returned by getopt(). */
|
||||
optopt = optchar;
|
||||
|
||||
/* The getopt() function shall return the next option character (if one is
|
||||
found) from argv that matches a character in optstring, if there is
|
||||
one that matches. */
|
||||
optdecl = strchr(optstring, optchar);
|
||||
if (optdecl) {
|
||||
/* [I]f a character is followed by a colon, the option takes an
|
||||
argument. */
|
||||
if (optdecl[1] == ':') {
|
||||
optarg = ++optcursor;
|
||||
if (*optarg == '\0') {
|
||||
/* GNU extension: Two colons mean an option takes an
|
||||
optional arg; if there is text in the current argv-element
|
||||
(i.e., in the same word as the option name itself, for example,
|
||||
"-oarg"), then it is returned in optarg, otherwise optarg is set
|
||||
to zero. */
|
||||
if (optdecl[2] != ':') {
|
||||
/* If the option was the last character in the string pointed to by
|
||||
an element of argv, then optarg shall contain the next element
|
||||
of argv, and optind shall be incremented by 2. If the resulting
|
||||
value of optind is greater than argc, this indicates a missing
|
||||
option-argument, and getopt() shall return an error indication.
|
||||
|
||||
Otherwise, optarg shall point to the string following the
|
||||
option character in that element of argv, and optind shall be
|
||||
incremented by 1.
|
||||
*/
|
||||
if (++optind < argc) {
|
||||
optarg = argv[optind];
|
||||
} else {
|
||||
/* If it detects a missing option-argument, it shall return the
|
||||
colon character ( ':' ) if the first character of optstring
|
||||
was a colon, or a question-mark character ( '?' ) otherwise.
|
||||
*/
|
||||
optarg = NULL;
|
||||
optchar = (optstring[0] == ':') ? ':' : '?';
|
||||
}
|
||||
} else {
|
||||
optarg = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
optcursor = NULL;
|
||||
}
|
||||
} else {
|
||||
/* If getopt() encounters an option character that is not contained in
|
||||
optstring, it shall return the question-mark ( '?' ) character. */
|
||||
optchar = '?';
|
||||
}
|
||||
|
||||
if (optcursor == NULL || *++optcursor == '\0')
|
||||
++optind;
|
||||
|
||||
return optchar;
|
||||
|
||||
no_more_optchars:
|
||||
optcursor = NULL;
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* Implementation based on [1].
|
||||
|
||||
[1] http://www.kernel.org/doc/man-pages/online/pages/man3/getopt.3.html
|
||||
*/
|
||||
int getopt_long(int argc, char* const argv[], const char* optstring,
|
||||
const struct option* longopts, int* longindex) {
|
||||
const struct option* o = longopts;
|
||||
const struct option* match = NULL;
|
||||
int num_matches = 0;
|
||||
size_t argument_name_length = 0;
|
||||
const char* current_argument = NULL;
|
||||
int retval = -1;
|
||||
|
||||
optarg = NULL;
|
||||
optopt = 0;
|
||||
|
||||
if (optind >= argc)
|
||||
return -1;
|
||||
|
||||
if (strlen(argv[optind]) < 3 || strncmp(argv[optind], "--", 2) != 0)
|
||||
return getopt(argc, argv, optstring);
|
||||
|
||||
/* It's an option; starts with -- and is longer than two chars. */
|
||||
current_argument = argv[optind] + 2;
|
||||
argument_name_length = strcspn(current_argument, "=");
|
||||
for (; o->name; ++o) {
|
||||
if (strncmp(o->name, current_argument, argument_name_length) == 0) {
|
||||
match = o;
|
||||
++num_matches;
|
||||
}
|
||||
}
|
||||
|
||||
if (num_matches == 1) {
|
||||
/* If longindex is not NULL, it points to a variable which is set to the
|
||||
index of the long option relative to longopts. */
|
||||
if (longindex)
|
||||
*longindex = (int) (match - longopts);
|
||||
|
||||
/* If flag is NULL, then getopt_long() shall return val.
|
||||
Otherwise, getopt_long() returns 0, and flag shall point to a variable
|
||||
which shall be set to val if the option is found, but left unchanged if
|
||||
the option is not found. */
|
||||
if (match->flag)
|
||||
*(match->flag) = match->val;
|
||||
|
||||
retval = match->flag ? 0 : match->val;
|
||||
|
||||
if (match->has_arg != no_argument) {
|
||||
optarg = strchr(argv[optind], '=');
|
||||
if (optarg != NULL)
|
||||
++optarg;
|
||||
|
||||
if (match->has_arg == required_argument) {
|
||||
/* Only scan the next argv for required arguments. Behavior is not
|
||||
specified, but has been observed with Ubuntu and Mac OSX. */
|
||||
if (optarg == NULL && ++optind < argc) {
|
||||
optarg = argv[optind];
|
||||
}
|
||||
|
||||
if (optarg == NULL)
|
||||
retval = ':';
|
||||
}
|
||||
} else if (strchr(argv[optind], '=')) {
|
||||
/* An argument was provided to a non-argument option.
|
||||
I haven't seen this specified explicitly, but both GNU and BSD-based
|
||||
implementations show this behavior.
|
||||
*/
|
||||
retval = '?';
|
||||
}
|
||||
} else {
|
||||
/* Unknown option or ambiguous match. */
|
||||
retval = '?';
|
||||
}
|
||||
|
||||
++optind;
|
||||
return retval;
|
||||
}
|
57
dependencies/glfw-3.4/deps/getopt.h
vendored
57
dependencies/glfw-3.4/deps/getopt.h
vendored
@ -1,57 +0,0 @@
|
||||
/* Copyright (c) 2012, Kim Gräsman
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions are met:
|
||||
* * Redistributions of source code must retain the above copyright notice,
|
||||
* this list of conditions and the following disclaimer.
|
||||
* * Redistributions in binary form must reproduce the above copyright notice,
|
||||
* this list of conditions and the following disclaimer in the documentation
|
||||
* and/or other materials provided with the distribution.
|
||||
* * Neither the name of Kim Gräsman nor the names of contributors may be used
|
||||
* to endorse or promote products derived from this software without specific
|
||||
* prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||
* ARE DISCLAIMED. IN NO EVENT SHALL KIM GRÄSMAN BE LIABLE FOR ANY DIRECT,
|
||||
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#ifndef INCLUDED_GETOPT_PORT_H
|
||||
#define INCLUDED_GETOPT_PORT_H
|
||||
|
||||
#if defined(__cplusplus)
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
extern const int no_argument;
|
||||
extern const int required_argument;
|
||||
extern const int optional_argument;
|
||||
|
||||
extern char* optarg;
|
||||
extern int optind, opterr, optopt;
|
||||
|
||||
struct option {
|
||||
const char* name;
|
||||
int has_arg;
|
||||
int* flag;
|
||||
int val;
|
||||
};
|
||||
|
||||
int getopt(int argc, char* const argv[], const char* optstring);
|
||||
|
||||
int getopt_long(int argc, char* const argv[],
|
||||
const char* optstring, const struct option* longopts, int* longindex);
|
||||
|
||||
#if defined(__cplusplus)
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif // INCLUDED_GETOPT_PORT_H
|
5996
dependencies/glfw-3.4/deps/glad/gl.h
vendored
5996
dependencies/glfw-3.4/deps/glad/gl.h
vendored
File diff suppressed because it is too large
Load Diff
1805
dependencies/glfw-3.4/deps/glad/gles2.h
vendored
1805
dependencies/glfw-3.4/deps/glad/gles2.h
vendored
File diff suppressed because it is too large
Load Diff
6330
dependencies/glfw-3.4/deps/glad/vulkan.h
vendored
6330
dependencies/glfw-3.4/deps/glad/vulkan.h
vendored
File diff suppressed because it is too large
Load Diff
606
dependencies/glfw-3.4/deps/linmath.h
vendored
606
dependencies/glfw-3.4/deps/linmath.h
vendored
@ -1,606 +0,0 @@
|
||||
#ifndef LINMATH_H
|
||||
#define LINMATH_H
|
||||
|
||||
#include <string.h>
|
||||
#include <math.h>
|
||||
#include <string.h>
|
||||
|
||||
/* 2021-03-21 Camilla Löwy <elmindreda@elmindreda.org>
|
||||
* - Replaced double constants with float equivalents
|
||||
*/
|
||||
|
||||
#ifdef LINMATH_NO_INLINE
|
||||
#define LINMATH_H_FUNC static
|
||||
#else
|
||||
#define LINMATH_H_FUNC static inline
|
||||
#endif
|
||||
|
||||
#define LINMATH_H_DEFINE_VEC(n) \
|
||||
typedef float vec##n[n]; \
|
||||
LINMATH_H_FUNC void vec##n##_add(vec##n r, vec##n const a, vec##n const b) \
|
||||
{ \
|
||||
int i; \
|
||||
for(i=0; i<n; ++i) \
|
||||
r[i] = a[i] + b[i]; \
|
||||
} \
|
||||
LINMATH_H_FUNC void vec##n##_sub(vec##n r, vec##n const a, vec##n const b) \
|
||||
{ \
|
||||
int i; \
|
||||
for(i=0; i<n; ++i) \
|
||||
r[i] = a[i] - b[i]; \
|
||||
} \
|
||||
LINMATH_H_FUNC void vec##n##_scale(vec##n r, vec##n const v, float const s) \
|
||||
{ \
|
||||
int i; \
|
||||
for(i=0; i<n; ++i) \
|
||||
r[i] = v[i] * s; \
|
||||
} \
|
||||
LINMATH_H_FUNC float vec##n##_mul_inner(vec##n const a, vec##n const b) \
|
||||
{ \
|
||||
float p = 0.f; \
|
||||
int i; \
|
||||
for(i=0; i<n; ++i) \
|
||||
p += b[i]*a[i]; \
|
||||
return p; \
|
||||
} \
|
||||
LINMATH_H_FUNC float vec##n##_len(vec##n const v) \
|
||||
{ \
|
||||
return sqrtf(vec##n##_mul_inner(v,v)); \
|
||||
} \
|
||||
LINMATH_H_FUNC void vec##n##_norm(vec##n r, vec##n const v) \
|
||||
{ \
|
||||
float k = 1.f / vec##n##_len(v); \
|
||||
vec##n##_scale(r, v, k); \
|
||||
} \
|
||||
LINMATH_H_FUNC void vec##n##_min(vec##n r, vec##n const a, vec##n const b) \
|
||||
{ \
|
||||
int i; \
|
||||
for(i=0; i<n; ++i) \
|
||||
r[i] = a[i]<b[i] ? a[i] : b[i]; \
|
||||
} \
|
||||
LINMATH_H_FUNC void vec##n##_max(vec##n r, vec##n const a, vec##n const b) \
|
||||
{ \
|
||||
int i; \
|
||||
for(i=0; i<n; ++i) \
|
||||
r[i] = a[i]>b[i] ? a[i] : b[i]; \
|
||||
} \
|
||||
LINMATH_H_FUNC void vec##n##_dup(vec##n r, vec##n const src) \
|
||||
{ \
|
||||
int i; \
|
||||
for(i=0; i<n; ++i) \
|
||||
r[i] = src[i]; \
|
||||
}
|
||||
|
||||
LINMATH_H_DEFINE_VEC(2)
|
||||
LINMATH_H_DEFINE_VEC(3)
|
||||
LINMATH_H_DEFINE_VEC(4)
|
||||
|
||||
LINMATH_H_FUNC void vec3_mul_cross(vec3 r, vec3 const a, vec3 const b)
|
||||
{
|
||||
r[0] = a[1]*b[2] - a[2]*b[1];
|
||||
r[1] = a[2]*b[0] - a[0]*b[2];
|
||||
r[2] = a[0]*b[1] - a[1]*b[0];
|
||||
}
|
||||
|
||||
LINMATH_H_FUNC void vec3_reflect(vec3 r, vec3 const v, vec3 const n)
|
||||
{
|
||||
float p = 2.f * vec3_mul_inner(v, n);
|
||||
int i;
|
||||
for(i=0;i<3;++i)
|
||||
r[i] = v[i] - p*n[i];
|
||||
}
|
||||
|
||||
LINMATH_H_FUNC void vec4_mul_cross(vec4 r, vec4 const a, vec4 const b)
|
||||
{
|
||||
r[0] = a[1]*b[2] - a[2]*b[1];
|
||||
r[1] = a[2]*b[0] - a[0]*b[2];
|
||||
r[2] = a[0]*b[1] - a[1]*b[0];
|
||||
r[3] = 1.f;
|
||||
}
|
||||
|
||||
LINMATH_H_FUNC void vec4_reflect(vec4 r, vec4 const v, vec4 const n)
|
||||
{
|
||||
float p = 2.f*vec4_mul_inner(v, n);
|
||||
int i;
|
||||
for(i=0;i<4;++i)
|
||||
r[i] = v[i] - p*n[i];
|
||||
}
|
||||
|
||||
typedef vec4 mat4x4[4];
|
||||
LINMATH_H_FUNC void mat4x4_identity(mat4x4 M)
|
||||
{
|
||||
int i, j;
|
||||
for(i=0; i<4; ++i)
|
||||
for(j=0; j<4; ++j)
|
||||
M[i][j] = i==j ? 1.f : 0.f;
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_dup(mat4x4 M, mat4x4 const N)
|
||||
{
|
||||
int i;
|
||||
for(i=0; i<4; ++i)
|
||||
vec4_dup(M[i], N[i]);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_row(vec4 r, mat4x4 const M, int i)
|
||||
{
|
||||
int k;
|
||||
for(k=0; k<4; ++k)
|
||||
r[k] = M[k][i];
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_col(vec4 r, mat4x4 const M, int i)
|
||||
{
|
||||
int k;
|
||||
for(k=0; k<4; ++k)
|
||||
r[k] = M[i][k];
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_transpose(mat4x4 M, mat4x4 const N)
|
||||
{
|
||||
// Note: if M and N are the same, the user has to
|
||||
// explicitly make a copy of M and set it to N.
|
||||
int i, j;
|
||||
for(j=0; j<4; ++j)
|
||||
for(i=0; i<4; ++i)
|
||||
M[i][j] = N[j][i];
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_add(mat4x4 M, mat4x4 const a, mat4x4 const b)
|
||||
{
|
||||
int i;
|
||||
for(i=0; i<4; ++i)
|
||||
vec4_add(M[i], a[i], b[i]);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_sub(mat4x4 M, mat4x4 const a, mat4x4 const b)
|
||||
{
|
||||
int i;
|
||||
for(i=0; i<4; ++i)
|
||||
vec4_sub(M[i], a[i], b[i]);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_scale(mat4x4 M, mat4x4 const a, float k)
|
||||
{
|
||||
int i;
|
||||
for(i=0; i<4; ++i)
|
||||
vec4_scale(M[i], a[i], k);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_scale_aniso(mat4x4 M, mat4x4 const a, float x, float y, float z)
|
||||
{
|
||||
vec4_scale(M[0], a[0], x);
|
||||
vec4_scale(M[1], a[1], y);
|
||||
vec4_scale(M[2], a[2], z);
|
||||
vec4_dup(M[3], a[3]);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_mul(mat4x4 M, mat4x4 const a, mat4x4 const b)
|
||||
{
|
||||
mat4x4 temp;
|
||||
int k, r, c;
|
||||
for(c=0; c<4; ++c) for(r=0; r<4; ++r) {
|
||||
temp[c][r] = 0.f;
|
||||
for(k=0; k<4; ++k)
|
||||
temp[c][r] += a[k][r] * b[c][k];
|
||||
}
|
||||
mat4x4_dup(M, temp);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_mul_vec4(vec4 r, mat4x4 const M, vec4 const v)
|
||||
{
|
||||
int i, j;
|
||||
for(j=0; j<4; ++j) {
|
||||
r[j] = 0.f;
|
||||
for(i=0; i<4; ++i)
|
||||
r[j] += M[i][j] * v[i];
|
||||
}
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_translate(mat4x4 T, float x, float y, float z)
|
||||
{
|
||||
mat4x4_identity(T);
|
||||
T[3][0] = x;
|
||||
T[3][1] = y;
|
||||
T[3][2] = z;
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_translate_in_place(mat4x4 M, float x, float y, float z)
|
||||
{
|
||||
vec4 t = {x, y, z, 0};
|
||||
vec4 r;
|
||||
int i;
|
||||
for (i = 0; i < 4; ++i) {
|
||||
mat4x4_row(r, M, i);
|
||||
M[3][i] += vec4_mul_inner(r, t);
|
||||
}
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_from_vec3_mul_outer(mat4x4 M, vec3 const a, vec3 const b)
|
||||
{
|
||||
int i, j;
|
||||
for(i=0; i<4; ++i) for(j=0; j<4; ++j)
|
||||
M[i][j] = i<3 && j<3 ? a[i] * b[j] : 0.f;
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_rotate(mat4x4 R, mat4x4 const M, float x, float y, float z, float angle)
|
||||
{
|
||||
float s = sinf(angle);
|
||||
float c = cosf(angle);
|
||||
vec3 u = {x, y, z};
|
||||
|
||||
if(vec3_len(u) > 1e-4) {
|
||||
vec3_norm(u, u);
|
||||
mat4x4 T;
|
||||
mat4x4_from_vec3_mul_outer(T, u, u);
|
||||
|
||||
mat4x4 S = {
|
||||
{ 0, u[2], -u[1], 0},
|
||||
{-u[2], 0, u[0], 0},
|
||||
{ u[1], -u[0], 0, 0},
|
||||
{ 0, 0, 0, 0}
|
||||
};
|
||||
mat4x4_scale(S, S, s);
|
||||
|
||||
mat4x4 C;
|
||||
mat4x4_identity(C);
|
||||
mat4x4_sub(C, C, T);
|
||||
|
||||
mat4x4_scale(C, C, c);
|
||||
|
||||
mat4x4_add(T, T, C);
|
||||
mat4x4_add(T, T, S);
|
||||
|
||||
T[3][3] = 1.f;
|
||||
mat4x4_mul(R, M, T);
|
||||
} else {
|
||||
mat4x4_dup(R, M);
|
||||
}
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_rotate_X(mat4x4 Q, mat4x4 const M, float angle)
|
||||
{
|
||||
float s = sinf(angle);
|
||||
float c = cosf(angle);
|
||||
mat4x4 R = {
|
||||
{1.f, 0.f, 0.f, 0.f},
|
||||
{0.f, c, s, 0.f},
|
||||
{0.f, -s, c, 0.f},
|
||||
{0.f, 0.f, 0.f, 1.f}
|
||||
};
|
||||
mat4x4_mul(Q, M, R);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_rotate_Y(mat4x4 Q, mat4x4 const M, float angle)
|
||||
{
|
||||
float s = sinf(angle);
|
||||
float c = cosf(angle);
|
||||
mat4x4 R = {
|
||||
{ c, 0.f, -s, 0.f},
|
||||
{ 0.f, 1.f, 0.f, 0.f},
|
||||
{ s, 0.f, c, 0.f},
|
||||
{ 0.f, 0.f, 0.f, 1.f}
|
||||
};
|
||||
mat4x4_mul(Q, M, R);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_rotate_Z(mat4x4 Q, mat4x4 const M, float angle)
|
||||
{
|
||||
float s = sinf(angle);
|
||||
float c = cosf(angle);
|
||||
mat4x4 R = {
|
||||
{ c, s, 0.f, 0.f},
|
||||
{ -s, c, 0.f, 0.f},
|
||||
{ 0.f, 0.f, 1.f, 0.f},
|
||||
{ 0.f, 0.f, 0.f, 1.f}
|
||||
};
|
||||
mat4x4_mul(Q, M, R);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_invert(mat4x4 T, mat4x4 const M)
|
||||
{
|
||||
float s[6];
|
||||
float c[6];
|
||||
s[0] = M[0][0]*M[1][1] - M[1][0]*M[0][1];
|
||||
s[1] = M[0][0]*M[1][2] - M[1][0]*M[0][2];
|
||||
s[2] = M[0][0]*M[1][3] - M[1][0]*M[0][3];
|
||||
s[3] = M[0][1]*M[1][2] - M[1][1]*M[0][2];
|
||||
s[4] = M[0][1]*M[1][3] - M[1][1]*M[0][3];
|
||||
s[5] = M[0][2]*M[1][3] - M[1][2]*M[0][3];
|
||||
|
||||
c[0] = M[2][0]*M[3][1] - M[3][0]*M[2][1];
|
||||
c[1] = M[2][0]*M[3][2] - M[3][0]*M[2][2];
|
||||
c[2] = M[2][0]*M[3][3] - M[3][0]*M[2][3];
|
||||
c[3] = M[2][1]*M[3][2] - M[3][1]*M[2][2];
|
||||
c[4] = M[2][1]*M[3][3] - M[3][1]*M[2][3];
|
||||
c[5] = M[2][2]*M[3][3] - M[3][2]*M[2][3];
|
||||
|
||||
/* Assumes it is invertible */
|
||||
float idet = 1.0f/( s[0]*c[5]-s[1]*c[4]+s[2]*c[3]+s[3]*c[2]-s[4]*c[1]+s[5]*c[0] );
|
||||
|
||||
T[0][0] = ( M[1][1] * c[5] - M[1][2] * c[4] + M[1][3] * c[3]) * idet;
|
||||
T[0][1] = (-M[0][1] * c[5] + M[0][2] * c[4] - M[0][3] * c[3]) * idet;
|
||||
T[0][2] = ( M[3][1] * s[5] - M[3][2] * s[4] + M[3][3] * s[3]) * idet;
|
||||
T[0][3] = (-M[2][1] * s[5] + M[2][2] * s[4] - M[2][3] * s[3]) * idet;
|
||||
|
||||
T[1][0] = (-M[1][0] * c[5] + M[1][2] * c[2] - M[1][3] * c[1]) * idet;
|
||||
T[1][1] = ( M[0][0] * c[5] - M[0][2] * c[2] + M[0][3] * c[1]) * idet;
|
||||
T[1][2] = (-M[3][0] * s[5] + M[3][2] * s[2] - M[3][3] * s[1]) * idet;
|
||||
T[1][3] = ( M[2][0] * s[5] - M[2][2] * s[2] + M[2][3] * s[1]) * idet;
|
||||
|
||||
T[2][0] = ( M[1][0] * c[4] - M[1][1] * c[2] + M[1][3] * c[0]) * idet;
|
||||
T[2][1] = (-M[0][0] * c[4] + M[0][1] * c[2] - M[0][3] * c[0]) * idet;
|
||||
T[2][2] = ( M[3][0] * s[4] - M[3][1] * s[2] + M[3][3] * s[0]) * idet;
|
||||
T[2][3] = (-M[2][0] * s[4] + M[2][1] * s[2] - M[2][3] * s[0]) * idet;
|
||||
|
||||
T[3][0] = (-M[1][0] * c[3] + M[1][1] * c[1] - M[1][2] * c[0]) * idet;
|
||||
T[3][1] = ( M[0][0] * c[3] - M[0][1] * c[1] + M[0][2] * c[0]) * idet;
|
||||
T[3][2] = (-M[3][0] * s[3] + M[3][1] * s[1] - M[3][2] * s[0]) * idet;
|
||||
T[3][3] = ( M[2][0] * s[3] - M[2][1] * s[1] + M[2][2] * s[0]) * idet;
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_orthonormalize(mat4x4 R, mat4x4 const M)
|
||||
{
|
||||
mat4x4_dup(R, M);
|
||||
float s = 1.f;
|
||||
vec3 h;
|
||||
|
||||
vec3_norm(R[2], R[2]);
|
||||
|
||||
s = vec3_mul_inner(R[1], R[2]);
|
||||
vec3_scale(h, R[2], s);
|
||||
vec3_sub(R[1], R[1], h);
|
||||
vec3_norm(R[1], R[1]);
|
||||
|
||||
s = vec3_mul_inner(R[0], R[2]);
|
||||
vec3_scale(h, R[2], s);
|
||||
vec3_sub(R[0], R[0], h);
|
||||
|
||||
s = vec3_mul_inner(R[0], R[1]);
|
||||
vec3_scale(h, R[1], s);
|
||||
vec3_sub(R[0], R[0], h);
|
||||
vec3_norm(R[0], R[0]);
|
||||
}
|
||||
|
||||
LINMATH_H_FUNC void mat4x4_frustum(mat4x4 M, float l, float r, float b, float t, float n, float f)
|
||||
{
|
||||
M[0][0] = 2.f*n/(r-l);
|
||||
M[0][1] = M[0][2] = M[0][3] = 0.f;
|
||||
|
||||
M[1][1] = 2.f*n/(t-b);
|
||||
M[1][0] = M[1][2] = M[1][3] = 0.f;
|
||||
|
||||
M[2][0] = (r+l)/(r-l);
|
||||
M[2][1] = (t+b)/(t-b);
|
||||
M[2][2] = -(f+n)/(f-n);
|
||||
M[2][3] = -1.f;
|
||||
|
||||
M[3][2] = -2.f*(f*n)/(f-n);
|
||||
M[3][0] = M[3][1] = M[3][3] = 0.f;
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_ortho(mat4x4 M, float l, float r, float b, float t, float n, float f)
|
||||
{
|
||||
M[0][0] = 2.f/(r-l);
|
||||
M[0][1] = M[0][2] = M[0][3] = 0.f;
|
||||
|
||||
M[1][1] = 2.f/(t-b);
|
||||
M[1][0] = M[1][2] = M[1][3] = 0.f;
|
||||
|
||||
M[2][2] = -2.f/(f-n);
|
||||
M[2][0] = M[2][1] = M[2][3] = 0.f;
|
||||
|
||||
M[3][0] = -(r+l)/(r-l);
|
||||
M[3][1] = -(t+b)/(t-b);
|
||||
M[3][2] = -(f+n)/(f-n);
|
||||
M[3][3] = 1.f;
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_perspective(mat4x4 m, float y_fov, float aspect, float n, float f)
|
||||
{
|
||||
/* NOTE: Degrees are an unhandy unit to work with.
|
||||
* linmath.h uses radians for everything! */
|
||||
float const a = 1.f / tanf(y_fov / 2.f);
|
||||
|
||||
m[0][0] = a / aspect;
|
||||
m[0][1] = 0.f;
|
||||
m[0][2] = 0.f;
|
||||
m[0][3] = 0.f;
|
||||
|
||||
m[1][0] = 0.f;
|
||||
m[1][1] = a;
|
||||
m[1][2] = 0.f;
|
||||
m[1][3] = 0.f;
|
||||
|
||||
m[2][0] = 0.f;
|
||||
m[2][1] = 0.f;
|
||||
m[2][2] = -((f + n) / (f - n));
|
||||
m[2][3] = -1.f;
|
||||
|
||||
m[3][0] = 0.f;
|
||||
m[3][1] = 0.f;
|
||||
m[3][2] = -((2.f * f * n) / (f - n));
|
||||
m[3][3] = 0.f;
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_look_at(mat4x4 m, vec3 const eye, vec3 const center, vec3 const up)
|
||||
{
|
||||
/* Adapted from Android's OpenGL Matrix.java. */
|
||||
/* See the OpenGL GLUT documentation for gluLookAt for a description */
|
||||
/* of the algorithm. We implement it in a straightforward way: */
|
||||
|
||||
/* TODO: The negation of of can be spared by swapping the order of
|
||||
* operands in the following cross products in the right way. */
|
||||
vec3 f;
|
||||
vec3_sub(f, center, eye);
|
||||
vec3_norm(f, f);
|
||||
|
||||
vec3 s;
|
||||
vec3_mul_cross(s, f, up);
|
||||
vec3_norm(s, s);
|
||||
|
||||
vec3 t;
|
||||
vec3_mul_cross(t, s, f);
|
||||
|
||||
m[0][0] = s[0];
|
||||
m[0][1] = t[0];
|
||||
m[0][2] = -f[0];
|
||||
m[0][3] = 0.f;
|
||||
|
||||
m[1][0] = s[1];
|
||||
m[1][1] = t[1];
|
||||
m[1][2] = -f[1];
|
||||
m[1][3] = 0.f;
|
||||
|
||||
m[2][0] = s[2];
|
||||
m[2][1] = t[2];
|
||||
m[2][2] = -f[2];
|
||||
m[2][3] = 0.f;
|
||||
|
||||
m[3][0] = 0.f;
|
||||
m[3][1] = 0.f;
|
||||
m[3][2] = 0.f;
|
||||
m[3][3] = 1.f;
|
||||
|
||||
mat4x4_translate_in_place(m, -eye[0], -eye[1], -eye[2]);
|
||||
}
|
||||
|
||||
typedef float quat[4];
|
||||
#define quat_add vec4_add
|
||||
#define quat_sub vec4_sub
|
||||
#define quat_norm vec4_norm
|
||||
#define quat_scale vec4_scale
|
||||
#define quat_mul_inner vec4_mul_inner
|
||||
|
||||
LINMATH_H_FUNC void quat_identity(quat q)
|
||||
{
|
||||
q[0] = q[1] = q[2] = 0.f;
|
||||
q[3] = 1.f;
|
||||
}
|
||||
LINMATH_H_FUNC void quat_mul(quat r, quat const p, quat const q)
|
||||
{
|
||||
vec3 w;
|
||||
vec3_mul_cross(r, p, q);
|
||||
vec3_scale(w, p, q[3]);
|
||||
vec3_add(r, r, w);
|
||||
vec3_scale(w, q, p[3]);
|
||||
vec3_add(r, r, w);
|
||||
r[3] = p[3]*q[3] - vec3_mul_inner(p, q);
|
||||
}
|
||||
LINMATH_H_FUNC void quat_conj(quat r, quat const q)
|
||||
{
|
||||
int i;
|
||||
for(i=0; i<3; ++i)
|
||||
r[i] = -q[i];
|
||||
r[3] = q[3];
|
||||
}
|
||||
LINMATH_H_FUNC void quat_rotate(quat r, float angle, vec3 const axis) {
|
||||
vec3 axis_norm;
|
||||
vec3_norm(axis_norm, axis);
|
||||
float s = sinf(angle / 2);
|
||||
float c = cosf(angle / 2);
|
||||
vec3_scale(r, axis_norm, s);
|
||||
r[3] = c;
|
||||
}
|
||||
LINMATH_H_FUNC void quat_mul_vec3(vec3 r, quat const q, vec3 const v)
|
||||
{
|
||||
/*
|
||||
* Method by Fabian 'ryg' Giessen (of Farbrausch)
|
||||
t = 2 * cross(q.xyz, v)
|
||||
v' = v + q.w * t + cross(q.xyz, t)
|
||||
*/
|
||||
vec3 t;
|
||||
vec3 q_xyz = {q[0], q[1], q[2]};
|
||||
vec3 u = {q[0], q[1], q[2]};
|
||||
|
||||
vec3_mul_cross(t, q_xyz, v);
|
||||
vec3_scale(t, t, 2);
|
||||
|
||||
vec3_mul_cross(u, q_xyz, t);
|
||||
vec3_scale(t, t, q[3]);
|
||||
|
||||
vec3_add(r, v, t);
|
||||
vec3_add(r, r, u);
|
||||
}
|
||||
LINMATH_H_FUNC void mat4x4_from_quat(mat4x4 M, quat const q)
|
||||
{
|
||||
float a = q[3];
|
||||
float b = q[0];
|
||||
float c = q[1];
|
||||
float d = q[2];
|
||||
float a2 = a*a;
|
||||
float b2 = b*b;
|
||||
float c2 = c*c;
|
||||
float d2 = d*d;
|
||||
|
||||
M[0][0] = a2 + b2 - c2 - d2;
|
||||
M[0][1] = 2.f*(b*c + a*d);
|
||||
M[0][2] = 2.f*(b*d - a*c);
|
||||
M[0][3] = 0.f;
|
||||
|
||||
M[1][0] = 2*(b*c - a*d);
|
||||
M[1][1] = a2 - b2 + c2 - d2;
|
||||
M[1][2] = 2.f*(c*d + a*b);
|
||||
M[1][3] = 0.f;
|
||||
|
||||
M[2][0] = 2.f*(b*d + a*c);
|
||||
M[2][1] = 2.f*(c*d - a*b);
|
||||
M[2][2] = a2 - b2 - c2 + d2;
|
||||
M[2][3] = 0.f;
|
||||
|
||||
M[3][0] = M[3][1] = M[3][2] = 0.f;
|
||||
M[3][3] = 1.f;
|
||||
}
|
||||
|
||||
LINMATH_H_FUNC void mat4x4o_mul_quat(mat4x4 R, mat4x4 const M, quat const q)
|
||||
{
|
||||
/* XXX: The way this is written only works for orthogonal matrices. */
|
||||
/* TODO: Take care of non-orthogonal case. */
|
||||
quat_mul_vec3(R[0], q, M[0]);
|
||||
quat_mul_vec3(R[1], q, M[1]);
|
||||
quat_mul_vec3(R[2], q, M[2]);
|
||||
|
||||
R[3][0] = R[3][1] = R[3][2] = 0.f;
|
||||
R[0][3] = M[0][3];
|
||||
R[1][3] = M[1][3];
|
||||
R[2][3] = M[2][3];
|
||||
R[3][3] = M[3][3]; // typically 1.0, but here we make it general
|
||||
}
|
||||
LINMATH_H_FUNC void quat_from_mat4x4(quat q, mat4x4 const M)
|
||||
{
|
||||
float r=0.f;
|
||||
int i;
|
||||
|
||||
int perm[] = { 0, 1, 2, 0, 1 };
|
||||
int *p = perm;
|
||||
|
||||
for(i = 0; i<3; i++) {
|
||||
float m = M[i][i];
|
||||
if( m < r )
|
||||
continue;
|
||||
m = r;
|
||||
p = &perm[i];
|
||||
}
|
||||
|
||||
r = sqrtf(1.f + M[p[0]][p[0]] - M[p[1]][p[1]] - M[p[2]][p[2]] );
|
||||
|
||||
if(r < 1e-6) {
|
||||
q[0] = 1.f;
|
||||
q[1] = q[2] = q[3] = 0.f;
|
||||
return;
|
||||
}
|
||||
|
||||
q[0] = r/2.f;
|
||||
q[1] = (M[p[0]][p[1]] - M[p[1]][p[0]])/(2.f*r);
|
||||
q[2] = (M[p[2]][p[0]] - M[p[0]][p[2]])/(2.f*r);
|
||||
q[3] = (M[p[2]][p[1]] - M[p[1]][p[2]])/(2.f*r);
|
||||
}
|
||||
|
||||
LINMATH_H_FUNC void mat4x4_arcball(mat4x4 R, mat4x4 const M, vec2 const _a, vec2 const _b, float s)
|
||||
{
|
||||
vec2 a; memcpy(a, _a, sizeof(a));
|
||||
vec2 b; memcpy(b, _b, sizeof(b));
|
||||
|
||||
float z_a = 0.f;
|
||||
float z_b = 0.f;
|
||||
|
||||
if(vec2_len(a) < 1.f) {
|
||||
z_a = sqrtf(1.f - vec2_mul_inner(a, a));
|
||||
} else {
|
||||
vec2_norm(a, a);
|
||||
}
|
||||
|
||||
if(vec2_len(b) < 1.f) {
|
||||
z_b = sqrtf(1.f - vec2_mul_inner(b, b));
|
||||
} else {
|
||||
vec2_norm(b, b);
|
||||
}
|
||||
|
||||
vec3 a_ = {a[0], a[1], z_a};
|
||||
vec3 b_ = {b[0], b[1], z_b};
|
||||
|
||||
vec3 c_;
|
||||
vec3_mul_cross(c_, a_, b_);
|
||||
|
||||
float const angle = acos(vec3_mul_inner(a_, b_)) * s;
|
||||
mat4x4_rotate(R, M, c_[0], c_[1], c_[2], angle);
|
||||
}
|
||||
#endif
|
117
dependencies/glfw-3.4/deps/mingw/_mingw_dxhelper.h
vendored
117
dependencies/glfw-3.4/deps/mingw/_mingw_dxhelper.h
vendored
@ -1,117 +0,0 @@
|
||||
/**
|
||||
* This file has no copyright assigned and is placed in the Public Domain.
|
||||
* This file is part of the mingw-w64 runtime package.
|
||||
* No warranty is given; refer to the file DISCLAIMER within this package.
|
||||
*/
|
||||
|
||||
#if defined(_MSC_VER) && !defined(_MSC_EXTENSIONS)
|
||||
#define NONAMELESSUNION 1
|
||||
#endif
|
||||
#if defined(NONAMELESSSTRUCT) && \
|
||||
!defined(NONAMELESSUNION)
|
||||
#define NONAMELESSUNION 1
|
||||
#endif
|
||||
#if defined(NONAMELESSUNION) && \
|
||||
!defined(NONAMELESSSTRUCT)
|
||||
#define NONAMELESSSTRUCT 1
|
||||
#endif
|
||||
#if !defined(__GNU_EXTENSION)
|
||||
#if defined(__GNUC__) || defined(__GNUG__)
|
||||
#define __GNU_EXTENSION __extension__
|
||||
#else
|
||||
#define __GNU_EXTENSION
|
||||
#endif
|
||||
#endif /* __extension__ */
|
||||
|
||||
#ifndef __ANONYMOUS_DEFINED
|
||||
#define __ANONYMOUS_DEFINED
|
||||
#if defined(__GNUC__) || defined(__GNUG__)
|
||||
#define _ANONYMOUS_UNION __extension__
|
||||
#define _ANONYMOUS_STRUCT __extension__
|
||||
#else
|
||||
#define _ANONYMOUS_UNION
|
||||
#define _ANONYMOUS_STRUCT
|
||||
#endif
|
||||
#ifndef NONAMELESSUNION
|
||||
#define _UNION_NAME(x)
|
||||
#define _STRUCT_NAME(x)
|
||||
#else /* NONAMELESSUNION */
|
||||
#define _UNION_NAME(x) x
|
||||
#define _STRUCT_NAME(x) x
|
||||
#endif
|
||||
#endif /* __ANONYMOUS_DEFINED */
|
||||
|
||||
#ifndef DUMMYUNIONNAME
|
||||
# ifdef NONAMELESSUNION
|
||||
# define DUMMYUNIONNAME u
|
||||
# define DUMMYUNIONNAME1 u1 /* Wine uses this variant */
|
||||
# define DUMMYUNIONNAME2 u2
|
||||
# define DUMMYUNIONNAME3 u3
|
||||
# define DUMMYUNIONNAME4 u4
|
||||
# define DUMMYUNIONNAME5 u5
|
||||
# define DUMMYUNIONNAME6 u6
|
||||
# define DUMMYUNIONNAME7 u7
|
||||
# define DUMMYUNIONNAME8 u8
|
||||
# define DUMMYUNIONNAME9 u9
|
||||
# else /* NONAMELESSUNION */
|
||||
# define DUMMYUNIONNAME
|
||||
# define DUMMYUNIONNAME1 /* Wine uses this variant */
|
||||
# define DUMMYUNIONNAME2
|
||||
# define DUMMYUNIONNAME3
|
||||
# define DUMMYUNIONNAME4
|
||||
# define DUMMYUNIONNAME5
|
||||
# define DUMMYUNIONNAME6
|
||||
# define DUMMYUNIONNAME7
|
||||
# define DUMMYUNIONNAME8
|
||||
# define DUMMYUNIONNAME9
|
||||
# endif
|
||||
#endif /* DUMMYUNIONNAME */
|
||||
|
||||
#if !defined(DUMMYUNIONNAME1) /* MinGW does not define this one */
|
||||
# ifdef NONAMELESSUNION
|
||||
# define DUMMYUNIONNAME1 u1 /* Wine uses this variant */
|
||||
# else
|
||||
# define DUMMYUNIONNAME1 /* Wine uses this variant */
|
||||
# endif
|
||||
#endif /* DUMMYUNIONNAME1 */
|
||||
|
||||
#ifndef DUMMYSTRUCTNAME
|
||||
# ifdef NONAMELESSUNION
|
||||
# define DUMMYSTRUCTNAME s
|
||||
# define DUMMYSTRUCTNAME1 s1 /* Wine uses this variant */
|
||||
# define DUMMYSTRUCTNAME2 s2
|
||||
# define DUMMYSTRUCTNAME3 s3
|
||||
# define DUMMYSTRUCTNAME4 s4
|
||||
# define DUMMYSTRUCTNAME5 s5
|
||||
# else
|
||||
# define DUMMYSTRUCTNAME
|
||||
# define DUMMYSTRUCTNAME1 /* Wine uses this variant */
|
||||
# define DUMMYSTRUCTNAME2
|
||||
# define DUMMYSTRUCTNAME3
|
||||
# define DUMMYSTRUCTNAME4
|
||||
# define DUMMYSTRUCTNAME5
|
||||
# endif
|
||||
#endif /* DUMMYSTRUCTNAME */
|
||||
|
||||
/* These are for compatibility with the Wine source tree */
|
||||
|
||||
#ifndef WINELIB_NAME_AW
|
||||
# ifdef __MINGW_NAME_AW
|
||||
# define WINELIB_NAME_AW __MINGW_NAME_AW
|
||||
# else
|
||||
# ifdef UNICODE
|
||||
# define WINELIB_NAME_AW(func) func##W
|
||||
# else
|
||||
# define WINELIB_NAME_AW(func) func##A
|
||||
# endif
|
||||
# endif
|
||||
#endif /* WINELIB_NAME_AW */
|
||||
|
||||
#ifndef DECL_WINELIB_TYPE_AW
|
||||
# ifdef __MINGW_TYPEDEF_AW
|
||||
# define DECL_WINELIB_TYPE_AW __MINGW_TYPEDEF_AW
|
||||
# else
|
||||
# define DECL_WINELIB_TYPE_AW(type) typedef WINELIB_NAME_AW(type) type;
|
||||
# endif
|
||||
#endif /* DECL_WINELIB_TYPE_AW */
|
||||
|
2467
dependencies/glfw-3.4/deps/mingw/dinput.h
vendored
2467
dependencies/glfw-3.4/deps/mingw/dinput.h
vendored
File diff suppressed because it is too large
Load Diff
239
dependencies/glfw-3.4/deps/mingw/xinput.h
vendored
239
dependencies/glfw-3.4/deps/mingw/xinput.h
vendored
@ -1,239 +0,0 @@
|
||||
/*
|
||||
* The Wine project - Xinput Joystick Library
|
||||
* Copyright 2008 Andrew Fenn
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2.1 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* Lesser General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
||||
*/
|
||||
|
||||
#ifndef __WINE_XINPUT_H
|
||||
#define __WINE_XINPUT_H
|
||||
|
||||
#include <windef.h>
|
||||
|
||||
/*
|
||||
* Bitmasks for the joysticks buttons, determines what has
|
||||
* been pressed on the joystick, these need to be mapped
|
||||
* to whatever device you're using instead of an xbox 360
|
||||
* joystick
|
||||
*/
|
||||
|
||||
#define XINPUT_GAMEPAD_DPAD_UP 0x0001
|
||||
#define XINPUT_GAMEPAD_DPAD_DOWN 0x0002
|
||||
#define XINPUT_GAMEPAD_DPAD_LEFT 0x0004
|
||||
#define XINPUT_GAMEPAD_DPAD_RIGHT 0x0008
|
||||
#define XINPUT_GAMEPAD_START 0x0010
|
||||
#define XINPUT_GAMEPAD_BACK 0x0020
|
||||
#define XINPUT_GAMEPAD_LEFT_THUMB 0x0040
|
||||
#define XINPUT_GAMEPAD_RIGHT_THUMB 0x0080
|
||||
#define XINPUT_GAMEPAD_LEFT_SHOULDER 0x0100
|
||||
#define XINPUT_GAMEPAD_RIGHT_SHOULDER 0x0200
|
||||
#define XINPUT_GAMEPAD_A 0x1000
|
||||
#define XINPUT_GAMEPAD_B 0x2000
|
||||
#define XINPUT_GAMEPAD_X 0x4000
|
||||
#define XINPUT_GAMEPAD_Y 0x8000
|
||||
|
||||
/*
|
||||
* Defines the flags used to determine if the user is pushing
|
||||
* down on a button, not holding a button, etc
|
||||
*/
|
||||
|
||||
#define XINPUT_KEYSTROKE_KEYDOWN 0x0001
|
||||
#define XINPUT_KEYSTROKE_KEYUP 0x0002
|
||||
#define XINPUT_KEYSTROKE_REPEAT 0x0004
|
||||
|
||||
/*
|
||||
* Defines the codes which are returned by XInputGetKeystroke
|
||||
*/
|
||||
|
||||
#define VK_PAD_A 0x5800
|
||||
#define VK_PAD_B 0x5801
|
||||
#define VK_PAD_X 0x5802
|
||||
#define VK_PAD_Y 0x5803
|
||||
#define VK_PAD_RSHOULDER 0x5804
|
||||
#define VK_PAD_LSHOULDER 0x5805
|
||||
#define VK_PAD_LTRIGGER 0x5806
|
||||
#define VK_PAD_RTRIGGER 0x5807
|
||||
#define VK_PAD_DPAD_UP 0x5810
|
||||
#define VK_PAD_DPAD_DOWN 0x5811
|
||||
#define VK_PAD_DPAD_LEFT 0x5812
|
||||
#define VK_PAD_DPAD_RIGHT 0x5813
|
||||
#define VK_PAD_START 0x5814
|
||||
#define VK_PAD_BACK 0x5815
|
||||
#define VK_PAD_LTHUMB_PRESS 0x5816
|
||||
#define VK_PAD_RTHUMB_PRESS 0x5817
|
||||
#define VK_PAD_LTHUMB_UP 0x5820
|
||||
#define VK_PAD_LTHUMB_DOWN 0x5821
|
||||
#define VK_PAD_LTHUMB_RIGHT 0x5822
|
||||
#define VK_PAD_LTHUMB_LEFT 0x5823
|
||||
#define VK_PAD_LTHUMB_UPLEFT 0x5824
|
||||
#define VK_PAD_LTHUMB_UPRIGHT 0x5825
|
||||
#define VK_PAD_LTHUMB_DOWNRIGHT 0x5826
|
||||
#define VK_PAD_LTHUMB_DOWNLEFT 0x5827
|
||||
#define VK_PAD_RTHUMB_UP 0x5830
|
||||
#define VK_PAD_RTHUMB_DOWN 0x5831
|
||||
#define VK_PAD_RTHUMB_RIGHT 0x5832
|
||||
#define VK_PAD_RTHUMB_LEFT 0x5833
|
||||
#define VK_PAD_RTHUMB_UPLEFT 0x5834
|
||||
#define VK_PAD_RTHUMB_UPRIGHT 0x5835
|
||||
#define VK_PAD_RTHUMB_DOWNRIGHT 0x5836
|
||||
#define VK_PAD_RTHUMB_DOWNLEFT 0x5837
|
||||
|
||||
/*
|
||||
* Deadzones are for analogue joystick controls on the joypad
|
||||
* which determine when input should be assumed to be in the
|
||||
* middle of the pad. This is a threshold to stop a joypad
|
||||
* controlling the game when the player isn't touching the
|
||||
* controls.
|
||||
*/
|
||||
|
||||
#define XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE 7849
|
||||
#define XINPUT_GAMEPAD_RIGHT_THUMB_DEADZONE 8689
|
||||
#define XINPUT_GAMEPAD_TRIGGER_THRESHOLD 30
|
||||
|
||||
|
||||
/*
|
||||
* Defines what type of abilities the type of joystick has
|
||||
* DEVTYPE_GAMEPAD is available for all joysticks, however
|
||||
* there may be more specific identifiers for other joysticks
|
||||
* which are being used.
|
||||
*/
|
||||
|
||||
#define XINPUT_DEVTYPE_GAMEPAD 0x01
|
||||
#define XINPUT_DEVSUBTYPE_GAMEPAD 0x01
|
||||
#define XINPUT_DEVSUBTYPE_WHEEL 0x02
|
||||
#define XINPUT_DEVSUBTYPE_ARCADE_STICK 0x03
|
||||
#define XINPUT_DEVSUBTYPE_FLIGHT_SICK 0x04
|
||||
#define XINPUT_DEVSUBTYPE_DANCE_PAD 0x05
|
||||
#define XINPUT_DEVSUBTYPE_GUITAR 0x06
|
||||
#define XINPUT_DEVSUBTYPE_DRUM_KIT 0x08
|
||||
|
||||
/*
|
||||
* These are used with the XInputGetCapabilities function to
|
||||
* determine the abilities to the joystick which has been
|
||||
* plugged in.
|
||||
*/
|
||||
|
||||
#define XINPUT_CAPS_VOICE_SUPPORTED 0x0004
|
||||
#define XINPUT_FLAG_GAMEPAD 0x00000001
|
||||
|
||||
/*
|
||||
* Defines the status of the battery if one is used in the
|
||||
* attached joystick. The first two define if the joystick
|
||||
* supports a battery. Disconnected means that the joystick
|
||||
* isn't connected. Wired shows that the joystick is a wired
|
||||
* joystick.
|
||||
*/
|
||||
|
||||
#define BATTERY_DEVTYPE_GAMEPAD 0x00
|
||||
#define BATTERY_DEVTYPE_HEADSET 0x01
|
||||
#define BATTERY_TYPE_DISCONNECTED 0x00
|
||||
#define BATTERY_TYPE_WIRED 0x01
|
||||
#define BATTERY_TYPE_ALKALINE 0x02
|
||||
#define BATTERY_TYPE_NIMH 0x03
|
||||
#define BATTERY_TYPE_UNKNOWN 0xFF
|
||||
#define BATTERY_LEVEL_EMPTY 0x00
|
||||
#define BATTERY_LEVEL_LOW 0x01
|
||||
#define BATTERY_LEVEL_MEDIUM 0x02
|
||||
#define BATTERY_LEVEL_FULL 0x03
|
||||
|
||||
/*
|
||||
* How many joysticks can be used with this library. Games that
|
||||
* use the xinput library will not go over this number.
|
||||
*/
|
||||
|
||||
#define XUSER_MAX_COUNT 4
|
||||
#define XUSER_INDEX_ANY 0x000000FF
|
||||
|
||||
/*
|
||||
* Defines the structure of an xbox 360 joystick.
|
||||
*/
|
||||
|
||||
typedef struct _XINPUT_GAMEPAD {
|
||||
WORD wButtons;
|
||||
BYTE bLeftTrigger;
|
||||
BYTE bRightTrigger;
|
||||
SHORT sThumbLX;
|
||||
SHORT sThumbLY;
|
||||
SHORT sThumbRX;
|
||||
SHORT sThumbRY;
|
||||
} XINPUT_GAMEPAD, *PXINPUT_GAMEPAD;
|
||||
|
||||
typedef struct _XINPUT_STATE {
|
||||
DWORD dwPacketNumber;
|
||||
XINPUT_GAMEPAD Gamepad;
|
||||
} XINPUT_STATE, *PXINPUT_STATE;
|
||||
|
||||
/*
|
||||
* Defines the structure of how much vibration is set on both the
|
||||
* right and left motors in a joystick. If you're not using a 360
|
||||
* joystick you will have to map these to your device.
|
||||
*/
|
||||
|
||||
typedef struct _XINPUT_VIBRATION {
|
||||
WORD wLeftMotorSpeed;
|
||||
WORD wRightMotorSpeed;
|
||||
} XINPUT_VIBRATION, *PXINPUT_VIBRATION;
|
||||
|
||||
/*
|
||||
* Defines the structure for what kind of abilities the joystick has
|
||||
* such abilities are things such as if the joystick has the ability
|
||||
* to send and receive audio, if the joystick is in fact a driving
|
||||
* wheel or perhaps if the joystick is some kind of dance pad or
|
||||
* guitar.
|
||||
*/
|
||||
|
||||
typedef struct _XINPUT_CAPABILITIES {
|
||||
BYTE Type;
|
||||
BYTE SubType;
|
||||
WORD Flags;
|
||||
XINPUT_GAMEPAD Gamepad;
|
||||
XINPUT_VIBRATION Vibration;
|
||||
} XINPUT_CAPABILITIES, *PXINPUT_CAPABILITIES;
|
||||
|
||||
/*
|
||||
* Defines the structure for a joystick input event which is
|
||||
* retrieved using the function XInputGetKeystroke
|
||||
*/
|
||||
typedef struct _XINPUT_KEYSTROKE {
|
||||
WORD VirtualKey;
|
||||
WCHAR Unicode;
|
||||
WORD Flags;
|
||||
BYTE UserIndex;
|
||||
BYTE HidCode;
|
||||
} XINPUT_KEYSTROKE, *PXINPUT_KEYSTROKE;
|
||||
|
||||
typedef struct _XINPUT_BATTERY_INFORMATION
|
||||
{
|
||||
BYTE BatteryType;
|
||||
BYTE BatteryLevel;
|
||||
} XINPUT_BATTERY_INFORMATION, *PXINPUT_BATTERY_INFORMATION;
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
void WINAPI XInputEnable(WINBOOL);
|
||||
DWORD WINAPI XInputSetState(DWORD, XINPUT_VIBRATION*);
|
||||
DWORD WINAPI XInputGetState(DWORD, XINPUT_STATE*);
|
||||
DWORD WINAPI XInputGetKeystroke(DWORD, DWORD, PXINPUT_KEYSTROKE);
|
||||
DWORD WINAPI XInputGetCapabilities(DWORD, DWORD, XINPUT_CAPABILITIES*);
|
||||
DWORD WINAPI XInputGetDSoundAudioDeviceGuids(DWORD, GUID*, GUID*);
|
||||
DWORD WINAPI XInputGetBatteryInformation(DWORD, BYTE, XINPUT_BATTERY_INFORMATION*);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* __WINE_XINPUT_H */
|
25778
dependencies/glfw-3.4/deps/nuklear.h
vendored
25778
dependencies/glfw-3.4/deps/nuklear.h
vendored
File diff suppressed because it is too large
Load Diff
381
dependencies/glfw-3.4/deps/nuklear_glfw_gl2.h
vendored
381
dependencies/glfw-3.4/deps/nuklear_glfw_gl2.h
vendored
@ -1,381 +0,0 @@
|
||||
/*
|
||||
* Nuklear - v1.32.0 - public domain
|
||||
* no warrenty implied; use at your own risk.
|
||||
* authored from 2015-2017 by Micha Mettke
|
||||
*/
|
||||
/*
|
||||
* ==============================================================
|
||||
*
|
||||
* API
|
||||
*
|
||||
* ===============================================================
|
||||
*/
|
||||
#ifndef NK_GLFW_GL2_H_
|
||||
#define NK_GLFW_GL2_H_
|
||||
|
||||
#include <GLFW/glfw3.h>
|
||||
|
||||
enum nk_glfw_init_state{
|
||||
NK_GLFW3_DEFAULT = 0,
|
||||
NK_GLFW3_INSTALL_CALLBACKS
|
||||
};
|
||||
NK_API struct nk_context* nk_glfw3_init(GLFWwindow *win, enum nk_glfw_init_state);
|
||||
NK_API void nk_glfw3_font_stash_begin(struct nk_font_atlas **atlas);
|
||||
NK_API void nk_glfw3_font_stash_end(void);
|
||||
|
||||
NK_API void nk_glfw3_new_frame(void);
|
||||
NK_API void nk_glfw3_render(enum nk_anti_aliasing);
|
||||
NK_API void nk_glfw3_shutdown(void);
|
||||
|
||||
NK_API void nk_glfw3_char_callback(GLFWwindow *win, unsigned int codepoint);
|
||||
NK_API void nk_gflw3_scroll_callback(GLFWwindow *win, double xoff, double yoff);
|
||||
|
||||
#endif
|
||||
|
||||
/*
|
||||
* ==============================================================
|
||||
*
|
||||
* IMPLEMENTATION
|
||||
*
|
||||
* ===============================================================
|
||||
*/
|
||||
#ifdef NK_GLFW_GL2_IMPLEMENTATION
|
||||
|
||||
#ifndef NK_GLFW_TEXT_MAX
|
||||
#define NK_GLFW_TEXT_MAX 256
|
||||
#endif
|
||||
#ifndef NK_GLFW_DOUBLE_CLICK_LO
|
||||
#define NK_GLFW_DOUBLE_CLICK_LO 0.02
|
||||
#endif
|
||||
#ifndef NK_GLFW_DOUBLE_CLICK_HI
|
||||
#define NK_GLFW_DOUBLE_CLICK_HI 0.2
|
||||
#endif
|
||||
|
||||
struct nk_glfw_device {
|
||||
struct nk_buffer cmds;
|
||||
struct nk_draw_null_texture null;
|
||||
GLuint font_tex;
|
||||
};
|
||||
|
||||
struct nk_glfw_vertex {
|
||||
float position[2];
|
||||
float uv[2];
|
||||
nk_byte col[4];
|
||||
};
|
||||
|
||||
static struct nk_glfw {
|
||||
GLFWwindow *win;
|
||||
int width, height;
|
||||
int display_width, display_height;
|
||||
struct nk_glfw_device ogl;
|
||||
struct nk_context ctx;
|
||||
struct nk_font_atlas atlas;
|
||||
struct nk_vec2 fb_scale;
|
||||
unsigned int text[NK_GLFW_TEXT_MAX];
|
||||
int text_len;
|
||||
struct nk_vec2 scroll;
|
||||
double last_button_click;
|
||||
int is_double_click_down;
|
||||
struct nk_vec2 double_click_pos;
|
||||
} glfw;
|
||||
|
||||
NK_INTERN void
|
||||
nk_glfw3_device_upload_atlas(const void *image, int width, int height)
|
||||
{
|
||||
struct nk_glfw_device *dev = &glfw.ogl;
|
||||
glGenTextures(1, &dev->font_tex);
|
||||
glBindTexture(GL_TEXTURE_2D, dev->font_tex);
|
||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
||||
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, (GLsizei)width, (GLsizei)height, 0,
|
||||
GL_RGBA, GL_UNSIGNED_BYTE, image);
|
||||
}
|
||||
|
||||
NK_API void
|
||||
nk_glfw3_render(enum nk_anti_aliasing AA)
|
||||
{
|
||||
/* setup global state */
|
||||
struct nk_glfw_device *dev = &glfw.ogl;
|
||||
glPushAttrib(GL_ENABLE_BIT|GL_COLOR_BUFFER_BIT|GL_TRANSFORM_BIT);
|
||||
glDisable(GL_CULL_FACE);
|
||||
glDisable(GL_DEPTH_TEST);
|
||||
glEnable(GL_SCISSOR_TEST);
|
||||
glEnable(GL_BLEND);
|
||||
glEnable(GL_TEXTURE_2D);
|
||||
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
||||
|
||||
/* setup viewport/project */
|
||||
glViewport(0,0,(GLsizei)glfw.display_width,(GLsizei)glfw.display_height);
|
||||
glMatrixMode(GL_PROJECTION);
|
||||
glPushMatrix();
|
||||
glLoadIdentity();
|
||||
glOrtho(0.0f, glfw.width, glfw.height, 0.0f, -1.0f, 1.0f);
|
||||
glMatrixMode(GL_MODELVIEW);
|
||||
glPushMatrix();
|
||||
glLoadIdentity();
|
||||
|
||||
glEnableClientState(GL_VERTEX_ARRAY);
|
||||
glEnableClientState(GL_TEXTURE_COORD_ARRAY);
|
||||
glEnableClientState(GL_COLOR_ARRAY);
|
||||
{
|
||||
GLsizei vs = sizeof(struct nk_glfw_vertex);
|
||||
size_t vp = offsetof(struct nk_glfw_vertex, position);
|
||||
size_t vt = offsetof(struct nk_glfw_vertex, uv);
|
||||
size_t vc = offsetof(struct nk_glfw_vertex, col);
|
||||
|
||||
/* convert from command queue into draw list and draw to screen */
|
||||
const struct nk_draw_command *cmd;
|
||||
const nk_draw_index *offset = NULL;
|
||||
struct nk_buffer vbuf, ebuf;
|
||||
|
||||
/* fill convert configuration */
|
||||
struct nk_convert_config config;
|
||||
static const struct nk_draw_vertex_layout_element vertex_layout[] = {
|
||||
{NK_VERTEX_POSITION, NK_FORMAT_FLOAT, NK_OFFSETOF(struct nk_glfw_vertex, position)},
|
||||
{NK_VERTEX_TEXCOORD, NK_FORMAT_FLOAT, NK_OFFSETOF(struct nk_glfw_vertex, uv)},
|
||||
{NK_VERTEX_COLOR, NK_FORMAT_R8G8B8A8, NK_OFFSETOF(struct nk_glfw_vertex, col)},
|
||||
{NK_VERTEX_LAYOUT_END}
|
||||
};
|
||||
NK_MEMSET(&config, 0, sizeof(config));
|
||||
config.vertex_layout = vertex_layout;
|
||||
config.vertex_size = sizeof(struct nk_glfw_vertex);
|
||||
config.vertex_alignment = NK_ALIGNOF(struct nk_glfw_vertex);
|
||||
config.null = dev->null;
|
||||
config.circle_segment_count = 22;
|
||||
config.curve_segment_count = 22;
|
||||
config.arc_segment_count = 22;
|
||||
config.global_alpha = 1.0f;
|
||||
config.shape_AA = AA;
|
||||
config.line_AA = AA;
|
||||
|
||||
/* convert shapes into vertexes */
|
||||
nk_buffer_init_default(&vbuf);
|
||||
nk_buffer_init_default(&ebuf);
|
||||
nk_convert(&glfw.ctx, &dev->cmds, &vbuf, &ebuf, &config);
|
||||
|
||||
/* setup vertex buffer pointer */
|
||||
{const void *vertices = nk_buffer_memory_const(&vbuf);
|
||||
glVertexPointer(2, GL_FLOAT, vs, (const void*)((const nk_byte*)vertices + vp));
|
||||
glTexCoordPointer(2, GL_FLOAT, vs, (const void*)((const nk_byte*)vertices + vt));
|
||||
glColorPointer(4, GL_UNSIGNED_BYTE, vs, (const void*)((const nk_byte*)vertices + vc));}
|
||||
|
||||
/* iterate over and execute each draw command */
|
||||
offset = (const nk_draw_index*)nk_buffer_memory_const(&ebuf);
|
||||
nk_draw_foreach(cmd, &glfw.ctx, &dev->cmds)
|
||||
{
|
||||
if (!cmd->elem_count) continue;
|
||||
glBindTexture(GL_TEXTURE_2D, (GLuint)cmd->texture.id);
|
||||
glScissor(
|
||||
(GLint)(cmd->clip_rect.x * glfw.fb_scale.x),
|
||||
(GLint)((glfw.height - (GLint)(cmd->clip_rect.y + cmd->clip_rect.h)) * glfw.fb_scale.y),
|
||||
(GLint)(cmd->clip_rect.w * glfw.fb_scale.x),
|
||||
(GLint)(cmd->clip_rect.h * glfw.fb_scale.y));
|
||||
glDrawElements(GL_TRIANGLES, (GLsizei)cmd->elem_count, GL_UNSIGNED_SHORT, offset);
|
||||
offset += cmd->elem_count;
|
||||
}
|
||||
nk_clear(&glfw.ctx);
|
||||
nk_buffer_free(&vbuf);
|
||||
nk_buffer_free(&ebuf);
|
||||
}
|
||||
|
||||
/* default OpenGL state */
|
||||
glDisableClientState(GL_VERTEX_ARRAY);
|
||||
glDisableClientState(GL_TEXTURE_COORD_ARRAY);
|
||||
glDisableClientState(GL_COLOR_ARRAY);
|
||||
|
||||
glDisable(GL_CULL_FACE);
|
||||
glDisable(GL_DEPTH_TEST);
|
||||
glDisable(GL_SCISSOR_TEST);
|
||||
glDisable(GL_BLEND);
|
||||
glDisable(GL_TEXTURE_2D);
|
||||
|
||||
glBindTexture(GL_TEXTURE_2D, 0);
|
||||
glMatrixMode(GL_MODELVIEW);
|
||||
glPopMatrix();
|
||||
glMatrixMode(GL_PROJECTION);
|
||||
glPopMatrix();
|
||||
glPopAttrib();
|
||||
}
|
||||
|
||||
NK_API void
|
||||
nk_glfw3_char_callback(GLFWwindow *win, unsigned int codepoint)
|
||||
{
|
||||
(void)win;
|
||||
if (glfw.text_len < NK_GLFW_TEXT_MAX)
|
||||
glfw.text[glfw.text_len++] = codepoint;
|
||||
}
|
||||
|
||||
NK_API void
|
||||
nk_gflw3_scroll_callback(GLFWwindow *win, double xoff, double yoff)
|
||||
{
|
||||
(void)win; (void)xoff;
|
||||
glfw.scroll.x += (float)xoff;
|
||||
glfw.scroll.y += (float)yoff;
|
||||
}
|
||||
|
||||
NK_API void
|
||||
nk_glfw3_mouse_button_callback(GLFWwindow* window, int button, int action, int mods)
|
||||
{
|
||||
double x, y;
|
||||
if (button != GLFW_MOUSE_BUTTON_LEFT) return;
|
||||
glfwGetCursorPos(window, &x, &y);
|
||||
if (action == GLFW_PRESS) {
|
||||
double dt = glfwGetTime() - glfw.last_button_click;
|
||||
if (dt > NK_GLFW_DOUBLE_CLICK_LO && dt < NK_GLFW_DOUBLE_CLICK_HI) {
|
||||
glfw.is_double_click_down = nk_true;
|
||||
glfw.double_click_pos = nk_vec2((float)x, (float)y);
|
||||
}
|
||||
glfw.last_button_click = glfwGetTime();
|
||||
} else glfw.is_double_click_down = nk_false;
|
||||
}
|
||||
|
||||
NK_INTERN void
|
||||
nk_glfw3_clipboard_paste(nk_handle usr, struct nk_text_edit *edit)
|
||||
{
|
||||
const char *text = glfwGetClipboardString(glfw.win);
|
||||
if (text) nk_textedit_paste(edit, text, nk_strlen(text));
|
||||
(void)usr;
|
||||
}
|
||||
|
||||
NK_INTERN void
|
||||
nk_glfw3_clipboard_copy(nk_handle usr, const char *text, int len)
|
||||
{
|
||||
char *str = 0;
|
||||
(void)usr;
|
||||
if (!len) return;
|
||||
str = (char*)malloc((size_t)len+1);
|
||||
if (!str) return;
|
||||
NK_MEMCPY(str, text, (size_t)len);
|
||||
str[len] = '\0';
|
||||
glfwSetClipboardString(glfw.win, str);
|
||||
free(str);
|
||||
}
|
||||
|
||||
NK_API struct nk_context*
|
||||
nk_glfw3_init(GLFWwindow *win, enum nk_glfw_init_state init_state)
|
||||
{
|
||||
glfw.win = win;
|
||||
if (init_state == NK_GLFW3_INSTALL_CALLBACKS) {
|
||||
glfwSetScrollCallback(win, nk_gflw3_scroll_callback);
|
||||
glfwSetCharCallback(win, nk_glfw3_char_callback);
|
||||
glfwSetMouseButtonCallback(win, nk_glfw3_mouse_button_callback);
|
||||
}
|
||||
nk_init_default(&glfw.ctx, 0);
|
||||
glfw.ctx.clip.copy = nk_glfw3_clipboard_copy;
|
||||
glfw.ctx.clip.paste = nk_glfw3_clipboard_paste;
|
||||
glfw.ctx.clip.userdata = nk_handle_ptr(0);
|
||||
nk_buffer_init_default(&glfw.ogl.cmds);
|
||||
|
||||
glfw.is_double_click_down = nk_false;
|
||||
glfw.double_click_pos = nk_vec2(0, 0);
|
||||
|
||||
return &glfw.ctx;
|
||||
}
|
||||
|
||||
NK_API void
|
||||
nk_glfw3_font_stash_begin(struct nk_font_atlas **atlas)
|
||||
{
|
||||
nk_font_atlas_init_default(&glfw.atlas);
|
||||
nk_font_atlas_begin(&glfw.atlas);
|
||||
*atlas = &glfw.atlas;
|
||||
}
|
||||
|
||||
NK_API void
|
||||
nk_glfw3_font_stash_end(void)
|
||||
{
|
||||
const void *image; int w, h;
|
||||
image = nk_font_atlas_bake(&glfw.atlas, &w, &h, NK_FONT_ATLAS_RGBA32);
|
||||
nk_glfw3_device_upload_atlas(image, w, h);
|
||||
nk_font_atlas_end(&glfw.atlas, nk_handle_id((int)glfw.ogl.font_tex), &glfw.ogl.null);
|
||||
if (glfw.atlas.default_font)
|
||||
nk_style_set_font(&glfw.ctx, &glfw.atlas.default_font->handle);
|
||||
}
|
||||
|
||||
NK_API void
|
||||
nk_glfw3_new_frame(void)
|
||||
{
|
||||
int i;
|
||||
double x, y;
|
||||
struct nk_context *ctx = &glfw.ctx;
|
||||
struct GLFWwindow *win = glfw.win;
|
||||
|
||||
glfwGetWindowSize(win, &glfw.width, &glfw.height);
|
||||
glfwGetFramebufferSize(win, &glfw.display_width, &glfw.display_height);
|
||||
glfw.fb_scale.x = (float)glfw.display_width/(float)glfw.width;
|
||||
glfw.fb_scale.y = (float)glfw.display_height/(float)glfw.height;
|
||||
|
||||
nk_input_begin(ctx);
|
||||
for (i = 0; i < glfw.text_len; ++i)
|
||||
nk_input_unicode(ctx, glfw.text[i]);
|
||||
|
||||
/* optional grabbing behavior */
|
||||
if (ctx->input.mouse.grab)
|
||||
glfwSetInputMode(glfw.win, GLFW_CURSOR, GLFW_CURSOR_HIDDEN);
|
||||
else if (ctx->input.mouse.ungrab)
|
||||
glfwSetInputMode(glfw.win, GLFW_CURSOR, GLFW_CURSOR_NORMAL);
|
||||
|
||||
nk_input_key(ctx, NK_KEY_DEL, glfwGetKey(win, GLFW_KEY_DELETE) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_ENTER, glfwGetKey(win, GLFW_KEY_ENTER) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TAB, glfwGetKey(win, GLFW_KEY_TAB) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_BACKSPACE, glfwGetKey(win, GLFW_KEY_BACKSPACE) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_UP, glfwGetKey(win, GLFW_KEY_UP) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_DOWN, glfwGetKey(win, GLFW_KEY_DOWN) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TEXT_START, glfwGetKey(win, GLFW_KEY_HOME) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TEXT_END, glfwGetKey(win, GLFW_KEY_END) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_SCROLL_START, glfwGetKey(win, GLFW_KEY_HOME) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_SCROLL_END, glfwGetKey(win, GLFW_KEY_END) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_SCROLL_DOWN, glfwGetKey(win, GLFW_KEY_PAGE_DOWN) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_SCROLL_UP, glfwGetKey(win, GLFW_KEY_PAGE_UP) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_SHIFT, glfwGetKey(win, GLFW_KEY_LEFT_SHIFT) == GLFW_PRESS||
|
||||
glfwGetKey(win, GLFW_KEY_RIGHT_SHIFT) == GLFW_PRESS);
|
||||
|
||||
if (glfwGetKey(win, GLFW_KEY_LEFT_CONTROL) == GLFW_PRESS ||
|
||||
glfwGetKey(win, GLFW_KEY_RIGHT_CONTROL) == GLFW_PRESS) {
|
||||
nk_input_key(ctx, NK_KEY_COPY, glfwGetKey(win, GLFW_KEY_C) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_PASTE, glfwGetKey(win, GLFW_KEY_V) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_CUT, glfwGetKey(win, GLFW_KEY_X) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TEXT_UNDO, glfwGetKey(win, GLFW_KEY_Z) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TEXT_REDO, glfwGetKey(win, GLFW_KEY_R) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TEXT_WORD_LEFT, glfwGetKey(win, GLFW_KEY_LEFT) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TEXT_WORD_RIGHT, glfwGetKey(win, GLFW_KEY_RIGHT) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TEXT_LINE_START, glfwGetKey(win, GLFW_KEY_B) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_TEXT_LINE_END, glfwGetKey(win, GLFW_KEY_E) == GLFW_PRESS);
|
||||
} else {
|
||||
nk_input_key(ctx, NK_KEY_LEFT, glfwGetKey(win, GLFW_KEY_LEFT) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_RIGHT, glfwGetKey(win, GLFW_KEY_RIGHT) == GLFW_PRESS);
|
||||
nk_input_key(ctx, NK_KEY_COPY, 0);
|
||||
nk_input_key(ctx, NK_KEY_PASTE, 0);
|
||||
nk_input_key(ctx, NK_KEY_CUT, 0);
|
||||
nk_input_key(ctx, NK_KEY_SHIFT, 0);
|
||||
}
|
||||
|
||||
glfwGetCursorPos(win, &x, &y);
|
||||
nk_input_motion(ctx, (int)x, (int)y);
|
||||
if (ctx->input.mouse.grabbed) {
|
||||
glfwSetCursorPos(glfw.win, (double)ctx->input.mouse.prev.x, (double)ctx->input.mouse.prev.y);
|
||||
ctx->input.mouse.pos.x = ctx->input.mouse.prev.x;
|
||||
ctx->input.mouse.pos.y = ctx->input.mouse.prev.y;
|
||||
}
|
||||
|
||||
nk_input_button(ctx, NK_BUTTON_LEFT, (int)x, (int)y, glfwGetMouseButton(win, GLFW_MOUSE_BUTTON_LEFT) == GLFW_PRESS);
|
||||
nk_input_button(ctx, NK_BUTTON_MIDDLE, (int)x, (int)y, glfwGetMouseButton(win, GLFW_MOUSE_BUTTON_MIDDLE) == GLFW_PRESS);
|
||||
nk_input_button(ctx, NK_BUTTON_RIGHT, (int)x, (int)y, glfwGetMouseButton(win, GLFW_MOUSE_BUTTON_RIGHT) == GLFW_PRESS);
|
||||
nk_input_button(ctx, NK_BUTTON_DOUBLE, (int)glfw.double_click_pos.x, (int)glfw.double_click_pos.y, glfw.is_double_click_down);
|
||||
nk_input_scroll(ctx, glfw.scroll);
|
||||
nk_input_end(&glfw.ctx);
|
||||
glfw.text_len = 0;
|
||||
glfw.scroll = nk_vec2(0,0);
|
||||
}
|
||||
|
||||
NK_API
|
||||
void nk_glfw3_shutdown(void)
|
||||
{
|
||||
struct nk_glfw_device *dev = &glfw.ogl;
|
||||
nk_font_atlas_clear(&glfw.atlas);
|
||||
nk_free(&glfw.ctx);
|
||||
glDeleteTextures(1, &dev->font_tex);
|
||||
nk_buffer_free(&dev->cmds);
|
||||
NK_MEMSET(&glfw, 0, sizeof(glfw));
|
||||
}
|
||||
|
||||
#endif
|
1724
dependencies/glfw-3.4/deps/stb_image_write.h
vendored
1724
dependencies/glfw-3.4/deps/stb_image_write.h
vendored
File diff suppressed because it is too large
Load Diff
594
dependencies/glfw-3.4/deps/tinycthread.c
vendored
594
dependencies/glfw-3.4/deps/tinycthread.c
vendored
@ -1,594 +0,0 @@
|
||||
/* -*- mode: c; tab-width: 2; indent-tabs-mode: nil; -*-
|
||||
Copyright (c) 2012 Marcus Geelnard
|
||||
|
||||
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.
|
||||
*/
|
||||
|
||||
/* 2013-01-06 Camilla Löwy <elmindreda@glfw.org>
|
||||
*
|
||||
* Added casts from time_t to DWORD to avoid warnings on VC++.
|
||||
* Fixed time retrieval on POSIX systems.
|
||||
*/
|
||||
|
||||
#include "tinycthread.h"
|
||||
#include <stdlib.h>
|
||||
|
||||
/* Platform specific includes */
|
||||
#if defined(_TTHREAD_POSIX_)
|
||||
#include <signal.h>
|
||||
#include <sched.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/time.h>
|
||||
#include <errno.h>
|
||||
#elif defined(_TTHREAD_WIN32_)
|
||||
#include <process.h>
|
||||
#include <sys/timeb.h>
|
||||
#endif
|
||||
|
||||
/* Standard, good-to-have defines */
|
||||
#ifndef NULL
|
||||
#define NULL (void*)0
|
||||
#endif
|
||||
#ifndef TRUE
|
||||
#define TRUE 1
|
||||
#endif
|
||||
#ifndef FALSE
|
||||
#define FALSE 0
|
||||
#endif
|
||||
|
||||
int mtx_init(mtx_t *mtx, int type)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
mtx->mAlreadyLocked = FALSE;
|
||||
mtx->mRecursive = type & mtx_recursive;
|
||||
InitializeCriticalSection(&mtx->mHandle);
|
||||
return thrd_success;
|
||||
#else
|
||||
int ret;
|
||||
pthread_mutexattr_t attr;
|
||||
pthread_mutexattr_init(&attr);
|
||||
if (type & mtx_recursive)
|
||||
{
|
||||
pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
|
||||
}
|
||||
ret = pthread_mutex_init(mtx, &attr);
|
||||
pthread_mutexattr_destroy(&attr);
|
||||
return ret == 0 ? thrd_success : thrd_error;
|
||||
#endif
|
||||
}
|
||||
|
||||
void mtx_destroy(mtx_t *mtx)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
DeleteCriticalSection(&mtx->mHandle);
|
||||
#else
|
||||
pthread_mutex_destroy(mtx);
|
||||
#endif
|
||||
}
|
||||
|
||||
int mtx_lock(mtx_t *mtx)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
EnterCriticalSection(&mtx->mHandle);
|
||||
if (!mtx->mRecursive)
|
||||
{
|
||||
while(mtx->mAlreadyLocked) Sleep(1000); /* Simulate deadlock... */
|
||||
mtx->mAlreadyLocked = TRUE;
|
||||
}
|
||||
return thrd_success;
|
||||
#else
|
||||
return pthread_mutex_lock(mtx) == 0 ? thrd_success : thrd_error;
|
||||
#endif
|
||||
}
|
||||
|
||||
int mtx_timedlock(mtx_t *mtx, const struct timespec *ts)
|
||||
{
|
||||
/* FIXME! */
|
||||
(void)mtx;
|
||||
(void)ts;
|
||||
return thrd_error;
|
||||
}
|
||||
|
||||
int mtx_trylock(mtx_t *mtx)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
int ret = TryEnterCriticalSection(&mtx->mHandle) ? thrd_success : thrd_busy;
|
||||
if ((!mtx->mRecursive) && (ret == thrd_success) && mtx->mAlreadyLocked)
|
||||
{
|
||||
LeaveCriticalSection(&mtx->mHandle);
|
||||
ret = thrd_busy;
|
||||
}
|
||||
return ret;
|
||||
#else
|
||||
return (pthread_mutex_trylock(mtx) == 0) ? thrd_success : thrd_busy;
|
||||
#endif
|
||||
}
|
||||
|
||||
int mtx_unlock(mtx_t *mtx)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
mtx->mAlreadyLocked = FALSE;
|
||||
LeaveCriticalSection(&mtx->mHandle);
|
||||
return thrd_success;
|
||||
#else
|
||||
return pthread_mutex_unlock(mtx) == 0 ? thrd_success : thrd_error;;
|
||||
#endif
|
||||
}
|
||||
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
#define _CONDITION_EVENT_ONE 0
|
||||
#define _CONDITION_EVENT_ALL 1
|
||||
#endif
|
||||
|
||||
int cnd_init(cnd_t *cond)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
cond->mWaitersCount = 0;
|
||||
|
||||
/* Init critical section */
|
||||
InitializeCriticalSection(&cond->mWaitersCountLock);
|
||||
|
||||
/* Init events */
|
||||
cond->mEvents[_CONDITION_EVENT_ONE] = CreateEvent(NULL, FALSE, FALSE, NULL);
|
||||
if (cond->mEvents[_CONDITION_EVENT_ONE] == NULL)
|
||||
{
|
||||
cond->mEvents[_CONDITION_EVENT_ALL] = NULL;
|
||||
return thrd_error;
|
||||
}
|
||||
cond->mEvents[_CONDITION_EVENT_ALL] = CreateEvent(NULL, TRUE, FALSE, NULL);
|
||||
if (cond->mEvents[_CONDITION_EVENT_ALL] == NULL)
|
||||
{
|
||||
CloseHandle(cond->mEvents[_CONDITION_EVENT_ONE]);
|
||||
cond->mEvents[_CONDITION_EVENT_ONE] = NULL;
|
||||
return thrd_error;
|
||||
}
|
||||
|
||||
return thrd_success;
|
||||
#else
|
||||
return pthread_cond_init(cond, NULL) == 0 ? thrd_success : thrd_error;
|
||||
#endif
|
||||
}
|
||||
|
||||
void cnd_destroy(cnd_t *cond)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
if (cond->mEvents[_CONDITION_EVENT_ONE] != NULL)
|
||||
{
|
||||
CloseHandle(cond->mEvents[_CONDITION_EVENT_ONE]);
|
||||
}
|
||||
if (cond->mEvents[_CONDITION_EVENT_ALL] != NULL)
|
||||
{
|
||||
CloseHandle(cond->mEvents[_CONDITION_EVENT_ALL]);
|
||||
}
|
||||
DeleteCriticalSection(&cond->mWaitersCountLock);
|
||||
#else
|
||||
pthread_cond_destroy(cond);
|
||||
#endif
|
||||
}
|
||||
|
||||
int cnd_signal(cnd_t *cond)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
int haveWaiters;
|
||||
|
||||
/* Are there any waiters? */
|
||||
EnterCriticalSection(&cond->mWaitersCountLock);
|
||||
haveWaiters = (cond->mWaitersCount > 0);
|
||||
LeaveCriticalSection(&cond->mWaitersCountLock);
|
||||
|
||||
/* If we have any waiting threads, send them a signal */
|
||||
if(haveWaiters)
|
||||
{
|
||||
if (SetEvent(cond->mEvents[_CONDITION_EVENT_ONE]) == 0)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
}
|
||||
|
||||
return thrd_success;
|
||||
#else
|
||||
return pthread_cond_signal(cond) == 0 ? thrd_success : thrd_error;
|
||||
#endif
|
||||
}
|
||||
|
||||
int cnd_broadcast(cnd_t *cond)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
int haveWaiters;
|
||||
|
||||
/* Are there any waiters? */
|
||||
EnterCriticalSection(&cond->mWaitersCountLock);
|
||||
haveWaiters = (cond->mWaitersCount > 0);
|
||||
LeaveCriticalSection(&cond->mWaitersCountLock);
|
||||
|
||||
/* If we have any waiting threads, send them a signal */
|
||||
if(haveWaiters)
|
||||
{
|
||||
if (SetEvent(cond->mEvents[_CONDITION_EVENT_ALL]) == 0)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
}
|
||||
|
||||
return thrd_success;
|
||||
#else
|
||||
return pthread_cond_signal(cond) == 0 ? thrd_success : thrd_error;
|
||||
#endif
|
||||
}
|
||||
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
static int _cnd_timedwait_win32(cnd_t *cond, mtx_t *mtx, DWORD timeout)
|
||||
{
|
||||
int result, lastWaiter;
|
||||
|
||||
/* Increment number of waiters */
|
||||
EnterCriticalSection(&cond->mWaitersCountLock);
|
||||
++ cond->mWaitersCount;
|
||||
LeaveCriticalSection(&cond->mWaitersCountLock);
|
||||
|
||||
/* Release the mutex while waiting for the condition (will decrease
|
||||
the number of waiters when done)... */
|
||||
mtx_unlock(mtx);
|
||||
|
||||
/* Wait for either event to become signaled due to cnd_signal() or
|
||||
cnd_broadcast() being called */
|
||||
result = WaitForMultipleObjects(2, cond->mEvents, FALSE, timeout);
|
||||
if (result == WAIT_TIMEOUT)
|
||||
{
|
||||
return thrd_timeout;
|
||||
}
|
||||
else if (result == (int)WAIT_FAILED)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
|
||||
/* Check if we are the last waiter */
|
||||
EnterCriticalSection(&cond->mWaitersCountLock);
|
||||
-- cond->mWaitersCount;
|
||||
lastWaiter = (result == (WAIT_OBJECT_0 + _CONDITION_EVENT_ALL)) &&
|
||||
(cond->mWaitersCount == 0);
|
||||
LeaveCriticalSection(&cond->mWaitersCountLock);
|
||||
|
||||
/* If we are the last waiter to be notified to stop waiting, reset the event */
|
||||
if (lastWaiter)
|
||||
{
|
||||
if (ResetEvent(cond->mEvents[_CONDITION_EVENT_ALL]) == 0)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
}
|
||||
|
||||
/* Re-acquire the mutex */
|
||||
mtx_lock(mtx);
|
||||
|
||||
return thrd_success;
|
||||
}
|
||||
#endif
|
||||
|
||||
int cnd_wait(cnd_t *cond, mtx_t *mtx)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
return _cnd_timedwait_win32(cond, mtx, INFINITE);
|
||||
#else
|
||||
return pthread_cond_wait(cond, mtx) == 0 ? thrd_success : thrd_error;
|
||||
#endif
|
||||
}
|
||||
|
||||
int cnd_timedwait(cnd_t *cond, mtx_t *mtx, const struct timespec *ts)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
struct timespec now;
|
||||
if (clock_gettime(CLOCK_REALTIME, &now) == 0)
|
||||
{
|
||||
DWORD delta = (DWORD) ((ts->tv_sec - now.tv_sec) * 1000 +
|
||||
(ts->tv_nsec - now.tv_nsec + 500000) / 1000000);
|
||||
return _cnd_timedwait_win32(cond, mtx, delta);
|
||||
}
|
||||
else
|
||||
return thrd_error;
|
||||
#else
|
||||
int ret;
|
||||
ret = pthread_cond_timedwait(cond, mtx, ts);
|
||||
if (ret == ETIMEDOUT)
|
||||
{
|
||||
return thrd_timeout;
|
||||
}
|
||||
return ret == 0 ? thrd_success : thrd_error;
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
/** Information to pass to the new thread (what to run). */
|
||||
typedef struct {
|
||||
thrd_start_t mFunction; /**< Pointer to the function to be executed. */
|
||||
void * mArg; /**< Function argument for the thread function. */
|
||||
} _thread_start_info;
|
||||
|
||||
/* Thread wrapper function. */
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
static unsigned WINAPI _thrd_wrapper_function(void * aArg)
|
||||
#elif defined(_TTHREAD_POSIX_)
|
||||
static void * _thrd_wrapper_function(void * aArg)
|
||||
#endif
|
||||
{
|
||||
thrd_start_t fun;
|
||||
void *arg;
|
||||
int res;
|
||||
#if defined(_TTHREAD_POSIX_)
|
||||
void *pres;
|
||||
#endif
|
||||
|
||||
/* Get thread startup information */
|
||||
_thread_start_info *ti = (_thread_start_info *) aArg;
|
||||
fun = ti->mFunction;
|
||||
arg = ti->mArg;
|
||||
|
||||
/* The thread is responsible for freeing the startup information */
|
||||
free((void *)ti);
|
||||
|
||||
/* Call the actual client thread function */
|
||||
res = fun(arg);
|
||||
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
return res;
|
||||
#else
|
||||
pres = malloc(sizeof(int));
|
||||
if (pres != NULL)
|
||||
{
|
||||
*(int*)pres = res;
|
||||
}
|
||||
return pres;
|
||||
#endif
|
||||
}
|
||||
|
||||
int thrd_create(thrd_t *thr, thrd_start_t func, void *arg)
|
||||
{
|
||||
/* Fill out the thread startup information (passed to the thread wrapper,
|
||||
which will eventually free it) */
|
||||
_thread_start_info* ti = (_thread_start_info*)malloc(sizeof(_thread_start_info));
|
||||
if (ti == NULL)
|
||||
{
|
||||
return thrd_nomem;
|
||||
}
|
||||
ti->mFunction = func;
|
||||
ti->mArg = arg;
|
||||
|
||||
/* Create the thread */
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
*thr = (HANDLE)_beginthreadex(NULL, 0, _thrd_wrapper_function, (void *)ti, 0, NULL);
|
||||
#elif defined(_TTHREAD_POSIX_)
|
||||
if(pthread_create(thr, NULL, _thrd_wrapper_function, (void *)ti) != 0)
|
||||
{
|
||||
*thr = 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
/* Did we fail to create the thread? */
|
||||
if(!*thr)
|
||||
{
|
||||
free(ti);
|
||||
return thrd_error;
|
||||
}
|
||||
|
||||
return thrd_success;
|
||||
}
|
||||
|
||||
thrd_t thrd_current(void)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
return GetCurrentThread();
|
||||
#else
|
||||
return pthread_self();
|
||||
#endif
|
||||
}
|
||||
|
||||
int thrd_detach(thrd_t thr)
|
||||
{
|
||||
/* FIXME! */
|
||||
(void)thr;
|
||||
return thrd_error;
|
||||
}
|
||||
|
||||
int thrd_equal(thrd_t thr0, thrd_t thr1)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
return thr0 == thr1;
|
||||
#else
|
||||
return pthread_equal(thr0, thr1);
|
||||
#endif
|
||||
}
|
||||
|
||||
void thrd_exit(int res)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
ExitThread(res);
|
||||
#else
|
||||
void *pres = malloc(sizeof(int));
|
||||
if (pres != NULL)
|
||||
{
|
||||
*(int*)pres = res;
|
||||
}
|
||||
pthread_exit(pres);
|
||||
#endif
|
||||
}
|
||||
|
||||
int thrd_join(thrd_t thr, int *res)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
if (WaitForSingleObject(thr, INFINITE) == WAIT_FAILED)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
if (res != NULL)
|
||||
{
|
||||
DWORD dwRes;
|
||||
GetExitCodeThread(thr, &dwRes);
|
||||
*res = dwRes;
|
||||
}
|
||||
#elif defined(_TTHREAD_POSIX_)
|
||||
void *pres;
|
||||
int ires = 0;
|
||||
if (pthread_join(thr, &pres) != 0)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
if (pres != NULL)
|
||||
{
|
||||
ires = *(int*)pres;
|
||||
free(pres);
|
||||
}
|
||||
if (res != NULL)
|
||||
{
|
||||
*res = ires;
|
||||
}
|
||||
#endif
|
||||
return thrd_success;
|
||||
}
|
||||
|
||||
int thrd_sleep(const struct timespec *time_point, struct timespec *remaining)
|
||||
{
|
||||
struct timespec now;
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
DWORD delta;
|
||||
#else
|
||||
long delta;
|
||||
#endif
|
||||
|
||||
/* Get the current time */
|
||||
if (clock_gettime(CLOCK_REALTIME, &now) != 0)
|
||||
return -2; // FIXME: Some specific error code?
|
||||
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
/* Delta in milliseconds */
|
||||
delta = (DWORD) ((time_point->tv_sec - now.tv_sec) * 1000 +
|
||||
(time_point->tv_nsec - now.tv_nsec + 500000) / 1000000);
|
||||
if (delta > 0)
|
||||
{
|
||||
Sleep(delta);
|
||||
}
|
||||
#else
|
||||
/* Delta in microseconds */
|
||||
delta = (time_point->tv_sec - now.tv_sec) * 1000000L +
|
||||
(time_point->tv_nsec - now.tv_nsec + 500L) / 1000L;
|
||||
|
||||
/* On some systems, the usleep argument must be < 1000000 */
|
||||
while (delta > 999999L)
|
||||
{
|
||||
usleep(999999);
|
||||
delta -= 999999L;
|
||||
}
|
||||
if (delta > 0L)
|
||||
{
|
||||
usleep((useconds_t)delta);
|
||||
}
|
||||
#endif
|
||||
|
||||
/* We don't support waking up prematurely (yet) */
|
||||
if (remaining)
|
||||
{
|
||||
remaining->tv_sec = 0;
|
||||
remaining->tv_nsec = 0;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
void thrd_yield(void)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
Sleep(0);
|
||||
#else
|
||||
sched_yield();
|
||||
#endif
|
||||
}
|
||||
|
||||
int tss_create(tss_t *key, tss_dtor_t dtor)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
/* FIXME: The destructor function is not supported yet... */
|
||||
if (dtor != NULL)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
*key = TlsAlloc();
|
||||
if (*key == TLS_OUT_OF_INDEXES)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
#else
|
||||
if (pthread_key_create(key, dtor) != 0)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
#endif
|
||||
return thrd_success;
|
||||
}
|
||||
|
||||
void tss_delete(tss_t key)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
TlsFree(key);
|
||||
#else
|
||||
pthread_key_delete(key);
|
||||
#endif
|
||||
}
|
||||
|
||||
void *tss_get(tss_t key)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
return TlsGetValue(key);
|
||||
#else
|
||||
return pthread_getspecific(key);
|
||||
#endif
|
||||
}
|
||||
|
||||
int tss_set(tss_t key, void *val)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
if (TlsSetValue(key, val) == 0)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
#else
|
||||
if (pthread_setspecific(key, val) != 0)
|
||||
{
|
||||
return thrd_error;
|
||||
}
|
||||
#endif
|
||||
return thrd_success;
|
||||
}
|
||||
|
||||
#if defined(_TTHREAD_EMULATE_CLOCK_GETTIME_)
|
||||
int _tthread_clock_gettime(clockid_t clk_id, struct timespec *ts)
|
||||
{
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
struct _timeb tb;
|
||||
_ftime(&tb);
|
||||
ts->tv_sec = (time_t)tb.time;
|
||||
ts->tv_nsec = 1000000L * (long)tb.millitm;
|
||||
#else
|
||||
struct timeval tv;
|
||||
gettimeofday(&tv, NULL);
|
||||
ts->tv_sec = (time_t)tv.tv_sec;
|
||||
ts->tv_nsec = 1000L * (long)tv.tv_usec;
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
#endif // _TTHREAD_EMULATE_CLOCK_GETTIME_
|
||||
|
443
dependencies/glfw-3.4/deps/tinycthread.h
vendored
443
dependencies/glfw-3.4/deps/tinycthread.h
vendored
@ -1,443 +0,0 @@
|
||||
/* -*- mode: c; tab-width: 2; indent-tabs-mode: nil; -*-
|
||||
Copyright (c) 2012 Marcus Geelnard
|
||||
|
||||
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.
|
||||
*/
|
||||
|
||||
#ifndef _TINYCTHREAD_H_
|
||||
#define _TINYCTHREAD_H_
|
||||
|
||||
/**
|
||||
* @file
|
||||
* @mainpage TinyCThread API Reference
|
||||
*
|
||||
* @section intro_sec Introduction
|
||||
* TinyCThread is a minimal, portable implementation of basic threading
|
||||
* classes for C.
|
||||
*
|
||||
* They closely mimic the functionality and naming of the C11 standard, and
|
||||
* should be easily replaceable with the corresponding standard variants.
|
||||
*
|
||||
* @section port_sec Portability
|
||||
* The Win32 variant uses the native Win32 API for implementing the thread
|
||||
* classes, while for other systems, the POSIX threads API (pthread) is used.
|
||||
*
|
||||
* @section misc_sec Miscellaneous
|
||||
* The following special keywords are available: #_Thread_local.
|
||||
*
|
||||
* For more detailed information, browse the different sections of this
|
||||
* documentation. A good place to start is:
|
||||
* tinycthread.h.
|
||||
*/
|
||||
|
||||
/* Which platform are we on? */
|
||||
#if !defined(_TTHREAD_PLATFORM_DEFINED_)
|
||||
#if defined(_WIN32) || defined(__WIN32__) || defined(__WINDOWS__)
|
||||
#define _TTHREAD_WIN32_
|
||||
#else
|
||||
#define _TTHREAD_POSIX_
|
||||
#endif
|
||||
#define _TTHREAD_PLATFORM_DEFINED_
|
||||
#endif
|
||||
|
||||
/* Activate some POSIX functionality (e.g. clock_gettime and recursive mutexes) */
|
||||
#if defined(_TTHREAD_POSIX_)
|
||||
#undef _FEATURES_H
|
||||
#if !defined(_GNU_SOURCE)
|
||||
#define _GNU_SOURCE
|
||||
#endif
|
||||
#if !defined(_POSIX_C_SOURCE) || ((_POSIX_C_SOURCE - 0) < 199309L)
|
||||
#undef _POSIX_C_SOURCE
|
||||
#define _POSIX_C_SOURCE 199309L
|
||||
#endif
|
||||
#if !defined(_XOPEN_SOURCE) || ((_XOPEN_SOURCE - 0) < 500)
|
||||
#undef _XOPEN_SOURCE
|
||||
#define _XOPEN_SOURCE 500
|
||||
#endif
|
||||
#endif
|
||||
|
||||
/* Generic includes */
|
||||
#include <time.h>
|
||||
|
||||
/* Platform specific includes */
|
||||
#if defined(_TTHREAD_POSIX_)
|
||||
#include <sys/time.h>
|
||||
#include <pthread.h>
|
||||
#elif defined(_TTHREAD_WIN32_)
|
||||
#ifndef WIN32_LEAN_AND_MEAN
|
||||
#define WIN32_LEAN_AND_MEAN
|
||||
#define __UNDEF_LEAN_AND_MEAN
|
||||
#endif
|
||||
#include <windows.h>
|
||||
#ifdef __UNDEF_LEAN_AND_MEAN
|
||||
#undef WIN32_LEAN_AND_MEAN
|
||||
#undef __UNDEF_LEAN_AND_MEAN
|
||||
#endif
|
||||
#endif
|
||||
|
||||
/* Workaround for missing TIME_UTC: If time.h doesn't provide TIME_UTC,
|
||||
it's quite likely that libc does not support it either. Hence, fall back to
|
||||
the only other supported time specifier: CLOCK_REALTIME (and if that fails,
|
||||
we're probably emulating clock_gettime anyway, so anything goes). */
|
||||
#ifndef TIME_UTC
|
||||
#ifdef CLOCK_REALTIME
|
||||
#define TIME_UTC CLOCK_REALTIME
|
||||
#else
|
||||
#define TIME_UTC 0
|
||||
#endif
|
||||
#endif
|
||||
|
||||
/* Workaround for missing clock_gettime (most Windows compilers, afaik) */
|
||||
#if defined(_TTHREAD_WIN32_) || defined(__APPLE_CC__)
|
||||
#define _TTHREAD_EMULATE_CLOCK_GETTIME_
|
||||
/* Emulate struct timespec */
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
struct _ttherad_timespec {
|
||||
time_t tv_sec;
|
||||
long tv_nsec;
|
||||
};
|
||||
#define timespec _ttherad_timespec
|
||||
#endif
|
||||
|
||||
/* Emulate clockid_t */
|
||||
typedef int _tthread_clockid_t;
|
||||
#define clockid_t _tthread_clockid_t
|
||||
|
||||
/* Emulate clock_gettime */
|
||||
int _tthread_clock_gettime(clockid_t clk_id, struct timespec *ts);
|
||||
#define clock_gettime _tthread_clock_gettime
|
||||
#ifndef CLOCK_REALTIME
|
||||
#define CLOCK_REALTIME 0
|
||||
#endif
|
||||
#endif
|
||||
|
||||
|
||||
/** TinyCThread version (major number). */
|
||||
#define TINYCTHREAD_VERSION_MAJOR 1
|
||||
/** TinyCThread version (minor number). */
|
||||
#define TINYCTHREAD_VERSION_MINOR 1
|
||||
/** TinyCThread version (full version). */
|
||||
#define TINYCTHREAD_VERSION (TINYCTHREAD_VERSION_MAJOR * 100 + TINYCTHREAD_VERSION_MINOR)
|
||||
|
||||
/**
|
||||
* @def _Thread_local
|
||||
* Thread local storage keyword.
|
||||
* A variable that is declared with the @c _Thread_local keyword makes the
|
||||
* value of the variable local to each thread (known as thread-local storage,
|
||||
* or TLS). Example usage:
|
||||
* @code
|
||||
* // This variable is local to each thread.
|
||||
* _Thread_local int variable;
|
||||
* @endcode
|
||||
* @note The @c _Thread_local keyword is a macro that maps to the corresponding
|
||||
* compiler directive (e.g. @c __declspec(thread)).
|
||||
* @note This directive is currently not supported on Mac OS X (it will give
|
||||
* a compiler error), since compile-time TLS is not supported in the Mac OS X
|
||||
* executable format. Also, some older versions of MinGW (before GCC 4.x) do
|
||||
* not support this directive.
|
||||
* @hideinitializer
|
||||
*/
|
||||
|
||||
/* FIXME: Check for a PROPER value of __STDC_VERSION__ to know if we have C11 */
|
||||
#if !(defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201102L)) && !defined(_Thread_local)
|
||||
#if defined(__GNUC__) || defined(__INTEL_COMPILER) || defined(__SUNPRO_CC) || defined(__IBMCPP__)
|
||||
#define _Thread_local __thread
|
||||
#else
|
||||
#define _Thread_local __declspec(thread)
|
||||
#endif
|
||||
#endif
|
||||
|
||||
/* Macros */
|
||||
#define TSS_DTOR_ITERATIONS 0
|
||||
|
||||
/* Function return values */
|
||||
#define thrd_error 0 /**< The requested operation failed */
|
||||
#define thrd_success 1 /**< The requested operation succeeded */
|
||||
#define thrd_timeout 2 /**< The time specified in the call was reached without acquiring the requested resource */
|
||||
#define thrd_busy 3 /**< The requested operation failed because a tesource requested by a test and return function is already in use */
|
||||
#define thrd_nomem 4 /**< The requested operation failed because it was unable to allocate memory */
|
||||
|
||||
/* Mutex types */
|
||||
#define mtx_plain 1
|
||||
#define mtx_timed 2
|
||||
#define mtx_try 4
|
||||
#define mtx_recursive 8
|
||||
|
||||
/* Mutex */
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
typedef struct {
|
||||
CRITICAL_SECTION mHandle; /* Critical section handle */
|
||||
int mAlreadyLocked; /* TRUE if the mutex is already locked */
|
||||
int mRecursive; /* TRUE if the mutex is recursive */
|
||||
} mtx_t;
|
||||
#else
|
||||
typedef pthread_mutex_t mtx_t;
|
||||
#endif
|
||||
|
||||
/** Create a mutex object.
|
||||
* @param mtx A mutex object.
|
||||
* @param type Bit-mask that must have one of the following six values:
|
||||
* @li @c mtx_plain for a simple non-recursive mutex
|
||||
* @li @c mtx_timed for a non-recursive mutex that supports timeout
|
||||
* @li @c mtx_try for a non-recursive mutex that supports test and return
|
||||
* @li @c mtx_plain | @c mtx_recursive (same as @c mtx_plain, but recursive)
|
||||
* @li @c mtx_timed | @c mtx_recursive (same as @c mtx_timed, but recursive)
|
||||
* @li @c mtx_try | @c mtx_recursive (same as @c mtx_try, but recursive)
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int mtx_init(mtx_t *mtx, int type);
|
||||
|
||||
/** Release any resources used by the given mutex.
|
||||
* @param mtx A mutex object.
|
||||
*/
|
||||
void mtx_destroy(mtx_t *mtx);
|
||||
|
||||
/** Lock the given mutex.
|
||||
* Blocks until the given mutex can be locked. If the mutex is non-recursive, and
|
||||
* the calling thread already has a lock on the mutex, this call will block
|
||||
* forever.
|
||||
* @param mtx A mutex object.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int mtx_lock(mtx_t *mtx);
|
||||
|
||||
/** NOT YET IMPLEMENTED.
|
||||
*/
|
||||
int mtx_timedlock(mtx_t *mtx, const struct timespec *ts);
|
||||
|
||||
/** Try to lock the given mutex.
|
||||
* The specified mutex shall support either test and return or timeout. If the
|
||||
* mutex is already locked, the function returns without blocking.
|
||||
* @param mtx A mutex object.
|
||||
* @return @ref thrd_success on success, or @ref thrd_busy if the resource
|
||||
* requested is already in use, or @ref thrd_error if the request could not be
|
||||
* honored.
|
||||
*/
|
||||
int mtx_trylock(mtx_t *mtx);
|
||||
|
||||
/** Unlock the given mutex.
|
||||
* @param mtx A mutex object.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int mtx_unlock(mtx_t *mtx);
|
||||
|
||||
/* Condition variable */
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
typedef struct {
|
||||
HANDLE mEvents[2]; /* Signal and broadcast event HANDLEs. */
|
||||
unsigned int mWaitersCount; /* Count of the number of waiters. */
|
||||
CRITICAL_SECTION mWaitersCountLock; /* Serialize access to mWaitersCount. */
|
||||
} cnd_t;
|
||||
#else
|
||||
typedef pthread_cond_t cnd_t;
|
||||
#endif
|
||||
|
||||
/** Create a condition variable object.
|
||||
* @param cond A condition variable object.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int cnd_init(cnd_t *cond);
|
||||
|
||||
/** Release any resources used by the given condition variable.
|
||||
* @param cond A condition variable object.
|
||||
*/
|
||||
void cnd_destroy(cnd_t *cond);
|
||||
|
||||
/** Signal a condition variable.
|
||||
* Unblocks one of the threads that are blocked on the given condition variable
|
||||
* at the time of the call. If no threads are blocked on the condition variable
|
||||
* at the time of the call, the function does nothing and return success.
|
||||
* @param cond A condition variable object.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int cnd_signal(cnd_t *cond);
|
||||
|
||||
/** Broadcast a condition variable.
|
||||
* Unblocks all of the threads that are blocked on the given condition variable
|
||||
* at the time of the call. If no threads are blocked on the condition variable
|
||||
* at the time of the call, the function does nothing and return success.
|
||||
* @param cond A condition variable object.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int cnd_broadcast(cnd_t *cond);
|
||||
|
||||
/** Wait for a condition variable to become signaled.
|
||||
* The function atomically unlocks the given mutex and endeavors to block until
|
||||
* the given condition variable is signaled by a call to cnd_signal or to
|
||||
* cnd_broadcast. When the calling thread becomes unblocked it locks the mutex
|
||||
* before it returns.
|
||||
* @param cond A condition variable object.
|
||||
* @param mtx A mutex object.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int cnd_wait(cnd_t *cond, mtx_t *mtx);
|
||||
|
||||
/** Wait for a condition variable to become signaled.
|
||||
* The function atomically unlocks the given mutex and endeavors to block until
|
||||
* the given condition variable is signaled by a call to cnd_signal or to
|
||||
* cnd_broadcast, or until after the specified time. When the calling thread
|
||||
* becomes unblocked it locks the mutex before it returns.
|
||||
* @param cond A condition variable object.
|
||||
* @param mtx A mutex object.
|
||||
* @param xt A point in time at which the request will time out (absolute time).
|
||||
* @return @ref thrd_success upon success, or @ref thrd_timeout if the time
|
||||
* specified in the call was reached without acquiring the requested resource, or
|
||||
* @ref thrd_error if the request could not be honored.
|
||||
*/
|
||||
int cnd_timedwait(cnd_t *cond, mtx_t *mtx, const struct timespec *ts);
|
||||
|
||||
/* Thread */
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
typedef HANDLE thrd_t;
|
||||
#else
|
||||
typedef pthread_t thrd_t;
|
||||
#endif
|
||||
|
||||
/** Thread start function.
|
||||
* Any thread that is started with the @ref thrd_create() function must be
|
||||
* started through a function of this type.
|
||||
* @param arg The thread argument (the @c arg argument of the corresponding
|
||||
* @ref thrd_create() call).
|
||||
* @return The thread return value, which can be obtained by another thread
|
||||
* by using the @ref thrd_join() function.
|
||||
*/
|
||||
typedef int (*thrd_start_t)(void *arg);
|
||||
|
||||
/** Create a new thread.
|
||||
* @param thr Identifier of the newly created thread.
|
||||
* @param func A function pointer to the function that will be executed in
|
||||
* the new thread.
|
||||
* @param arg An argument to the thread function.
|
||||
* @return @ref thrd_success on success, or @ref thrd_nomem if no memory could
|
||||
* be allocated for the thread requested, or @ref thrd_error if the request
|
||||
* could not be honored.
|
||||
* @note A thread’s identifier may be reused for a different thread once the
|
||||
* original thread has exited and either been detached or joined to another
|
||||
* thread.
|
||||
*/
|
||||
int thrd_create(thrd_t *thr, thrd_start_t func, void *arg);
|
||||
|
||||
/** Identify the calling thread.
|
||||
* @return The identifier of the calling thread.
|
||||
*/
|
||||
thrd_t thrd_current(void);
|
||||
|
||||
/** NOT YET IMPLEMENTED.
|
||||
*/
|
||||
int thrd_detach(thrd_t thr);
|
||||
|
||||
/** Compare two thread identifiers.
|
||||
* The function determines if two thread identifiers refer to the same thread.
|
||||
* @return Zero if the two thread identifiers refer to different threads.
|
||||
* Otherwise a nonzero value is returned.
|
||||
*/
|
||||
int thrd_equal(thrd_t thr0, thrd_t thr1);
|
||||
|
||||
/** Terminate execution of the calling thread.
|
||||
* @param res Result code of the calling thread.
|
||||
*/
|
||||
void thrd_exit(int res);
|
||||
|
||||
/** Wait for a thread to terminate.
|
||||
* The function joins the given thread with the current thread by blocking
|
||||
* until the other thread has terminated.
|
||||
* @param thr The thread to join with.
|
||||
* @param res If this pointer is not NULL, the function will store the result
|
||||
* code of the given thread in the integer pointed to by @c res.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int thrd_join(thrd_t thr, int *res);
|
||||
|
||||
/** Put the calling thread to sleep.
|
||||
* Suspend execution of the calling thread.
|
||||
* @param time_point A point in time at which the thread will resume (absolute time).
|
||||
* @param remaining If non-NULL, this parameter will hold the remaining time until
|
||||
* time_point upon return. This will typically be zero, but if
|
||||
* the thread was woken up by a signal that is not ignored before
|
||||
* time_point was reached @c remaining will hold a positive
|
||||
* time.
|
||||
* @return 0 (zero) on successful sleep, or -1 if an interrupt occurred.
|
||||
*/
|
||||
int thrd_sleep(const struct timespec *time_point, struct timespec *remaining);
|
||||
|
||||
/** Yield execution to another thread.
|
||||
* Permit other threads to run, even if the current thread would ordinarily
|
||||
* continue to run.
|
||||
*/
|
||||
void thrd_yield(void);
|
||||
|
||||
/* Thread local storage */
|
||||
#if defined(_TTHREAD_WIN32_)
|
||||
typedef DWORD tss_t;
|
||||
#else
|
||||
typedef pthread_key_t tss_t;
|
||||
#endif
|
||||
|
||||
/** Destructor function for a thread-specific storage.
|
||||
* @param val The value of the destructed thread-specific storage.
|
||||
*/
|
||||
typedef void (*tss_dtor_t)(void *val);
|
||||
|
||||
/** Create a thread-specific storage.
|
||||
* @param key The unique key identifier that will be set if the function is
|
||||
* successful.
|
||||
* @param dtor Destructor function. This can be NULL.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
* @note The destructor function is not supported under Windows. If @c dtor is
|
||||
* not NULL when calling this function under Windows, the function will fail
|
||||
* and return @ref thrd_error.
|
||||
*/
|
||||
int tss_create(tss_t *key, tss_dtor_t dtor);
|
||||
|
||||
/** Delete a thread-specific storage.
|
||||
* The function releases any resources used by the given thread-specific
|
||||
* storage.
|
||||
* @param key The key that shall be deleted.
|
||||
*/
|
||||
void tss_delete(tss_t key);
|
||||
|
||||
/** Get the value for a thread-specific storage.
|
||||
* @param key The thread-specific storage identifier.
|
||||
* @return The value for the current thread held in the given thread-specific
|
||||
* storage.
|
||||
*/
|
||||
void *tss_get(tss_t key);
|
||||
|
||||
/** Set the value for a thread-specific storage.
|
||||
* @param key The thread-specific storage identifier.
|
||||
* @param val The value of the thread-specific storage to set for the current
|
||||
* thread.
|
||||
* @return @ref thrd_success on success, or @ref thrd_error if the request could
|
||||
* not be honored.
|
||||
*/
|
||||
int tss_set(tss_t key, void *val);
|
||||
|
||||
|
||||
#endif /* _TINYTHREAD_H_ */
|
||||
|
@ -1,102 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<protocol name="fractional_scale_v1">
|
||||
<copyright>
|
||||
Copyright © 2022 Kenny Levinsen
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a
|
||||
copy of this software and associated documentation files (the "Software"),
|
||||
to deal in the Software without restriction, including without limitation
|
||||
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
and/or sell copies of the Software, and to permit persons to whom the
|
||||
Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice (including the next
|
||||
paragraph) shall be included in all copies or substantial portions of the
|
||||
Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
DEALINGS IN THE SOFTWARE.
|
||||
</copyright>
|
||||
|
||||
<description summary="Protocol for requesting fractional surface scales">
|
||||
This protocol allows a compositor to suggest for surfaces to render at
|
||||
fractional scales.
|
||||
|
||||
A client can submit scaled content by utilizing wp_viewport. This is done by
|
||||
creating a wp_viewport object for the surface and setting the destination
|
||||
rectangle to the surface size before the scale factor is applied.
|
||||
|
||||
The buffer size is calculated by multiplying the surface size by the
|
||||
intended scale.
|
||||
|
||||
The wl_surface buffer scale should remain set to 1.
|
||||
|
||||
If a surface has a surface-local size of 100 px by 50 px and wishes to
|
||||
submit buffers with a scale of 1.5, then a buffer of 150px by 75 px should
|
||||
be used and the wp_viewport destination rectangle should be 100 px by 50 px.
|
||||
|
||||
For toplevel surfaces, the size is rounded halfway away from zero. The
|
||||
rounding algorithm for subsurface position and size is not defined.
|
||||
</description>
|
||||
|
||||
<interface name="wp_fractional_scale_manager_v1" version="1">
|
||||
<description summary="fractional surface scale information">
|
||||
A global interface for requesting surfaces to use fractional scales.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="unbind the fractional surface scale interface">
|
||||
Informs the server that the client will not be using this protocol
|
||||
object anymore. This does not affect any other objects,
|
||||
wp_fractional_scale_v1 objects included.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<enum name="error">
|
||||
<entry name="fractional_scale_exists" value="0"
|
||||
summary="the surface already has a fractional_scale object associated"/>
|
||||
</enum>
|
||||
|
||||
<request name="get_fractional_scale">
|
||||
<description summary="extend surface interface for scale information">
|
||||
Create an add-on object for the the wl_surface to let the compositor
|
||||
request fractional scales. If the given wl_surface already has a
|
||||
wp_fractional_scale_v1 object associated, the fractional_scale_exists
|
||||
protocol error is raised.
|
||||
</description>
|
||||
<arg name="id" type="new_id" interface="wp_fractional_scale_v1"
|
||||
summary="the new surface scale info interface id"/>
|
||||
<arg name="surface" type="object" interface="wl_surface"
|
||||
summary="the surface"/>
|
||||
</request>
|
||||
</interface>
|
||||
|
||||
<interface name="wp_fractional_scale_v1" version="1">
|
||||
<description summary="fractional scale interface to a wl_surface">
|
||||
An additional interface to a wl_surface object which allows the compositor
|
||||
to inform the client of the preferred scale.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="remove surface scale information for surface">
|
||||
Destroy the fractional scale object. When this object is destroyed,
|
||||
preferred_scale events will no longer be sent.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<event name="preferred_scale">
|
||||
<description summary="notify of new preferred scale">
|
||||
Notification of a new preferred scale for this surface that the
|
||||
compositor suggests that the client should use.
|
||||
|
||||
The sent scale is the numerator of a fraction with a denominator of 120.
|
||||
</description>
|
||||
<arg name="scale" type="uint" summary="the new preferred scale"/>
|
||||
</event>
|
||||
</interface>
|
||||
</protocol>
|
@ -1,83 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<protocol name="idle_inhibit_unstable_v1">
|
||||
|
||||
<copyright>
|
||||
Copyright © 2015 Samsung Electronics Co., Ltd
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a
|
||||
copy of this software and associated documentation files (the "Software"),
|
||||
to deal in the Software without restriction, including without limitation
|
||||
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
and/or sell copies of the Software, and to permit persons to whom the
|
||||
Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice (including the next
|
||||
paragraph) shall be included in all copies or substantial portions of the
|
||||
Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
DEALINGS IN THE SOFTWARE.
|
||||
</copyright>
|
||||
|
||||
<interface name="zwp_idle_inhibit_manager_v1" version="1">
|
||||
<description summary="control behavior when display idles">
|
||||
This interface permits inhibiting the idle behavior such as screen
|
||||
blanking, locking, and screensaving. The client binds the idle manager
|
||||
globally, then creates idle-inhibitor objects for each surface.
|
||||
|
||||
Warning! The protocol described in this file is experimental and
|
||||
backward incompatible changes may be made. Backward compatible changes
|
||||
may be added together with the corresponding interface version bump.
|
||||
Backward incompatible changes are done by bumping the version number in
|
||||
the protocol and interface names and resetting the interface version.
|
||||
Once the protocol is to be declared stable, the 'z' prefix and the
|
||||
version number in the protocol and interface names are removed and the
|
||||
interface version number is reset.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the idle inhibitor object">
|
||||
Destroy the inhibit manager.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<request name="create_inhibitor">
|
||||
<description summary="create a new inhibitor object">
|
||||
Create a new inhibitor object associated with the given surface.
|
||||
</description>
|
||||
<arg name="id" type="new_id" interface="zwp_idle_inhibitor_v1"/>
|
||||
<arg name="surface" type="object" interface="wl_surface"
|
||||
summary="the surface that inhibits the idle behavior"/>
|
||||
</request>
|
||||
|
||||
</interface>
|
||||
|
||||
<interface name="zwp_idle_inhibitor_v1" version="1">
|
||||
<description summary="context object for inhibiting idle behavior">
|
||||
An idle inhibitor prevents the output that the associated surface is
|
||||
visible on from being set to a state where it is not visually usable due
|
||||
to lack of user interaction (e.g. blanked, dimmed, locked, set to power
|
||||
save, etc.) Any screensaver processes are also blocked from displaying.
|
||||
|
||||
If the surface is destroyed, unmapped, becomes occluded, loses
|
||||
visibility, or otherwise becomes not visually relevant for the user, the
|
||||
idle inhibitor will not be honored by the compositor; if the surface
|
||||
subsequently regains visibility the inhibitor takes effect once again.
|
||||
Likewise, the inhibitor isn't honored if the system was already idled at
|
||||
the time the inhibitor was established, although if the system later
|
||||
de-idles and re-idles the inhibitor will take effect.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the idle inhibitor object">
|
||||
Remove the inhibitor effect from the associated wl_surface.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
</interface>
|
||||
</protocol>
|
@ -1,339 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<protocol name="pointer_constraints_unstable_v1">
|
||||
|
||||
<copyright>
|
||||
Copyright © 2014 Jonas Ådahl
|
||||
Copyright © 2015 Red Hat Inc.
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a
|
||||
copy of this software and associated documentation files (the "Software"),
|
||||
to deal in the Software without restriction, including without limitation
|
||||
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
and/or sell copies of the Software, and to permit persons to whom the
|
||||
Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice (including the next
|
||||
paragraph) shall be included in all copies or substantial portions of the
|
||||
Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
DEALINGS IN THE SOFTWARE.
|
||||
</copyright>
|
||||
|
||||
<description summary="protocol for constraining pointer motions">
|
||||
This protocol specifies a set of interfaces used for adding constraints to
|
||||
the motion of a pointer. Possible constraints include confining pointer
|
||||
motions to a given region, or locking it to its current position.
|
||||
|
||||
In order to constrain the pointer, a client must first bind the global
|
||||
interface "wp_pointer_constraints" which, if a compositor supports pointer
|
||||
constraints, is exposed by the registry. Using the bound global object, the
|
||||
client uses the request that corresponds to the type of constraint it wants
|
||||
to make. See wp_pointer_constraints for more details.
|
||||
|
||||
Warning! The protocol described in this file is experimental and backward
|
||||
incompatible changes may be made. Backward compatible changes may be added
|
||||
together with the corresponding interface version bump. Backward
|
||||
incompatible changes are done by bumping the version number in the protocol
|
||||
and interface names and resetting the interface version. Once the protocol
|
||||
is to be declared stable, the 'z' prefix and the version number in the
|
||||
protocol and interface names are removed and the interface version number is
|
||||
reset.
|
||||
</description>
|
||||
|
||||
<interface name="zwp_pointer_constraints_v1" version="1">
|
||||
<description summary="constrain the movement of a pointer">
|
||||
The global interface exposing pointer constraining functionality. It
|
||||
exposes two requests: lock_pointer for locking the pointer to its
|
||||
position, and confine_pointer for locking the pointer to a region.
|
||||
|
||||
The lock_pointer and confine_pointer requests create the objects
|
||||
wp_locked_pointer and wp_confined_pointer respectively, and the client can
|
||||
use these objects to interact with the lock.
|
||||
|
||||
For any surface, only one lock or confinement may be active across all
|
||||
wl_pointer objects of the same seat. If a lock or confinement is requested
|
||||
when another lock or confinement is active or requested on the same surface
|
||||
and with any of the wl_pointer objects of the same seat, an
|
||||
'already_constrained' error will be raised.
|
||||
</description>
|
||||
|
||||
<enum name="error">
|
||||
<description summary="wp_pointer_constraints error values">
|
||||
These errors can be emitted in response to wp_pointer_constraints
|
||||
requests.
|
||||
</description>
|
||||
<entry name="already_constrained" value="1"
|
||||
summary="pointer constraint already requested on that surface"/>
|
||||
</enum>
|
||||
|
||||
<enum name="lifetime">
|
||||
<description summary="constraint lifetime">
|
||||
These values represent different lifetime semantics. They are passed
|
||||
as arguments to the factory requests to specify how the constraint
|
||||
lifetimes should be managed.
|
||||
</description>
|
||||
<entry name="oneshot" value="1">
|
||||
<description summary="the pointer constraint is defunct once deactivated">
|
||||
A oneshot pointer constraint will never reactivate once it has been
|
||||
deactivated. See the corresponding deactivation event
|
||||
(wp_locked_pointer.unlocked and wp_confined_pointer.unconfined) for
|
||||
details.
|
||||
</description>
|
||||
</entry>
|
||||
<entry name="persistent" value="2">
|
||||
<description summary="the pointer constraint may reactivate">
|
||||
A persistent pointer constraint may again reactivate once it has
|
||||
been deactivated. See the corresponding deactivation event
|
||||
(wp_locked_pointer.unlocked and wp_confined_pointer.unconfined) for
|
||||
details.
|
||||
</description>
|
||||
</entry>
|
||||
</enum>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the pointer constraints manager object">
|
||||
Used by the client to notify the server that it will no longer use this
|
||||
pointer constraints object.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<request name="lock_pointer">
|
||||
<description summary="lock pointer to a position">
|
||||
The lock_pointer request lets the client request to disable movements of
|
||||
the virtual pointer (i.e. the cursor), effectively locking the pointer
|
||||
to a position. This request may not take effect immediately; in the
|
||||
future, when the compositor deems implementation-specific constraints
|
||||
are satisfied, the pointer lock will be activated and the compositor
|
||||
sends a locked event.
|
||||
|
||||
The protocol provides no guarantee that the constraints are ever
|
||||
satisfied, and does not require the compositor to send an error if the
|
||||
constraints cannot ever be satisfied. It is thus possible to request a
|
||||
lock that will never activate.
|
||||
|
||||
There may not be another pointer constraint of any kind requested or
|
||||
active on the surface for any of the wl_pointer objects of the seat of
|
||||
the passed pointer when requesting a lock. If there is, an error will be
|
||||
raised. See general pointer lock documentation for more details.
|
||||
|
||||
The intersection of the region passed with this request and the input
|
||||
region of the surface is used to determine where the pointer must be
|
||||
in order for the lock to activate. It is up to the compositor whether to
|
||||
warp the pointer or require some kind of user interaction for the lock
|
||||
to activate. If the region is null the surface input region is used.
|
||||
|
||||
A surface may receive pointer focus without the lock being activated.
|
||||
|
||||
The request creates a new object wp_locked_pointer which is used to
|
||||
interact with the lock as well as receive updates about its state. See
|
||||
the the description of wp_locked_pointer for further information.
|
||||
|
||||
Note that while a pointer is locked, the wl_pointer objects of the
|
||||
corresponding seat will not emit any wl_pointer.motion events, but
|
||||
relative motion events will still be emitted via wp_relative_pointer
|
||||
objects of the same seat. wl_pointer.axis and wl_pointer.button events
|
||||
are unaffected.
|
||||
</description>
|
||||
<arg name="id" type="new_id" interface="zwp_locked_pointer_v1"/>
|
||||
<arg name="surface" type="object" interface="wl_surface"
|
||||
summary="surface to lock pointer to"/>
|
||||
<arg name="pointer" type="object" interface="wl_pointer"
|
||||
summary="the pointer that should be locked"/>
|
||||
<arg name="region" type="object" interface="wl_region" allow-null="true"
|
||||
summary="region of surface"/>
|
||||
<arg name="lifetime" type="uint" enum="lifetime" summary="lock lifetime"/>
|
||||
</request>
|
||||
|
||||
<request name="confine_pointer">
|
||||
<description summary="confine pointer to a region">
|
||||
The confine_pointer request lets the client request to confine the
|
||||
pointer cursor to a given region. This request may not take effect
|
||||
immediately; in the future, when the compositor deems implementation-
|
||||
specific constraints are satisfied, the pointer confinement will be
|
||||
activated and the compositor sends a confined event.
|
||||
|
||||
The intersection of the region passed with this request and the input
|
||||
region of the surface is used to determine where the pointer must be
|
||||
in order for the confinement to activate. It is up to the compositor
|
||||
whether to warp the pointer or require some kind of user interaction for
|
||||
the confinement to activate. If the region is null the surface input
|
||||
region is used.
|
||||
|
||||
The request will create a new object wp_confined_pointer which is used
|
||||
to interact with the confinement as well as receive updates about its
|
||||
state. See the the description of wp_confined_pointer for further
|
||||
information.
|
||||
</description>
|
||||
<arg name="id" type="new_id" interface="zwp_confined_pointer_v1"/>
|
||||
<arg name="surface" type="object" interface="wl_surface"
|
||||
summary="surface to lock pointer to"/>
|
||||
<arg name="pointer" type="object" interface="wl_pointer"
|
||||
summary="the pointer that should be confined"/>
|
||||
<arg name="region" type="object" interface="wl_region" allow-null="true"
|
||||
summary="region of surface"/>
|
||||
<arg name="lifetime" type="uint" enum="lifetime" summary="confinement lifetime"/>
|
||||
</request>
|
||||
</interface>
|
||||
|
||||
<interface name="zwp_locked_pointer_v1" version="1">
|
||||
<description summary="receive relative pointer motion events">
|
||||
The wp_locked_pointer interface represents a locked pointer state.
|
||||
|
||||
While the lock of this object is active, the wl_pointer objects of the
|
||||
associated seat will not emit any wl_pointer.motion events.
|
||||
|
||||
This object will send the event 'locked' when the lock is activated.
|
||||
Whenever the lock is activated, it is guaranteed that the locked surface
|
||||
will already have received pointer focus and that the pointer will be
|
||||
within the region passed to the request creating this object.
|
||||
|
||||
To unlock the pointer, send the destroy request. This will also destroy
|
||||
the wp_locked_pointer object.
|
||||
|
||||
If the compositor decides to unlock the pointer the unlocked event is
|
||||
sent. See wp_locked_pointer.unlock for details.
|
||||
|
||||
When unlocking, the compositor may warp the cursor position to the set
|
||||
cursor position hint. If it does, it will not result in any relative
|
||||
motion events emitted via wp_relative_pointer.
|
||||
|
||||
If the surface the lock was requested on is destroyed and the lock is not
|
||||
yet activated, the wp_locked_pointer object is now defunct and must be
|
||||
destroyed.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the locked pointer object">
|
||||
Destroy the locked pointer object. If applicable, the compositor will
|
||||
unlock the pointer.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<request name="set_cursor_position_hint">
|
||||
<description summary="set the pointer cursor position hint">
|
||||
Set the cursor position hint relative to the top left corner of the
|
||||
surface.
|
||||
|
||||
If the client is drawing its own cursor, it should update the position
|
||||
hint to the position of its own cursor. A compositor may use this
|
||||
information to warp the pointer upon unlock in order to avoid pointer
|
||||
jumps.
|
||||
|
||||
The cursor position hint is double buffered. The new hint will only take
|
||||
effect when the associated surface gets it pending state applied. See
|
||||
wl_surface.commit for details.
|
||||
</description>
|
||||
<arg name="surface_x" type="fixed"
|
||||
summary="surface-local x coordinate"/>
|
||||
<arg name="surface_y" type="fixed"
|
||||
summary="surface-local y coordinate"/>
|
||||
</request>
|
||||
|
||||
<request name="set_region">
|
||||
<description summary="set a new lock region">
|
||||
Set a new region used to lock the pointer.
|
||||
|
||||
The new lock region is double-buffered. The new lock region will
|
||||
only take effect when the associated surface gets its pending state
|
||||
applied. See wl_surface.commit for details.
|
||||
|
||||
For details about the lock region, see wp_locked_pointer.
|
||||
</description>
|
||||
<arg name="region" type="object" interface="wl_region" allow-null="true"
|
||||
summary="region of surface"/>
|
||||
</request>
|
||||
|
||||
<event name="locked">
|
||||
<description summary="lock activation event">
|
||||
Notification that the pointer lock of the seat's pointer is activated.
|
||||
</description>
|
||||
</event>
|
||||
|
||||
<event name="unlocked">
|
||||
<description summary="lock deactivation event">
|
||||
Notification that the pointer lock of the seat's pointer is no longer
|
||||
active. If this is a oneshot pointer lock (see
|
||||
wp_pointer_constraints.lifetime) this object is now defunct and should
|
||||
be destroyed. If this is a persistent pointer lock (see
|
||||
wp_pointer_constraints.lifetime) this pointer lock may again
|
||||
reactivate in the future.
|
||||
</description>
|
||||
</event>
|
||||
</interface>
|
||||
|
||||
<interface name="zwp_confined_pointer_v1" version="1">
|
||||
<description summary="confined pointer object">
|
||||
The wp_confined_pointer interface represents a confined pointer state.
|
||||
|
||||
This object will send the event 'confined' when the confinement is
|
||||
activated. Whenever the confinement is activated, it is guaranteed that
|
||||
the surface the pointer is confined to will already have received pointer
|
||||
focus and that the pointer will be within the region passed to the request
|
||||
creating this object. It is up to the compositor to decide whether this
|
||||
requires some user interaction and if the pointer will warp to within the
|
||||
passed region if outside.
|
||||
|
||||
To unconfine the pointer, send the destroy request. This will also destroy
|
||||
the wp_confined_pointer object.
|
||||
|
||||
If the compositor decides to unconfine the pointer the unconfined event is
|
||||
sent. The wp_confined_pointer object is at this point defunct and should
|
||||
be destroyed.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the confined pointer object">
|
||||
Destroy the confined pointer object. If applicable, the compositor will
|
||||
unconfine the pointer.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<request name="set_region">
|
||||
<description summary="set a new confine region">
|
||||
Set a new region used to confine the pointer.
|
||||
|
||||
The new confine region is double-buffered. The new confine region will
|
||||
only take effect when the associated surface gets its pending state
|
||||
applied. See wl_surface.commit for details.
|
||||
|
||||
If the confinement is active when the new confinement region is applied
|
||||
and the pointer ends up outside of newly applied region, the pointer may
|
||||
warped to a position within the new confinement region. If warped, a
|
||||
wl_pointer.motion event will be emitted, but no
|
||||
wp_relative_pointer.relative_motion event.
|
||||
|
||||
The compositor may also, instead of using the new region, unconfine the
|
||||
pointer.
|
||||
|
||||
For details about the confine region, see wp_confined_pointer.
|
||||
</description>
|
||||
<arg name="region" type="object" interface="wl_region" allow-null="true"
|
||||
summary="region of surface"/>
|
||||
</request>
|
||||
|
||||
<event name="confined">
|
||||
<description summary="pointer confined">
|
||||
Notification that the pointer confinement of the seat's pointer is
|
||||
activated.
|
||||
</description>
|
||||
</event>
|
||||
|
||||
<event name="unconfined">
|
||||
<description summary="pointer unconfined">
|
||||
Notification that the pointer confinement of the seat's pointer is no
|
||||
longer active. If this is a oneshot pointer confinement (see
|
||||
wp_pointer_constraints.lifetime) this object is now defunct and should
|
||||
be destroyed. If this is a persistent pointer confinement (see
|
||||
wp_pointer_constraints.lifetime) this pointer confinement may again
|
||||
reactivate in the future.
|
||||
</description>
|
||||
</event>
|
||||
</interface>
|
||||
|
||||
</protocol>
|
@ -1,136 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<protocol name="relative_pointer_unstable_v1">
|
||||
|
||||
<copyright>
|
||||
Copyright © 2014 Jonas Ådahl
|
||||
Copyright © 2015 Red Hat Inc.
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a
|
||||
copy of this software and associated documentation files (the "Software"),
|
||||
to deal in the Software without restriction, including without limitation
|
||||
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
and/or sell copies of the Software, and to permit persons to whom the
|
||||
Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice (including the next
|
||||
paragraph) shall be included in all copies or substantial portions of the
|
||||
Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
DEALINGS IN THE SOFTWARE.
|
||||
</copyright>
|
||||
|
||||
<description summary="protocol for relative pointer motion events">
|
||||
This protocol specifies a set of interfaces used for making clients able to
|
||||
receive relative pointer events not obstructed by barriers (such as the
|
||||
monitor edge or other pointer barriers).
|
||||
|
||||
To start receiving relative pointer events, a client must first bind the
|
||||
global interface "wp_relative_pointer_manager" which, if a compositor
|
||||
supports relative pointer motion events, is exposed by the registry. After
|
||||
having created the relative pointer manager proxy object, the client uses
|
||||
it to create the actual relative pointer object using the
|
||||
"get_relative_pointer" request given a wl_pointer. The relative pointer
|
||||
motion events will then, when applicable, be transmitted via the proxy of
|
||||
the newly created relative pointer object. See the documentation of the
|
||||
relative pointer interface for more details.
|
||||
|
||||
Warning! The protocol described in this file is experimental and backward
|
||||
incompatible changes may be made. Backward compatible changes may be added
|
||||
together with the corresponding interface version bump. Backward
|
||||
incompatible changes are done by bumping the version number in the protocol
|
||||
and interface names and resetting the interface version. Once the protocol
|
||||
is to be declared stable, the 'z' prefix and the version number in the
|
||||
protocol and interface names are removed and the interface version number is
|
||||
reset.
|
||||
</description>
|
||||
|
||||
<interface name="zwp_relative_pointer_manager_v1" version="1">
|
||||
<description summary="get relative pointer objects">
|
||||
A global interface used for getting the relative pointer object for a
|
||||
given pointer.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the relative pointer manager object">
|
||||
Used by the client to notify the server that it will no longer use this
|
||||
relative pointer manager object.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<request name="get_relative_pointer">
|
||||
<description summary="get a relative pointer object">
|
||||
Create a relative pointer interface given a wl_pointer object. See the
|
||||
wp_relative_pointer interface for more details.
|
||||
</description>
|
||||
<arg name="id" type="new_id" interface="zwp_relative_pointer_v1"/>
|
||||
<arg name="pointer" type="object" interface="wl_pointer"/>
|
||||
</request>
|
||||
</interface>
|
||||
|
||||
<interface name="zwp_relative_pointer_v1" version="1">
|
||||
<description summary="relative pointer object">
|
||||
A wp_relative_pointer object is an extension to the wl_pointer interface
|
||||
used for emitting relative pointer events. It shares the same focus as
|
||||
wl_pointer objects of the same seat and will only emit events when it has
|
||||
focus.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="release the relative pointer object"/>
|
||||
</request>
|
||||
|
||||
<event name="relative_motion">
|
||||
<description summary="relative pointer motion">
|
||||
Relative x/y pointer motion from the pointer of the seat associated with
|
||||
this object.
|
||||
|
||||
A relative motion is in the same dimension as regular wl_pointer motion
|
||||
events, except they do not represent an absolute position. For example,
|
||||
moving a pointer from (x, y) to (x', y') would have the equivalent
|
||||
relative motion (x' - x, y' - y). If a pointer motion caused the
|
||||
absolute pointer position to be clipped by for example the edge of the
|
||||
monitor, the relative motion is unaffected by the clipping and will
|
||||
represent the unclipped motion.
|
||||
|
||||
This event also contains non-accelerated motion deltas. The
|
||||
non-accelerated delta is, when applicable, the regular pointer motion
|
||||
delta as it was before having applied motion acceleration and other
|
||||
transformations such as normalization.
|
||||
|
||||
Note that the non-accelerated delta does not represent 'raw' events as
|
||||
they were read from some device. Pointer motion acceleration is device-
|
||||
and configuration-specific and non-accelerated deltas and accelerated
|
||||
deltas may have the same value on some devices.
|
||||
|
||||
Relative motions are not coupled to wl_pointer.motion events, and can be
|
||||
sent in combination with such events, but also independently. There may
|
||||
also be scenarios where wl_pointer.motion is sent, but there is no
|
||||
relative motion. The order of an absolute and relative motion event
|
||||
originating from the same physical motion is not guaranteed.
|
||||
|
||||
If the client needs button events or focus state, it can receive them
|
||||
from a wl_pointer object of the same seat that the wp_relative_pointer
|
||||
object is associated with.
|
||||
</description>
|
||||
<arg name="utime_hi" type="uint"
|
||||
summary="high 32 bits of a 64 bit timestamp with microsecond granularity"/>
|
||||
<arg name="utime_lo" type="uint"
|
||||
summary="low 32 bits of a 64 bit timestamp with microsecond granularity"/>
|
||||
<arg name="dx" type="fixed"
|
||||
summary="the x component of the motion vector"/>
|
||||
<arg name="dy" type="fixed"
|
||||
summary="the y component of the motion vector"/>
|
||||
<arg name="dx_unaccel" type="fixed"
|
||||
summary="the x component of the unaccelerated motion vector"/>
|
||||
<arg name="dy_unaccel" type="fixed"
|
||||
summary="the y component of the unaccelerated motion vector"/>
|
||||
</event>
|
||||
</interface>
|
||||
|
||||
</protocol>
|
180
dependencies/glfw-3.4/deps/wayland/viewporter.xml
vendored
180
dependencies/glfw-3.4/deps/wayland/viewporter.xml
vendored
@ -1,180 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<protocol name="viewporter">
|
||||
|
||||
<copyright>
|
||||
Copyright © 2013-2016 Collabora, Ltd.
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a
|
||||
copy of this software and associated documentation files (the "Software"),
|
||||
to deal in the Software without restriction, including without limitation
|
||||
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
and/or sell copies of the Software, and to permit persons to whom the
|
||||
Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice (including the next
|
||||
paragraph) shall be included in all copies or substantial portions of the
|
||||
Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
DEALINGS IN THE SOFTWARE.
|
||||
</copyright>
|
||||
|
||||
<interface name="wp_viewporter" version="1">
|
||||
<description summary="surface cropping and scaling">
|
||||
The global interface exposing surface cropping and scaling
|
||||
capabilities is used to instantiate an interface extension for a
|
||||
wl_surface object. This extended interface will then allow
|
||||
cropping and scaling the surface contents, effectively
|
||||
disconnecting the direct relationship between the buffer and the
|
||||
surface size.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="unbind from the cropping and scaling interface">
|
||||
Informs the server that the client will not be using this
|
||||
protocol object anymore. This does not affect any other objects,
|
||||
wp_viewport objects included.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<enum name="error">
|
||||
<entry name="viewport_exists" value="0"
|
||||
summary="the surface already has a viewport object associated"/>
|
||||
</enum>
|
||||
|
||||
<request name="get_viewport">
|
||||
<description summary="extend surface interface for crop and scale">
|
||||
Instantiate an interface extension for the given wl_surface to
|
||||
crop and scale its content. If the given wl_surface already has
|
||||
a wp_viewport object associated, the viewport_exists
|
||||
protocol error is raised.
|
||||
</description>
|
||||
<arg name="id" type="new_id" interface="wp_viewport"
|
||||
summary="the new viewport interface id"/>
|
||||
<arg name="surface" type="object" interface="wl_surface"
|
||||
summary="the surface"/>
|
||||
</request>
|
||||
</interface>
|
||||
|
||||
<interface name="wp_viewport" version="1">
|
||||
<description summary="crop and scale interface to a wl_surface">
|
||||
An additional interface to a wl_surface object, which allows the
|
||||
client to specify the cropping and scaling of the surface
|
||||
contents.
|
||||
|
||||
This interface works with two concepts: the source rectangle (src_x,
|
||||
src_y, src_width, src_height), and the destination size (dst_width,
|
||||
dst_height). The contents of the source rectangle are scaled to the
|
||||
destination size, and content outside the source rectangle is ignored.
|
||||
This state is double-buffered, and is applied on the next
|
||||
wl_surface.commit.
|
||||
|
||||
The two parts of crop and scale state are independent: the source
|
||||
rectangle, and the destination size. Initially both are unset, that
|
||||
is, no scaling is applied. The whole of the current wl_buffer is
|
||||
used as the source, and the surface size is as defined in
|
||||
wl_surface.attach.
|
||||
|
||||
If the destination size is set, it causes the surface size to become
|
||||
dst_width, dst_height. The source (rectangle) is scaled to exactly
|
||||
this size. This overrides whatever the attached wl_buffer size is,
|
||||
unless the wl_buffer is NULL. If the wl_buffer is NULL, the surface
|
||||
has no content and therefore no size. Otherwise, the size is always
|
||||
at least 1x1 in surface local coordinates.
|
||||
|
||||
If the source rectangle is set, it defines what area of the wl_buffer is
|
||||
taken as the source. If the source rectangle is set and the destination
|
||||
size is not set, then src_width and src_height must be integers, and the
|
||||
surface size becomes the source rectangle size. This results in cropping
|
||||
without scaling. If src_width or src_height are not integers and
|
||||
destination size is not set, the bad_size protocol error is raised when
|
||||
the surface state is applied.
|
||||
|
||||
The coordinate transformations from buffer pixel coordinates up to
|
||||
the surface-local coordinates happen in the following order:
|
||||
1. buffer_transform (wl_surface.set_buffer_transform)
|
||||
2. buffer_scale (wl_surface.set_buffer_scale)
|
||||
3. crop and scale (wp_viewport.set*)
|
||||
This means, that the source rectangle coordinates of crop and scale
|
||||
are given in the coordinates after the buffer transform and scale,
|
||||
i.e. in the coordinates that would be the surface-local coordinates
|
||||
if the crop and scale was not applied.
|
||||
|
||||
If src_x or src_y are negative, the bad_value protocol error is raised.
|
||||
Otherwise, if the source rectangle is partially or completely outside of
|
||||
the non-NULL wl_buffer, then the out_of_buffer protocol error is raised
|
||||
when the surface state is applied. A NULL wl_buffer does not raise the
|
||||
out_of_buffer error.
|
||||
|
||||
If the wl_surface associated with the wp_viewport is destroyed,
|
||||
all wp_viewport requests except 'destroy' raise the protocol error
|
||||
no_surface.
|
||||
|
||||
If the wp_viewport object is destroyed, the crop and scale
|
||||
state is removed from the wl_surface. The change will be applied
|
||||
on the next wl_surface.commit.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="remove scaling and cropping from the surface">
|
||||
The associated wl_surface's crop and scale state is removed.
|
||||
The change is applied on the next wl_surface.commit.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<enum name="error">
|
||||
<entry name="bad_value" value="0"
|
||||
summary="negative or zero values in width or height"/>
|
||||
<entry name="bad_size" value="1"
|
||||
summary="destination size is not integer"/>
|
||||
<entry name="out_of_buffer" value="2"
|
||||
summary="source rectangle extends outside of the content area"/>
|
||||
<entry name="no_surface" value="3"
|
||||
summary="the wl_surface was destroyed"/>
|
||||
</enum>
|
||||
|
||||
<request name="set_source">
|
||||
<description summary="set the source rectangle for cropping">
|
||||
Set the source rectangle of the associated wl_surface. See
|
||||
wp_viewport for the description, and relation to the wl_buffer
|
||||
size.
|
||||
|
||||
If all of x, y, width and height are -1.0, the source rectangle is
|
||||
unset instead. Any other set of values where width or height are zero
|
||||
or negative, or x or y are negative, raise the bad_value protocol
|
||||
error.
|
||||
|
||||
The crop and scale state is double-buffered state, and will be
|
||||
applied on the next wl_surface.commit.
|
||||
</description>
|
||||
<arg name="x" type="fixed" summary="source rectangle x"/>
|
||||
<arg name="y" type="fixed" summary="source rectangle y"/>
|
||||
<arg name="width" type="fixed" summary="source rectangle width"/>
|
||||
<arg name="height" type="fixed" summary="source rectangle height"/>
|
||||
</request>
|
||||
|
||||
<request name="set_destination">
|
||||
<description summary="set the surface size for scaling">
|
||||
Set the destination size of the associated wl_surface. See
|
||||
wp_viewport for the description, and relation to the wl_buffer
|
||||
size.
|
||||
|
||||
If width is -1 and height is -1, the destination size is unset
|
||||
instead. Any other pair of values for width and height that
|
||||
contains zero or negative values raises the bad_value protocol
|
||||
error.
|
||||
|
||||
The crop and scale state is double-buffered state, and will be
|
||||
applied on the next wl_surface.commit.
|
||||
</description>
|
||||
<arg name="width" type="int" summary="surface width"/>
|
||||
<arg name="height" type="int" summary="surface height"/>
|
||||
</request>
|
||||
</interface>
|
||||
|
||||
</protocol>
|
3151
dependencies/glfw-3.4/deps/wayland/wayland.xml
vendored
3151
dependencies/glfw-3.4/deps/wayland/wayland.xml
vendored
File diff suppressed because it is too large
Load Diff
@ -1,200 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<protocol name="xdg_activation_v1">
|
||||
|
||||
<copyright>
|
||||
Copyright © 2020 Aleix Pol Gonzalez <aleixpol@kde.org>
|
||||
Copyright © 2020 Carlos Garnacho <carlosg@gnome.org>
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a
|
||||
copy of this software and associated documentation files (the "Software"),
|
||||
to deal in the Software without restriction, including without limitation
|
||||
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
and/or sell copies of the Software, and to permit persons to whom the
|
||||
Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice (including the next
|
||||
paragraph) shall be included in all copies or substantial portions of the
|
||||
Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
DEALINGS IN THE SOFTWARE.
|
||||
</copyright>
|
||||
|
||||
<description summary="Protocol for requesting activation of surfaces">
|
||||
The way for a client to pass focus to another toplevel is as follows.
|
||||
|
||||
The client that intends to activate another toplevel uses the
|
||||
xdg_activation_v1.get_activation_token request to get an activation token.
|
||||
This token is then forwarded to the client, which is supposed to activate
|
||||
one of its surfaces, through a separate band of communication.
|
||||
|
||||
One established way of doing this is through the XDG_ACTIVATION_TOKEN
|
||||
environment variable of a newly launched child process. The child process
|
||||
should unset the environment variable again right after reading it out in
|
||||
order to avoid propagating it to other child processes.
|
||||
|
||||
Another established way exists for Applications implementing the D-Bus
|
||||
interface org.freedesktop.Application, which should get their token under
|
||||
activation-token on their platform_data.
|
||||
|
||||
In general activation tokens may be transferred across clients through
|
||||
means not described in this protocol.
|
||||
|
||||
The client to be activated will then pass the token
|
||||
it received to the xdg_activation_v1.activate request. The compositor can
|
||||
then use this token to decide how to react to the activation request.
|
||||
|
||||
The token the activating client gets may be ineffective either already at
|
||||
the time it receives it, for example if it was not focused, for focus
|
||||
stealing prevention. The activating client will have no way to discover
|
||||
the validity of the token, and may still forward it to the to be activated
|
||||
client.
|
||||
|
||||
The created activation token may optionally get information attached to it
|
||||
that can be used by the compositor to identify the application that we
|
||||
intend to activate. This can for example be used to display a visual hint
|
||||
about what application is being started.
|
||||
|
||||
Warning! The protocol described in this file is currently in the testing
|
||||
phase. Backward compatible changes may be added together with the
|
||||
corresponding interface version bump. Backward incompatible changes can
|
||||
only be done by creating a new major version of the extension.
|
||||
</description>
|
||||
|
||||
<interface name="xdg_activation_v1" version="1">
|
||||
<description summary="interface for activating surfaces">
|
||||
A global interface used for informing the compositor about applications
|
||||
being activated or started, or for applications to request to be
|
||||
activated.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the xdg_activation object">
|
||||
Notify the compositor that the xdg_activation object will no longer be
|
||||
used.
|
||||
|
||||
The child objects created via this interface are unaffected and should
|
||||
be destroyed separately.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<request name="get_activation_token">
|
||||
<description summary="requests a token">
|
||||
Creates an xdg_activation_token_v1 object that will provide
|
||||
the initiating client with a unique token for this activation. This
|
||||
token should be offered to the clients to be activated.
|
||||
</description>
|
||||
|
||||
<arg name="id" type="new_id" interface="xdg_activation_token_v1"/>
|
||||
</request>
|
||||
|
||||
<request name="activate">
|
||||
<description summary="notify new interaction being available">
|
||||
Requests surface activation. It's up to the compositor to display
|
||||
this information as desired, for example by placing the surface above
|
||||
the rest.
|
||||
|
||||
The compositor may know who requested this by checking the activation
|
||||
token and might decide not to follow through with the activation if it's
|
||||
considered unwanted.
|
||||
|
||||
Compositors can ignore unknown activation tokens when an invalid
|
||||
token is passed.
|
||||
</description>
|
||||
<arg name="token" type="string" summary="the activation token of the initiating client"/>
|
||||
<arg name="surface" type="object" interface="wl_surface"
|
||||
summary="the wl_surface to activate"/>
|
||||
</request>
|
||||
</interface>
|
||||
|
||||
<interface name="xdg_activation_token_v1" version="1">
|
||||
<description summary="an exported activation handle">
|
||||
An object for setting up a token and receiving a token handle that can
|
||||
be passed as an activation token to another client.
|
||||
|
||||
The object is created using the xdg_activation_v1.get_activation_token
|
||||
request. This object should then be populated with the app_id, surface
|
||||
and serial information and committed. The compositor shall then issue a
|
||||
done event with the token. In case the request's parameters are invalid,
|
||||
the compositor will provide an invalid token.
|
||||
</description>
|
||||
|
||||
<enum name="error">
|
||||
<entry name="already_used" value="0"
|
||||
summary="The token has already been used previously"/>
|
||||
</enum>
|
||||
|
||||
<request name="set_serial">
|
||||
<description summary="specifies the seat and serial of the activating event">
|
||||
Provides information about the seat and serial event that requested the
|
||||
token.
|
||||
|
||||
The serial can come from an input or focus event. For instance, if a
|
||||
click triggers the launch of a third-party client, the launcher client
|
||||
should send a set_serial request with the serial and seat from the
|
||||
wl_pointer.button event.
|
||||
|
||||
Some compositors might refuse to activate toplevels when the token
|
||||
doesn't have a valid and recent enough event serial.
|
||||
|
||||
Must be sent before commit. This information is optional.
|
||||
</description>
|
||||
<arg name="serial" type="uint"
|
||||
summary="the serial of the event that triggered the activation"/>
|
||||
<arg name="seat" type="object" interface="wl_seat"
|
||||
summary="the wl_seat of the event"/>
|
||||
</request>
|
||||
|
||||
<request name="set_app_id">
|
||||
<description summary="specifies the application being activated">
|
||||
The requesting client can specify an app_id to associate the token
|
||||
being created with it.
|
||||
|
||||
Must be sent before commit. This information is optional.
|
||||
</description>
|
||||
<arg name="app_id" type="string"
|
||||
summary="the application id of the client being activated."/>
|
||||
</request>
|
||||
|
||||
<request name="set_surface">
|
||||
<description summary="specifies the surface requesting activation">
|
||||
This request sets the surface requesting the activation. Note, this is
|
||||
different from the surface that will be activated.
|
||||
|
||||
Some compositors might refuse to activate toplevels when the token
|
||||
doesn't have a requesting surface.
|
||||
|
||||
Must be sent before commit. This information is optional.
|
||||
</description>
|
||||
<arg name="surface" type="object" interface="wl_surface"
|
||||
summary="the requesting surface"/>
|
||||
</request>
|
||||
|
||||
<request name="commit">
|
||||
<description summary="issues the token request">
|
||||
Requests an activation token based on the different parameters that
|
||||
have been offered through set_serial, set_surface and set_app_id.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<event name="done">
|
||||
<description summary="the exported activation token">
|
||||
The 'done' event contains the unique token of this activation request
|
||||
and notifies that the provider is done.
|
||||
</description>
|
||||
<arg name="token" type="string" summary="the exported activation token"/>
|
||||
</event>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the xdg_activation_token_v1 object">
|
||||
Notify the compositor that the xdg_activation_token_v1 object will no
|
||||
longer be used. The received token stays valid.
|
||||
</description>
|
||||
</request>
|
||||
</interface>
|
||||
</protocol>
|
@ -1,156 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<protocol name="xdg_decoration_unstable_v1">
|
||||
<copyright>
|
||||
Copyright © 2018 Simon Ser
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a
|
||||
copy of this software and associated documentation files (the "Software"),
|
||||
to deal in the Software without restriction, including without limitation
|
||||
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
and/or sell copies of the Software, and to permit persons to whom the
|
||||
Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice (including the next
|
||||
paragraph) shall be included in all copies or substantial portions of the
|
||||
Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
DEALINGS IN THE SOFTWARE.
|
||||
</copyright>
|
||||
|
||||
<interface name="zxdg_decoration_manager_v1" version="1">
|
||||
<description summary="window decoration manager">
|
||||
This interface allows a compositor to announce support for server-side
|
||||
decorations.
|
||||
|
||||
A window decoration is a set of window controls as deemed appropriate by
|
||||
the party managing them, such as user interface components used to move,
|
||||
resize and change a window's state.
|
||||
|
||||
A client can use this protocol to request being decorated by a supporting
|
||||
compositor.
|
||||
|
||||
If compositor and client do not negotiate the use of a server-side
|
||||
decoration using this protocol, clients continue to self-decorate as they
|
||||
see fit.
|
||||
|
||||
Warning! The protocol described in this file is experimental and
|
||||
backward incompatible changes may be made. Backward compatible changes
|
||||
may be added together with the corresponding interface version bump.
|
||||
Backward incompatible changes are done by bumping the version number in
|
||||
the protocol and interface names and resetting the interface version.
|
||||
Once the protocol is to be declared stable, the 'z' prefix and the
|
||||
version number in the protocol and interface names are removed and the
|
||||
interface version number is reset.
|
||||
</description>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the decoration manager object">
|
||||
Destroy the decoration manager. This doesn't destroy objects created
|
||||
with the manager.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<request name="get_toplevel_decoration">
|
||||
<description summary="create a new toplevel decoration object">
|
||||
Create a new decoration object associated with the given toplevel.
|
||||
|
||||
Creating an xdg_toplevel_decoration from an xdg_toplevel which has a
|
||||
buffer attached or committed is a client error, and any attempts by a
|
||||
client to attach or manipulate a buffer prior to the first
|
||||
xdg_toplevel_decoration.configure event must also be treated as
|
||||
errors.
|
||||
</description>
|
||||
<arg name="id" type="new_id" interface="zxdg_toplevel_decoration_v1"/>
|
||||
<arg name="toplevel" type="object" interface="xdg_toplevel"/>
|
||||
</request>
|
||||
</interface>
|
||||
|
||||
<interface name="zxdg_toplevel_decoration_v1" version="1">
|
||||
<description summary="decoration object for a toplevel surface">
|
||||
The decoration object allows the compositor to toggle server-side window
|
||||
decorations for a toplevel surface. The client can request to switch to
|
||||
another mode.
|
||||
|
||||
The xdg_toplevel_decoration object must be destroyed before its
|
||||
xdg_toplevel.
|
||||
</description>
|
||||
|
||||
<enum name="error">
|
||||
<entry name="unconfigured_buffer" value="0"
|
||||
summary="xdg_toplevel has a buffer attached before configure"/>
|
||||
<entry name="already_constructed" value="1"
|
||||
summary="xdg_toplevel already has a decoration object"/>
|
||||
<entry name="orphaned" value="2"
|
||||
summary="xdg_toplevel destroyed before the decoration object"/>
|
||||
</enum>
|
||||
|
||||
<request name="destroy" type="destructor">
|
||||
<description summary="destroy the decoration object">
|
||||
Switch back to a mode without any server-side decorations at the next
|
||||
commit.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<enum name="mode">
|
||||
<description summary="window decoration modes">
|
||||
These values describe window decoration modes.
|
||||
</description>
|
||||
<entry name="client_side" value="1"
|
||||
summary="no server-side window decoration"/>
|
||||
<entry name="server_side" value="2"
|
||||
summary="server-side window decoration"/>
|
||||
</enum>
|
||||
|
||||
<request name="set_mode">
|
||||
<description summary="set the decoration mode">
|
||||
Set the toplevel surface decoration mode. This informs the compositor
|
||||
that the client prefers the provided decoration mode.
|
||||
|
||||
After requesting a decoration mode, the compositor will respond by
|
||||
emitting an xdg_surface.configure event. The client should then update
|
||||
its content, drawing it without decorations if the received mode is
|
||||
server-side decorations. The client must also acknowledge the configure
|
||||
when committing the new content (see xdg_surface.ack_configure).
|
||||
|
||||
The compositor can decide not to use the client's mode and enforce a
|
||||
different mode instead.
|
||||
|
||||
Clients whose decoration mode depend on the xdg_toplevel state may send
|
||||
a set_mode request in response to an xdg_surface.configure event and wait
|
||||
for the next xdg_surface.configure event to prevent unwanted state.
|
||||
Such clients are responsible for preventing configure loops and must
|
||||
make sure not to send multiple successive set_mode requests with the
|
||||
same decoration mode.
|
||||
</description>
|
||||
<arg name="mode" type="uint" enum="mode" summary="the decoration mode"/>
|
||||
</request>
|
||||
|
||||
<request name="unset_mode">
|
||||
<description summary="unset the decoration mode">
|
||||
Unset the toplevel surface decoration mode. This informs the compositor
|
||||
that the client doesn't prefer a particular decoration mode.
|
||||
|
||||
This request has the same semantics as set_mode.
|
||||
</description>
|
||||
</request>
|
||||
|
||||
<event name="configure">
|
||||
<description summary="suggest a surface change">
|
||||
The configure event asks the client to change its decoration mode. The
|
||||
configured state should not be applied immediately. Clients must send an
|
||||
ack_configure in response to this event. See xdg_surface.configure and
|
||||
xdg_surface.ack_configure for details.
|
||||
|
||||
A configure event can be sent at any time. The specified mode must be
|
||||
obeyed by the client.
|
||||
</description>
|
||||
<arg name="mode" type="uint" enum="mode" summary="the decoration mode"/>
|
||||
</event>
|
||||
</interface>
|
||||
</protocol>
|
1370
dependencies/glfw-3.4/deps/wayland/xdg-shell.xml
vendored
1370
dependencies/glfw-3.4/deps/wayland/xdg-shell.xml
vendored
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user