diff --git a/.github/workflows/build-all.yaml b/.github/workflows/build-all.yaml index a53a28fa..9ab3e69e 100644 --- a/.github/workflows/build-all.yaml +++ b/.github/workflows/build-all.yaml @@ -1,11 +1,25 @@ name: build-all on: + workflow_dispatch: + inputs: + version: + type: string + required: false + ignore-cache: + required: false + type: string + default: false workflow_call: inputs: version: type: string required: false + ignore-cache: + required: false + type: string + default: false + push: branches: - main @@ -22,18 +36,40 @@ jobs: build-angle-mac: uses: ./.github/workflows/build-angle-mac.yaml + with: + ignore-cache: build-angle-win: uses: ./.github/workflows/build-angle-win.yaml + determine-version: + runs-on: ubuntu-latest + outputs: + version: ${{ steps.determine-version.outputs.version }} + steps: + - name: Determine version + id: determine-version + run: | + if [ "${{ inputs.version }}" != '' ] ; then + echo Selecting version: ${{ inputs.version }} + echo "version='${{ inputs.version }}'" >> "$GITHUB_OUTPUT" + else + echo Selecting version: ${{ github.ref_name }} + echo "version='${{ github.ref_name }}'" >> "$GITHUB_OUTPUT" + fi + build-win: - needs: [build-dawn-win, build-angle-win] + needs: [build-dawn-win, build-angle-win, determine-version] runs-on: windows-2022 steps: - uses: actions/setup-python@v4 with: python-version: '3.10' + - uses: mlugg/setup-zig@v1 + with: + version: 0.14.0 + - uses: actions/checkout@v4 - uses: actions/download-artifact@v4 @@ -50,22 +86,21 @@ jobs: run: | xcopy artifact\angle-windows-x64 build\angle.out\ /s /y - - name: Build + - name: Build and run tests shell: cmd run: | - call "C:\Program Files\Microsoft Visual Studio\2022\Enterprise\VC\Auxiliary\Build\vcvars64.bat" - orcadev.bat build --release --version ${{github.ref_name}} || exit 1 + zig build test - - name: Package + - name: Build sketches shell: cmd run: | - if [${{ inputs.version }}]==[] ( - orcadev.bat install orca || exit 1 - tar --format=ustar -cvzf orca-windows.tar.gz orca - ) else ( - orcadev.bat install --version ${{ inputs.version }} orca || exit 1 + zig build sketches + + - name: Build and Package + shell: cmd + run: | + zig build -Doptimize=ReleaseFast -Dversion=${{ needs.determine-version.outputs.version }} -Dsdk-path=orca --summary all tar --format=ustar -cvzf orca-windows.tar.gz orca - ) - uses: actions/upload-artifact@v4 with: @@ -73,13 +108,17 @@ jobs: path: orca-windows.tar.gz build-macos-x64: - needs: [build-dawn-mac, build-angle-mac] + needs: [build-dawn-mac, build-angle-mac, determine-version] runs-on: macos-13 steps: - uses: actions/setup-python@v4 with: python-version: '3.10' + - uses: mlugg/setup-zig@v1 + with: + version: 0.14.0 + - uses: actions/checkout@v4 - uses: actions/download-artifact@v4 @@ -96,17 +135,17 @@ jobs: mkdir -p build cp -r artifact/angle-mac-x64 build/angle.out - - name: Build + - name: Build and run tests run: | - ./orcadev build --release --version ${{github.ref_name}} + zig build test - - name: Package + - name: Build sketches run: | - if [ "${{ inputs.version }}" != '' ] ; then - ./orcadev install --version ${{ inputs.version }} orca - else - ./orcadev install orca - fi + zig build sketches + + - name: Build and Package + run: | + zig build -Doptimize=ReleaseFast -Dversion=${{ needs.determine-version.outputs.version }} -Dsdk-path=orca --summary all tar --format=ustar -czf orca-mac-x64.tar.gz orca - uses: actions/upload-artifact@v4 @@ -115,13 +154,17 @@ jobs: path: orca-mac-x64.tar.gz build-macos-arm64: - needs: [build-dawn-mac, build-angle-mac] + needs: [build-dawn-mac, build-angle-mac, determine-version] runs-on: macos-14 steps: - uses: actions/setup-python@v4 with: python-version: '3.10' + - uses: mlugg/setup-zig@v1 + with: + version: 0.14.0 + - uses: actions/checkout@v4 - uses: actions/download-artifact@v4 @@ -138,17 +181,17 @@ jobs: mkdir -p build cp -r artifact/angle-mac-arm64 build/angle.out - - name: Build + - name: Build and run tests run: | - ./orcadev build --release --version ${{github.ref_name}} + zig build test - - name: Package + - name: Build sketches run: | - if [ "${{ inputs.version }}" != '' ] ; then - ./orcadev install --version ${{ inputs.version }} orca - else - ./orcadev install orca - fi + zig build sketches + + - name: Build and Package + run: | + zig build -Doptimize=ReleaseFast -Dversion=${{ needs.determine-version.outputs.version }} -Dsdk-path=orca --summary all tar --format=ustar -czf orca-mac-arm64.tar.gz orca - uses: actions/upload-artifact@v4 diff --git a/.github/workflows/build-angle-mac.yaml b/.github/workflows/build-angle-mac.yaml index 54377dd1..140dcc8e 100644 --- a/.github/workflows/build-angle-mac.yaml +++ b/.github/workflows/build-angle-mac.yaml @@ -2,7 +2,17 @@ name: build-angle-mac on: workflow_dispatch: + inputs: + ignore-cache: + required: false + type: string + default: false workflow_call: + inputs: + ignore-cache: + required: false + type: string + default: false jobs: build-mac: @@ -17,24 +27,29 @@ jobs: with: python-version: '3.10' + - uses: mlugg/setup-zig@v1 + with: + version: 0.14.0 + - name: Angle version id: angle-version run: | - echo "ANGLE_COMMIT=$(cat deps/angle-commit.txt)" >> $GITHUB_OUTPUT + echo "ANGLE_COMMIT=$(zig build print-sha-angle)" >> $GITHUB_OUTPUT ARCH=${{ runner.arch }} echo "LOWERCASE_ARCH=$(echo $ARCH | tr '[:upper:]' '[:lower:]')" >> $GITHUB_OUTPUT - name: Cache id: cache-angle uses: actions/cache@v3 + if: inputs.ignore-cache != 'true' with: path: build/angle.out key: angle-${{ runner.os }}-${{ runner.arch }}-${{ steps.angle-version.outputs.ANGLE_COMMIT }} - name: Build Angle - if: steps.cache-angle.outputs.cache-hit != 'true' + if: steps.cache-angle.outputs.cache-hit != 'true' || inputs.ignore-cache == 'true' run: | - ./orcadev build-angle --release + zig build angle -Doptimize=ReleaseFast - uses: actions/upload-artifact@v4 with: diff --git a/.github/workflows/build-angle-win.yaml b/.github/workflows/build-angle-win.yaml index da4f69b9..784c9d0e 100644 --- a/.github/workflows/build-angle-win.yaml +++ b/.github/workflows/build-angle-win.yaml @@ -2,7 +2,17 @@ name: build-angle-win on: workflow_dispatch: + inputs: + ignore-cache: + required: false + type: string + default: false workflow_call: + inputs: + ignore-cache: + required: false + type: string + default: false jobs: build-win: @@ -14,26 +24,32 @@ jobs: with: python-version: '3.10' + - uses: mlugg/setup-zig@v1 + with: + version: 0.14.0 + - name: Angle version id: angle-version shell: cmd run: | @chcp 65001>nul - set /p ANGLE_COMMIT= angle-commit.txt + set /p ANGLE_COMMIT=> %GITHUB_OUTPUT% - name: Cache id: cache-angle uses: actions/cache@v3 + if: inputs.ignore-cache != 'true' with: path: build/angle.out key: angle-${{ runner.os }}-${{ runner.arch }}-${{ steps.angle-version.outputs.ANGLE_COMMIT }} - name: Build Angle - if: steps.cache-angle.outputs.cache-hit != 'true' + if: steps.cache-angle.outputs.cache-hit != 'true' || inputs.ignore-cache == 'true' shell: cmd run: | - orcadev build-angle --release + zig build angle -Doptimize=ReleaseFast - uses: actions/upload-artifact@v4 with: diff --git a/.github/workflows/build-dawn-mac.yaml b/.github/workflows/build-dawn-mac.yaml index 644ade00..a3316262 100644 --- a/.github/workflows/build-dawn-mac.yaml +++ b/.github/workflows/build-dawn-mac.yaml @@ -2,7 +2,17 @@ name: build-dawn-mac on: workflow_dispatch: + inputs: + ignore-cache: + required: false + type: string + default: false workflow_call: + inputs: + ignore-cache: + required: false + type: string + default: false jobs: build-mac: @@ -16,25 +26,30 @@ jobs: - uses: actions/setup-python@v4 with: python-version: '3.10' + + - uses: mlugg/setup-zig@v1 + with: + version: 0.14.0 - name: Dawn version id: dawn-version run: | - echo "DAWN_COMMIT=$(cat deps/dawn-commit.txt)" >> $GITHUB_OUTPUT + echo "DAWN_COMMIT=$(zig build print-sha-dawn)" >> $GITHUB_OUTPUT ARCH=${{ runner.arch }} echo "LOWERCASE_ARCH=$(echo $ARCH | tr '[:upper:]' '[:lower:]')" >> $GITHUB_OUTPUT - name: Cache id: cache-dawn uses: actions/cache@v3 + if: inputs.ignore-cache != 'true' with: path: build/dawn.out key: dawn-${{ runner.os }}-${{ runner.arch }}-${{ steps.dawn-version.outputs.DAWN_COMMIT }} - name: Build Dawn - if: steps.cache-dawn.outputs.cache-hit != 'true' + if: steps.cache-dawn.outputs.cache-hit != 'true' || inputs.ignore-cache == 'true' run: | - ./orcadev build-dawn --release --parallel 4 + zig build dawn -Doptimize=ReleaseFast - uses: actions/upload-artifact@v4 with: diff --git a/.github/workflows/build-dawn-win.yaml b/.github/workflows/build-dawn-win.yaml index 7208e90a..fad5d658 100644 --- a/.github/workflows/build-dawn-win.yaml +++ b/.github/workflows/build-dawn-win.yaml @@ -2,7 +2,17 @@ name: build-dawn-win on: workflow_dispatch: + inputs: + ignore-cache: + required: false + type: string + default: false workflow_call: + inputs: + ignore-cache: + required: false + type: string + default: false jobs: build-win: @@ -19,26 +29,32 @@ jobs: with: python-version: '3.10' + - uses: mlugg/setup-zig@v1 + with: + version: 0.14.0 + - name: Dawn version id: dawn-version shell: cmd run: | @chcp 65001>nul - set /p DAWN_COMMIT= dawn-commit.txt + set /p DAWN_COMMIT=> %GITHUB_OUTPUT% - name: Cache id: cache-dawn uses: actions/cache@v3 + if: inputs.ignore-cache != 'true' with: path: build/dawn.out key: dawn-${{ runner.os }}-${{ runner.arch }}-${{ steps.dawn-version.outputs.DAWN_COMMIT }} - name: Build Dawn - if: steps.cache-dawn.outputs.cache-hit != 'true' + if: steps.cache-dawn.outputs.cache-hit != 'true' || inputs.ignore-cache == 'true' shell: cmd run: | - orcadev build-dawn --release --parallel 4 + zig build dawn -Doptimize=ReleaseFast - uses: actions/upload-artifact@v4 with: diff --git a/.gitignore b/.gitignore index 6e90a335..d53b9d3d 100644 --- a/.gitignore +++ b/.gitignore @@ -35,6 +35,8 @@ src/orca_surface.c src/graphics/orca_gl31.h *bind_gen.c *_stubs.c +.zig-cache +zig-out src/ext/bytebox/zig-cache src/ext/bytebox/zig-out @@ -60,4 +62,7 @@ src/ext/curl/winbuild/*.idb src/ext/zlib/build/ doc/mkdocs/docs/api -doc/mkdocs/mkdocs.yml \ No newline at end of file +doc/mkdocs/mkdocs.yml + +# build folder exclusion is a little too aggressive +!src/build diff --git a/build.zig b/build.zig new file mode 100644 index 00000000..b650b8f0 --- /dev/null +++ b/build.zig @@ -0,0 +1,1473 @@ +const std = @import("std"); + +const Build = std.Build; +const LazyPath = Build.LazyPath; +const Step = Build.Step; +const Module = Build.Module; +const ModuleImport = Module.Import; +const CrossTarget = std.zig.CrossTarget; +const CompileStep = std.Build.Step.Compile; + +const MACOS_VERSION_MIN = "13.0.0"; + +const CSources = struct { + files: std.ArrayList([]const u8), + b: *Build, + + fn init(b: *Build) CSources { + return .{ + .files = std.ArrayList([]const u8).init(b.allocator), + .b = b, + }; + } + + fn collect(sources: *CSources, path: []const u8) !void { + const cwd: std.fs.Dir = sources.b.build_root.handle; + const dir: std.fs.Dir = try cwd.openDir(path, .{ .iterate = true }); + var iter: std.fs.Dir.Iterator = dir.iterate(); + while (try iter.next()) |entry| { + if (entry.kind == .file and std.mem.endsWith(u8, entry.name, ".c")) { + const filepath = try std.fs.path.resolve(sources.b.allocator, &.{ path, entry.name }); + try sources.files.append(filepath); + } + } + } + + fn deinit(sources: CSources) void { + for (sources.files.items) |path| { + sources.b.allocator.free(path); + } + sources.files.deinit(); + } +}; + +fn makeDir(path: []const u8) !void { + const cwd = std.fs.cwd(); + cwd.makeDir(path) catch |e| { + if (e != error.PathAlreadyExists) { + return e; + } + }; +} + +fn pathExists(dir: std.fs.Dir, path: []const u8) bool { + dir.access(path, .{}) catch { + return false; + }; + return true; +} + +const LibShas = struct { + angle: []const u8, + dawn: []const u8, + + fn find(cwd: std.fs.Dir, allocator: std.mem.Allocator) !LibShas { + const Helpers = struct { + fn findFieldIndex(ast: *const std.zig.Ast, current_node: u32, search: []const u8) ?u32 { + var buf: [2]std.zig.Ast.Node.Index = undefined; + if (ast.fullStructInit(&buf, current_node)) |struct_ast| { + for (struct_ast.ast.fields) |i| { + const field_name = ast.tokenSlice(ast.firstToken(i) - 2); + if (std.mem.eql(u8, field_name, search)) { + return i; + } + } + } + return null; + } + + // "https://chromium.googlesource.com/angle/angle.git/+archive/8a8c8fc280d74b34731e0e417b19bff7c967388a.tar.gz" + fn extractCommitFromUrl(url: []const u8) []const u8 { + const basename = std.fs.path.basename; + const stem = std.fs.path.stem; + return stem(stem(basename(url))); + } + }; + + const contents: []const u8 = try cwd.readFileAlloc(allocator, "build.zig.zon", 1024 * 1024 * 1); + const contents_z = try allocator.alloc(u8, contents.len + 1); + @memcpy(contents_z[0..contents.len], contents); + contents_z[contents.len] = 0; + + // NOTE: there will eventually be a ZON parser in the stdlib, but until then we'll just get by + // with a hacky version that only looks for the libs we care about + var ast = try std.zig.Ast.parse(allocator, contents_z[0..contents.len :0], .zon); + + var angle_sha: []const u8 = ""; + var dawn_sha: []const u8 = ""; + + const root_index = ast.nodes.items(.data)[0].lhs; + if (Helpers.findFieldIndex(&ast, root_index, "dependencies")) |deps_index| { + if (Helpers.findFieldIndex(&ast, deps_index, "angle")) |angle_index| { + if (Helpers.findFieldIndex(&ast, angle_index, "url")) |url_index| { + const url = ast.tokenSlice(ast.nodes.items(.main_token)[url_index]); + angle_sha = Helpers.extractCommitFromUrl(url); + } + } + if (Helpers.findFieldIndex(&ast, deps_index, "dawn")) |angle_index| { + if (Helpers.findFieldIndex(&ast, angle_index, "url")) |url_index| { + const url = ast.tokenSlice(ast.nodes.items(.main_token)[url_index]); + dawn_sha = Helpers.extractCommitFromUrl(url); + } + } + } + + return LibShas{ + .angle = angle_sha, + .dawn = dawn_sha, + }; + } +}; + +const RunHelpers = struct { + fn addPythonArg(run: *Build.Step.Run, target: Build.ResolvedTarget, b: *Build) void { + if (target.result.os.tag == .windows) { + if (b.lazyDependency("python3-win64", .{})) |python3_dep| { + const python_path = python3_dep.path("python.exe"); + run.addPrefixedFileArg("--python=", python_path); + } + } else { + // Unfortunately there don't seem to be any public archives of python binaries for python3 so + // we have to depend on a system install available in the PATH :( + run.addArg("--python=python3"); + } + } + + fn addCmakeArg(run: *Build.Step.Run, target: Build.ResolvedTarget, b: *Build) void { + if (target.result.os.tag == .windows) { + if (b.lazyDependency("cmake-win64", .{})) |dep| { + const subpath = std.fs.path.join(b.allocator, &.{ "bin", "cmake.exe" }) catch @panic("OOM"); + const cmake_path = dep.path(subpath); + run.addPrefixedFileArg("--cmake=", cmake_path); + } + } else if (target.result.os.tag.isDarwin()) { + if (b.lazyDependency("cmake-macos", .{})) |dep| { + const subpath = std.fs.path.join(b.allocator, &.{ "CMake.app", "Contents", "bin", "cmake" }) catch @panic("OOM"); + const cmake_path = dep.path(subpath); + run.addPrefixedFileArg("--cmake=", cmake_path); + } + } + } +}; + +const GenerateWasmBindingsParams = struct { + exe: *Build.Step.Compile, + api: []const u8, + spec_path: []const u8, + host_bindings_path: []const u8, + guest_bindings_path: ?[]const u8 = null, + guest_include_path: ?[]const u8 = null, + deps: []const *Build.Step = &.{}, +}; + +fn generateWasmBindings(b: *Build, params: GenerateWasmBindingsParams) *Build.Step.UpdateSourceFiles { + var copy_outputs_to_src: *Build.Step.UpdateSourceFiles = b.addUpdateSourceFiles(); + + var run = b.addRunArtifact(params.exe); + run.addArg(std.mem.join(b.allocator, "", &.{ "--api-name=", params.api }) catch @panic("OOM")); + run.addPrefixedFileArg("--spec-path=", b.path(params.spec_path)); + const host_bindings_path = run.addPrefixedOutputFileArg("--bindings-path=", params.host_bindings_path); + copy_outputs_to_src.addCopyFileToSource(host_bindings_path, params.host_bindings_path); + if (params.guest_bindings_path) |path| { + const guest_bindings_path = run.addPrefixedOutputFileArg("--guest-stubs-path=", path); + copy_outputs_to_src.addCopyFileToSource(guest_bindings_path, path); + } + if (params.guest_include_path) |path| { + run.addArg(std.mem.join(b.allocator, "", &.{ "--guest-include-path=", path }) catch @panic("OOM")); + } + + for (params.deps) |dep| { + run.step.dependOn(dep); + } + + copy_outputs_to_src.step.dependOn(&run.step); + + return copy_outputs_to_src; +} + +pub fn build(b: *Build) !void { + const git_version_opt: ?[]const u8 = b.option([]const u8, "version", "Specify the specific git version you want to package") orelse null; + + const cwd = b.build_root.handle; + + // set artifact output directory - a bit of a hack since we're directly overriding + // the prefix, including whatever the user may have specified with -p, but in Orca's + // case we always want everything going into build/ + { + const default_install_dirs = Build.DirList{ + .exe_dir = "bin", + // orca just simplifies everything by having both exes and libs go to the bin directory + .lib_dir = "bin", + // by default headers go into /headers, but we want to be able to parent the libc + // headers to orca-libc/, so by default the header lib is the root of the prefix + .include_dir = "", + }; + try makeDir("build"); + const output_dir = try cwd.realpathAlloc(b.allocator, "build"); + b.resolveInstallPrefix(output_dir, default_install_dirs); + } + + const target = b.standardTargetOptions(.{}); + const optimize = b.standardOptimizeOption(.{}); + + const compile_flag_min_macos_version: []const u8 = b.fmt("-mmacos-version-min={s}", .{MACOS_VERSION_MIN}); + + const shas: LibShas = try LibShas.find(cwd, b.allocator); + + try makeDir("build"); + const deps_intermediate_path = try cwd.realpathAlloc(b.allocator, "build"); + + ///////////////////////////////////////////////////////// + // build_dependencies helper program + + const build_deps_exe: *Build.Step.Compile = b.addExecutable(.{ + .name = "build_dependencies", + .root_source_file = b.path("src/build/build_dependencies.zig"), + .target = target, + .optimize = .Debug, + }); + + ///////////////////////////////////////////////////////// + // angle build + check + + var run_angle_build: *Build.Step.Run = b.addRunArtifact(build_deps_exe); + run_angle_build.addArg("--lib=angle"); + run_angle_build.addArg(b.fmt("--sha={s}", .{shas.angle})); + run_angle_build.addArg(b.fmt("--intermediate={s}", .{deps_intermediate_path})); + RunHelpers.addPythonArg(run_angle_build, target, b); + RunHelpers.addCmakeArg(run_angle_build, target, b); + + const build_angle_step = b.step("angle", "Build Angle libs"); + build_angle_step.dependOn(&run_angle_build.step); + + var run_angle_uptodate: *Build.Step.Run = b.addRunArtifact(build_deps_exe); + run_angle_uptodate.addArg("--check"); + run_angle_uptodate.addArg("--lib=angle"); + run_angle_uptodate.addArg(b.fmt("--sha={s}", .{shas.angle})); + run_angle_uptodate.addArg(b.fmt("--intermediate={s}", .{deps_intermediate_path})); + RunHelpers.addPythonArg(run_angle_uptodate, target, b); + RunHelpers.addCmakeArg(run_angle_uptodate, target, b); + + ///////////////////////////////////////////////////////// + // dawn build + check + + var run_dawn_build: *Build.Step.Run = b.addRunArtifact(build_deps_exe); + run_dawn_build.addArg("--lib=dawn"); + run_dawn_build.addArg(b.fmt("--sha={s}", .{shas.dawn})); + run_dawn_build.addArg(b.fmt("--intermediate={s}", .{deps_intermediate_path})); + RunHelpers.addPythonArg(run_dawn_build, target, b); + RunHelpers.addCmakeArg(run_dawn_build, target, b); + + const build_dawn_step = b.step("dawn", "Build Dawn libs"); + build_dawn_step.dependOn(&run_dawn_build.step); + + var run_dawn_uptodate: *Build.Step.Run = b.addRunArtifact(build_deps_exe); + run_dawn_uptodate.addArg("--check"); + run_dawn_uptodate.addArg("--lib=dawn"); + run_dawn_uptodate.addArg(b.fmt("--sha={s}", .{shas.dawn})); + run_dawn_uptodate.addArg(b.fmt("--intermediate={s}", .{deps_intermediate_path})); + RunHelpers.addPythonArg(run_dawn_uptodate, target, b); + RunHelpers.addCmakeArg(run_dawn_uptodate, target, b); + + ///////////////////////////////////////////////////////// + // CI helper to print the desired angle/dawn shas to stdout + + const echo_exe: *Build.Step.Compile = b.addExecutable(.{ + .name = "echo", + .root_source_file = b.path("src/build/echo.zig"), + .target = target, + .optimize = .Debug, + }); + + var print_version_angle: *Build.Step.Run = b.addRunArtifact(echo_exe); + print_version_angle.addArg(shas.angle); + + const print_version_angle_step = b.step("print-sha-angle", "Prints the desired Angle git commit sha to stdout"); + print_version_angle_step.dependOn(&print_version_angle.step); + + var print_version_dawn: *Build.Step.Run = b.addRunArtifact(echo_exe); + print_version_dawn.addArg(shas.dawn); + + const print_version_dawn_step = b.step("print-sha-dawn", "Prints the desired Dawn git commit sha to stdout"); + print_version_dawn_step.dependOn(&print_version_dawn.step); + + ///////////////////////////////////////////////////////// + // binding generator + + const bindgen_exe: *Build.Step.Compile = b.addExecutable(.{ + .name = "bindgen", + .root_source_file = b.path("src/build/bindgen.zig"), + .target = target, + .optimize = .Debug, + }); + + const bindgen_install = b.addInstallArtifact(bindgen_exe, .{}); + + const bindgen_run: *Build.Step.Run = b.addRunArtifact(bindgen_exe); + if (b.args) |args| { + bindgen_run.addArgs(args); + } + bindgen_run.step.dependOn(&bindgen_install.step); + + const bindgen_step = b.step("run-bindgen", "Generate wasm bindings from a json spec file"); + bindgen_step.dependOn(&bindgen_run.step); + + ///////////////////////////////////////////////////////// + // Orca runtime and dependencies + + // copy angle + dawn libs to output directory + var stage_angle_artifacts = b.addUpdateSourceFiles(); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/EGL/egl.h"), "src/ext/angle/include/EGL/egl.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/EGL/eglext.h"), "src/ext/angle/include/EGL/eglext.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/EGL/eglext_angle.h"), "src/ext/angle/include/EGL/eglext_angle.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/EGL/eglplatform.h"), "src/ext/angle/include/EGL/eglplatform.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES/egl.h"), "src/ext/angle/include/GLES/egl.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES/gl.h"), "src/ext/angle/include/GLES/gl.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES/glext.h"), "src/ext/angle/include/GLES/glext.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES/glplatform.h"), "src/ext/angle/include/GLES/glplatform.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES2/gl2.h"), "src/ext/angle/include/GLES2/gl2.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES2/gl2ext.h"), "src/ext/angle/include/GLES2/gl2ext.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES2/gl2ext_angle.h"), "src/ext/angle/include/GLES2/gl2ext_angle.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES2/gl2platform.h"), "src/ext/angle/include/GLES2/gl2platform.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES3/gl3.h"), "src/ext/angle/include/GLES3/gl3.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES3/gl31.h"), "src/ext/angle/include/GLES3/gl31.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES3/gl32.h"), "src/ext/angle/include/GLES3/gl32.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/GLES3/gl3platform.h"), "src/ext/angle/include/GLES3/gl3platform.h"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/include/KHR/khrplatform.h"), "src/ext/angle/include/KHR/khrplatform.h"); + if (target.result.os.tag == .windows) { + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/d3dcompiler_47.dll"), "build/bin/d3dcompiler_47.dll"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dll"), "build/bin/libEGL.dll"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dll.lib"), "build/bin/libEGL.dll.lib"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dll"), "build/bin/libGLESv2.dll"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dll.lib"), "build/bin/libGLESv2.dll.lib"); + } else { + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dylib"), "build/bin/libEGL.dylib"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dylib"), "build/bin/libGLESv2.dylib"); + } + stage_angle_artifacts.step.dependOn(&run_angle_uptodate.step); + + var stage_dawn_artifacts = b.addUpdateSourceFiles(); + stage_dawn_artifacts.addCopyFileToSource(b.path("build/dawn.out/include/webgpu.h"), "src/ext/dawn/include/webgpu.h"); + if (target.result.os.tag == .windows) { + stage_angle_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/webgpu.dll"), "build/bin/webgpu.dll"); + stage_angle_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/webgpu.lib"), "build/bin/webgpu.lib"); + } else { + stage_angle_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/libwebgpu.dylib"), "build/bin/libwebgpu.dylib"); + } + stage_dawn_artifacts.step.dependOn(&run_dawn_uptodate.step); + + // generate GLES API spec from OpenGL XML registry + // TODO port this to C or Zig + // TODO use python package dependency to avoid system dependency + const python_exe_name = if (target.result.os.tag == .windows) "python.exe" else "python3"; + var python_gen_gles_spec_run: *Build.Step.Run = b.addSystemCommand(&.{python_exe_name}); + python_gen_gles_spec_run.addArg("scripts/gles_gen.py"); + python_gen_gles_spec_run.addPrefixedFileArg("--spec=", b.path("src/ext/gl.xml")); + const gles_api_header = python_gen_gles_spec_run.addPrefixedOutputFileArg("--header=", "orca_gl31.h"); + const gles_api_json = python_gen_gles_spec_run.addPrefixedOutputFileArg("--json=", "gles_api.json"); + const gles_api_log = python_gen_gles_spec_run.addPrefixedOutputFileArg("--log=", "gles_gen.log"); + + var stage_gles_api_spec_artifacts = b.addUpdateSourceFiles(); + stage_gles_api_spec_artifacts.step.dependOn(&python_gen_gles_spec_run.step); + stage_gles_api_spec_artifacts.addCopyFileToSource(gles_api_header, "src/graphics/orca_gl31.h"); + stage_gles_api_spec_artifacts.addCopyFileToSource(gles_api_json, "src/wasmbind/gles_api.json"); + stage_gles_api_spec_artifacts.addCopyFileToSource(gles_api_log, "build/gles_gen.log"); + + // generate wasm bindings + + const orca_runtime_bindgen_core: *Build.Step.UpdateSourceFiles = generateWasmBindings(b, .{ + .exe = bindgen_exe, + .api = "core", + .spec_path = "src/wasmbind/core_api.json", + .host_bindings_path = "src/wasmbind/core_api_bind_gen.c", + .guest_bindings_path = "src/wasmbind/core_api_stubs.c", + }); + + const orca_runtime_bindgen_surface: *Build.Step.UpdateSourceFiles = generateWasmBindings(b, .{ + .exe = bindgen_exe, + .api = "surface", + .spec_path = "src/wasmbind/surface_api.json", + .host_bindings_path = "src/wasmbind/surface_api_bind_gen.c", + .guest_bindings_path = "src/graphics/orca_surface_stubs.c", + .guest_include_path = "graphics/graphics.h", + }); + + const orca_runtime_bindgen_clock: *Build.Step.UpdateSourceFiles = generateWasmBindings(b, .{ + .exe = bindgen_exe, + .api = "clock", + .spec_path = "src/wasmbind/clock_api.json", + .host_bindings_path = "src/wasmbind/clock_api_bind_gen.c", + .guest_include_path = "platform/platform_clock.h", + }); + + const orca_runtime_bindgen_io: *Build.Step.UpdateSourceFiles = generateWasmBindings(b, .{ + .exe = bindgen_exe, + .api = "io", + .spec_path = "src/wasmbind/io_api.json", + .host_bindings_path = "src/wasmbind/io_api_bind_gen.c", + .guest_bindings_path = "src/platform/orca_io_stubs.c", + .guest_include_path = "platform/platform_io_dialog.h", + }); + + const orca_runtime_bindgen_gles: *Build.Step.UpdateSourceFiles = generateWasmBindings(b, .{ + .exe = bindgen_exe, + .api = "gles", + .spec_path = "src/wasmbind/gles_api.json", + .host_bindings_path = "src/wasmbind/gles_api_bind_gen.c", + .deps = &.{&stage_gles_api_spec_artifacts.step}, + }); + + // wgpu shaders header + + const gen_header_exe: *Build.Step.Compile = b.addExecutable(.{ + .name = "gen_header", + .root_source_file = b.path("src/build/gen_header_from_files.zig"), + .target = target, + .optimize = .Debug, + }); + + const wgpu_shaders: []const []const u8 = &.{ + "src/graphics/wgsl_shaders/common.wgsl", + "src/graphics/wgsl_shaders/path_setup.wgsl", + "src/graphics/wgsl_shaders/segment_setup.wgsl", + "src/graphics/wgsl_shaders/backprop.wgsl", + "src/graphics/wgsl_shaders/chunk.wgsl", + "src/graphics/wgsl_shaders/merge.wgsl", + "src/graphics/wgsl_shaders/balance_workgroups.wgsl", + "src/graphics/wgsl_shaders/raster.wgsl", + "src/graphics/wgsl_shaders/blit.wgsl", + "src/graphics/wgsl_shaders/final_blit.wgsl", + }; + + var run_gen_wgpu_header: *Build.Step.Run = b.addRunArtifact(gen_header_exe); + for (wgpu_shaders) |path| { + run_gen_wgpu_header.addPrefixedFileArg("--file=", b.path(path)); + } + run_gen_wgpu_header.addArg("--namespace=oc_wgsl_"); + run_gen_wgpu_header.addPrefixedDirectoryArg("--root=", b.path("")); + const wgpu_header_path = run_gen_wgpu_header.addPrefixedOutputFileArg("--output=", "generated_headers/wgpu_renderer_shaders.h"); + + var update_wgpu_header: *Build.Step.UpdateSourceFiles = b.addUpdateSourceFiles(); + update_wgpu_header.addCopyFileToSource(wgpu_header_path, "src/graphics/wgpu_renderer_shaders.h"); + update_wgpu_header.step.dependOn(&run_gen_wgpu_header.step); + + // platform lib + + var orca_platform_compile_flags = std.ArrayList([]const u8).init(b.allocator); + defer orca_platform_compile_flags.deinit(); + try orca_platform_compile_flags.append("-std=c11"); + try orca_platform_compile_flags.append("-DOC_BUILD_DLL"); + try orca_platform_compile_flags.append("-D_USE_MATH_DEFINES"); + if (optimize == .Debug) { + try orca_platform_compile_flags.append("-DOC_DEBUG"); + try orca_platform_compile_flags.append("-DOC_LOG_COMPILE_DEBUG"); + } + if (target.result.os.tag.isDarwin()) { + try orca_platform_compile_flags.append(compile_flag_min_macos_version); + } + // if (target.result.os.tag == .windows) { + // try orca_platform_compile_flags.append("-Wl,--delayload=libEGL.dll"); + // try orca_platform_compile_flags.append("-Wl,--delayload=libGLESv2.dll"); + // try orca_platform_compile_flags.append("-Wl,--delayload=webgpu.dll"); + // } + + var orca_platform_lib = b.addSharedLibrary(.{ + .name = "orca_platform", + .target = target, + .optimize = optimize, + .win32_manifest = b.path("src/app/win32_manifest.manifest"), + }); + + orca_platform_lib.addIncludePath(b.path("src")); + orca_platform_lib.addIncludePath(b.path("src/ext")); + orca_platform_lib.addIncludePath(b.path("src/ext/angle/include")); + orca_platform_lib.addIncludePath(b.path("src/ext/dawn/include")); + + var orca_platform_files = std.ArrayList([]const u8).init(b.allocator); + try orca_platform_files.append("src/orca.c"); + + orca_platform_lib.addLibraryPath(b.path("build/bin")); + + if (target.result.os.tag == .windows) { + orca_platform_lib.linkSystemLibrary("user32"); + orca_platform_lib.linkSystemLibrary("opengl32"); + orca_platform_lib.linkSystemLibrary("gdi32"); + orca_platform_lib.linkSystemLibrary("dxgi"); + orca_platform_lib.linkSystemLibrary("dxguid"); + orca_platform_lib.linkSystemLibrary("d3d11"); + orca_platform_lib.linkSystemLibrary("dcomp"); + orca_platform_lib.linkSystemLibrary("shcore"); + // orca_platform_lib.linkSystemLibrary("delayimp"); + orca_platform_lib.linkSystemLibrary("dwmapi"); + orca_platform_lib.linkSystemLibrary("comctl32"); + orca_platform_lib.linkSystemLibrary("ole32"); + orca_platform_lib.linkSystemLibrary("shell32"); + orca_platform_lib.linkSystemLibrary("shlwapi"); + + orca_platform_lib.linkSystemLibrary("libEGL.dll"); // todo DELAYLOAD? + orca_platform_lib.linkSystemLibrary("libGLESv2.dll"); // todo DELAYLOAD? + orca_platform_lib.linkSystemLibrary("webgpu"); // todo DELAYLOAD? + } else if (target.result.os.tag.isDarwin()) { + orca_platform_lib.linkFramework("Carbon"); + orca_platform_lib.linkFramework("Cocoa"); + orca_platform_lib.linkFramework("Metal"); + orca_platform_lib.linkFramework("QuartzCore"); + orca_platform_lib.linkFramework("UniformTypeIdentifiers"); + + // TODO add @rpath stuff? + // orca_platform_lib.addRPath("@rpath"); + orca_platform_lib.linkSystemLibrary2("EGL", .{ .weak = true }); + orca_platform_lib.linkSystemLibrary2("GLESv2", .{ .weak = true }); + orca_platform_lib.linkSystemLibrary2("webgpu", .{ .weak = true }); + + try orca_platform_files.append("src/orca.m"); + } + + orca_platform_lib.addCSourceFiles(.{ + .files = orca_platform_files.items, + .flags = orca_platform_compile_flags.items, + }); + + orca_platform_lib.step.dependOn(&stage_angle_artifacts.step); + orca_platform_lib.step.dependOn(&stage_dawn_artifacts.step); + orca_platform_lib.step.dependOn(&update_wgpu_header.step); + + orca_platform_lib.step.dependOn(&orca_runtime_bindgen_core.step); + orca_platform_lib.step.dependOn(&orca_runtime_bindgen_surface.step); + orca_platform_lib.step.dependOn(&orca_runtime_bindgen_clock.step); + orca_platform_lib.step.dependOn(&orca_runtime_bindgen_io.step); + orca_platform_lib.step.dependOn(&orca_runtime_bindgen_gles.step); + + const orca_platform_install: *Build.Step.InstallArtifact = b.addInstallArtifact(orca_platform_lib, .{}); + + const build_orca_platform_step = b.step("orca-platform", "Build the Orca platform layer from source."); + build_orca_platform_step.dependOn(&orca_platform_install.step); + + // wasm3 + + var wasm3_sources = CSources.init(b); + defer wasm3_sources.deinit(); + try wasm3_sources.collect("src/ext/wasm3/source"); + + var wasm3_lib = b.addStaticLibrary(.{ + .name = "wasm3", + .target = target, + .optimize = optimize, + }); + + var wasm3_compile_flags = std.ArrayList([]const u8).init(b.allocator); + try wasm3_compile_flags.append("-fno-sanitize=undefined"); + if (target.result.os.tag.isDarwin()) { + try wasm3_compile_flags.append("-foptimize-sibling-calls"); + try wasm3_compile_flags.append("-Wno-extern-initializer"); + try wasm3_compile_flags.append("-Dd_m3VerboseErrorMessages"); + try wasm3_compile_flags.append(compile_flag_min_macos_version); + } + + wasm3_lib.addIncludePath(b.path("src/ext/wasm3/source")); + wasm3_lib.addCSourceFiles(.{ + .files = wasm3_sources.files.items, + .flags = wasm3_compile_flags.items, + }); + wasm3_lib.linkLibC(); + + // orca runtime exe + + var orca_runtime_compile_flags = std.ArrayList([]const u8).init(b.allocator); + defer orca_runtime_compile_flags.deinit(); + try orca_runtime_compile_flags.append("-DOC_WASM_BACKEND_WASM3=1"); + try orca_runtime_compile_flags.append("-DOC_WASM_BACKEND_BYTEBOX=0"); // TODO remove bytebox support + if (optimize == .Debug) { + try orca_runtime_compile_flags.append("-DOC_DEBUG"); + try orca_runtime_compile_flags.append("-DOC_LOG_COMPILE_DEBUG"); + } + + const orca_runtime_exe = b.addExecutable(.{ + .name = "orca_runtime", + .target = target, + .optimize = optimize, + }); + + orca_runtime_exe.addIncludePath(b.path("src")); + orca_runtime_exe.addIncludePath(b.path("src/ext")); + orca_runtime_exe.addIncludePath(b.path("build/angle.out/include")); + orca_runtime_exe.addIncludePath(b.path("src/ext/wasm3/source")); + + orca_runtime_exe.addCSourceFiles(.{ + .files = &.{"src/runtime.c"}, + .flags = orca_runtime_compile_flags.items, + }); + + orca_runtime_exe.linkLibrary(wasm3_lib); + orca_runtime_exe.linkLibrary(orca_platform_lib); + orca_runtime_exe.linkLibC(); + + orca_runtime_exe.step.dependOn(&stage_angle_artifacts.step); + orca_runtime_exe.step.dependOn(&stage_dawn_artifacts.step); + + orca_runtime_exe.step.dependOn(&orca_runtime_bindgen_core.step); + orca_runtime_exe.step.dependOn(&orca_runtime_bindgen_surface.step); + orca_runtime_exe.step.dependOn(&orca_runtime_bindgen_clock.step); + orca_runtime_exe.step.dependOn(&orca_runtime_bindgen_io.step); + orca_runtime_exe.step.dependOn(&orca_runtime_bindgen_gles.step); + + const install_runtime_exe: *Build.Step.InstallArtifact = b.addInstallArtifact(orca_runtime_exe, .{}); + + const build_runtime_step = b.step("orca-runtime", "Build the Orca runtime from source."); + build_runtime_step.dependOn(&install_runtime_exe.step); + + /////////////////////////////////////////////////////// + // orca wasm libc + + var wasm_target_query: std.Target.Query = .{ + .cpu_arch = std.Target.Cpu.Arch.wasm32, + .os_tag = std.Target.Os.Tag.freestanding, + }; + wasm_target_query.cpu_features_add.addFeature(@intFromEnum(std.Target.wasm.Feature.bulk_memory)); + wasm_target_query.cpu_features_add.addFeature(@intFromEnum(std.Target.wasm.Feature.nontrapping_fptoint)); + + const wasm_target: Build.ResolvedTarget = b.resolveTargetQuery(wasm_target_query); + + const wasm_optimize: std.builtin.OptimizeMode = .ReleaseSmall; + + // zig fmt: off + const libc_flags: []const []const u8 = &.{ + // need to provide absolute paths to these since we're overriding the default zig lib dir + b.fmt("-I{s}", .{b.pathFromRoot("src")}), + "-isystem", b.pathFromRoot("src/orca-libc/src/include"), + "-isystem", b.pathFromRoot("src/orca-libc/src/include/private"), + b.fmt("-I{s}", .{b.pathFromRoot("src/orca-libc/src/arch")}), + b.fmt("-I{s}", .{b.pathFromRoot("src/orca-libc/src/internal")}), + + // warnings + "-Wall", + "-Wextra", + "-Werror", + "-Wno-null-pointer-arithmetic", + "-Wno-unused-parameter", + "-Wno-sign-compare", + "-Wno-unused-variable", + "-Wno-unused-function", + "-Wno-ignored-attributes", + "-Wno-missing-braces", + "-Wno-ignored-pragmas", + "-Wno-unused-but-set-variable", + "-Wno-unknown-warning-option", + "-Wno-parentheses", + "-Wno-shift-op-parentheses", + "-Wno-bitwise-op-parentheses", + "-Wno-logical-op-parentheses", + "-Wno-string-plus-int", + "-Wno-dangling-else", + "-Wno-unknown-pragmas", + + // defines + "-D__ORCA__", + "-DBULK_MEMORY_THRESHOLD=32", + + // other flags + "--std=c11", + }; + // zig fmt: on + + // dummy crt1 object for sysroot folder + + var dummy_crt_obj = b.addObject(.{ + .name = "crt1", + .target = wasm_target, + .optimize = wasm_optimize, + .link_libc = false, + }); + dummy_crt_obj.addCSourceFiles(.{ + .files = &.{"src/orca-libc/src/crt/crt1.c"}, + .flags = libc_flags, + }); + + const libc_install_opts = Build.Step.InstallArtifact.Options{ + .dest_dir = .{ .override = .{ .custom = "orca-libc/lib" } }, + }; + + const dummy_crt_install: *Build.Step.InstallArtifact = b.addInstallArtifact(dummy_crt_obj, libc_install_opts); + + // wasm libc + // + // NOTE - libc must be built in a 2-stage pass by compiling all the c files into .objs individually and then linking them + // all together at the end into a static lib. There are a couple reasons for this: + // 1. The build runner invokes zig.exe with commandline args corresponding to its inputs, and the commandline gets too + // long if all the C files are added directly to the static lib. :( + // 2. Generating a unity build file doesn't work because the .c files have been written assuming individual compilation + // and there are lots of constants with conflicting names in different files. For example, see "huge" in csinh.c + // and csinhf.c + // 3. Only one .c file is allowed to correspond to an obj file. We can't combine multiple C files into one obj. + + const wasm_libc_source_paths: []const []const u8 = &.{ + "src/orca-libc/src/complex", + "src/orca-libc/src/crt", + "src/orca-libc/src/ctype", + "src/orca-libc/src/errno", + "src/orca-libc/src/exit", + "src/orca-libc/src/fenv", + "src/orca-libc/src/internal", + "src/orca-libc/src/malloc", + "src/orca-libc/src/math", + "src/orca-libc/src/multibyte", + "src/orca-libc/src/prng", + "src/orca-libc/src/stdio", + "src/orca-libc/src/stdlib", + "src/orca-libc/src/string", + }; + + var wasm_libc_sources = CSources.init(b); + defer wasm_libc_sources.deinit(); + + var wasm_libc_objs = std.ArrayList(*Build.Step.Compile).init(b.allocator); + try wasm_libc_objs.ensureUnusedCapacity(1024); // there are 496 .c files in the libc so this should be enough + + for (wasm_libc_source_paths) |path| { + wasm_libc_sources.files.shrinkRetainingCapacity(0); + try wasm_libc_sources.collect(path); + + const libc_group: []const u8 = std.fs.path.basename(path); + + for (wasm_libc_sources.files.items) |filepath| { + const filename: []const u8 = std.fs.path.basename(filepath); + const obj_name: []const u8 = try std.mem.join(b.allocator, "_", &.{ "libc", libc_group, filename }); + var obj = b.addObject(.{ + .name = obj_name, + .target = wasm_target, + .optimize = wasm_optimize, + .single_threaded = true, + .link_libc = false, + .zig_lib_dir = b.path("src/orca-libc"), // ensures c stdlib headers bundled with zig are ignored + }); + obj.addCSourceFiles(.{ + .files = &.{filepath}, + .flags = libc_flags, + }); + try wasm_libc_objs.append(obj); + } + } + + var wasm_libc_lib = b.addStaticLibrary(.{ + .name = "c", + .target = wasm_target, + .optimize = wasm_optimize, + .link_libc = false, + .single_threaded = true, + }); + for (wasm_libc_objs.items) |obj| { + wasm_libc_lib.addObject(obj); + } + + // wasm_libc_lib.rdynamic = true; + // wasm_libc_lib.entry = .disabled; + + wasm_libc_lib.installHeadersDirectory(b.path("src/orca-libc/include"), "orca-libc/include", .{}); + + const libc_install: *Build.Step.InstallArtifact = b.addInstallArtifact(wasm_libc_lib, libc_install_opts); + + const build_libc_step = b.step("orca-libc", "Build the Orca libC from source."); + build_libc_step.dependOn(&libc_install.step); + build_libc_step.dependOn(&dummy_crt_install.step); + + ///////////////////////////////////////////////////////// + // Orca wasm SDK + + const wasm_sdk_flags: []const []const u8 = &.{ + // "-Isrc", + // "-Isrc/ext", + // "-Isrc/orca-libc/include", + b.fmt("-I{s}", .{b.pathFromRoot("src")}), + b.fmt("-I{s}", .{b.pathFromRoot("src/ext")}), + b.fmt("-I{s}", .{b.pathFromRoot("src/orca-libc/include")}), + "--no-standard-libraries", + "-D__ORCA__", + // "-Wl,--no-entry", + // "-Wl,--export-dynamic", + // "-Wl,--relocatable" + }; + + var wasm_sdk_obj = b.addObject(.{ + .name = "orca_wasm", + .target = wasm_target, + .optimize = wasm_optimize, + .link_libc = false, + .single_threaded = true, + .zig_lib_dir = b.path("src/orca-libc"), + }); + wasm_sdk_obj.addCSourceFile(.{ + .file = b.path("src/orca.c"), + .flags = wasm_sdk_flags, + }); + + wasm_sdk_obj.step.dependOn(&stage_angle_artifacts.step); + wasm_sdk_obj.step.dependOn(&stage_dawn_artifacts.step); + + wasm_sdk_obj.step.dependOn(&orca_runtime_bindgen_core.step); + wasm_sdk_obj.step.dependOn(&orca_runtime_bindgen_surface.step); + wasm_sdk_obj.step.dependOn(&orca_runtime_bindgen_clock.step); + wasm_sdk_obj.step.dependOn(&orca_runtime_bindgen_io.step); + wasm_sdk_obj.step.dependOn(&orca_runtime_bindgen_gles.step); + + var wasm_sdk_lib = b.addStaticLibrary(.{ + .name = "orca_wasm", + .target = wasm_target, + .optimize = wasm_optimize, + .link_libc = false, + .single_threaded = true, + }); + wasm_sdk_lib.addObject(wasm_sdk_obj); + + const wasm_sdk_install: *Build.Step.InstallArtifact = b.addInstallArtifact(wasm_sdk_lib, .{}); + + const build_wasm_sdk_step = b.step("orca-wasm-sdk", "Build the Orca wasm sdk from source."); + build_wasm_sdk_step.dependOn(&wasm_sdk_install.step); + + ///////////////////////////////////////////////////////// + // Orca CLI tool and dependencies + + // zlib + + var z_sources = CSources.init(b); + defer z_sources.deinit(); + try z_sources.collect("src/ext/zlib/"); + + var z_lib = b.addStaticLibrary(.{ + .name = "z", + .target = target, + .optimize = optimize, + }); + z_lib.addIncludePath(b.path("src/ext/zlib/")); + z_lib.addCSourceFiles(.{ + .files = z_sources.files.items, + .flags = &.{ + "-DHAVE_SYS_TYPES_H", + "-DHAVE_STDINT_H", + "-DHAVE_STDDEF_H", + "-DZ_HAVE_UNISTD_H", + }, + }); + z_lib.linkLibC(); + + // curl + + // Original code MIT licensed from: https://github.com/jiacai2050/zig-curl/blob/main/libs/curl.zig + const curl_lib = b.addStaticLibrary(.{ + .name = "curl", + .target = target, + .optimize = optimize, + .link_libc = true, + }); + + var curl_sources = CSources.init(b); + try curl_sources.collect("src/ext/curl/lib"); + try curl_sources.collect("src/ext/curl/lib/vauth"); + try curl_sources.collect("src/ext/curl/lib/vtls"); + try curl_sources.collect("src/ext/curl/lib/vssh"); + try curl_sources.collect("src/ext/curl/lib/vquic"); + + for (curl_sources.files.items) |path| { + curl_lib.addCSourceFile(.{ + .file = b.path(path), + .flags = &.{"-std=gnu89"}, + }); + } + + curl_lib.addIncludePath(b.path("src/ext/curl/lib")); + curl_lib.addIncludePath(b.path("src/ext/curl/include")); + curl_lib.addIncludePath(b.path("src/ext/zlib")); + + curl_lib.root_module.addCMacro("BUILDING_LIBCURL", "1"); + curl_lib.root_module.addCMacro("CURL_STATICLIB", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_LDAP", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_LDAPS", "1"); + // curl_lib.root_module.addCMacro("USE_MBEDTLS", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_DICT", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_FILE", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_FTP", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_GOPHER", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_IMAP", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_MQTT", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_POP3", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_RTSP", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_SMB", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_SMTP", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_TELNET", "1"); + curl_lib.root_module.addCMacro("CURL_DISABLE_TFTP", "1"); + curl_lib.root_module.addCMacro("HAVE_LIBZ", "1"); + curl_lib.root_module.addCMacro("HAVE_ZLIB_H", "1"); + + if (target.result.os.tag == .windows) { + curl_lib.linkSystemLibrary("ws2_32"); + curl_lib.linkSystemLibrary("wldap32"); + curl_lib.linkSystemLibrary("advapi32"); + curl_lib.linkSystemLibrary("crypt32"); + curl_lib.linkSystemLibrary("gdi32"); + curl_lib.linkSystemLibrary("user32"); + curl_lib.linkSystemLibrary("bcrypt"); + } else { + curl_lib.root_module.addCMacro("CURL_EXTERN_SYMBOL", "__attribute__ ((__visibility__ (\"default\"))"); + + if (target.result.os.tag.isDarwin() == false) { + curl_lib.root_module.addCMacro("ENABLE_IPV6", "1"); + curl_lib.root_module.addCMacro("HAVE_GETHOSTBYNAME_R", "1"); + curl_lib.root_module.addCMacro("HAVE_MSG_NOSIGNAL", "1"); + } + + if (target.result.os.tag == .linux) { + curl_lib.root_module.addCMacro("HAVE_LINUX_TCP_H", "1"); + } + + curl_lib.root_module.addCMacro("HAVE_ALARM", "1"); + curl_lib.root_module.addCMacro("HAVE_ALLOCA_H", "1"); + curl_lib.root_module.addCMacro("HAVE_ARPA_INET_H", "1"); + curl_lib.root_module.addCMacro("HAVE_ARPA_TFTP_H", "1"); + curl_lib.root_module.addCMacro("HAVE_ASSERT_H", "1"); + curl_lib.root_module.addCMacro("HAVE_BASENAME", "1"); + curl_lib.root_module.addCMacro("HAVE_BOOL_T", "1"); + curl_lib.root_module.addCMacro("HAVE_BUILTIN_AVAILABLE", "1"); + curl_lib.root_module.addCMacro("HAVE_CLOCK_GETTIME_MONOTONIC", "1"); + curl_lib.root_module.addCMacro("HAVE_DLFCN_H", "1"); + curl_lib.root_module.addCMacro("HAVE_ERRNO_H", "1"); + curl_lib.root_module.addCMacro("HAVE_FCNTL", "1"); + curl_lib.root_module.addCMacro("HAVE_FCNTL_H", "1"); + curl_lib.root_module.addCMacro("HAVE_FCNTL_O_NONBLOCK", "1"); + curl_lib.root_module.addCMacro("HAVE_FREEADDRINFO", "1"); + curl_lib.root_module.addCMacro("HAVE_FTRUNCATE", "1"); + curl_lib.root_module.addCMacro("HAVE_GETADDRINFO", "1"); + curl_lib.root_module.addCMacro("HAVE_GETEUID", "1"); + curl_lib.root_module.addCMacro("HAVE_GETPPID", "1"); + curl_lib.root_module.addCMacro("HAVE_GETHOSTBYNAME", "1"); + curl_lib.root_module.addCMacro("HAVE_GETHOSTBYNAME_R_6", "1"); + curl_lib.root_module.addCMacro("HAVE_GETHOSTNAME", "1"); + curl_lib.root_module.addCMacro("HAVE_GETPPID", "1"); + curl_lib.root_module.addCMacro("HAVE_GETPROTOBYNAME", "1"); + curl_lib.root_module.addCMacro("HAVE_GETPEERNAME", "1"); + curl_lib.root_module.addCMacro("HAVE_GETSOCKNAME", "1"); + curl_lib.root_module.addCMacro("HAVE_IF_NAMETOINDEX", "1"); + curl_lib.root_module.addCMacro("HAVE_GETPWUID", "1"); + curl_lib.root_module.addCMacro("HAVE_GETPWUID_R", "1"); + curl_lib.root_module.addCMacro("HAVE_GETRLIMIT", "1"); + curl_lib.root_module.addCMacro("HAVE_GETTIMEOFDAY", "1"); + curl_lib.root_module.addCMacro("HAVE_GMTIME_R", "1"); + curl_lib.root_module.addCMacro("HAVE_IFADDRS_H", "1"); + curl_lib.root_module.addCMacro("HAVE_INET_ADDR", "1"); + curl_lib.root_module.addCMacro("HAVE_INET_PTON", "1"); + curl_lib.root_module.addCMacro("HAVE_SA_FAMILY_T", "1"); + curl_lib.root_module.addCMacro("HAVE_INTTYPES_H", "1"); + curl_lib.root_module.addCMacro("HAVE_IOCTL", "1"); + curl_lib.root_module.addCMacro("HAVE_IOCTL_FIONBIO", "1"); + curl_lib.root_module.addCMacro("HAVE_IOCTL_SIOCGIFADDR", "1"); + curl_lib.root_module.addCMacro("HAVE_LDAP_URL_PARSE", "1"); + curl_lib.root_module.addCMacro("HAVE_LIBGEN_H", "1"); + curl_lib.root_module.addCMacro("HAVE_IDN2_H", "1"); + curl_lib.root_module.addCMacro("HAVE_LL", "1"); + curl_lib.root_module.addCMacro("HAVE_LOCALE_H", "1"); + curl_lib.root_module.addCMacro("HAVE_LOCALTIME_R", "1"); + curl_lib.root_module.addCMacro("HAVE_LONGLONG", "1"); + curl_lib.root_module.addCMacro("HAVE_MALLOC_H", "1"); + curl_lib.root_module.addCMacro("HAVE_MEMORY_H", "1"); + curl_lib.root_module.addCMacro("HAVE_NETDB_H", "1"); + curl_lib.root_module.addCMacro("HAVE_NETINET_IN_H", "1"); + curl_lib.root_module.addCMacro("HAVE_NETINET_TCP_H", "1"); + curl_lib.root_module.addCMacro("HAVE_NET_IF_H", "1"); + curl_lib.root_module.addCMacro("HAVE_PIPE", "1"); + curl_lib.root_module.addCMacro("HAVE_POLL", "1"); + curl_lib.root_module.addCMacro("HAVE_POLL_FINE", "1"); + curl_lib.root_module.addCMacro("HAVE_POLL_H", "1"); + curl_lib.root_module.addCMacro("HAVE_POSIX_STRERROR_R", "1"); + curl_lib.root_module.addCMacro("HAVE_PTHREAD_H", "1"); + curl_lib.root_module.addCMacro("HAVE_PWD_H", "1"); + curl_lib.root_module.addCMacro("HAVE_RECV", "1"); + curl_lib.root_module.addCMacro("HAVE_SELECT", "1"); + curl_lib.root_module.addCMacro("HAVE_SEND", "1"); + curl_lib.root_module.addCMacro("HAVE_FSETXATTR", "1"); + curl_lib.root_module.addCMacro("HAVE_FSETXATTR_5", "1"); + curl_lib.root_module.addCMacro("HAVE_SETJMP_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SETLOCALE", "1"); + curl_lib.root_module.addCMacro("HAVE_SETRLIMIT", "1"); + curl_lib.root_module.addCMacro("HAVE_SETSOCKOPT", "1"); + curl_lib.root_module.addCMacro("HAVE_SIGACTION", "1"); + curl_lib.root_module.addCMacro("HAVE_SIGINTERRUPT", "1"); + curl_lib.root_module.addCMacro("HAVE_SIGNAL", "1"); + curl_lib.root_module.addCMacro("HAVE_SIGNAL_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SIGSETJMP", "1"); + curl_lib.root_module.addCMacro("HAVE_SOCKADDR_IN6_SIN6_SCOPE_ID", "1"); + curl_lib.root_module.addCMacro("HAVE_SOCKET", "1"); + curl_lib.root_module.addCMacro("HAVE_STDBOOL_H", "1"); + curl_lib.root_module.addCMacro("HAVE_STDINT_H", "1"); + curl_lib.root_module.addCMacro("HAVE_STDIO_H", "1"); + curl_lib.root_module.addCMacro("HAVE_STDLIB_H", "1"); + curl_lib.root_module.addCMacro("HAVE_STRCASECMP", "1"); + curl_lib.root_module.addCMacro("HAVE_STRDUP", "1"); + curl_lib.root_module.addCMacro("HAVE_STRERROR_R", "1"); + curl_lib.root_module.addCMacro("HAVE_STRINGS_H", "1"); + curl_lib.root_module.addCMacro("HAVE_STRING_H", "1"); + curl_lib.root_module.addCMacro("HAVE_STRSTR", "1"); + curl_lib.root_module.addCMacro("HAVE_STRTOK_R", "1"); + curl_lib.root_module.addCMacro("HAVE_STRTOLL", "1"); + curl_lib.root_module.addCMacro("HAVE_STRUCT_SOCKADDR_STORAGE", "1"); + curl_lib.root_module.addCMacro("HAVE_STRUCT_TIMEVAL", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_IOCTL_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_PARAM_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_POLL_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_RESOURCE_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_SELECT_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_SOCKET_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_STAT_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_TIME_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_TYPES_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_UIO_H", "1"); + curl_lib.root_module.addCMacro("HAVE_SYS_UN_H", "1"); + curl_lib.root_module.addCMacro("HAVE_TERMIOS_H", "1"); + curl_lib.root_module.addCMacro("HAVE_TERMIO_H", "1"); + curl_lib.root_module.addCMacro("HAVE_TIME_H", "1"); + curl_lib.root_module.addCMacro("HAVE_UNAME", "1"); + curl_lib.root_module.addCMacro("HAVE_UNISTD_H", "1"); + curl_lib.root_module.addCMacro("HAVE_UTIME", "1"); + curl_lib.root_module.addCMacro("HAVE_UTIMES", "1"); + curl_lib.root_module.addCMacro("HAVE_UTIME_H", "1"); + curl_lib.root_module.addCMacro("HAVE_VARIADIC_MACROS_C99", "1"); + curl_lib.root_module.addCMacro("HAVE_VARIADIC_MACROS_GCC", "1"); + curl_lib.root_module.addCMacro("OS", "\"Linux\""); + curl_lib.root_module.addCMacro("RANDOM_FILE", "\"/dev/urandom\""); + curl_lib.root_module.addCMacro("RECV_TYPE_ARG1", "int"); + curl_lib.root_module.addCMacro("RECV_TYPE_ARG2", "void *"); + curl_lib.root_module.addCMacro("RECV_TYPE_ARG3", "size_t"); + curl_lib.root_module.addCMacro("RECV_TYPE_ARG4", "int"); + curl_lib.root_module.addCMacro("RECV_TYPE_RETV", "ssize_t"); + curl_lib.root_module.addCMacro("SEND_QUAL_ARG2", "const"); + curl_lib.root_module.addCMacro("SEND_TYPE_ARG1", "int"); + curl_lib.root_module.addCMacro("SEND_TYPE_ARG2", "void *"); + curl_lib.root_module.addCMacro("SEND_TYPE_ARG3", "size_t"); + curl_lib.root_module.addCMacro("SEND_TYPE_ARG4", "int"); + curl_lib.root_module.addCMacro("SEND_TYPE_RETV", "ssize_t"); + curl_lib.root_module.addCMacro("SIZEOF_INT", "4"); + curl_lib.root_module.addCMacro("SIZEOF_SHORT", "2"); + curl_lib.root_module.addCMacro("SIZEOF_LONG", "8"); + curl_lib.root_module.addCMacro("SIZEOF_OFF_T", "8"); + curl_lib.root_module.addCMacro("SIZEOF_CURL_OFF_T", "8"); + curl_lib.root_module.addCMacro("SIZEOF_SIZE_T", "8"); + curl_lib.root_module.addCMacro("SIZEOF_TIME_T", "8"); + curl_lib.root_module.addCMacro("STDC_HEADERS", "1"); + curl_lib.root_module.addCMacro("TIME_WITH_SYS_TIME", "1"); + curl_lib.root_module.addCMacro("USE_THREADS_POSIX", "1"); + curl_lib.root_module.addCMacro("USE_UNIX_SOCKETS", "1"); + curl_lib.root_module.addCMacro("_FILE_OFFSET_BITS", "64"); + } + + // orca cli tool + + const git_version_tool: []const u8 = blk: { + if (git_version_opt) |git_version| { + break :blk try b.allocator.dupe(u8, git_version); + } else { + const git_version = b.run(&.{ "git", "rev-parse", "--short", "HEAD" }); + break :blk std.mem.trimRight(u8, git_version, "\n"); + } + }; + + var orca_tool_compile_flags = std.ArrayList([]const u8).init(b.allocator); + defer orca_tool_compile_flags.deinit(); + try orca_tool_compile_flags.append("-DFLAG_IMPLEMENTATION"); + try orca_tool_compile_flags.append("-DOC_NO_APP_LAYER"); + try orca_tool_compile_flags.append("-DOC_BUILD_DLL"); + try orca_tool_compile_flags.append("-DCURL_STATICLIB"); + try orca_tool_compile_flags.append(b.fmt("-DORCA_TOOL_VERSION={s}", .{git_version_tool})); + + if (optimize == .Debug) { + try orca_tool_compile_flags.append("-DOC_DEBUG"); + try orca_tool_compile_flags.append("-DOC_LOG_COMPILE_DEBUG"); + } + + const orca_tool_exe = b.addExecutable(.{ + .name = "orca_tool", + .target = target, + .optimize = optimize, + }); + orca_tool_exe.addIncludePath(b.path("src/")); + orca_tool_exe.addIncludePath(b.path("src/tool")); + orca_tool_exe.addIncludePath(b.path("src/ext/stb")); + orca_tool_exe.addIncludePath(b.path("src/ext/curl/include")); + orca_tool_exe.addIncludePath(b.path("src/ext/zlib")); + orca_tool_exe.addIncludePath(b.path("src/ext/microtar")); + + orca_tool_exe.addCSourceFiles(.{ + .files = &.{"src/tool/main.c"}, + .flags = orca_tool_compile_flags.items, + }); + + orca_tool_exe.linkLibrary(curl_lib); + orca_tool_exe.linkLibrary(z_lib); + if (target.result.os.tag == .windows) { + orca_tool_exe.linkSystemLibrary("shlwapi"); + orca_tool_exe.linkSystemLibrary("shell32"); + orca_tool_exe.linkSystemLibrary("ole32"); + orca_tool_exe.linkSystemLibrary("kernel32"); + } else if (target.result.os.tag.isDarwin()) { + orca_tool_exe.linkFramework("Cocoa"); + orca_tool_exe.linkFramework("SystemConfiguration"); + orca_tool_exe.linkFramework("CoreFoundation"); + orca_tool_exe.linkFramework("CoreServices"); + orca_tool_exe.linkFramework("SystemConfiguration"); + orca_tool_exe.linkFramework("Security"); + } + + orca_tool_exe.step.dependOn(&curl_lib.step); + orca_tool_exe.step.dependOn(&z_lib.step); + orca_tool_exe.linkLibC(); + + const orca_tool_install: *Build.Step.InstallArtifact = b.addInstallArtifact(orca_tool_exe, .{}); + + const build_tool_step = b.step("orca-tool", "Build the Orca CLI tool from source."); + build_tool_step.dependOn(&orca_tool_install.step); + + /////////////////////////////////////////////////////////////// + // zig build - default install step builds and installs a dev build of orca + + const build_orca = b.step("orca", "Build all orca binaries"); + build_orca.dependOn(build_orca_platform_step); + build_orca.dependOn(build_runtime_step); + build_orca.dependOn(build_libc_step); + build_orca.dependOn(build_wasm_sdk_step); + build_orca.dependOn(build_tool_step); + + const package_sdk_exe: *Build.Step.Compile = b.addExecutable(.{ + .name = "package_sdk", + .root_source_file = b.path("src/build/package_sdk.zig"), + .target = target, + .optimize = .Debug, + }); + + const sdk_install_dir_opt = b.option([]const u8, "sdk-path", "Specify absolute path for installing the Orca SDK."); + + var orca_install = b.addRunArtifact(package_sdk_exe); + orca_install.addArg("--dev-install"); + orca_install.addPrefixedFileArg("--artifacts-path=", b.path("build")); + orca_install.addPrefixedFileArg("--resources-path=", b.path("resources")); + orca_install.addPrefixedFileArg("--src-path=", b.path("src")); + + if (sdk_install_dir_opt) |sdk_install_dir| { + var sdk_install_path_absolute = sdk_install_dir; + if (std.fs.path.isAbsolute(sdk_install_path_absolute) == false) { + sdk_install_path_absolute = b.pathFromRoot(sdk_install_dir); + } + std.debug.assert(std.fs.path.isAbsolute(sdk_install_path_absolute)); + + const sdk_path = try std.mem.join(b.allocator, "", &.{ "--sdk-path=", sdk_install_path_absolute }); + orca_install.addArg(sdk_path); + } + + if (git_version_opt) |git_version| { + orca_install.addArg(b.fmt("--version={s}", .{git_version})); + } + + orca_install.step.dependOn(build_orca); + + const opt_sdk_version = b.option([]const u8, "sdk-version", "Override current git version for sdk packaging."); + if (opt_sdk_version) |sdk_version| { + const version = try std.mem.join(b.allocator, "", &.{ "--version=", sdk_version }); + orca_install.addArg(version); + } + + b.getInstallStep().dependOn(&orca_install.step); + + /////////////////////////////////////////////////////////////// + // zig build package-sdk + + var package_sdk_to_dir = b.addRunArtifact(package_sdk_exe); + package_sdk_to_dir.addPrefixedFileArg("--artifacts-path=", b.path("build")); + package_sdk_to_dir.addPrefixedFileArg("--resources-path=", b.path("resources")); + package_sdk_to_dir.addPrefixedFileArg("--src-path=", b.path("src")); + + if (sdk_install_dir_opt) |sdk_install_dir| { + var sdk_install_path_absolute = sdk_install_dir; + if (std.fs.path.isAbsolute(sdk_install_path_absolute) == false) { + sdk_install_path_absolute = b.pathFromRoot(sdk_install_dir); + } + std.debug.assert(std.fs.path.isAbsolute(sdk_install_path_absolute)); + + const sdk_path = try std.mem.join(b.allocator, "", &.{ "--sdk-path=", sdk_install_path_absolute }); + package_sdk_to_dir.addArg(sdk_path); + } else { + const fail = b.addFail("package-sdk requires -Dsdk-path"); + package_sdk_to_dir.step.dependOn(&fail.step); + } + + if (git_version_opt) |git_version| { + package_sdk_to_dir.addArg(b.fmt("--version={s}", .{git_version})); + } + + package_sdk_to_dir.step.dependOn(build_orca); + + // package command + const package_sdk_step = b.step("package-sdk", "Packages the Orca SDK for a release."); + package_sdk_step.dependOn(&package_sdk_to_dir.step); + + /////////////////////////////////////////////////////////////// + // TODO bundle command ? + + // python_build_libc.step.dependOn(&orca_runtime_exe.step); + // python_build_sdk.step.dependOn(&python_build_libc.step); + // orca_tool_exe.step.dependOn(&python_build_sdk.step); + // python_build_tool.step.dependOn(&python_build_sdk.step); + + // install_step.dependOn(&orca_tool_exe.step); + + // ensure_programs() + + // build_runtime_internal(args.release, args.wasm_backend) # this also builds the platform layer + // build_libc_internal(args.release) + // build_sdk_internal(args.release) + // build_tool(args) + + // const python_install = + + // install_step.dependOn(&python_install.step); + + ///////////////////////////////////////////////////////////////// + // zig build clean + + const clean_step: *Build.Step = b.step("clean", "Delete all build artifacts and start fresh."); + + const clean_paths = [_][]const u8{ + // folders + "build/bin", + "build/orca-libc", + "build/gles_gen.log", + "build/sketches", + "build/tests", + "src/ext/angle", + "src/ext/dawn", + "scripts/files", + "scripts/__pycache", + + // files + "src/graphics/orca_surface_stubs.c", + "src/platform/orca_io_stubs.c", + "src/wasmbind/clock_api_bind_gen.c", + "src/wasmbind/core_api_bind_gen.c", + "src/wasmbind/core_api_stubs.c", + "src/wasmbind/gles_api.json", + "src/wasmbind/gles_api_bind_gen.c", + "src/wasmbind/io_api_bind_gen.c", + "src/wasmbind/surface_api_bind_gen.c", + }; + for (clean_paths) |path| { + var remove_dir = b.addRemoveDirTree(b.path(path)); + clean_step.dependOn(&remove_dir.step); + } + + b.getUninstallStep().dependOn(clean_step); + + ///////////////////////////////////////////////////////////////// + // sketches + + var sketches = b.step("sketches", "Build all sketches into build/sketches"); + + const sketches_folders: []const []const u8 = &.{ + //"atlas", // bitrotted + "canvas", + "canvas_test", + "canvas_triangle_stress", + "check-bleeding", + "colorspace", + "image", + "keyboard", + "minimalD3D12", + "minimalDawnWebGPU", + // "multi_surface", // bitrotted + "perf_text", + // "render_thread", // bitrotted + "simpleWindow", + "smiley", + //"smooth_resize", // bitrotted + // "test-clear", // wasm bundle test - probably should be in samples? + "tiger", + //"triangleGL", // openGL API no longer supported + "triangleGLES", + "triangleMetal", + // "triangleWGPU", // bitrotted + // "ui", // bitrotted + }; + + const sketches_install_opts = Build.Step.InstallArtifact.Options{ + .dest_dir = .{ .override = .{ .custom = "sketches" } }, + }; + + const orca_platform_sketches_install: *Build.Step.InstallArtifact = b.addInstallArtifact(orca_platform_lib, sketches_install_opts); + sketches.dependOn(&orca_platform_sketches_install.step); + + var stage_sketch_dependency_artifacts = b.addUpdateSourceFiles(); + stage_sketch_dependency_artifacts.step.dependOn(&run_angle_uptodate.step); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/CMUSerif-Roman.ttf"), "build/sketches/resources/CMUSerif-Roman.ttf"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/Courier.ttf"), "build/sketches/resources/Courier.ttf"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/gamma-1.0-or-2.2.png"), "build/sketches/resources/gamma-1.0-or-2.2.png"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/gamma_dalai_lama_gray.png"), "build/sketches/resources/gamma_dalai_lama_gray.png"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/gradient_srgb.png"), "build/sketches/resources/gradient_srgb.png"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/OpenSansLatinSubset.ttf"), "build/sketches/resources/OpenSansLatinSubset.ttf"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/OpenSans-Regular.ttf"), "build/sketches/resources/OpenSans-Regular.ttf"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/OpenSans-Bold.ttf"), "build/sketches/resources/OpenSans-Bold.ttf"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/Top512.png"), "build/sketches/resources/Top512.png"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("sketches/resources/triceratops.png"), "build/sketches/resources/triceratops.png"); + if (target.result.os.tag == .windows) { + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/d3dcompiler_47.dll"), "build/sketches/d3dcompiler_47.dll"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dll"), "build/sketches/libEGL.dll"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dll.lib"), "build/sketches/libEGL.dll.lib"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dll"), "build/sketches/libGLESv2.dll"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dll.lib"), "build/sketches/libGLESv2.dll.lib"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/webgpu.dll"), "build/sketches/webgpu.dll"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/webgpu.lib"), "build/sketches/webgpu.lib"); + } else if (target.result.os.tag.isDarwin()) { + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dylib"), "build/sketches/libEGL.dylib"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dylib"), "build/sketches/libGLESv2.dylib"); + stage_sketch_dependency_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/libwebgpu.dylib"), "build/sketches/libwebgpu.dylib"); + } + stage_sketch_dependency_artifacts.step.dependOn(&run_angle_uptodate.step); + stage_sketch_dependency_artifacts.step.dependOn(&run_dawn_uptodate.step); + sketches.dependOn(&stage_sketch_dependency_artifacts.step); + + for (sketches_folders) |sketch| { + const sketch_source: []const u8 = b.pathJoin(&.{ "sketches", sketch, "main.c" }); + if (pathExists(cwd, sketch_source) == false) { + continue; + } + + if (std.mem.eql(u8, "triangleMetal", sketch) and target.result.os.tag.isDarwin() == false) { + continue; + } + + const flags: []const []const u8 = &.{ + "-Isrc", + "-Isrc/ext", + "-Isrc/ext/angle/include", + "-Isrc/ext/dawn/include", + "-Isrc/util", + "-Isrc/platform", + }; + + var sketch_exe: *Build.Step.Compile = b.addExecutable(.{ + .name = sketch, + .target = target, + .optimize = optimize, + }); + sketch_exe.addCSourceFiles(.{ + .files = &.{sketch_source}, + .flags = flags, + }); + sketch_exe.linkLibC(); + sketch_exe.linkLibrary(orca_platform_lib); + + const install: *Build.Step.InstallArtifact = b.addInstallArtifact(sketch_exe, sketches_install_opts); + sketches.dependOn(&install.step); + } + + ///////////////////////////////////////////////////////////////// + // tests + + var tests = b.step("test", "Build and run all tests"); + + const TestConfig = struct { + name: []const u8, + testfile: []const u8 = "main.c", + run: bool = false, + wasm: bool = false, + }; + + // several tests require UI interactions so we won't run them all automatically, but configure + // only some of them to be run + const test_configs: []const TestConfig = &.{ + .{ + .name = "bulkmem", + .wasm = true, + }, + .{ + .name = "file_dialog", + }, + .{ + .name = "file_open_request", + }, + .{ + .name = "files", + .run = true, + }, + .{ + .name = "perf", + .testfile = "driver.c", + }, + .{ + .name = "wasm_tests", + .wasm = true, + }, + }; + + const tests_install_opts = Build.Step.InstallArtifact.Options{ + .dest_dir = .{ .override = .{ .custom = "tests" } }, + }; + + const orca_platform_tests_install: *Build.Step.InstallArtifact = b.addInstallArtifact(orca_platform_lib, tests_install_opts); + tests.dependOn(&orca_platform_tests_install.step); + + var stage_test_dependency_artifacts = b.addUpdateSourceFiles(); + if (target.result.os.tag == .windows) { + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/d3dcompiler_47.dll"), "build/tests/d3dcompiler_47.dll"); + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dll"), "build/tests/libEGL.dll"); + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dll.lib"), "build/tests/libEGL.dll.lib"); + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dll"), "build/tests/libGLESv2.dll"); + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dll.lib"), "build/tests/libGLESv2.dll.lib"); + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/webgpu.dll"), "build/tests/webgpu.dll"); + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/webgpu.lib"), "build/tests/webgpu.lib"); + } else { + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libEGL.dylib"), "build/tests/libEGL.dylib"); + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/angle.out/bin/libGLESv2.dylib"), "build/tests/libGLESv2.dylib"); + stage_test_dependency_artifacts.addCopyFileToSource(b.path("build/dawn.out/bin/libwebgpu.dylib"), "build/tests/libwebgpu.dylib"); + } + stage_test_dependency_artifacts.step.dependOn(&run_angle_uptodate.step); + stage_test_dependency_artifacts.step.dependOn(&run_dawn_uptodate.step); + tests.dependOn(&stage_test_dependency_artifacts.step); + + for (test_configs) |config| { + // TODO add support for building wasm samples + if (config.wasm) { + continue; + } + + const test_source: []const u8 = b.pathJoin(&.{ "tests", config.name, config.testfile }); + + var test_exe: *Build.Step.Compile = b.addExecutable(.{ + .name = config.name, + .target = target, + .optimize = optimize, + }); + test_exe.addCSourceFiles(.{ + .files = &.{test_source}, + .flags = &.{b.fmt("-I{s}", .{b.pathFromRoot("src")})}, + }); + test_exe.linkLibC(); + test_exe.linkLibrary(orca_platform_lib); + + if (target.result.os.tag == .windows) { + test_exe.linkSystemLibrary("shlwapi"); + } + + const install: *Build.Step.InstallArtifact = b.addInstallArtifact(test_exe, tests_install_opts); + tests.dependOn(&install.step); + + if (config.run) { + if (config.wasm) { + // TODO add support for running wasm tests + const fail = b.addFail("Running is currently not supported for wasm tests."); + tests.dependOn(&fail.step); + } else { + const test_dir_path = b.path(b.pathJoin(&.{ "tests", config.name })); + + const run_test = b.addRunArtifact(test_exe); + run_test.addPrefixedFileArg("--test-dir=", test_dir_path); // allows tests to access their data files + run_test.step.dependOn(&stage_test_dependency_artifacts.step); + run_test.step.dependOn(&install.step); // causes test exe working dir to be build\tests\ instead of zig-cache + + tests.dependOn(&run_test.step); + } + } + } +} diff --git a/build.zig.zon b/build.zig.zon new file mode 100644 index 00000000..45e774eb --- /dev/null +++ b/build.zig.zon @@ -0,0 +1,64 @@ +.{ + .name = .orca, + + // This is a [Semantic Version](https://semver.org/). + // In a future version of Zig it will be used for package deduplication. + .version = "0.0.1", + + // Together with name, this represents a globally unique package + // identifier. This field is generated by the Zig toolchain when the + // package is first created, and then *never changes*. This allows + // unambiguous detection of one package being an updated version of + // another. + // + // When forking a Zig project, this id should be regenerated (delete the + // field and run `zig build`) if the upstream project is still maintained. + // Otherwise, the fork is *hostile*, attempting to take control over the + // original project's identity. Thus it is recommended to leave the comment + // on the following line intact, so that it shows up in code reviews that + // modify the field. + .fingerprint = 0x6111edddc2c61a9f, // Changing this has security and trust implications. + + // Tracks the earliest Zig version that the package considers to be a + // supported use case. + .minimum_zig_version = "0.14.0", + + // + .dependencies = .{ + .angle = .{ + .url = "https://chromium.googlesource.com/angle/angle.git/+archive/8a8c8fc280d74b34731e0e417b19bff7c967388a.tar.gz", + .hash = "1220df877ce2ab2f8775207778ed97f1df3123447150066d5704cbf4678e8871e982", + .lazy = true, + }, + .dawn = .{ + .url = "https://dawn.googlesource.com/dawn.git/+archive/08035d488d9b94c2ec4a9e8f0d9d58a4a915431c.tar.gz", + .hash = "1220c7233b6391482bfff4b92ecc108f88aafaa4827319ad8b2f351740734e3d3051", + .lazy = true, + }, + .@"python3-win64" = .{ + .url = "https://www.python.org/ftp/python/3.12.5/python-3.12.5-embed-amd64.zip", + .hash = "1220f762a97c1e1613f7259ae688806289485fc5145e9453e7b6611a3f8afa0c0749", + .lazy = true, + }, + .@"cmake-win64" = .{ + .url = "https://github.com/Kitware/CMake/releases/download/v3.31.0-rc2/cmake-3.31.0-rc2-windows-x86_64.zip", + .hash = "1220965b619795414210eaaf0e84493ace28fb839dd72874492ca63455cbb425e9b9", + .lazy = true, + }, + .@"cmake-macos" = .{ + .url = "https://github.com/Kitware/CMake/releases/download/v3.31.4/cmake-3.31.4-macos-universal.tar.gz", + .hash = "1220f1f14a2f29ee35eb788474180a81f3175e88b5d1760a3625643cbf654eb176d4", + .lazy = true, + }, + }, + .paths = .{ + "build.zig", + "build.zig.zon", + "doc", + "LICENSE.txt", + "resources", + "sketches", + "src", + "tests", + }, +} diff --git a/deps/angle-commit.txt b/deps/angle-commit.txt deleted file mode 100644 index 23a420c0..00000000 --- a/deps/angle-commit.txt +++ /dev/null @@ -1 +0,0 @@ -8a8c8fc280d74b34731e0e417b19bff7c967388a diff --git a/deps/dawn-commit.txt b/deps/dawn-commit.txt deleted file mode 100644 index 044908fb..00000000 --- a/deps/dawn-commit.txt +++ /dev/null @@ -1 +0,0 @@ -08035d488d9b94c2ec4a9e8f0d9d58a4a915431c diff --git a/scripts/dev.py b/scripts/dev.py index 8fe48844..4cea1f4c 100644 --- a/scripts/dev.py +++ b/scripts/dev.py @@ -639,7 +639,7 @@ def build_runtime_mac(release, wasm_backend): defines += ["-DOC_WASM_BACKEND_WASM3=1", "-DOC_WASM_BACKEND_BYTEBOX=0"] libs += ["-lwasm3"] - debug_flags = ["-O2"] if release else ["-g", "-DOC_DEBUG -DOC_LOG_COMPILE_DEBUG"] + debug_flags = ["-O2"] if release else ["-g", "-DOC_DEBUG", "-DOC_LOG_COMPILE_DEBUG", "-fsanitize=address", "-fsanitize=undefined"] flags = [ *debug_flags, "--std=c11", @@ -797,7 +797,7 @@ def build_platform_layer_lib_win(release): *includes, "src/orca.c", "/Fo:build/bin/orca.o", "/LD", "/link", - "/MANIFEST:EMBED", "/MANIFESTINPUT:src/app/win32_manifest.xml", + "/MANIFEST:EMBED", "/MANIFESTINPUT:src/app/win32_manifest.manifest", *libs, "/OUT:build/bin/orca.dll", "/IMPLIB:build/bin/orca.dll.lib", @@ -824,8 +824,11 @@ def build_platform_layer_lib_mac(release): flags = [f"-mmacos-version-min={MACOS_VERSION_MIN}"] cflags = ["-std=c11"] - debug_flags = ["-O3"] if release else ["-g", "-DOC_DEBUG", "-DOC_LOG_COMPILE_DEBUG"] + debug_flags = ["-O3"] if release else ["-g", "-DOC_DEBUG", "-DOC_LOG_COMPILE_DEBUG", "-fsanitize=address", "-fsanitize=undefined"] ldflags = [f"-L{MAC_SDK_DIR}/usr/lib", f"-F{MAC_SDK_DIR}/System/Library/Frameworks/"] + if not release: + ldflags.extend(["-fsanitize=address", "-fsanitize=undefined"]) + includes = ["-Isrc", "-Isrc/ext", "-Isrc/ext/angle/include", "-Isrc/ext/dawn/include"] # compile platform layer. We use one compilation unit for all C code, and one @@ -846,9 +849,10 @@ def build_platform_layer_lib_mac(release): ], check=True) # build dynamic library + # Note: we must use clang instead of ld for ubsan to link the correct runtime libraries subprocess.run([ - "ld", - *ldflags, "-dylib", + "clang", + *ldflags, "-dynamiclib", "-o", "build/bin/liborca.dylib", "build/orca_c.o", "build/orca_objc.o", "-Lbuild/bin", "-lc", @@ -965,7 +969,7 @@ def build_sdk_internal(release): #------------------------------------------------------ def build_libc(args): ensure_programs() - build_lib_internal(args.release) + build_libc_internal(args.release) def build_libc_internal(release): print("Building orca-libc...") diff --git a/sketches/atlas/main.c b/sketches/atlas/main.c index c51fb916..35749c55 100644 --- a/sketches/atlas/main.c +++ b/sketches/atlas/main.c @@ -8,8 +8,6 @@ #include #include #include - -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include #include "orca.h" diff --git a/sketches/canvas_test/main.c b/sketches/canvas_test/main.c index f33b1264..455a903f 100644 --- a/sketches/canvas_test/main.c +++ b/sketches/canvas_test/main.c @@ -41,7 +41,7 @@ int main() }); oc_arena_scope scratch = oc_scratch_begin(); - oc_str8 path = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/triceratops.png")); + oc_str8 path = oc_path_executable_relative(scratch.arena, OC_STR8("resources/triceratops.png")); oc_image image = oc_image_create_from_path(renderer, path, false); if(oc_image_is_nil(image)) @@ -230,6 +230,7 @@ int main() oc_fill(); oc_set_gradient( + OC_GRADIENT_BLEND_LINEAR, (oc_color){ 0, 0, 0, 1 }, (oc_color){ 1, 1, 1, 1 }, (oc_color){ 0, 1, 0, 1 }, diff --git a/sketches/canvas_triangle_stress/main.c b/sketches/canvas_triangle_stress/main.c index 487bb029..7030fdcf 100644 --- a/sketches/canvas_triangle_stress/main.c +++ b/sketches/canvas_triangle_stress/main.c @@ -32,7 +32,7 @@ int main() } oc_arena_scope scratch = oc_scratch_begin(); - oc_str8 path = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/triceratops.png")); + oc_str8 path = oc_path_executable_relative(scratch.arena, OC_STR8("resources/triceratops.png")); oc_image image = oc_image_create_from_path(renderer, path, false); if(oc_image_is_nil(image)) diff --git a/sketches/check-bleeding/main.c b/sketches/check-bleeding/main.c index b5e4cd5d..c43a6b10 100644 --- a/sketches/check-bleeding/main.c +++ b/sketches/check-bleeding/main.c @@ -9,8 +9,6 @@ #include #include #include - -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include #include "orca.h" @@ -55,7 +53,7 @@ int main() //NOTE: create image oc_arena_scope scratch = oc_scratch_begin(); - oc_str8 imagePath = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/square_small.png")); + oc_str8 imagePath = oc_path_executable_relative(scratch.arena, OC_STR8("resources/square_small.png")); oc_image image = oc_image_create_from_path(renderer, imagePath, false); oc_vec2 imageSize = oc_image_size(image); diff --git a/sketches/colorspace/main.c b/sketches/colorspace/main.c index e876fb83..298e53a9 100644 --- a/sketches/colorspace/main.c +++ b/sketches/colorspace/main.c @@ -36,10 +36,10 @@ int main() oc_arena_scope scratch = oc_scratch_begin(); - oc_str8 dalaiPath = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/gamma_dalai_lama_gray.png")); + oc_str8 dalaiPath = oc_path_executable_relative(scratch.arena, OC_STR8("resources/gamma_dalai_lama_gray.png")); oc_image dalai = oc_image_create_from_path(renderer, dalaiPath, false); - oc_str8 offensivePath = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/gamma-1.0-or-2.2.png")); + oc_str8 offensivePath = oc_path_executable_relative(scratch.arena, OC_STR8("resources/gamma-1.0-or-2.2.png")); oc_image offensive = oc_image_create_from_path(renderer, offensivePath, false); oc_wgpu_canvas_debug_set_record_options(renderer, diff --git a/sketches/image/main.c b/sketches/image/main.c index 2b3203a9..8a6819c4 100644 --- a/sketches/image/main.c +++ b/sketches/image/main.c @@ -9,8 +9,6 @@ #include #include #include - -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include #include "orca.h" @@ -48,11 +46,11 @@ int main() //NOTE: create image oc_arena_scope scratch = oc_scratch_begin(); - oc_str8 imagePath = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/triceratops.png")); + oc_str8 imagePath = oc_path_executable_relative(scratch.arena, OC_STR8("resources/triceratops.png")); oc_image image = oc_image_create_from_path(renderer, imagePath, false); oc_vec2 imageSize = oc_image_size(image); - oc_str8 imagePath2 = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/Top512.png")); + oc_str8 imagePath2 = oc_path_executable_relative(scratch.arena, OC_STR8("resources/Top512.png")); oc_image image2 = oc_image_create_from_path(renderer, imagePath2, false); oc_vec2 imageSize2 = oc_image_size(image2); diff --git a/sketches/minimalD3D12/win32_d3d12.c b/sketches/minimalD3D12/win32_d3d12.c index 537ff57d..61a53752 100644 --- a/sketches/minimalD3D12/win32_d3d12.c +++ b/sketches/minimalD3D12/win32_d3d12.c @@ -7,9 +7,7 @@ #include #include #include -#include - -#define _USE_MATH_DEFINES +#include #include #include #include diff --git a/sketches/minimalD3D12/win32_d3d12_transparent.c b/sketches/minimalD3D12/win32_d3d12_transparent.c index bf5bcb11..6d5dfa12 100644 --- a/sketches/minimalD3D12/win32_d3d12_transparent.c +++ b/sketches/minimalD3D12/win32_d3d12_transparent.c @@ -16,7 +16,6 @@ EXTERN_C const IID IID_IDCompositionDevice; DEFINE_GUID(IID_IDCompositionDevice, 0xC37EA93A, 0xE7AA, 0x450D, 0xB1, 0x6F, 0x97, 0x46, 0xCB, 0x04, 0x07, 0xF3); -#define _USE_MATH_DEFINES #include #include #include diff --git a/sketches/minimalDawnWebGPU/win32_webgpu.c b/sketches/minimalDawnWebGPU/win32_webgpu.c index 49f57a84..949e70e9 100644 --- a/sketches/minimalDawnWebGPU/win32_webgpu.c +++ b/sketches/minimalDawnWebGPU/win32_webgpu.c @@ -4,8 +4,6 @@ #define WIN32_LEAN_AND_MEAN #include #include - -#define _USE_MATH_DEFINES #include #include #include diff --git a/sketches/perf_text/main.c b/sketches/perf_text/main.c index c05229a8..3ded11c6 100644 --- a/sketches/perf_text/main.c +++ b/sketches/perf_text/main.c @@ -134,9 +134,9 @@ int main() } int fontIndex = 0; - oc_font fonts[FONT_COUNT] = { create_font("../../resources/OpenSansLatinSubset.ttf"), - create_font("../../resources/CMUSerif-Roman.ttf"), - create_font("../../resources/Courier.ttf") }; + oc_font fonts[FONT_COUNT] = { create_font("resources/OpenSansLatinSubset.ttf"), + create_font("resources/CMUSerif-Roman.ttf"), + create_font("resources/Courier.ttf") }; oc_font_metrics extents[FONT_COUNT]; f32 fontScales[FONT_COUNT]; diff --git a/sketches/ui/OpenSans-Bold.ttf b/sketches/resources/OpenSans-Bold.ttf similarity index 100% rename from sketches/ui/OpenSans-Bold.ttf rename to sketches/resources/OpenSans-Bold.ttf diff --git a/sketches/ui/OpenSans-Regular.ttf b/sketches/resources/OpenSans-Regular.ttf similarity index 100% rename from sketches/ui/OpenSans-Regular.ttf rename to sketches/resources/OpenSans-Regular.ttf diff --git a/sketches/simpleWindow/main.c b/sketches/simpleWindow/main.c index 9ca76f4f..3b557899 100644 --- a/sketches/simpleWindow/main.c +++ b/sketches/simpleWindow/main.c @@ -99,7 +99,7 @@ int main() case OC_EVENT_MOUSE_BUTTON: { oc_log_info("mouse button %i: %i\n", - event->key.code, + event->key.keyCode, event->key.action == OC_KEY_PRESS ? 1 : 0); } break; @@ -107,7 +107,7 @@ int main() case OC_EVENT_KEYBOARD_KEY: { oc_log_info("key %i: %s\n", - event->key.code, + event->key.keyCode, event->key.action == OC_KEY_PRESS ? "press" : (event->key.action == OC_KEY_RELEASE ? "release" : "repeat")); } break; diff --git a/sketches/smiley/main.c b/sketches/smiley/main.c index 89f99acc..dcf9e41b 100644 --- a/sketches/smiley/main.c +++ b/sketches/smiley/main.c @@ -9,8 +9,6 @@ #include #include #include - -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include #include "orca.h" @@ -19,7 +17,7 @@ oc_font create_font() { //NOTE(martin): create font oc_arena_scope scratch = oc_scratch_begin(); - oc_str8 fontPath = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/OpenSansLatinSubset.ttf")); + oc_str8 fontPath = oc_path_executable_relative(scratch.arena, OC_STR8("resources/OpenSansLatinSubset.ttf")); char* fontPathCString = oc_str8_to_cstring(scratch.arena, fontPath); FILE* fontFile = fopen(fontPathCString, "r"); diff --git a/sketches/smooth_resize/main.c b/sketches/smooth_resize/main.c index 535a4ab3..262bfd19 100644 --- a/sketches/smooth_resize/main.c +++ b/sketches/smooth_resize/main.c @@ -9,11 +9,9 @@ #include #include #include - -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include -#define OC_INCLUDE_GL_API +#define OC_GRAPHICS_INCLUDE_GL_API 1 #include "orca.h" unsigned int program; diff --git a/sketches/tiger/main.c b/sketches/tiger/main.c index d47a0ac1..83862017 100644 --- a/sketches/tiger/main.c +++ b/sketches/tiger/main.c @@ -9,8 +9,6 @@ #include #include #include - -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include #include "orca.h" @@ -21,7 +19,7 @@ oc_font create_font() { //NOTE(martin): create font oc_arena_scope scratch = oc_scratch_begin(); - oc_str8 fontPath = oc_path_executable_relative(scratch.arena, OC_STR8("../../resources/OpenSansLatinSubset.ttf")); + oc_str8 fontPath = oc_path_executable_relative(scratch.arena, OC_STR8("resources/OpenSansLatinSubset.ttf")); char* fontPathCString = oc_str8_to_cstring(scratch.arena, fontPath); FILE* fontFile = fopen(fontPathCString, "r"); diff --git a/sketches/triangleGL/main.c b/sketches/triangleGL/main.c index 1454b739..eae6f770 100644 --- a/sketches/triangleGL/main.c +++ b/sketches/triangleGL/main.c @@ -7,11 +7,9 @@ **************************************************************************/ #include #include - -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include -#define OC_INCLUDE_GL_API +#define OC_GRAPHICS_INCLUDE_GL_API 1 #include "orca.h" #include "graphics/gl_surface.h" diff --git a/sketches/triangleGLES/main.c b/sketches/triangleGLES/main.c index d5374aec..cc146ef2 100644 --- a/sketches/triangleGLES/main.c +++ b/sketches/triangleGLES/main.c @@ -8,11 +8,9 @@ #include #include #include - -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include -#define OC_INCLUDE_GL_API 1 +#define OC_GRAPHICS_INCLUDE_GL_API 1 #include "orca.h" #include "graphics/gles_surface.h" diff --git a/sketches/triangleWGPU/main.c b/sketches/triangleWGPU/main.c index 8a980a3b..2ff95779 100644 --- a/sketches/triangleWGPU/main.c +++ b/sketches/triangleWGPU/main.c @@ -12,6 +12,7 @@ #define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include +#define OC_GRAPHICS_ENABLE_WEBGPU 1 #include "orca.h" #include "graphics/wgpu_surface.h" diff --git a/sketches/ui/main.c b/sketches/ui/main.c index a05e708c..62e42ef9 100644 --- a/sketches/ui/main.c +++ b/sketches/ui/main.c @@ -818,8 +818,8 @@ int main() oc_font* fonts[2] = { &fontRegular, &fontBold }; oc_str8 fontNames[2] = { - oc_path_executable_relative(scratch.arena, OC_STR8("../OpenSans-Regular.ttf")), - oc_path_executable_relative(scratch.arena, OC_STR8("../OpenSans-Bold.ttf")) + oc_path_executable_relative(scratch.arena, OC_STR8("resources/OpenSans-Regular.ttf")), + oc_path_executable_relative(scratch.arena, OC_STR8("resources/OpenSans-Bold.ttf")) }; for(int i = 0; i < 2; i++) diff --git a/src/app/win32_app.c b/src/app/win32_app.c index 748b475e..4bdd4b91 100644 --- a/src/app/win32_app.c +++ b/src/app/win32_app.c @@ -144,6 +144,8 @@ void oc_init_keys() oc_appData.scanCodes[0x04A] = OC_SCANCODE_KP_SUBTRACT; } +_Static_assert(OC_SCANCODE_COUNT == 349, "hmmm"); + void oc_win32_update_keyboard_layout() { memcpy(oc_appData.keyMap, oc_defaultKeyMap, sizeof(oc_key_code) * OC_SCANCODE_COUNT); @@ -245,7 +247,7 @@ void oc_init() hr = ID3D11Device_QueryInterface(d3d11Device, &IID_IDXGIDevice, (void**)&dxgiDevice); OC_ASSERT(hr == S_OK, "Failed to initialize Direct Composition: couldn't get DXGI device"); - hr = DCompositionCreateDevice(dxgiDevice, &IID_IDCompositionDevice, &oc_appData.win32.dcompDevice); + hr = DCompositionCreateDevice(dxgiDevice, &IID_IDCompositionDevice, (void**)&oc_appData.win32.dcompDevice); OC_ASSERT(hr == S_OK, "Failed to create DirectComposition device"); IDXGIDevice_Release(dxgiDevice); @@ -265,7 +267,7 @@ void oc_terminate() } } -static oc_key_code oc_convert_win32_key(int code) +static oc_scan_code oc_convert_win32_key(int code) { return (oc_appData.scanCodes[code]); } @@ -293,7 +295,7 @@ static oc_keymod_flags oc_get_mod_keys() return (mods); } -static void oc_win32_process_mouse_event(oc_window_data* window, oc_key_action action, oc_key_code button) +static void oc_win32_process_mouse_event(oc_window_data* window, oc_key_action action, oc_mouse_button button) { if(action == OC_KEY_PRESS) { @@ -565,8 +567,8 @@ LRESULT oc_win32_win_proc(HWND windowHandle, UINT message, WPARAM wParam, LPARAM event.mouse.deltaX = event.mouse.x - oc_appData.win32.lastMousePos.x; event.mouse.deltaY = event.mouse.y - oc_appData.win32.lastMousePos.y; } - if(abs(event.mouse.x - oc_appData.win32.lastMousePos.x) > GetSystemMetrics(SM_CXDOUBLECLK) / 2 - || abs(event.mouse.y - oc_appData.win32.lastMousePos.y) > GetSystemMetrics(SM_CYDOUBLECLK) / 2) + if(abs((int)(event.mouse.x - oc_appData.win32.lastMousePos.x)) > GetSystemMetrics(SM_CXDOUBLECLK) / 2 + || abs((int)(event.mouse.y - oc_appData.win32.lastMousePos.y)) > GetSystemMetrics(SM_CYDOUBLECLK) / 2) { for(int i = 0; i < OC_MOUSE_BUTTON_COUNT; i++) { @@ -1216,7 +1218,7 @@ void oc_window_center(oc_window window) MONITORINFO monitorInfo = { .cbSize = sizeof(MONITORINFO) }; GetMonitorInfoW(monitor, &monitorInfo); - int dpiX, dpiY; + UINT dpiX, dpiY; GetDpiForMonitor(monitor, MDT_EFFECTIVE_DPI, &dpiX, &dpiY); f32 scaleX = dpiX / 96.; f32 scaleY = dpiY / 96.; @@ -1457,7 +1459,7 @@ oc_file_dialog_result oc_file_dialog_for_table(oc_arena* arena, oc_file_dialog_d hr = ((IFileOpenDialog*)dialog)->lpVtbl->GetResults((IFileOpenDialog*)dialog, &array); if(SUCCEEDED(hr)) { - int count = 0; + DWORD count = 0; array->lpVtbl->GetCount(array, &count); for(int itemIndex = 0; itemIndex < count; itemIndex++) { @@ -1565,7 +1567,6 @@ int oc_alert_popup(oc_str8 title, .dwFlags = 0, .dwCommonButtons = 0, .pszWindowTitle = titleWide, - .hMainIcon = 0, .pszMainIcon = TD_WARNING_ICON, .pszMainInstruction = messageWide, .pszContent = NULL, diff --git a/src/app/win32_dcomp_c_api.h b/src/app/win32_dcomp_c_api.h index 272ca6fe..ae1230f2 100644 --- a/src/app/win32_dcomp_c_api.h +++ b/src/app/win32_dcomp_c_api.h @@ -18,6 +18,85 @@ #if (NTDDI_VERSION >= NTDDI_WIN8) + +/////////////////////////////////////////////////////////////////////////////////////////////////// +// dcomptypes.h + +// +// Composition Stats +// + +typedef enum COMPOSITION_FRAME_ID_TYPE +{ + COMPOSITION_FRAME_ID_CREATED = 0, + COMPOSITION_FRAME_ID_CONFIRMED = 1, + COMPOSITION_FRAME_ID_COMPLETED = 2 +} COMPOSITION_FRAME_ID_TYPE; + +typedef ULONG64 COMPOSITION_FRAME_ID; + +typedef struct tagCOMPOSITION_FRAME_STATS +{ + UINT64 startTime; + UINT64 targetTime; + UINT64 framePeriod; +} COMPOSITION_FRAME_STATS; + +typedef struct tagCOMPOSITION_TARGET_ID +{ +#if defined(__cplusplus) && !defined(SORTPP_PASS) + bool operator ==(const tagCOMPOSITION_TARGET_ID& rhs) const + { + return ((displayAdapterLuid.LowPart == rhs.displayAdapterLuid.LowPart) && + (displayAdapterLuid.HighPart == rhs.displayAdapterLuid.HighPart) && + (renderAdapterLuid.LowPart == rhs.renderAdapterLuid.LowPart) && + (renderAdapterLuid.HighPart == rhs.renderAdapterLuid.HighPart) && + (vidPnSourceId == rhs.vidPnSourceId) && + (vidPnTargetId == rhs.vidPnTargetId) && + ((uniqueId == rhs.uniqueId) || (uniqueId == 0) || (rhs.uniqueId == 0))); + } + + bool operator !=(const tagCOMPOSITION_TARGET_ID& rhs) const + { + return !(*this == rhs); + } +#endif + + LUID displayAdapterLuid; + LUID renderAdapterLuid; + UINT vidPnSourceId; + UINT vidPnTargetId; + UINT uniqueId; +} COMPOSITION_TARGET_ID; + +typedef struct tagCOMPOSITION_STATS +{ + UINT presentCount; + UINT refreshCount; + UINT virtualRefreshCount; + UINT64 time; +} COMPOSITION_STATS; + +typedef struct tagCOMPOSITION_TARGET_STATS +{ + UINT outstandingPresents; + UINT64 presentTime; + UINT64 vblankDuration; + + COMPOSITION_STATS presentedStats; + COMPOSITION_STATS completedStats; +} COMPOSITION_TARGET_STATS; + +// The maximum nubmer of objects we allow users to wait on the compositor clock +#define DCOMPOSITION_MAX_WAITFORCOMPOSITORCLOCK_OBJECTS 32 + +// Maximum number of targets kept per frame +#define COMPOSITION_STATS_MAX_TARGETS 256 + +/////////////////////////////////////////////////////////////////////////////////////////////////// +// dcomp.h +// + typedef interface IDCompositionAffineTransform2DEffect IDCompositionAffineTransform2DEffect; typedef interface IDCompositionAnimation IDCompositionAnimation; typedef interface IDCompositionArithmeticCompositeEffect IDCompositionArithmeticCompositeEffect; diff --git a/src/app/win32_manifest.xml b/src/app/win32_manifest.manifest similarity index 93% rename from src/app/win32_manifest.xml rename to src/app/win32_manifest.manifest index 6d43ef51..54c10a66 100644 --- a/src/app/win32_manifest.xml +++ b/src/app/win32_manifest.manifest @@ -1,4 +1,4 @@ - + diff --git a/src/build/bindgen.zig b/src/build/bindgen.zig new file mode 100644 index 00000000..fc1b4529 --- /dev/null +++ b/src/build/bindgen.zig @@ -0,0 +1,620 @@ +// This is a zig program run as part of the orca build process to generate bindings in C for wasm +// functions. + +const std = @import("std"); + +const MAX_FILE_SIZE = 1024 * 1024 * 128; + +const Options = struct { + arena: std.mem.Allocator, + api_name: []const u8, + spec_path: []const u8, + bindings_path: []const u8, + guest_stubs_path: ?[]const u8, + guest_include_path: ?[]const u8, + + fn parse(args: []const [:0]const u8, arena: std.mem.Allocator) !Options { + var api_name: ?[]const u8 = null; + var spec_path: ?[]const u8 = null; + var bindings_path: ?[]const u8 = null; + var guest_stubs_path: ?[]const u8 = null; + var guest_include_path: ?[]const u8 = null; + + for (args, 0..) |raw_arg, i| { + if (i == 0) { + continue; + } + + var split_iter = std.mem.splitScalar(u8, raw_arg, '='); + const arg: []const u8 = split_iter.next().?; + if (std.mem.eql(u8, arg, "--api-name")) { + api_name = split_iter.next(); + } else if (std.mem.eql(u8, arg, "--spec-path")) { + spec_path = split_iter.next(); + } else if (std.mem.eql(u8, arg, "--bindings-path")) { + bindings_path = split_iter.next(); + } else if (std.mem.eql(u8, arg, "--guest-stubs-path")) { + guest_stubs_path = split_iter.next(); + } else if (std.mem.eql(u8, arg, "--guest-include-path")) { + guest_include_path = split_iter.next(); + } + } + + var missing_arg: ?[]const u8 = null; + if (api_name == null) { + missing_arg = "api-name"; + } else if (spec_path == null) { + missing_arg = "spec-path"; + } else if (bindings_path == null) { + missing_arg = "bindings-path"; + } + + if (missing_arg) |arg| { + std.log.err("Missing required arg: {s}", .{arg}); + return error.MissingRequiredArgument; + } + + return Options{ + .arena = arena, + .api_name = api_name.?, + .spec_path = spec_path.?, + .bindings_path = bindings_path.?, + .guest_stubs_path = guest_stubs_path, + .guest_include_path = guest_include_path, + }; + } +}; + +const BindingUntyped = struct { + const CType = struct { + name: []const u8, + cname: ?[]const u8 = null, + tag: []const u8, + }; + + const Arg = struct { + // Basically a union + const Length = struct { + // call a function with these args to determine the length + proc: ?[]const u8 = null, + args: ?[]const []const u8 = null, + + // length specified by another argument + count: ?[]const u8 = null, + + // hardcoded length + components: ?u32 = null, + }; + + name: []const u8, + type: CType, + len: ?Length = null, + }; + + name: []const u8, + cname: []const u8, + ret: CType, + args: []Arg, +}; + +const Binding = struct { + const Tag = enum { + Void, + Struct, + Pointer, + Int32, + Int64, + Float32, + Float64, + + fn fromStr(tag: []const u8, binding_name: []const u8) !Tag { + if (std.mem.eql(u8, tag, "i")) { + return .Int32; + } else if (std.mem.eql(u8, tag, "I")) { + return .Int64; + } else if (std.mem.eql(u8, tag, "f")) { + return .Float32; + } else if (std.mem.eql(u8, tag, "F")) { + return .Float64; + } else if (std.mem.eql(u8, tag, "S")) { + return .Struct; + } else if (std.mem.eql(u8, tag, "p")) { + return .Pointer; + } else if (std.mem.eql(u8, tag, "v")) { + return .Void; + } + + std.log.err("Unknown tag type {s} in binding {s}", .{ tag, binding_name }); + return error.UnknownTag; + } + + fn toValtype(tag: Tag, binding_name: []const u8) ![]const u8 { + std.debug.assert(tag != .Struct); + + return switch (tag) { + .Int32 => "OC_WASM_VALTYPE_I32", + .Int64 => "OC_WASM_VALTYPE_I64", + .Float32 => "OC_WASM_VALTYPE_F32", + .Float64 => "OC_WASM_VALTYPE_F64", + else => { + std.log.err("Cannot convert {} tag to valtype in binding {s}", .{ tag, binding_name }); + return error.StructToValtype; + }, + }; + } + }; + + const CType = struct { + name: []const u8, + cname: []const u8, + tag: Tag, + + fn fromUntyped(untyped: BindingUntyped.CType, binding_name: []const u8) !CType { + return .{ + .name = untyped.name, + .cname = if (untyped.cname) |cname| cname else untyped.name, + .tag = try Tag.fromStr(untyped.tag, binding_name), + }; + } + }; + + const Arg = struct { + const LengthType = enum { + proc, + components, + count, + }; + const Length = union(LengthType) { + proc: struct { + name: []const u8, + args: []const []const u8, + }, + components: struct { + num: u32, + count_arg: ?[]const u8, + }, + count: []const u8, + }; + + name: []const u8, + type: CType, + len: ?Length, + }; + + name: []const u8, + cname: []const u8, + ret: CType, + args: []Arg, + needs_stub: bool, + + fn fromUntyped(opts: Options, untyped_bindings: []const BindingUntyped) ![]const Binding { + const bindings = try opts.arena.alloc(Binding, untyped_bindings.len); + for (untyped_bindings, bindings) |untyped, *binding| { + binding.name = untyped.name; + binding.cname = untyped.cname; + binding.ret = try CType.fromUntyped(untyped.ret, binding.name); + binding.args = try opts.arena.alloc(Arg, untyped.args.len); + for (untyped.args, binding.args) |untyped_arg, *binding_arg| { + binding_arg.name = untyped_arg.name; + binding_arg.type = try CType.fromUntyped(untyped_arg.type, binding.name); + binding_arg.len = null; + if (untyped_arg.len) |len| { + if (len.proc) |proc_name| { + const args: []const []const u8 = if (len.args) |len_args| len_args else &.{}; + binding_arg.len = Arg.Length{ + .proc = .{ + .name = proc_name, + .args = args, + }, + }; + } + + if (len.components) |components| { + if (binding_arg.len == null) { + binding_arg.len = Arg.Length{ + .components = .{ + .num = components, + .count_arg = len.count, + }, + }; + } else { + std.log.err("Binding {s} arg {s} has invalid length settings", .{ binding.name, binding_arg.name }); + return error.InvalidBindingArgLength; + } + } else if (len.count) |count| { + if (binding_arg.len == null) { + binding_arg.len = Arg.Length{ .count = count }; + } else { + std.log.err("Binding {s} arg {s} has invalid length settings", .{ binding.name, binding_arg.name }); + return error.InvalidBindingArgLength; + } + } + } + } + + binding.needs_stub = binding.ret.tag == .Struct; + if (binding.needs_stub == false) { + for (binding.args) |arg| { + if (arg.type.tag == .Struct) { + binding.needs_stub = true; + break; + } + } + } + } + + return bindings; + } +}; + +const GeneratedBindings = struct { + host: []const u8, + guest: []const u8, +}; + +fn generateBindings(opts: Options, bindings: []const Binding) !GeneratedBindings { + var bindings_host_array = std.ArrayList(u8).init(opts.arena); + try bindings_host_array.ensureUnusedCapacity(1024 * 1024); + var host = bindings_host_array.writer(); + + var bindings_guest_array = std.ArrayList(u8).init(opts.arena); + try bindings_guest_array.ensureUnusedCapacity(1024 * 1024); + var guest = bindings_guest_array.writer(); + + if (opts.guest_include_path) |path| { + for (bindings) |binding| { + if (binding.needs_stub) { + try guest.print("#include \"{s}\"\n\n", .{path}); + } + break; + } + } + + for (bindings) |binding| { + if (binding.needs_stub) { + const argptr_stub_name = try std.mem.join(opts.arena, "", &.{ binding.name, "_argptr_stub" }); + + // pointer arg stub declaration + if (binding.ret.tag == .Struct) { + try guest.writeAll("void"); + } else { + try guest.writeAll(binding.ret.name); + } + try guest.print(" ORCA_IMPORT({s}) (", .{argptr_stub_name}); + + if (binding.ret.tag == .Struct) { + try guest.writeAll(binding.ret.name); + try guest.writeAll("* __retArg"); + if (binding.args.len > 0) { + try guest.writeAll(", "); + } + } else if (binding.args.len == 0) { + try guest.writeAll("void"); + } + + for (binding.args, 0..) |arg, i| { + try guest.writeAll(arg.type.name); + if (arg.type.tag == .Struct) { + try guest.writeAll("*"); + } + try guest.print(" {s}", .{arg.name}); + if (i + 1 < binding.args.len) { + try guest.writeAll(", "); + } + } + try guest.writeAll(");\n\n"); + + // forward function to pointer arg stub declaration + + try guest.print("{s} {s}(", .{ binding.ret.name, binding.name }); + + if (binding.args.len == 0) { + try guest.writeAll("void"); + } + + for (binding.args, 0..) |arg, i| { + try guest.print("{s} {s}", .{ arg.type.name, arg.name }); + if (i + 1 < binding.args.len) { + try guest.writeAll(", "); + } + } + try guest.writeAll(")\n"); + try guest.writeAll("{\n"); + try guest.writeAll("\t"); + + if (binding.ret.tag == .Struct) { + try guest.print("{s} __ret;\n\t", .{binding.ret.name}); + } else if (binding.ret.tag != .Void) { + try guest.print("{s} __ret = ", .{binding.ret.name}); + } + try guest.print("{s}(", .{argptr_stub_name}); + + if (binding.ret.tag == .Struct) { + try guest.writeAll("&__ret"); + if (binding.args.len > 0) { + try guest.writeAll(", "); + } + } + + for (binding.args, 0..) |arg, i| { + if (arg.type.tag == .Struct) { + try guest.writeAll("&"); + } + + try guest.writeAll(arg.name); + if (i + 1 < binding.args.len) { + try guest.writeAll(", "); + } + } + try guest.writeAll(");\n"); + + if (binding.ret.tag != .Void) { + try guest.writeAll("\treturn(__ret);\n"); + } + try guest.writeAll("}\n\n"); + } + + // host-side stub + try host.print( + "void {s}_stub(const i64* restrict _params, i64* restrict _returns, u8* _mem, oc_wasm* wasm)\n", + .{binding.cname}, + ); + try host.writeAll("{\n"); + + const first_arg_index: u32 = if (binding.ret.tag == .Struct) 1 else 0; + if (binding.ret.tag == .Struct) { + try host.print("\t{s}* __retPtr = ({s}*)((char*)_mem + *(i32*)&_params[0]);\n", .{ binding.ret.cname, binding.ret.cname }); + + try host.writeAll("\t{\n"); + try host.writeAll("\t\tOC_ASSERT_DIALOG(((char*)__retPtr >= (char*)_mem) && (((char*)__retPtr - (char*)_mem) < oc_wasm_mem_size(wasm)), \"return pointer is out of bounds\");\n"); + try host.print( + "\t\tOC_ASSERT_DIALOG((char*)__retPtr + sizeof({s}) <= ((char*)_mem + oc_wasm_mem_size(wasm)), \"return pointer is out of bounds\");\n", + .{binding.ret.cname}, + ); + try host.writeAll("\t}\n"); + } + + for (binding.args, 0..) |arg, i| { + try host.writeAll("\t"); + + const argtype = arg.type; + + const arg_index = i + first_arg_index; + + switch (argtype.tag) { + .Int32 => try host.print("{s} {s} = ({s})*(i32*)&_params[{}];\n", .{ argtype.cname, arg.name, argtype.cname, arg_index }), + .Int64 => try host.print("{s} {s} = ({s})*(i64*)&_params[{}];\n", .{ argtype.cname, arg.name, argtype.cname, arg_index }), + .Float32 => try host.print("{s} {s} = ({s})*(f32*)&_params[{}];\n", .{ argtype.cname, arg.name, argtype.cname, arg_index }), + .Float64 => try host.print("{s} {s} = ({s})*(f64*)&_params[{}];\n", .{ argtype.cname, arg.name, argtype.cname, arg_index }), + .Pointer => try host.print("{s} {s} = ({s})((char*)_mem + *(u32*)&_params[{}]);\n", .{ argtype.cname, arg.name, argtype.cname, arg_index }), + .Struct => try host.print("{s} {s} = *({s}*)((char*)_mem + *(u32*)&_params[{}]);\n", .{ argtype.cname, arg.name, argtype.cname, arg_index }), + else => { + std.log.err("Found invalid void type for arg {s} in binding {s}", .{ arg.name, binding.name }); + return error.InvalidVoidArgType; + }, + } + } + + // check pointer arg length + for (binding.args) |arg| { + if (arg.type.tag == .Pointer) { + if (arg.len) |len| { + try host.writeAll("\t{\n"); + try host.print( + "\t\tOC_ASSERT_DIALOG(((char*){s} >= (char*)_mem) && (((char*){s} - (char*)_mem) < oc_wasm_mem_size(wasm)), \"parameter '{s}' is out of bounds\");\n", + .{ arg.name, arg.name, arg.name }, + ); + try host.print("\t\tOC_ASSERT_DIALOG((char*){s} + ", .{arg.name}); + + switch (len) { + .proc => |proc| { + try host.print("{s}(wasm, ", .{proc.name}); + for (proc.args, 0..) |proc_arg, i| { + try host.print("{s}", .{proc_arg}); + if (i + 1 < proc.args.len) { + try host.writeAll(", "); + } + } + try host.writeAll(")"); + }, + .components => |components| { + try host.print("{}", .{components.num}); + if (components.count_arg) |count_arg| { + try host.print("*{s}", .{count_arg}); + } + }, + .count => |count_arg| { + try host.print("{s}", .{count_arg}); + }, + } + + const cname = arg.type.cname; + if (std.mem.endsWith(u8, cname, "**") or (std.mem.startsWith(u8, cname, "void") == false and std.mem.startsWith(u8, cname, "const void") == false)) { + try host.print("*sizeof({s})", .{cname[0 .. cname.len - 1]}); + } + + try host.print(" <= ((char*)_mem + oc_wasm_mem_size(wasm)), \"parameter '{s}' is out of bounds\");\n", .{arg.name}); + try host.writeAll("\t}\n"); + } else { + std.log.err("Binding {s} missing pointer length decoration for param '{s}'", .{ binding.name, arg.name }); + return error.MissingRequiredPointerLength; + } + } + } + + try host.writeAll("\t"); + switch (binding.ret.tag) { + .Void => {}, + .Int32 => try host.writeAll("*((i32*)&_returns[0]) = (i32)"), + .Int64 => try host.writeAll("*((i64*)&_returns[0]) = (i64)"), + .Float32 => try host.writeAll("*((f32*)&_returns[0]) = (f32)"), + .Float64 => try host.writeAll("*((f64*)&_returns[0]) = (f64)"), + .Struct => try host.writeAll("*__retPtr = "), + .Pointer => { + std.log.err("Binding {s} has pointer return type, but this isn't supported yet.", .{binding.name}); + return error.UnsupportedPointerReturn; + }, + } + + try host.print("{s}(", .{binding.cname}); + + for (binding.args, 0..) |arg, i| { + try host.writeAll(arg.name); + if (i + 1 < binding.args.len) { + try host.writeAll(", "); + } + } + try host.writeAll(");\n}\n\n"); + } + + // link function + try host.print("int bindgen_link_{s}_api(oc_wasm* wasm)\n", .{opts.api_name}); + try host.writeAll("{\n\toc_wasm_status status;\n"); + try host.writeAll("\tint ret = 0;\n\n"); + + // for decl in data: + for (bindings) |binding| { + const name: []const u8 = if (binding.needs_stub) try std.mem.join(opts.arena, "", &.{ binding.name, "_argptr_stub" }) else binding.name; + const num_args: usize = binding.args.len + @as(usize, if (binding.ret.tag == .Struct) 1 else 0); + const num_returns: usize = if (binding.ret.tag == .Struct or binding.ret.tag == .Void) 0 else 1; + + const param_types: []const u8 = blk: { + var params_str = std.ArrayList(u8).init(opts.arena); + try params_str.ensureUnusedCapacity(1024 * 4); + var writer = params_str.writer(); + + if (num_args == 0) { + try writer.writeAll("\t\toc_wasm_valtype paramTypes[1];\n"); + } else { + try writer.writeAll("\t\toc_wasm_valtype paramTypes[] = {"); + + if (binding.ret.tag == .Struct) { + try writer.writeAll("OC_WASM_VALTYPE_I32, "); + } + for (binding.args) |arg| { + var tag = arg.type.tag; + if (tag == .Pointer or tag == .Struct) { + tag = .Int32; + } + const valtype_str: []const u8 = try tag.toValtype(name); + try writer.print("{s}, ", .{valtype_str}); + } + + try writer.writeAll("};\n"); + } + + break :blk params_str.items; + }; + + const return_types: []const u8 = blk: { + if (num_returns == 0) { + break :blk "\t\toc_wasm_valtype returnTypes[1];\n\n"; + } else { + const return_str = try binding.ret.tag.toValtype(name); + break :blk try std.fmt.allocPrint(opts.arena, "\t\toc_wasm_valtype returnTypes[] = {{ {s} }};\n\n", .{return_str}); + } + }; + + try host.writeAll("\t{\n"); + try host.print("{s}", .{param_types}); + try host.print("{s}", .{return_types}); + try host.print("\t\toc_wasm_binding binding = {{0}};\n", .{}); // double {{ }} escapes the zig format specifier to be = {0} + try host.print("\t\tbinding.importName = OC_STR8(\"{s}\");\n", .{name}); + try host.print("\t\tbinding.proc = {s}_stub;\n", .{binding.cname}); + try host.print("\t\tbinding.countParams = {};\n", .{num_args}); + try host.print("\t\tbinding.countReturns = {};\n", .{num_returns}); + try host.print("\t\tbinding.params = paramTypes;\n", .{}); + try host.print("\t\tbinding.returns = returnTypes;\n", .{}); + try host.print("\t\tstatus = oc_wasm_add_binding(wasm, &binding);\n", .{}); + try host.print("\t\tif(oc_wasm_status_is_fail(status))\n", .{}); + try host.writeAll("\t\t{\n"); + try host.print("\t\t\toc_log_error(\"Couldn't link function {s} (%s)\\n\", oc_wasm_status_str8(status).ptr);\n", .{name}); + try host.writeAll("\t\t\tret = -1;\n"); + try host.writeAll("\t\t}\n"); + try host.writeAll("\t}\n\n"); + } + + try host.writeAll("\treturn(ret);\n}\n"); + + return GeneratedBindings{ + .host = bindings_host_array.items, + .guest = bindings_guest_array.items, + }; +} + +fn writeBindings(path: []const u8, data: []const u8) !void { + const cwd = std.fs.cwd(); + + if (std.fs.path.dirname(path)) |dir_path| { + cwd.makeDir(dir_path) catch |e| { + if (e != error.PathAlreadyExists) { + return e; + } + }; + } + + const file: std.fs.File = try cwd.createFile(path, .{ .read = false, .truncate = true }); + defer file.close(); + try file.writeAll(data); +} + +pub fn main() !void { + var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator); + defer arena.deinit(); + const allocator: std.mem.Allocator = arena.allocator(); + + const args: []const [:0]u8 = try std.process.argsAlloc(allocator); + defer std.process.argsFree(allocator, args); + + const opts = try Options.parse(args, allocator); + + const bindings_json: []const u8 = std.fs.cwd().readFileAlloc(opts.arena, opts.spec_path, MAX_FILE_SIZE) catch |e| { + std.log.err("Failed to read bindings spec file from path {s}: {}", .{ opts.spec_path, e }); + return e; + }; + + const bindings_untyped = std.json.parseFromSliceLeaky([]BindingUntyped, opts.arena, bindings_json, .{}) catch |e| { + std.log.err("Failed to parse json. Was the json malformed, or does the binding generator need to be updated? Error: {}", .{e}); + return e; + }; + + const bindings: []const Binding = try Binding.fromUntyped(opts, bindings_untyped); + const generated: GeneratedBindings = try generateBindings(opts, bindings); + + std.debug.assert(generated.host.len > 0); + try writeBindings(opts.bindings_path, generated.host); + + if (generated.guest.len > 0) { + try writeBindings(opts.guest_stubs_path.?, generated.guest); + } +} + +test "serialization with nulls works" { + var arena = std.heap.ArenaAllocator.init(std.testing.allocator); + defer arena.deinit(); + const allocator: std.mem.Allocator = arena.allocator(); + + const Test1 = struct { + foo: u32 = 0, + bar: ?u32 = null, + }; + + const json = + \\[ + \\ { + \\ "foo": 1, + \\ "bar": 2 + \\ }, + \\ { + \\ "foo": 3 + \\ } + \\] + ; + const results = try std.json.parseFromSliceLeaky([]Test1, allocator, json, .{}); + try std.testing.expect(results.len == 2); + try std.testing.expect(results[0].foo == 1); + try std.testing.expect(results[0].bar != null); + try std.testing.expect(results[0].bar == 2); + try std.testing.expect(results[1].foo == 3); + try std.testing.expect(results[1].bar == null); +} diff --git a/src/build/build_dependencies.zig b/src/build/build_dependencies.zig new file mode 100644 index 00000000..6ffc5b99 --- /dev/null +++ b/src/build/build_dependencies.zig @@ -0,0 +1,689 @@ +// This is a zig program run as part of the orca build process to build angle and dawn +// libraries from source. + +const std = @import("std"); +const builtin = @import("builtin"); + +const MAX_FILE_SIZE = 1024 * 1024 * 128; + +const Lib = enum { + Angle, + Dawn, + + fn toStr(lib: Lib) []const u8 { + return switch (lib) { + .Angle => "angle", + .Dawn => "dawn", + }; + } +}; + +const Options = struct { + arena: std.mem.Allocator, + lib: Lib, + commit_sha: []const u8, + check_only: bool, + optimize: std.builtin.OptimizeMode, + paths: struct { + python: []const u8, + cmake: []const u8, + intermediate_dir: []const u8, + output_dir: []const u8, + }, + + fn parse(args: []const [:0]const u8, arena: std.mem.Allocator) !Options { + var lib: ?Lib = null; + var commit_sha: ?[]const u8 = null; + var check_only: bool = false; + var optimize: std.builtin.OptimizeMode = .ReleaseFast; + + var python: ?[]const u8 = null; + var cmake: ?[]const u8 = null; + var intermediate_dir: ?[]const u8 = null; + + for (args, 0..) |raw_arg, i| { + if (i == 0) { + continue; + } + + var splitIter = std.mem.splitScalar(u8, raw_arg, '='); + const arg: []const u8 = splitIter.next().?; + if (std.mem.eql(u8, arg, "--lib")) { + if (splitIter.next()) |lib_str| { + if (std.mem.eql(u8, lib_str, "angle")) { + lib = .Angle; + } else if (std.mem.eql(u8, lib_str, "dawn")) { + lib = .Dawn; + } else { + return error.InvalidArgument; + } + } else { + return error.InvalidArgument; + } + } else if (std.mem.eql(u8, arg, "--sha")) { + commit_sha = splitIter.next(); + } else if (std.mem.eql(u8, arg, "--check")) { + check_only = true; + } else if (std.mem.eql(u8, arg, "--debug")) { + optimize = .Debug; + } else if (std.mem.eql(u8, arg, "--python")) { + python = splitIter.next(); + } else if (std.mem.eql(u8, arg, "--cmake")) { + cmake = splitIter.next(); + } else if (std.mem.eql(u8, arg, "--intermediate")) { + intermediate_dir = splitIter.next(); + } + + // logic above should have consumed all tokens, if any are left it's an error + if (splitIter.next()) |last| { + std.log.err("Unexpected part of arg: {s}", .{last}); + return error.InvalidArgument; + } + } + + var missing_arg: ?[]const u8 = null; + if (lib == null) { + missing_arg = "lib"; + } else if (commit_sha == null) { + missing_arg = "sha"; + } else if (python == null) { + if (builtin.os.tag == .windows) { + missing_arg = "python"; + } else { + missing_arg = "python3"; + } + } else if (cmake == null) { + missing_arg = "cmake"; + } else if (intermediate_dir == null) { + missing_arg = "intermediate"; + } + + if (missing_arg) |arg| { + std.log.err("Missing required arg: {s}\n", .{arg}); + return error.MissingRequiredArgument; + } + + var bad_absolute_path: ?[]const u8 = null; + if (std.fs.path.isAbsolute(intermediate_dir.?) == false) { + bad_absolute_path = intermediate_dir; + } + + if (bad_absolute_path) |path| { + std.log.err("Path {s} must be absolute", .{path}); + } + + const output_folder: []const u8 = try std.fmt.allocPrint(arena, "{s}.out", .{lib.?.toStr()}); + const output_dir: []const u8 = try std.fs.path.join(arena, &.{ intermediate_dir.?, output_folder }); + + return .{ + .arena = arena, + .lib = lib.?, + .commit_sha = commit_sha.?, + .check_only = check_only, + .optimize = optimize, + .paths = .{ + .python = python.?, + .cmake = cmake.?, + .intermediate_dir = intermediate_dir.?, + .output_dir = output_dir, + }, + }; + } +}; + +const Sort = struct { + fn lessThanString(_: void, lhs: []const u8, rhs: []const u8) bool { + return std.mem.lessThan(u8, lhs, rhs); + } +}; + +fn exec(arena: std.mem.Allocator, argv: []const []const u8, cwd: []const u8, env: *const std.process.EnvMap) !void { + var log_msg = std.ArrayList(u8).init(arena); + var log_writer = log_msg.writer(); + try log_writer.print("running: ", .{}); + for (argv) |arg| { + try log_writer.print("{s} ", .{arg}); + } + try log_writer.print(" in dir {s}", .{cwd}); + std.log.info("{s}\n", .{log_msg.items}); + + var process = std.process.Child.init(argv, arena); + process.stdin_behavior = .Ignore; + process.cwd = cwd; + process.env_map = env; + process.stdout_behavior = .Inherit; + process.stderr_behavior = .Inherit; + + try process.spawn(); + + const term = try process.wait(); + + switch (term) { + .Exited => |v| { + if (v != 0) { + std.log.err("process {s} exited with nonzero exit code {}.", .{ argv[0], v }); + return error.NonZeroExitCode; + } + }, + else => { + std.log.err("process {s} exited abnormally.", .{argv[0]}); + return error.AbnormalExit; + }, + } + + std.debug.print("\n", .{}); +} + +fn execShell(arena: std.mem.Allocator, argv: []const []const u8, cwd: []const u8, env: *const std.process.EnvMap) !void { + var final_args = std.ArrayList([]const u8).init(arena); + if (builtin.os.tag == .windows) { + try final_args.append("cmd.exe"); + try final_args.append("/c"); + try final_args.append(try std.mem.join(arena, "", &.{ argv[0], ".bat" })); + try final_args.appendSlice(argv[1..]); + } else { + try final_args.appendSlice(argv); + } + try exec(arena, final_args.items, cwd, env); +} + +fn pathExists(dir: std.fs.Dir, path: []const u8) std.fs.Dir.AccessError!bool { + dir.access(path, .{}) catch |e| { + if (e == std.fs.Dir.AccessError.FileNotFound) { + return false; + } else { + return e; + } + }; + + return true; +} + +fn copyFolder(allocator: std.mem.Allocator, dest: []const u8, src: []const u8) !void { + std.log.info("copying '{s}' to '{s}'", .{ src, dest }); + + const cwd = std.fs.cwd(); + try cwd.makePath(dest); + + const src_dir: std.fs.Dir = try cwd.openDir(src, .{ .iterate = true }); + const dest_dir: std.fs.Dir = try cwd.openDir(dest, .{ .iterate = true }); + + var src_walker = try src_dir.walk(allocator); + while (try src_walker.next()) |src_entry| { + // std.debug.print("\t{s}\n", .{src_entry.path}); + _ = switch (src_entry.kind) { + .directory => try dest_dir.makePath(src_entry.path), + .file => try src_dir.updateFile(src_entry.path, dest_dir, src_entry.path, .{}), + else => {}, + }; + } +} + +const CommitStamp = struct { + commit: []const u8, + + fn write(opts: *const Options, path: []const u8) !void { + std.log.info("writing commit file to {s}", .{path}); + + const cwd = std.fs.cwd(); + + var json_buffer = std.ArrayList(u8).init(opts.arena); + var writer = std.json.writeStream(json_buffer.writer(), .{ .whitespace = .indent_4 }); + defer writer.deinit(); + + try writer.beginObject(); + { + try writer.objectField("commit"); + try writer.write(opts.commit_sha); + } + try writer.endObject(); + + try cwd.writeFile(.{ + .sub_path = path, + .data = json_buffer.items, + .flags = .{}, + }); + } +}; + +const ANGLE_COMMIT_FILENAME = "angle.json"; +const DAWN_COMMIT_FILENAME = "dawn.json"; + +fn ensureDepotTools(opts: *const Options) !std.process.EnvMap { + var env: std.process.EnvMap = try std.process.getEnvMap(opts.arena); + if (builtin.os.tag == .windows) { + try env.put("DEPOT_TOOLS_WIN_TOOLCHAIN", "0"); + } else if (builtin.os.tag.isDarwin()) { + try env.put("HOMEBREW_NO_AUTO_UPDATE", "1"); + } + + const depot_tools_path = try std.fs.path.join(opts.arena, &.{ opts.paths.intermediate_dir, "depot_tools" }); + if (try pathExists(std.fs.cwd(), depot_tools_path) == false) { + std.log.info("cloning depot_tools to intermediate '{s}'...", .{opts.paths.intermediate_dir}); + try exec(opts.arena, &.{ "git", "clone", "https://chromium.googlesource.com/chromium/tools/depot_tools.git" }, opts.paths.intermediate_dir, &env); + } else { + std.log.info("depot_tools already exists, skipping clone", .{}); + } + + const key = "PATH"; + if (env.get(key)) |env_path| { + const new_path = try std.fmt.allocPrint(opts.arena, "{s}" ++ [1]u8{std.fs.path.delimiter} ++ "{s}", .{ env_path, depot_tools_path }); + try env.put(key, new_path); + // std.debug.print(">>>>> old path: {s}\n", .{env_path}); + // std.debug.print(">>>>> new path: {s}\n", .{new_path}); + } else { + try env.put(key, depot_tools_path); + } + + // macos build needs some extra help installing needed packages for gclient. First try using the python3 + // pip directly - if that doesn't work, the user is likely using brew to manage python3 packages, so try + // that approach. + if (builtin.os.tag.isDarwin()) { + std.log.info("installing python-setuptools to python environment...", .{}); + _ = execShell(opts.arena, &.{ opts.paths.python, "-m", "pip", "install", "python-setuptools" }, depot_tools_path, &env) catch { + execShell(opts.arena, &.{ "brew", "install", "python-setuptools" }, depot_tools_path, &env) catch {}; + }; + } + + return env; +} + +fn noopLog(comptime _: []const u8, _: anytype) void {} + +const ShouldLogError = enum { + LogError, + NoError, +}; + +fn isLibUpToDate(opts: *const Options, comptime log_error: ShouldLogError) bool { + const logfn = if (log_error == .LogError) &std.log.err else &noopLog; + + const commit_filename: []const u8 = switch (opts.lib) { + .Angle => ANGLE_COMMIT_FILENAME, + .Dawn => DAWN_COMMIT_FILENAME, + }; + + const commit_stamp_path = std.fs.path.join(opts.arena, &.{ opts.paths.output_dir, commit_filename }) catch @panic("OOM"); + const json_data: []const u8 = std.fs.cwd().readFileAlloc(opts.arena, commit_stamp_path, MAX_FILE_SIZE) catch |e| { + logfn("Failed to read commit file from location '{s}': {}", .{ commit_stamp_path, e }); + return false; + }; + + const loaded_stamp = std.json.parseFromSliceLeaky(CommitStamp, opts.arena, json_data, .{}) catch |e| { + logfn("Failed to parse file '{s}' json: {}. Raw json data:\n{s}\n", .{ + commit_stamp_path, + e, + json_data, + }); + return false; + }; + if (std.mem.eql(u8, loaded_stamp.commit, opts.commit_sha) == false) { + logfn("{s} doesn't match the required angle commit. expected {s}, got {s}", .{ + commit_stamp_path, + opts.commit_sha, + loaded_stamp.commit, + }); + return false; + } + + return true; +} + +fn buildAngle(opts: *const Options) !void { + if (isLibUpToDate(opts, .NoError)) { + // std.log.info("angle is up to date - no rebuild needed.\n", .{}); + return; + } else if (opts.check_only) { + const msg = + \\Angle files are not present or don't match required commit. + \\Angle commit: {s} + \\ + \\You can build the required files by running 'zig build angle' + \\ + \\Alternatively you can trigger a CI run to build the binaries on github: + \\ * For Windows, go to https://github.com/orca-app/orca/actions/workflows/build-angle-win.yaml + \\ * For macOS, go to https://github.com/orca-app/orca/actions/workflows/build-angle-mac.yaml + \\ * Click on \"Run workflow\" to tigger a new run, or download artifacts from a previous run + \\ * Put the contents of the artifacts folder in './build/angle.out' + ; + std.log.err(msg, .{opts.commit_sha}); + return error.AngleOutOfDate; + } + + const cwd = std.fs.cwd(); + try cwd.makePath(opts.paths.intermediate_dir); + + std.log.info("angle is out of date - rebuilding", .{}); + + var env: std.process.EnvMap = try ensureDepotTools(opts); + defer env.deinit(); + + const src_path = try std.fs.path.join(opts.arena, &.{ opts.paths.intermediate_dir, opts.lib.toStr() }); + + if (try pathExists(std.fs.cwd(), src_path) == false) { + try exec( + opts.arena, + &.{ + "git", + "clone", + "--no-tags", + "--single-branch", + "https://chromium.googlesource.com/angle/angle", + src_path, + }, + opts.paths.intermediate_dir, + &env, + ); + } + + try exec(opts.arena, &.{ "git", "fetch", "--no-tags" }, src_path, &env); + try exec(opts.arena, &.{ "git", "reset", "--hard", opts.commit_sha }, src_path, &env); + try exec(opts.arena, &.{ opts.paths.python, "scripts/bootstrap.py" }, src_path, &env); + + const depot_tools_path = try std.fs.path.join(opts.arena, &.{ opts.paths.intermediate_dir, "depot_tools" }); + + const gclient_path = try std.fs.path.join(opts.arena, &.{ depot_tools_path, "gclient" }); + try execShell(opts.arena, &.{ gclient_path, "sync" }, src_path, &env); + + const optimize_str = if (opts.optimize == .Debug) "Debug" else "Release"; + const is_debug_str = if (opts.optimize == .Debug) "is_debug=true" else "is_debug=false"; + + var gn_args_list = std.ArrayList([]const u8).init(opts.arena); + try gn_args_list.append("angle_build_all=false"); + try gn_args_list.append("angle_build_tests=false"); + try gn_args_list.append("is_component_build=false"); + try gn_args_list.append(is_debug_str); + + if (builtin.os.tag == .windows) { + try gn_args_list.append("angle_enable_d3d9=false"); + try gn_args_list.append("angle_enable_gl=false"); + try gn_args_list.append("angle_enable_vulkan=false"); + try gn_args_list.append("angle_enable_null=false"); + try gn_args_list.append("angle_has_frame_capture=false"); + } else { + //NOTE(martin): oddly enough, this is needed to avoid deprecation errors when _not_ using OpenGL, + // because angle uses some CGL APIs to detect GPUs. + try gn_args_list.append("treat_warnings_as_errors=false"); + try gn_args_list.append("angle_enable_metal=true"); + try gn_args_list.append("angle_enable_gl=false"); + try gn_args_list.append("angle_enable_vulkan=false"); + try gn_args_list.append("angle_enable_null=false"); + } + const gn_all_args = try std.mem.join(opts.arena, " ", gn_args_list.items); + + const gn_args: []const u8 = try std.fmt.allocPrint(opts.arena, "--args={s}", .{gn_all_args}); + + const optimize_output_path = try std.fs.path.join(opts.arena, &.{ src_path, "out", optimize_str }); + try cwd.makePath(optimize_output_path); + + const gn_path = try std.fs.path.join(opts.arena, &.{ depot_tools_path, "gn" }); + try execShell(opts.arena, &.{ gn_path, "gen", optimize_output_path, gn_args }, src_path, &env); + + const autoninja_path = try std.fs.path.join(opts.arena, &.{ depot_tools_path, "autoninja" }); + try execShell(opts.arena, &.{ autoninja_path, "-C", optimize_output_path, "libEGL", "libGLESv2" }, src_path, &env); + + // copy artifacts to output dir + { + const join = std.fs.path.join; + const a = opts.arena; + const output_dir = opts.paths.output_dir; + + const bin_path = try join(a, &.{ output_dir, "bin" }); + + const inc_folders: []const []const u8 = &.{ + "include/KHR", + "include/EGL", + "include/GLES", + "include/GLES2", + "include/GLES3", + }; + + for (inc_folders) |folder| { + const src_include_path = try join(a, &.{ src_path, folder }); + const dest_include_path = try join(a, &.{ output_dir, folder }); + try cwd.deleteTree(dest_include_path); + try cwd.makePath(dest_include_path); + _ = try copyFolder(a, dest_include_path, src_include_path); + } + + var libs = std.ArrayList([]const u8).init(a); + if (builtin.os.tag == .windows) { + try libs.append("libEGL.dll"); + try libs.append("libEGL.dll.lib"); + try libs.append("libGLESv2.dll"); + try libs.append("libGLESv2.dll.lib"); + } else { + try libs.append("libEGL.dylib"); + try libs.append("libGLESv2.dylib"); + } + + var bin_src_dir: std.fs.Dir = try cwd.openDir(optimize_output_path, .{}); + + try cwd.deleteTree(bin_path); + try cwd.makePath(bin_path); + + const bin_dest_dir: std.fs.Dir = try cwd.openDir(bin_path, .{}); + for (libs.items) |filename| { + _ = bin_src_dir.updateFile(filename, bin_dest_dir, filename, .{}) catch |e| { + if (e == error.FileNotFound) { + const source_path = try std.fs.path.join(opts.arena, &.{ optimize_output_path, filename }); + std.log.err("Failed to copy {s} - not found.", .{source_path}); + return e; + } + }; + } + + if (builtin.os.tag == .windows) { + const windows_sdk = std.zig.WindowsSdk.find(opts.arena, builtin.cpu.arch) catch |e| { + std.log.err("Failed to find Windows SDK. Do you have the Windows 10 SDK installed?", .{}); + return e; + }; + + var windows_sdk_path: []const u8 = ""; + if (windows_sdk.windows10sdk) |install| { + windows_sdk_path = install.path; + } else if (windows_sdk.windows81sdk) |install| { + windows_sdk_path = install.path; + } else { + std.log.err("Failed to find Windows SDK. Do you have the Windows 10 SDK installed?", .{}); + return error.FailedToFindWindowsSdk; + } + + const src_d3dcompiler_path = try std.fs.path.join(opts.arena, &.{ + windows_sdk_path, + "Redist", + "D3D", + "x64", + }); + var src_d3dcompiler_dir: std.fs.Dir = try cwd.openDir(src_d3dcompiler_path, .{}); + _ = try src_d3dcompiler_dir.updateFile("d3dcompiler_47.dll", bin_dest_dir, "d3dcompiler_47.dll", .{}); + } + } + + // write commit stamp file + const commit_stamp_path = try std.fs.path.join(opts.arena, &.{ opts.paths.output_dir, ANGLE_COMMIT_FILENAME }); + try CommitStamp.write(opts, commit_stamp_path); + + std.log.info("angle build successful", .{}); +} + +fn buildDawn(opts: *const Options) !void { + if (isLibUpToDate(opts, .NoError)) { + // std.log.info("dawn is up to date - no rebuild needed.\n", .{}); + return; + } else if (opts.check_only) { + const msg = + \\Dawn files are not present or don't match required commit. + \\Dawn commit: {s} + \\You can build the required files by running 'zig build dawn' + \\ + \\Alternatively you can trigger a CI run to build the binaries on github: + \\ * For Windows, go to https://github.com/orca-app/orca/actions/workflows/build-dawn-win.yaml + \\ * For macOS, go to https://github.com/orca-app/orca/actions/workflows/build-dawn-mac.yaml + \\ * Click on "Run workflow" to tigger a new run, or download artifacts from a previous run + \\ * Put the contents of the artifacts folder in './build/dawn.out' + ; + std.log.err(msg, .{opts.commit_sha}); + + return error.DawnOutOfDate; + } + + std.log.info("dawn is out of date - rebuilding", .{}); + + var env: std.process.EnvMap = try ensureDepotTools(opts); + defer env.deinit(); + + const cwd = std.fs.cwd(); + + const src_path = try std.fs.path.join(opts.arena, &.{ opts.paths.intermediate_dir, opts.lib.toStr() }); + + if (try pathExists(std.fs.cwd(), src_path) == false) { + try exec(opts.arena, &.{ + "git", + "clone", + "--no-tags", + "--single-branch", + "https://dawn.googlesource.com/dawn", + src_path, + }, opts.paths.intermediate_dir, &env); + } + try exec(opts.arena, &.{ "git", "restore", "." }, src_path, &env); + try exec(opts.arena, &.{ "git", "pull", "--force", "--no-tags" }, src_path, &env); + try exec(opts.arena, &.{ "git", "checkout", "--force", opts.commit_sha }, src_path, &env); + + const src_dir = try cwd.openDir(src_path, .{}); + _ = try src_dir.updateFile("scripts/standalone.gclient", src_dir, ".gclient", .{}); + + const depot_tools_path = try std.fs.path.join(opts.arena, &.{ opts.paths.intermediate_dir, "depot_tools" }); + // const gclient_entrypoint = if (builtin.os.tag == .windows) "gclient" else "gclient"; + const gclient_path = try std.fs.path.join(opts.arena, &.{ depot_tools_path, "gclient" }); + try execShell(opts.arena, &.{ gclient_path, "sync" }, src_path, &env); + + { + const cmake_patch = + \\add_library(webgpu SHARED ${DAWN_PLACEHOLDER_FILE}) + \\common_compile_options(webgpu) + \\target_link_libraries(webgpu PRIVATE dawn_native) + \\target_link_libraries(webgpu PUBLIC dawn_headers) + \\target_compile_definitions(webgpu PRIVATE WGPU_IMPLEMENTATION WGPU_SHARED_LIBRARY) + \\target_sources(webgpu PRIVATE ${WEBGPU_DAWN_NATIVE_PROC_GEN}) + \\ + ; + const cmake_list_path = try std.fs.path.join(opts.arena, &.{ src_path, "src/dawn/native/CMakeLists.txt" }); + const cmake_list_file = try cwd.createFile(cmake_list_path, .{ + .read = false, + .truncate = false, + }); + defer cmake_list_file.close(); + + try cmake_list_file.seekFromEnd(0); + try cmake_list_file.writeAll(cmake_patch); + } + + const diff_file_path = try std.fs.path.join(opts.arena, &.{ src_path, "../../deps/dawn-d3d12-transparent.diff" }); + try exec(opts.arena, &.{ "git", "apply", "-v", diff_file_path }, src_path, &env); + + if (builtin.os.tag != .windows) { + try exec(opts.arena, &.{ "chmod", "+x", opts.paths.cmake }, src_path, &env); + } + + const optimize_str = if (opts.optimize == .Debug) "Debug" else "Release"; + const cmake_build_type = try std.fmt.allocPrint(opts.arena, "CMAKE_BUILD_TYPE={s}", .{optimize_str}); + + var cmake_args = std.ArrayList([]const u8).init(opts.arena); + // zig fmt: off + try cmake_args.appendSlice(&.{ + opts.paths.cmake, + "-S", "dawn", + "-B", "dawn.build", + "-D", cmake_build_type, + "-D", "CMAKE_POLICY_DEFAULT_CMP0091=NEW", + "-D", "BUILD_SHARED_LIBS=OFF", + "-D", "BUILD_SAMPLES=ON", + "-D", "DAWN_BUILD_SAMPLES=ON", + "-D", "TINT_BUILD_SAMPLES=OFF", + "-D", "TINT_BUILD_DOCS=OFF", + "-D", "TINT_BUILD_TESTS=OFF", + }); + // zig fmt: on + + // zig fmt: off + if (builtin.os.tag == .windows) { + try cmake_args.appendSlice(&.{ + "-D", "DAWN_ENABLE_D3D12=ON", + "-D", "DAWN_ENABLE_D3D11=OFF", + "-D", "DAWN_ENABLE_METAL=OFF", + "-D", "DAWN_ENABLE_NULL=OFF", + "-D", "DAWN_ENABLE_DESKTOP_GL=OFF", + "-D", "DAWN_ENABLE_OPENGLES=OFF", + "-D", "DAWN_ENABLE_VULKAN=OFF" + }); + } else { + try cmake_args.appendSlice(&.{ + "-D", "DAWN_ENABLE_METAL=ON", + "-D", "DAWN_ENABLE_NULL=OFF", + "-D", "DAWN_ENABLE_DESKTOP_GL=OFF", + "-D", "DAWN_ENABLE_OPENGLES=OFF", + "-D", "DAWN_ENABLE_VULKAN=OFF" + }); + } + // zig fmt: on + + try exec(opts.arena, cmake_args.items, opts.paths.intermediate_dir, &env); + + // TODO allow user customization of number of parallel jobs + // zig fmt: off + const cmake_build_args = &.{ + opts.paths.cmake, + "--build", "dawn.build", + "--config", optimize_str, + "--target", "webgpu", + "--parallel", + }; + // zig fmt: on + try exec(opts.arena, cmake_build_args, opts.paths.intermediate_dir, &env); + + // copy aftifacts to output dir + { + try cwd.makePath(opts.paths.output_dir); + + const build_path = try std.fs.path.join(opts.arena, &.{ opts.paths.intermediate_dir, "dawn.build" }); + const build_dir = try cwd.openDir(build_path, .{}); + const dest_dir = try cwd.openDir(opts.paths.output_dir, .{}); + + try dest_dir.makePath("include"); + try dest_dir.makePath("bin"); + + _ = try build_dir.updateFile("gen/include/dawn/webgpu.h", dest_dir, "include/webgpu.h", .{}); + + if (builtin.os.tag == .windows) { + const dll_path_src = try std.fs.path.join(opts.arena, &.{ optimize_str, "webgpu.dll" }); + const dll_path_dest = try std.fs.path.join(opts.arena, &.{ "bin", "webgpu.dll" }); + _ = try build_dir.updateFile(dll_path_src, dest_dir, dll_path_dest, .{}); + + const lib_path_src = try std.fs.path.join(opts.arena, &.{ "src", "dawn", "native", optimize_str, "webgpu.lib" }); + const lib_path_dest = try std.fs.path.join(opts.arena, &.{ "bin", "webgpu.lib" }); + _ = try build_dir.updateFile(lib_path_src, dest_dir, lib_path_dest, .{}); + } else { + _ = try build_dir.updateFile("src/dawn/native/libwebgpu.dylib", dest_dir, "bin/libwebgpu.dylib", .{}); + } + + const commit_stamp_path = try std.fs.path.join(opts.arena, &.{ opts.paths.output_dir, DAWN_COMMIT_FILENAME }); + try CommitStamp.write(opts, commit_stamp_path); + } +} + +pub fn main() !void { + var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator); + defer arena.deinit(); + const allocator: std.mem.Allocator = arena.allocator(); + + const args: []const [:0]u8 = try std.process.argsAlloc(allocator); + defer std.process.argsFree(allocator, args); + + const opts = try Options.parse(args, allocator); + switch (opts.lib) { + .Angle => try buildAngle(&opts), + .Dawn => try buildDawn(&opts), + } +} diff --git a/src/build/echo.zig b/src/build/echo.zig new file mode 100644 index 00000000..c2c02b1b --- /dev/null +++ b/src/build/echo.zig @@ -0,0 +1,24 @@ +const std = @import("std"); + +pub fn main() !void { + var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator); + defer arena.deinit(); + const allocator: std.mem.Allocator = arena.allocator(); + + const args: []const [:0]u8 = try std.process.argsAlloc(allocator); + defer std.process.argsFree(allocator, args); + + const stdout_file = std.io.getStdOut().writer(); + var bw = std.io.bufferedWriter(stdout_file); + + const stdout = bw.writer(); + + for (args, 0..) |arg, i| { + if (i == 0) { + continue; + } + try stdout.print("{s}", .{arg}); + } + + try bw.flush(); +} diff --git a/src/build/gen_header_from_files.zig b/src/build/gen_header_from_files.zig new file mode 100644 index 00000000..e947fd81 --- /dev/null +++ b/src/build/gen_header_from_files.zig @@ -0,0 +1,141 @@ +// This is a zig program run as part of the orca build process to generate a header file +// embedding strings of passed-along text files. + +const std = @import("std"); + +const Options = struct { + arena: std.mem.Allocator, + out_path: []const u8, + in_paths: []const []const u8, + namespace: []const u8, + root: []const u8, + + fn parse(args: []const [:0]const u8, arena: std.mem.Allocator) !Options { + var out_path: ?[]const u8 = null; + var in_paths = std.ArrayList([]const u8).init(arena); + var namespace: ?[]const u8 = null; + var root: ?[]const u8 = null; + + for (args, 0..) |raw_arg, i| { + if (i == 0) { + continue; + } + + var split_iter = std.mem.splitScalar(u8, raw_arg, '='); + const arg: []const u8 = split_iter.next().?; + if (std.mem.eql(u8, arg, "--output")) { + out_path = split_iter.next(); + } else if (std.mem.eql(u8, arg, "--file")) { + if (split_iter.next()) |path| { + try in_paths.append(path); + } + } else if (std.mem.eql(u8, arg, "--namespace")) { + namespace = split_iter.next(); + } else if (std.mem.eql(u8, arg, "--root")) { + root = split_iter.next(); + } + } + + var missing_arg: ?[]const u8 = null; + if (out_path == null) { + missing_arg = "output"; + } else if (in_paths.items.len == 0) { + missing_arg = "file"; + } else if (namespace == null) { + missing_arg = "namespace"; + } else if (root == null) { + missing_arg = "root"; + } + + if (missing_arg) |arg| { + std.log.err("Missing required arg: {s}", .{arg}); + return error.MissingRequiredArgument; + } + + return Options{ + .arena = arena, + .out_path = out_path.?, + .in_paths = in_paths.items, + .namespace = namespace.?, + .root = root.?, + }; + } +}; + +fn basenameNoExtension(path: []const u8) []const u8 { + const filename: []const u8 = std.fs.path.basename(path); + if (std.mem.lastIndexOf(u8, filename, ".")) |index| { + return filename[0..index]; + } + return filename; +} + +pub fn main() !void { + var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator); + defer arena.deinit(); + const allocator: std.mem.Allocator = arena.allocator(); + + const args: []const [:0]u8 = try std.process.argsAlloc(allocator); + defer std.process.argsFree(allocator, args); + + const opts = try Options.parse(args, allocator); + + var concat = std.ArrayList(u8).init(allocator); + defer concat.deinit(); + + const filename: []const u8 = std.fs.path.basename(opts.out_path); + const filename_no_ext: []const u8 = basenameNoExtension(filename); + const guard_name: []u8 = try allocator.dupe(u8, filename_no_ext); + for (guard_name) |*c| { + c.* = std.ascii.toUpper(c.*); + } + + var writer = concat.writer(); + + try writer.print("/*********************************************************************\n", .{}); + try writer.print("*\n", .{}); + try writer.print("*\tfile: {s}\n", .{filename}); + try writer.print("*\tnote: string literals auto-generated by gen_header_from_files.zig\n", .{}); + // writer.print("*\tdate: {s}\n", .{datetime.now().strftime("%d/%m%Y")}) // TODO datetime + try writer.print("*\n", .{}); + try writer.print("**********************************************************************/\n", .{}); + + try writer.print("#ifndef __{s}_H__\n", .{guard_name}); + try writer.print("#define __{s}_H__\n", .{guard_name}); + try writer.print("\n\n", .{}); + + const cwd = std.fs.cwd(); + for (opts.in_paths) |input_path| { + const file_contents = try cwd.readFileAlloc(allocator, input_path, 1024 * 1024 * 128); + defer allocator.free(file_contents); + + const input_path_filename: []const u8 = std.fs.path.basename(input_path); + const input_path_relative: []u8 = try allocator.dupe(u8, std.mem.trimLeft(u8, input_path, opts.root)); + std.mem.replaceScalar(u8, input_path_relative, '\\', '/'); + const input_path_no_ext: []const u8 = basenameNoExtension(input_path_filename); + + try writer.print("//NOTE: string imported from {s}\n", .{input_path_relative}); + try writer.print("const char* {s}{s} = \n", .{ opts.namespace, input_path_no_ext }); + + var split_iter = std.mem.tokenizeAny(u8, file_contents, "\r\n"); + while (split_iter.next()) |line| { + const line_fmt = + \\"{s}\n" + ; + try writer.print(line_fmt, .{line}); + if (split_iter.peek() != null) { + try writer.print("\n", .{}); + } + } + + try writer.print(";\n\n", .{}); + } + + try writer.print("#endif // __{s}_H__\n", .{guard_name}); + + try cwd.writeFile(.{ + .sub_path = opts.out_path, + .data = concat.items, + .flags = .{ .truncate = true }, + }); +} diff --git a/src/build/package_sdk.zig b/src/build/package_sdk.zig new file mode 100644 index 00000000..2be06cdd --- /dev/null +++ b/src/build/package_sdk.zig @@ -0,0 +1,320 @@ +// This is a zig program run as part of the orca build process to copy all appropriate +// build artifacts to an output location. Optionally the user can package the SDK +// for installation to the system orca path. + +const std = @import("std"); +const builtin = @import("builtin"); + +fn trimWhitespace(str0: []const u8) []const u8 { + const trim = std.mem.trim; + const str1 = trim(u8, str0, "\n"); + const str2 = trim(u8, str1, "\r"); + const str3 = trim(u8, str2, " "); + return str3; +} + +fn findGitVersion(arena: std.mem.Allocator) ![]const u8 { + const result: std.process.Child.RunResult = try std.process.Child.run(.{ + .allocator = arena, + .argv = &.{ "git", "rev-parse", "--short", "HEAD" }, + .cwd_dir = std.fs.cwd(), + }); + + switch (result.term) { + .Exited => |exit_code| { + if (exit_code != 0) { + std.log.err("git rev-parse failed with nonzero exit code {}", .{exit_code}); + return error.GitRevParseFailed; + } + }, + else => { + std.log.err("git rev-parse failed. stderr: {s}", .{result.stderr}); + return error.GitRevParseFailed; + }, + } + + const output = trimWhitespace(result.stdout); + if (output.len == 0) { + std.log.err("git rev-parse output was empty", .{}); + return error.GitRevParseUnexpectedOutput; + } + + return output; +} + +fn copyFolder( + allocator: std.mem.Allocator, + dest: []const u8, + src: []const u8, + include_extensions: []const []const u8, + ignore_patterns: []const []const u8, +) !void { + std.log.info("copying '{s}' to '{s}'", .{ src, dest }); + + const cwd = std.fs.cwd(); + try cwd.makePath(dest); + + const src_dir: std.fs.Dir = try cwd.openDir(src, .{ .iterate = true }); + const dest_dir: std.fs.Dir = try cwd.openDir(dest, .{ .iterate = true }); + + var normalized_ignore_patterns = std.ArrayList([]u8).init(allocator); + for (ignore_patterns) |pattern| { + try normalized_ignore_patterns.append(try std.fs.path.resolve(allocator, &.{pattern})); + } + + var src_walker = try src_dir.walk(allocator); + while (try src_walker.next()) |src_entry| { + var included: bool = true; + if (src_entry.kind != .directory) { + included = include_extensions.len == 0; + const extension = std.fs.path.extension(src_entry.basename); + for (include_extensions) |included_extension| { + if (std.mem.eql(u8, extension, included_extension)) { + included = true; + break; + } + } + } + + var ignored: bool = false; + for (normalized_ignore_patterns.items) |pattern| { + if (std.mem.indexOf(u8, src_entry.path, pattern) != null) { + ignored = true; + break; + } + } + + if (included and !ignored) { + _ = switch (src_entry.kind) { + .file => try src_dir.updateFile(src_entry.path, dest_dir, src_entry.path, .{}), + else => {}, + }; + } + } +} + +const ShouldLog = enum { + False, + True, +}; + +fn findSystemOrcaDir(allocator: std.mem.Allocator, should_log: ShouldLog) ![]const u8 { + const result: std.process.Child.RunResult = std.process.Child.run(.{ + .allocator = allocator, + .argv = &.{ "orca", "install-path" }, + .cwd_dir = std.fs.cwd(), + }) catch |e| { + if (should_log == .True) { + std.log.err("Failed to run orca executable in PATH.", .{}); + } + return e; + }; + + switch (result.term) { + .Exited => |exit_code| { + if (exit_code != 0) { + if (should_log == .True) { + std.log.err("orca install-path failed with nonzero exit code {}", .{exit_code}); + } + return error.OrcaInstallPathFailed; + } + }, + else => { + if (should_log == .True) { + std.log.err("orca install-path failed. stderr: {s}", .{result.stderr}); + } + return error.OrcaInstallPathFailed; + }, + } + + const output = trimWhitespace(result.stdout); + if (output.len == 0) { + if (should_log == .True) { + std.log.err("orca install-path output was empty", .{}); + } + return error.OrcaInstallPathUnexpectedOutput; + } + + return output; +} + +const Options = struct { + arena: std.mem.Allocator, + sdk_path: []const u8, + artifacts_path: []const u8, + resources_path: []const u8, + src_path: []const u8, + version: []const u8, + is_dev_install: bool, + + fn parse(args: []const [:0]const u8, arena: std.mem.Allocator) !Options { + var sdk_path: ?[]const u8 = null; + var artifacts_path: ?[]const u8 = null; + var resources_path: ?[]const u8 = null; + var src_path: ?[]const u8 = null; + var version: ?[]const u8 = null; + var is_dev_install: bool = false; + + for (args, 0..) |raw_arg, i| { + if (i == 0) { + continue; + } + + var splitIter = std.mem.splitScalar(u8, raw_arg, '='); + const arg: []const u8 = splitIter.next().?; + if (std.mem.eql(u8, arg, "--sdk-path")) { + sdk_path = splitIter.next(); + } else if (std.mem.eql(u8, arg, "--artifacts-path")) { + artifacts_path = splitIter.next(); + } else if (std.mem.eql(u8, arg, "--resources-path")) { + resources_path = splitIter.next(); + } else if (std.mem.eql(u8, arg, "--src-path")) { + src_path = splitIter.next(); + } else if (std.mem.eql(u8, arg, "--version")) { + version = splitIter.next(); + } else if (std.mem.eql(u8, arg, "--dev-install")) { + is_dev_install = true; + } + } + + var missing_arg: ?[]const u8 = null; + if (sdk_path == null and is_dev_install == false) { + missing_arg = "sdk-path"; + } else if (artifacts_path == null) { + missing_arg = "artifacts-path"; + } else if (resources_path == null) { + missing_arg = "resources-path"; + } else if (src_path == null) { + missing_arg = "src-path"; + } + + if (missing_arg) |arg| { + std.log.err("Missing required arg: {s}", .{arg}); + return error.MissingRequiredArgument; + } + + if (version == null or std.mem.eql(u8, version.?, "")) { + const git_version = try findGitVersion(arena); + version = try std.mem.join(arena, "", &.{ "dev-", git_version }); + std.log.info("No version supplied, using version derived from git sha: {s}", .{version.?}); + } + + if (is_dev_install) { + if (sdk_path == null) { + const orca_dir: []const u8 = findSystemOrcaDir(arena, ShouldLog.False) catch |e| { + std.log.err( + \\When performing a dev install, you must either have a version of the orca CLI tool installed on your + \\PATH so orca knows where to install itself, or you must explicitly supply zig build with the arg: + \\ -Dsdk-path= + \\ + , .{}); + return e; + }; + sdk_path = try std.fs.path.join(arena, &.{ orca_dir, version.? }); + } else { + sdk_path = try std.fs.path.join(arena, &.{ sdk_path.?, version.? }); + } + } + + return Options{ + .arena = arena, + .sdk_path = sdk_path.?, + .artifacts_path = artifacts_path.?, + .resources_path = resources_path.?, + .src_path = src_path.?, + .version = version.?, + .is_dev_install = is_dev_install, + }; + } +}; + +pub fn main() !void { + var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator); + defer arena.deinit(); + const allocator: std.mem.Allocator = arena.allocator(); + + const args: []const [:0]u8 = try std.process.argsAlloc(allocator); + defer std.process.argsFree(allocator, args); + + const opts = try Options.parse(args, allocator); + + std.fs.deleteTreeAbsolute(opts.sdk_path) catch {}; + + const src_paths: []const []const u8 = &.{ + try std.fs.path.join(opts.arena, &.{ opts.artifacts_path, "bin" }), + try std.fs.path.join(opts.arena, &.{ opts.artifacts_path, "orca-libc" }), + opts.resources_path, + }; + + const dest_paths: []const []const u8 = &.{ + try std.fs.path.join(opts.arena, &.{ opts.sdk_path, "bin" }), + try std.fs.path.join(opts.arena, &.{ opts.sdk_path, "orca-libc" }), + try std.fs.path.join(opts.arena, &.{ opts.sdk_path, "resources" }), + }; + + for (src_paths, dest_paths) |src, dest| { + try copyFolder(opts.arena, dest, src, &.{}, &.{}); + } + + const header_extensions: []const []const u8 = &.{ + ".h", + }; + // For copying the src directory, manually copy the curl and wasm3 dirs since we only care + // about the headers in a specific directory. Everything else is ok though + { + const dest_src_path = try std.fs.path.join(opts.arena, &.{ opts.sdk_path, "src" }); + const ignore_patterns: []const []const u8 = &.{ + "tool/", + "orca-libc/", + "wasm/", + "ext/curl/", + "ext/wasm3/", + "ext/zlib/build/", // copy all headers in zlib except zlib/build + }; + try copyFolder(opts.arena, dest_src_path, opts.src_path, header_extensions, ignore_patterns); + } + + { + const curl_src_path = try std.fs.path.resolve(opts.arena, &.{ opts.src_path, "ext/curl/include" }); + const curl_dest_path = try std.fs.path.resolve(opts.arena, &.{ opts.sdk_path, "src/ext/curl/include" }); + + try copyFolder(opts.arena, curl_dest_path, curl_src_path, header_extensions, &.{}); + } + + { + const wasm3_src_path = try std.fs.path.resolve(opts.arena, &.{ opts.src_path, "ext/wasm3/source" }); + const wasm3_dest_path = try std.fs.path.resolve(opts.arena, &.{ opts.sdk_path, "src/ext/wasm3/source" }); + try copyFolder(opts.arena, wasm3_dest_path, wasm3_src_path, header_extensions, &.{}); + } + + // dev installs need to overwrite the orca tool in case there were any changes + if (opts.is_dev_install) { + const orca_dir_path: []const u8 = if (std.fs.path.dirname(opts.sdk_path)) |orca_dir| orca_dir else opts.sdk_path; + + const src_orca_exe_name: []const u8 = if (builtin.os.tag == .windows) "orca_tool.exe" else "orca_tool"; + const src_tool_path: []const u8 = try std.fs.path.join(opts.arena, &.{ opts.artifacts_path, "bin", src_orca_exe_name }); + + const dest_orca_exe_name: []const u8 = if (builtin.os.tag == .windows) "orca.exe" else "orca"; + const dest_tool_path: []const u8 = try std.fs.path.join(opts.arena, &.{ orca_dir_path, dest_orca_exe_name }); + + std.log.info("copying '{s}' to '{s}'", .{ src_tool_path, dest_tool_path }); + const cwd = std.fs.cwd(); + const orca_dir: std.fs.Dir = try cwd.openDir(orca_dir_path, .{ .iterate = true }); + _ = try cwd.updateFile(src_tool_path, orca_dir, dest_tool_path, .{}); + + // copy pdb file as well since windows debuggers have a hard time finding the debug symbols otherwise + if (builtin.os.tag == .windows) { + const src_pdb_path: []const u8 = try std.fs.path.join(opts.arena, &.{ opts.artifacts_path, "bin", "orca_tool.pdb" }); + const dest_pdb_path: []const u8 = try std.fs.path.join(opts.arena, &.{ orca_dir_path, "orca.pdb" }); + std.log.info("copying '{s}' to '{s}'", .{ src_pdb_path, dest_pdb_path }); + _ = try cwd.updateFile(src_pdb_path, orca_dir, dest_pdb_path, .{}); + } + + try orca_dir.writeFile(.{ + .sub_path = "current_version", + .data = opts.version, + }); + } + + std.log.info("Packaged Orca SDK to {s}", .{opts.sdk_path}); +} diff --git a/src/graphics/graphics_common.c b/src/graphics/graphics_common.c index c14e86f5..ed22ccf3 100644 --- a/src/graphics/graphics_common.c +++ b/src/graphics/graphics_common.c @@ -6,7 +6,6 @@ * **************************************************************************/ -#define _USE_MATH_DEFINES //NOTE: necessary for MSVC #include "platform/platform.h" #include diff --git a/src/graphics/wgpu_renderer.c b/src/graphics/wgpu_renderer.c index 651387a3..968e215e 100644 --- a/src/graphics/wgpu_renderer.c +++ b/src/graphics/wgpu_renderer.c @@ -1527,18 +1527,26 @@ void oc_wgpu_canvas_encode_path(oc_wgpu_canvas_encoding_context* context, oc_pri { path->cmd = primitive->cmd; + //NOTE: + // We clamp boxes to int range so that they can be used to compute + // integer tile coordinate without cast overflow. Since extents + // specify top-left, bottom-right (not width/height), we can clamp + // each component separately. + + const f32 bound = (f32)(1 << 14); + path->box = (oc_vec4){ - context->pathScreenExtents.x * context->scale.x, - context->pathScreenExtents.y * context->scale.y, - context->pathScreenExtents.z * context->scale.x, - context->pathScreenExtents.w * context->scale.y, + oc_clamp(context->pathScreenExtents.x * context->scale.x, -bound, bound), + oc_clamp(context->pathScreenExtents.y * context->scale.y, -bound, bound), + oc_clamp(context->pathScreenExtents.z * context->scale.x, -bound, bound), + oc_clamp(context->pathScreenExtents.w * context->scale.y, -bound, bound), }; path->clip = (oc_vec4){ - primitive->attributes.clip.x * context->scale.x, - primitive->attributes.clip.y * context->scale.y, - (primitive->attributes.clip.x + primitive->attributes.clip.w) * context->scale.x, - (primitive->attributes.clip.y + primitive->attributes.clip.h) * context->scale.y, + oc_clamp(primitive->attributes.clip.x * context->scale.x, -bound, bound), + oc_clamp(primitive->attributes.clip.y * context->scale.y, -bound, bound), + oc_clamp((primitive->attributes.clip.x + primitive->attributes.clip.w) * context->scale.x, -bound, bound), + oc_clamp((primitive->attributes.clip.y + primitive->attributes.clip.h) * context->scale.y, -bound, bound), }; for(int i = 0; i < 4; i++) diff --git a/src/orca-libc/include/stdatomic.h b/src/orca-libc/include/stdatomic.h new file mode 100644 index 00000000..fcef0871 --- /dev/null +++ b/src/orca-libc/include/stdatomic.h @@ -0,0 +1,10 @@ +#ifndef _STDATOMIC_H +#define _STDATOMIC_H + +#ifndef __cplusplus + +#define _Atomic(type) type + +#endif // __cplusplus + +#endif // _STDATOMIC_H diff --git a/src/orca-libc/src/stdio/vfprintf.c b/src/orca-libc/src/stdio/vfprintf.c index 7da2e504..8f76f183 100644 --- a/src/orca-libc/src/stdio/vfprintf.c +++ b/src/orca-libc/src/stdio/vfprintf.c @@ -12,7 +12,7 @@ #include #ifdef __wasilibc_unmodified_upstream // Changes to optimize printf/scanf when long double isn't needed #else -#include "printscan.h" +#include #endif /* Some useful macros */ diff --git a/src/platform/win32_string_helpers.c b/src/platform/win32_string_helpers.c index 5d149640..858dd0da 100644 --- a/src/platform/win32_string_helpers.c +++ b/src/platform/win32_string_helpers.c @@ -25,7 +25,7 @@ oc_str8 oc_win32_wide_to_utf8(oc_arena* arena, oc_str16 s) { oc_str8 res = { 0 }; res.len = WideCharToMultiByte(CP_UTF8, 0, s.ptr, s.len, NULL, 0, NULL, NULL); - res.ptr = oc_arena_push_array(arena, u8, res.len + 1); + res.ptr = oc_arena_push_array(arena, char, res.len + 1); WideCharToMultiByte(CP_UTF8, 0, s.ptr, s.len, res.ptr, res.len, NULL, NULL); res.ptr[res.len] = '\0'; return (res); diff --git a/src/runtime.c b/src/runtime.c index d3d0e579..adcafd8b 100644 --- a/src/runtime.c +++ b/src/runtime.c @@ -231,7 +231,7 @@ void oc_bridge_log(oc_log_level level, if(!entry) { - char* mem = oc_arena_push(&debug->logArena, cap); + char* mem = oc_arena_push_aligned(&debug->logArena, cap, _Alignof(log_entry)); entry = (log_entry*)mem; entry->cap = cap; } diff --git a/src/tool/bundle.c b/src/tool/bundle.c index 39b09076..82c9377a 100644 --- a/src/tool/bundle.c +++ b/src/tool/bundle.c @@ -190,7 +190,7 @@ int winBundle( //----------------------------------------------------------- //NOTE: copy orca libraries //----------------------------------------------------------- - oc_str8 orcaLib = oc_path_append(a, sdkDir, OC_STR8("bin/orca.dll")); + oc_str8 orcaLib = oc_path_append(a, sdkDir, OC_STR8("bin/orca_platform.dll")); oc_str8 glesLib = oc_path_append(a, sdkDir, OC_STR8("bin/libGLESv2.dll")); oc_str8 eglLib = oc_path_append(a, sdkDir, OC_STR8("bin/libEGL.dll")); oc_str8 wgpuLib = oc_path_append(a, sdkDir, OC_STR8("bin/webgpu.dll")); @@ -290,7 +290,7 @@ int macBundle( //NOTE: copy orca runtime executable and libraries //----------------------------------------------------------- oc_str8 orcaExe = oc_path_append(a, sdkDir, OC_STR8("bin/orca_runtime")); - oc_str8 orcaLib = oc_path_append(a, sdkDir, OC_STR8("bin/liborca.dylib")); + oc_str8 orcaLib = oc_path_append(a, sdkDir, OC_STR8("bin/liborca_platform.dylib")); oc_str8 glesLib = oc_path_append(a, sdkDir, OC_STR8("bin/libGLESv2.dylib")); oc_str8 eglLib = oc_path_append(a, sdkDir, OC_STR8("bin/libEGL.dylib")); oc_str8 wgpu_lib = oc_path_append(a, sdkDir, OC_STR8("bin/libwebgpu.dylib")); diff --git a/src/tool/win32_icon.c b/src/tool/win32_icon.c index c9eb2489..007dce4f 100644 --- a/src/tool/win32_icon.c +++ b/src/tool/win32_icon.c @@ -81,7 +81,7 @@ bool icon_from_image(oc_arena* a, oc_str8 image_path, oc_str8 ico_path) //----------------------------------------------------------------------------- // Write .ico header //----------------------------------------------------------------------------- - oc_file_write(ico_file, sizeof(header), (u8*)&header); + oc_file_write(ico_file, sizeof(header), (char*)&header); if(oc_file_last_error(ico_file) != OC_IO_OK) { result = false; @@ -132,7 +132,7 @@ bool icon_from_image(oc_arena* a, oc_str8 image_path, oc_str8 ico_path) .image_size_in_bytes = image_sizes_bytes[i], .offset = data_offset, }; - oc_file_write(ico_file, sizeof(entry), (u8*)&entry); + oc_file_write(ico_file, sizeof(entry), (char*)&entry); if(oc_file_last_error(ico_file) != OC_IO_OK) { result = false; @@ -146,7 +146,7 @@ bool icon_from_image(oc_arena* a, oc_str8 image_path, oc_str8 ico_path) //----------------------------------------------------------------------------- for(i32 i = 0; i < num_sizes; ++i) { - oc_file_write(ico_file, image_sizes_bytes[i], images[i]); + oc_file_write(ico_file, image_sizes_bytes[i], (char*)images[i]); if(oc_file_last_error(ico_file) != OC_IO_OK) { result = false; @@ -179,7 +179,7 @@ bool embed_icon_into_exe(oc_arena* a, oc_str8 exe_path, oc_str8 ico_path) u64 ico_file_size = oc_file_size(ico_file); u8* ico_file_data = oc_arena_push_array(a, u8, ico_file_size); - u64 total_read = oc_file_read(ico_file, ico_file_size, ico_file_data); + u64 total_read = oc_file_read(ico_file, ico_file_size, (char*)ico_file_data); if (total_read < ico_file_size) { result = false; @@ -198,7 +198,7 @@ bool embed_icon_into_exe(oc_arena* a, oc_str8 exe_path, oc_str8 ico_path) ico_header->image_type = ico_header_disk->image_type; ico_header->num_images = ico_header_disk->num_images; - void** images = oc_arena_push_array(a, u8*, ico_header->num_images); + void** images = (void**)oc_arena_push_array(a, u8*, ico_header->num_images); for (int i = 0; i < ico_header->num_images; ++i) { diff --git a/src/util/hash.c b/src/util/hash.c index 372c0b37..b52cee72 100644 --- a/src/util/hash.c +++ b/src/util/hash.c @@ -8,7 +8,7 @@ #include "hash.h" #include "platform/platform.h" -//xxhash64, copy-pasted from https://github.com/demetri/scribbles/blob/master/hashing/hash_functions.c +//xxhash64, copy-pasted from https://github.com/demetri/scribbles/blob/master/hashing/ub_aware_hash_functions.c // Thanks to Demetri Spanos uint64_t xxh_64(const void* key, int len, uint64_t h) @@ -21,10 +21,11 @@ uint64_t xxh_64(const void* key, int len, uint64_t h) uint64_t s[4] = { h + p1 + p2, h + p2, h, h - p1 }; // bulk work: process all 32 byte blocks - uint64_t* k32 = (uint64_t*)key; - for(int i = 0; i < (len / 32) * 4; i += 4) + for(int i = 0; i < (len / 32); i++) { - uint64_t b[4] = { k32[i + 0], k32[i + 1], k32[i + 2], k32[i + 3] }; + uint64_t b[4]; + memcpy(b, (const char*)key + 4 * i, sizeof(b)); + for(int j = 0; j < 4; j++) b[j] = b[j] * p2 + s[j]; for(int j = 0; j < 4; j++) @@ -45,10 +46,13 @@ uint64_t xxh_64(const void* key, int len, uint64_t h) s64 += len; // up to 31 bytes remain, process 0-3 8 byte blocks - uint8_t* tail = (uint8_t*)(((char*)key) + (len / 32) * 32); + uint8_t* tail = (uint8_t*)((const char*)key + (len / 32) * 32); for(int i = 0; i < (len & 31) / 8; i++, tail += 8) { - uint64_t b = (*((uint64_t*)tail)) * p2; + uint64_t b; + memcpy(&b, tail, sizeof(uint64_t)); + + b *= p2; b = (((b << 31) | (b >> 33)) * p1) ^ s64; s64 = ((b << 27) | (b >> 37)) * p1 + p4; } @@ -56,7 +60,11 @@ uint64_t xxh_64(const void* key, int len, uint64_t h) // up to 7 bytes remain, process 0-1 4 byte block for(int i = 0; i < (len & 7) / 4; i++, tail += 4) { - uint64_t b = s64 ^ (*(uint32_t*)tail) * p1; + uint32_t b32; + memcpy(&b32, tail, sizeof(b32)); + uint64_t b = b32; + + b = (s64 ^ b) * p1; s64 = ((b << 23) | (b >> 41)) * p2 + p3; } diff --git a/src/wasm/backend_wasm3.c b/src/wasm/backend_wasm3.c index 69e6efaa..9faf60bd 100644 --- a/src/wasm/backend_wasm3.c +++ b/src/wasm/backend_wasm3.c @@ -399,7 +399,7 @@ oc_wasm_function_info oc_wasm_function_get_info(oc_arena* scratch, oc_wasm* wasm info.countParams = countParams; info.countReturns = countReturns; info.params = types; - info.returns = types + info.countParams; + info.returns = types ? types + info.countParams : 0; for(u32 i = 0; i < info.countParams; ++i) { diff --git a/tests/file_open_request/main.c b/tests/file_open_request/main.c index 0ac9e8d6..cad4c5b4 100644 --- a/tests/file_open_request/main.c +++ b/tests/file_open_request/main.c @@ -8,11 +8,33 @@ #include "orca.h" -int main(int argc, char** argv) +oc_str8 parseTestDir(int argc, const char** argv, oc_arena* arena) +{ + const char* test_dir_arg_prefix = "--test-dir="; + for (int i = 1; i < argc; ++i) + { + const char* arg = argv[i]; + const size_t arglen = strlen(arg); + + if (strstr(arg, test_dir_arg_prefix) == arg) + { + const char* slice = arg + strlen(test_dir_arg_prefix); + return OC_STR8(slice); + } + } + + return OC_STR8(""); +} + +int main(int argc, const char** argv) { oc_init(); - oc_file file = oc_file_open_with_request(OC_STR8("./test.txt"), OC_FILE_ACCESS_READ, 0); + oc_arena_scope arena_scope = oc_scratch_begin(); + oc_str8 test_dir = parseTestDir(argc, argv, arena_scope.arena); + oc_str8 test_txt_path = oc_path_append(arena_scope.arena, test_dir, OC_STR8("test.txt")); + + oc_file file = oc_file_open_with_request(test_txt_path, OC_FILE_ACCESS_READ, 0); if(oc_file_is_nil(file)) { oc_log_error("Couldn't open file\n"); @@ -24,5 +46,7 @@ int main(int argc, char** argv) oc_log_info("file contents: %.*s\n", (int)len, buffer); } + oc_scratch_end(arena_scope); + return (0); } diff --git a/tests/files/main.c b/tests/files/main.c index f918a0f1..37e75b3b 100644 --- a/tests/files/main.c +++ b/tests/files/main.c @@ -9,14 +9,13 @@ #define OC_NO_APP_LAYER #include "orca.c" -int test_write() +oc_str8 TEST_DIR; + +int test_write(oc_arena* arena) { oc_log_info("writing\n"); - oc_arena_scope scratch = oc_scratch_begin(); - oc_arena* arena = scratch.arena; - - oc_str8 path = OC_STR8("./data/write_test.txt"); + oc_str8 path = oc_path_append(arena, TEST_DIR, OC_STR8("data/write_test.txt")); oc_str8 test_string = OC_STR8("Hello from write_test.txt"); oc_file f = oc_file_open(path, OC_FILE_ACCESS_WRITE, OC_FILE_OPEN_CREATE | OC_FILE_OPEN_TRUNCATE); @@ -51,6 +50,8 @@ int test_write() } fclose(file); + remove(path.ptr); + return (0); } @@ -72,11 +73,11 @@ int check_string(oc_file f, oc_str8 test_string) return (0); } -int test_read() +int test_read(oc_arena* arena) { oc_log_info("reading\n"); - oc_str8 path = OC_STR8("./data/regular.txt"); + oc_str8 path = oc_path_append(arena, TEST_DIR, OC_STR8("data/regular.txt")); oc_str8 test_string = OC_STR8("Hello from regular.txt"); oc_file f = oc_file_open(path, OC_FILE_ACCESS_READ, 0); @@ -96,11 +97,11 @@ int test_read() return (0); } -int test_stat_size() +int test_stat_size(oc_arena* arena) { oc_log_info("stat size\n"); - oc_str8 path = OC_STR8("./data/regular.txt"); + oc_str8 path = oc_path_append(arena, TEST_DIR, OC_STR8("data/regular.txt")); oc_str8 test_string = OC_STR8("Hello from regular.txt"); u64 size = test_string.len; @@ -128,11 +129,11 @@ int test_stat_size() return (0); } -int test_stat_type() +int test_stat_type(oc_arena* arena) { - oc_str8 regular = OC_STR8("./data/regular.txt"); - oc_str8 dir = OC_STR8("./data/directory"); - oc_str8 link = OC_STR8("./data/symlink"); + oc_str8 regular = oc_path_append(arena, TEST_DIR, OC_STR8("data/regular.txt")); + oc_str8 dir = oc_path_append(arena, TEST_DIR, OC_STR8("data/directory")); + oc_str8 link = oc_path_append(arena, TEST_DIR, OC_STR8("data/symlink")); oc_log_info("stat type, regular\n"); @@ -203,14 +204,16 @@ int test_stat_type() return (0); } -int test_symlinks() +int test_symlinks(oc_arena* arena) { + oc_str8 path = oc_path_append(arena, TEST_DIR, OC_STR8("data/symlink")); + // open symlink target oc_log_info("open symlink target\n"); - oc_file f = oc_file_open_at(oc_file_nil(), OC_STR8("./data/symlink"), OC_FILE_ACCESS_READ, 0); + oc_file f = oc_file_open_at(oc_file_nil(), path, OC_FILE_ACCESS_READ, 0); if(oc_file_last_error(f)) { - oc_log_error("failed to open ./data/symlink\n"); + oc_log_error("failed to open %s\n", path.ptr); return (-1); } if(check_string(f, OC_STR8("Hello from regular.txt"))) @@ -222,10 +225,10 @@ int test_symlinks() // open symlink file oc_log_info("open symlink file\n"); - f = oc_file_open_at(oc_file_nil(), OC_STR8("./data/symlink"), OC_FILE_ACCESS_READ, OC_FILE_OPEN_SYMLINK); + f = oc_file_open_at(oc_file_nil(), path, OC_FILE_ACCESS_READ, OC_FILE_OPEN_SYMLINK); if(oc_file_last_error(f)) { - oc_log_error("failed to open ./data/symlink\n"); + oc_log_error("failed to open %s\n", path.ptr); return (-1); } oc_file_status status = oc_file_get_status(f); @@ -254,11 +257,13 @@ int test_symlinks() return (0); } -int test_args() +int test_args(oc_arena* arena) { + oc_str8 path = oc_path_append(arena, TEST_DIR, OC_STR8("data/regular.txt")); + //NOTE: nil handle oc_log_info("check open_at with nil handle\n"); - oc_file f = oc_file_open_at(oc_file_nil(), OC_STR8("./data/regular.txt"), OC_FILE_ACCESS_READ, 0); + oc_file f = oc_file_open_at(oc_file_nil(), path, OC_FILE_ACCESS_READ, 0); if(oc_file_last_error(f)) { oc_log_error("oc_file_open_at() with nil handle failed\n"); @@ -275,7 +280,7 @@ int test_args() oc_log_info("check open_at with nil handle\n"); oc_file wrongHandle = { .h = 123456789 }; - f = oc_file_open_at(wrongHandle, OC_STR8("./data/regular.txt"), OC_FILE_ACCESS_READ, 0); + f = oc_file_open_at(wrongHandle, path, OC_FILE_ACCESS_READ, 0); if(oc_file_last_error(f) != OC_IO_ERR_HANDLE) { oc_log_error("oc_file_open_at() with non-nil invalid handle should return OC_IO_ERR_HANDLE\n"); @@ -286,7 +291,7 @@ int test_args() //NOTE: nil/wrong handle and OC_FILE_OPEN_RESTRICT oc_log_info("check open_at with nil handle and OC_FILE_OPEN_RESTRICT\n"); - f = oc_file_open_at(oc_file_nil(), OC_STR8("./data/regular.txt"), OC_FILE_ACCESS_READ, OC_FILE_OPEN_RESTRICT); + f = oc_file_open_at(oc_file_nil(), path, OC_FILE_ACCESS_READ, OC_FILE_OPEN_RESTRICT); if(oc_file_last_error(f) != OC_IO_ERR_HANDLE) { oc_log_error("oc_file_open_at() with nil handle and OC_FILE_OPEN_RESTRICT should return OC_IO_ERR_HANDLE\n"); @@ -294,7 +299,7 @@ int test_args() } oc_file_close(f); - f = oc_file_open_at(wrongHandle, OC_STR8("./data/regular.txt"), OC_FILE_ACCESS_READ, OC_FILE_OPEN_RESTRICT); + f = oc_file_open_at(wrongHandle, path, OC_FILE_ACCESS_READ, OC_FILE_OPEN_RESTRICT); if(oc_file_last_error(f) != OC_IO_ERR_HANDLE) { oc_log_error("oc_file_open_at() with invalid handle and OC_FILE_OPEN_RESTRICT should return OC_IO_ERR_HANDLE\n"); @@ -316,11 +321,13 @@ int test_args() return (0); } -int test_jail() +int test_jail(oc_arena* arena) { oc_log_info("test jail\n"); - oc_file jail = oc_file_open(OC_STR8("./data/jail"), OC_FILE_ACCESS_READ, 0); + oc_str8 jailPath = oc_path_append(arena, TEST_DIR, OC_STR8("data/jail")); + + oc_file jail = oc_file_open(jailPath, OC_FILE_ACCESS_READ, 0); if(oc_file_last_error(jail)) { oc_log_error("Can't open jail directory\n"); @@ -388,7 +395,8 @@ int test_jail() //NOTE: escape with bad root handle oc_file wrong_handle = { 0 }; - f = oc_file_open_at(wrong_handle, OC_STR8("./data/regular.txt"), OC_FILE_ACCESS_READ, OC_FILE_OPEN_RESTRICT); + oc_str8 regularPath = oc_path_append(arena, TEST_DIR, OC_STR8("data/regular.txt")); + f = oc_file_open_at(wrong_handle, regularPath, OC_FILE_ACCESS_READ, OC_FILE_OPEN_RESTRICT); if(oc_file_last_error(f) == OC_IO_OK) { oc_log_error("Escaped jail with nil root handle\n"); @@ -470,11 +478,11 @@ int test_jail() return (0); } -int test_rights() +int test_rights(oc_arena* arena) { oc_log_info("test rights\n"); - oc_str8 dirPath = OC_STR8("./data"); + oc_str8 dirPath = oc_path_append(arena, TEST_DIR, OC_STR8("data")); //-------------------------------------------------------------------------------------- // base dir with no access @@ -483,7 +491,7 @@ int test_rights() oc_file dir = oc_file_open(dirPath, OC_FILE_ACCESS_NONE, 0); if(oc_file_last_error(dir)) { - oc_log_error("Couldn't open ./data with no access rights\n"); + oc_log_error("Couldn't open data with no access rights\n"); return (-1); } @@ -547,7 +555,7 @@ int test_rights() oc_file dir = oc_file_open(dirPath, OC_FILE_ACCESS_WRITE, 0); if(oc_file_last_error(dir)) { - oc_log_error("Couldn't open ./data with write rights\n"); + oc_log_error("Couldn't open %s with write rights\n", dirPath.ptr); return (-1); } @@ -618,46 +626,64 @@ int test_rights() return (0); } -int main(int argc, char** argv) +oc_str8 parseTestDir(int argc, const char** argv, oc_arena* arena) +{ + const char* test_dir_arg_prefix = "--test-dir="; + for (int i = 1; i < argc; ++i) + { + const char* arg = argv[i]; + const size_t arglen = strlen(arg); + + if (strstr(arg, test_dir_arg_prefix) == arg) + { + const char* slice = arg + strlen(test_dir_arg_prefix); + return OC_STR8(slice); + } + } + + return OC_STR8(""); +} + +int main(int argc, const char** argv) { oc_arena_scope scratch = oc_scratch_begin(); oc_arena* arena = scratch.arena; - if(test_write()) + TEST_DIR = parseTestDir(argc, argv, arena); + + if(test_write(arena)) { return (-1); } - if(test_read()) + if(test_read(arena)) { return (-1); } - if(test_stat_size()) + if(test_stat_size(arena)) { return (-1); } - if(test_stat_type()) + if(test_stat_type(arena)) { return (-1); } - if(test_args()) + if(test_args(arena)) { return (-1); } - if(test_symlinks()) + if(test_symlinks(arena)) { return (-1); } - if(test_rights()) + if(test_rights(arena)) { return (-1); } - if(test_jail()) + if(test_jail(arena)) { return (-1); } - remove("./data/write_test.txt"); - oc_log_info("OK\n"); return (0);