Clear-Host $target_arch = Join-Path $PSScriptRoot 'helpers/target_arch.psm1' $devshell = Join-Path $PSScriptRoot 'helpers/devshell.ps1' $format_cpp = Join-Path $PSScriptRoot 'helpers/format_cpp.psm1' $config_toolchain = Join-Path $PSScriptRoot 'helpers/configure_toolchain.ps1' $path_root = git rev-parse --show-toplevel $path_build = Join-Path $path_root 'build' Import-Module $target_arch Import-Module $format_cpp #region Arguments $vendor = $null $optimize = $null $debug = $null $analysis = $false $dev = $false $verbose = $null $platform = $null $engine = $null $game = $null $module_specified = $false [array] $vendors = @( "clang", "msvc" ) # This is a really lazy way of parsing the args, could use actual params down the line... if ( $args ) { $args | ForEach-Object { switch ($_){ { $_ -in $vendors } { $vendor = $_; break } "optimize" { $optimize = $true } "debug" { $debug = $true } "analysis" { $analysis = $true } "dev" { $dev = $true } "verbose" { $verbose = $true } "platform" { $platform = $true } "engine" { $engine = $true; $module_specified = $true } "game" { $game = $true; $module_specified = $true } } }} #endregion Argument if ( -not $module_specified ) { $platform = $true $engine = $true $game = $true } # Load up toolchain configuraion . $config_toolchain function check-FileForChanges { param( [Parameter(Mandatory=$true)] [string]$path_file ) if (-not (Test-Path $path_file -PathType Leaf)) { Write-Error "The provided path is not a valid file: $path_file" return $false } $file_name = Split-Path $path_file -Leaf $path_csv = Join-Path $path_build ($file_name + "_file_hash.csv") $csv_file_hash = $null if (Test-Path $path_csv) { $csv_file_hash = Import-Csv $path_csv | Select-Object -ExpandProperty value } $current_hash_info = Get-FileHash -Path $path_file -Algorithm MD5 $current_file_hash = $current_hash_info.Hash # Save the current hash to the CSV [PSCustomObject]@{ name = $path_file value = $current_file_hash } | Export-Csv $path_csv -NoTypeInformation if ($csv_file_hash -and $csv_file_hash -eq $current_file_hash) { return $false } else { return $true } } # Check to see if the module has changed files since the last build function check-ModuleForChanges { param( [string]$path_module, [array]$excludes ) $module_name = split-path $path_module -leaf $path_csv = Join-Path $path_build ($module_name + "_module_hashes.csv") $csv_file_hashes = $null if ( test-path $path_csv ) { $csv_file_hashes = @{} import-csv $path_csv | foreach-object { $csv_file_hashes[ $_.name ] = $_.value } } $file_hashes = @{} get-childitem -path $path_module -recurse -file -Exclude $excludes | foreach-object { $id = $_.fullname $hash_info = get-filehash -path $id -Algorithm MD5 $file_hashes[ $id ] = $hash_info.Hash } $file_hashes.GetEnumerator() | foreach-object { [PSCustomObject]$_ } | export-csv $path_csv -NoTypeInformation if ( -not $csv_file_hashes ) { return $true } if ( $csv_file_hashes.Count -ne $file_hashes.Count ) { return $true } foreach ( $key in $csv_file_hashes.Keys ) { if ( $csv_file_hashes[ $key ] -ne $file_hashes[ $key ] ) { return $true } } return $false } #region Building write-host "Building HandmadeHero with $vendor" $path_project = Join-Path $path_root 'project' $path_scripts = Join-Path $path_root 'scripts' $path_data = Join-Path $path_root 'data' $path_binaries = Join-Path $path_data 'binaries' $path_deps = Join-Path $path_project 'dependencies' $path_codegen = Join-Path $path_project 'codegen' $path_platform = Join-Path $path_project 'platform' $path_engine = Join-Path $path_project 'engine' $path_engine_gen = Join-Path $path_engine 'gen' $update_deps = Join-Path $PSScriptRoot 'update_deps.ps1' $handmade_process_active = Get-Process | Where-Object {$_.Name -like 'handmade_win32*'} if ( (Test-Path $path_build) -eq $false ) { New-Item $path_build -ItemType Directory } if ( (Test-Path $path_deps) -eq $false ) { & $update_deps } if ( (Test-Path $path_binaries) -eq $false ) { New-Item $path_binaries -ItemType Directory } #region Handmade Runtime $includes = @( $path_project ) # Microsoft $lib_gdi32 = 'Gdi32.lib' $lib_xinput = 'Xinput.lib' $lib_user32 = 'User32.lib' $lib_winmm = 'Winmm.lib' # Github $lib_jsl = Join-Path $path_deps 'JoyShockLibrary/x64/JoyShockLibrary.lib' $stack_size = 1024 * 1024 * 4 $compiler_args = @( ($flag_define + 'UNICODE'), ($flag_define + '_UNICODE') # ($flag_set_stack_size + $stack_size) $flag_wall $flag_warnings_as_errors $flag_optimize_intrinsics ) if ( $dev ) { $compiler_args += ( $flag_define + 'Build_Development=1' ) } else { $compiler_args += ( $flag_define + 'Build_Development=0' ) } function build-engine { $unit = Join-Path $path_codegen 'engine_gen.cpp' $should_build = (check-FileForChanges $unit) -eq $true if ( $should_build ) { $executable = Join-Path $path_build 'engine_gen.exe' $compiler_args = @() $compiler_args += ( $flag_define + 'GEN_TIME' ) $linker_args = @( $flag_link_win_subsystem_console ) $build_result = build-simple $path_build $includes $compiler_args $linker_args $unit $executable if ( $build_result -eq $false ) { return $false } Push-Location $path_engine_gen $time_taken = Measure-Command { & $executable 2>&1 | ForEach-Object { write-host `t $_ -ForegroundColor Green } } Pop-Location push-location $path_scripts $include = @( '*.cpp' '*.hpp' ) format-cpp $path_engine_gen $include pop-location } else { write-host "No changes detected in engine gen, skipping build" -ForegroundColor Yellow } $should_build = check-ModuleForChanges $path_engine if ( $should_build -eq $false ) { write-host "No changes detected in engine module, skipping build" -ForegroundColor Yellow return $true } write-host "`nBuilding Engine Module" -ForegroundColor Green $path_pdb_lock = Join-Path $path_binaries 'handmade_engine.pdb.lock' New-Item $path_pdb_lock -ItemType File -Force > $null # Delete old PDBs [Array]$pdb_files = Get-ChildItem -Path $path_binaries -Filter "handmade_engine_*.pdb" foreach ($file in $pdb_files) { Remove-Item -Path $file.FullName -Force if ( $verbose ) { Write-Host "Deleted $file" -ForegroundColor Green } } #region Building Engine Module Runtime $local:includes = $script:includes $includes += $path_engine $local:compiler_args = $script:compiler_args $compiler_args += ($flag_define + 'Build_DLL=1' ) if ( $IsWindows ) { $compiler_args += ($flag_define + 'Engine_API=__declspec(dllexport)') } else { $compiler_args += ($flag_define + 'Engine_API=__attribute__((visibility("default")))') } $local:linker_args = @( $flag_link_dll # $flag_link_optimize_references ) $unit = Join-Path $path_project 'handmade_engine.cpp' $dynamic_library = Join-Path $path_binaries 'handmade_engine.dll' $build_result = build-simple $path_build $includes $compiler_args $linker_args $unit $dynamic_library Remove-Item $path_pdb_lock -Force if ( $build_result -eq $false ) { return $false } #endregion #CodeGen Post-Build if ( -not $handmade_process_active -and $build_result ) { $path_engine_symbols = Join-Path $path_build 'handmade_engine.symbols' # Create the symbol table if ( Test-Path $dynamic_library ) { # $data_path = Join-Path $path_data 'handmade_engine.dll' # move-item $dynamic_library $data_path -Force $path_lib = $dynamic_library -replace '\.dll', '.lib' $path_exp = $dynamic_library -replace '\.dll', '.exp' Remove-Item $path_lib -Force if ( Test-Path $path_exp ) { Remove-Item $path_exp -Force } # We need to generate the symbol table so that we can lookup the symbols we need when loading/reloading the library at runtime. # This is done by sifting through the emitter.map file from the linker for the base symbol names # and mapping them to their found decorated name # Initialize the hashtable with the desired order of symbols $engine_symbols = [ordered]@{ 'on_module_reload' = $null 'startup' = $null 'shutdown' = $null 'update_and_render' = $null 'update_audio' = $null } $path_engine_obj = Join-Path $path_build 'handmade_engine.obj' $path_engine_map = Join-Path $path_build 'handmade_engine.map' $maxNameLength = ($engine_symbols.Keys | Measure-Object -Property Length -Maximum).Maximum Get-Content -Path $path_engine_map | ForEach-Object { # If all symbols are found, exit the loop if ($engine_symbols.Values -notcontains $null) { return } # Split the line into tokens $tokens = $_ -split '\s+', 4 # Extract only the decorated name using regex for both MSVC and Clang conventions $decoratedName = ($tokens[2] -match '(\?[\w@]+|_Z[\w@]+)') ? $matches[1] : $null # Check the origin of the symbol # If the origin matches 'handmade_engine.obj', then process the symbol $originParts = $tokens[3] -split '\s+' $origin = if ($originParts.Count -eq 3) { $originParts[2] } else { $originParts[1] } # Diagnostic output if ( $false -and $decoratedName) { write-host "Found decorated name: $decoratedName" -ForegroundColor Yellow write-host "Origin : $origin" -ForegroundColor Yellow } if ($origin -like 'handmade_engine.obj') { # Check each regular name against the current line $engine_symbols.Keys | Where-Object { $engine_symbols[$_] -eq $null } | ForEach-Object { if ($decoratedName -like "*$_*") { $engine_symbols[$_] = $decoratedName } } } } if ($verbose) { write-host "Engine Symbol Table:" -ForegroundColor Green } $engine_symbols.GetEnumerator() | ForEach-Object { $paddedName = $_.Key.PadRight($maxNameLength) $decoratedName = $_.Value if ($verbose ) { write-host "`t$paddedName, $decoratedName" -ForegroundColor Green } } # Write the symbol table to a file $engine_symbols.Values | Out-File -Path $path_engine_symbols } else { return $false } $unit = Join-Path $path_codegen 'engine_postbuild_gen.cpp' $executable = Join-Path $path_build 'engine_postbuild_gen.exe' $should_build = (check-FileForChanges $unit) -eq $true if ( $should_build ) { $compiler_args = @() $compiler_args += ( $flag_define + 'GEN_TIME' ) $linker_args = @( $flag_link_win_subsystem_console ) $build_result = build-simple $path_build $local:includes $compiler_args $linker_args $unit $executable if ( $build_result -eq $false ) { return $false } } else { write-host "No changes detected in engine post-build gen, skipping build" -ForegroundColor Yellow } Push-Location $path_build $time_taken = Measure-Command { & $executable 2>&1 | ForEach-Object { write-host `t $_ -ForegroundColor Green } } Pop-Location $path_generated_file = Join-Path $path_build 'engine_symbols.gen.hpp' move-item $path_generated_file (join-path $path_engine_gen (split-path $path_generated_file -leaf)) -Force push-location $path_scripts $include = @( '*.cpp' '*.hpp' ) format-cpp $path_engine_gen $include pop-location return $true } } if ( $engine ) { $build_result = build-engine if ( $build_result -eq $false ) { $path_csv = Join-Path $path_build 'engine_module_hashes.csv' Remove-Item $path_csv -Force } } function build-platform { if ( $handmade_process_active ) { write-host "Handmade process is active, skipping platform build" -ForegroundColor Yellow return $true } $should_build = check-ModuleForChanges $path_platform if ( $should_build -eq $false ) { write-host "No changes detected in platform module, skipping build" -ForegroundColor Yellow return $true } write-host "Building Platform Module" -ForegroundColor Green $local:includes = $script:includes $includes += $path_platform # CodeGen Pre-Build $path_engine_symbols = Join-Path $path_engine_gen 'engine_symbols.gen.hpp' $unit = Join-Path $path_codegen 'platform_gen.cpp' $should_build = (check-FileForChanges $unit) -eq $true -or (check-FileForChanges $path_engine_symbols) -eq $true if ( $should_build ) { # Delete old PDBs $pdb_files = Get-ChildItem -Path $path_build -Filter "platform_gen_*.pdb" foreach ($file in $pdb_files) { Remove-Item -Path $file.FullName -Force if ( $verbose ) { Write-Host "Deleted $file" -ForegroundColor Green } } $path_platform_gen = Join-Path $path_platform 'gen' if ( -not (Test-Path $path_platform_gen) ) { New-Item $path_platform_gen -ItemType Directory > $null } $local:compiler_args = @() $compiler_args += ( $flag_define + 'GEN_TIME' ) $local:linker_args = @( $flag_link_win_subsystem_console ) $unit = Join-Path $path_codegen 'platform_gen.cpp' $executable = Join-Path $path_build 'platform_gen.exe' $build_result = build-simple $path_build $includes $compiler_args $linker_args $unit $executable $path_build if ( $build_result -eq $false ) { return $false } Push-Location $path_platform $time_taken = Measure-Command { & $executable 2>&1 | ForEach-Object { write-host `t $_ -ForegroundColor Green } } Pop-Location push-location $path_scripts $include = @( '*.cpp' '*.hpp' ) format-cpp (Join-Path $path_platform 'gen' ) $include pop-location } else { write-host "No changes detected in platform gen, skipping build" -ForegroundColor Yellow } # Delete old PDBs $pdb_files = Get-ChildItem -Path $path_binaries -Filter "handmade_win32_*.pdb" foreach ($file in $pdb_files) { Remove-Item -Path $file.FullName -Force if ( $verbose ) { Write-Host "Deleted $file" -ForegroundColor Green } } $local:compiler_args = $script:compiler_args $compiler_args += ($flag_define + 'Build_DLL=0' ) $local:linker_args = @( $lib_gdi32, # $lib_xinput, $lib_user32, $lib_winmm, $lib_jsl, $flag_link_win_subsystem_windows, $flag_link_optimize_references ) $unit = Join-Path $path_project 'handmade_win32.cpp' $executable = Join-Path $path_binaries 'handmade_win32.exe' return build-simple $path_build $includes $compiler_args $linker_args $unit $executable } if ( $platform ) { $build_result = build-platform if ( -not $build_result ) { $path_csv = Join-Path $path_build 'platform_module_hashes.csv' Remove-Item $path_csv -Force } } $path_jsl_dll = Join-Path $path_binaries 'JoyShockLibrary.dll' if ( (Test-Path $path_jsl_dll) -eq $false ) { $path_jsl_dep_dll = Join-Path $path_deps 'JoyShockLibrary/x64/JoyShockLibrary.dll' Copy-Item $path_jsl_dep_dll $path_jsl_dll } #endregion Handmade Runtime Pop-Location #endregion Building