422 lines
14 KiB
PowerShell
422 lines
14 KiB
PowerShell
#Region Variables
|
|
$input_rom_file = "bin/m12fresh.gba"
|
|
$output_rom_file = "bin/m12.gba"
|
|
$eb_rom_file = "bin/eb.smc"
|
|
$working_dir = "working"
|
|
$src_dir = "src"
|
|
$compiled_asm_file = "src/m2-compiled.asm"
|
|
$includes_asm_file = "m12-includes.asm" # implicitly rooted in working_dir
|
|
$hack_asm_file = "m2-hack.asm" # implicitly rooted in src_dir
|
|
|
|
$input_c_files =
|
|
"src/c/ext.c",
|
|
"src/c/vwf.c",
|
|
"src/c/locs.c",
|
|
"src/c/goods.c"
|
|
|
|
$base_c_address = 0x8100000;
|
|
$scripttool_cmd = "bin/ScriptTool/ScriptTool.dll"
|
|
$gcc_cmd = "arm-none-eabi-gcc"
|
|
$ld_cmd = "arm-none-eabi-ld"
|
|
$objdump_cmd = "arm-none-eabi-objdump"
|
|
$readelf_cmd = "arm-none-eabi-readelf"
|
|
$combined_obj_file = "src/c/combined.o"
|
|
$linked_obj_file = "src/c/linked.o"
|
|
$combine_script = "src/c/combine.ld"
|
|
$link_script = "src/c/link.ld"
|
|
$undefine_obj_file = "src/c/ext.o"
|
|
|
|
If ($IsWindows) { $asm_cmd = "bin/armips.exe" }
|
|
ElseIf ($IsLinux) { $asm_cmd = "bin/armips" }
|
|
Else {
|
|
Write-Host "TODO: what's the Mac version of armips?"
|
|
Exit -1
|
|
}
|
|
|
|
$includes_sym_file = [IO.Path]::ChangeExtension($includes_asm_file, "sym")
|
|
$output_rom_sym_file = [IO.Path]::ChangeExtension($output_rom_file, "sym")
|
|
$hack_sym_file = [IO.Path]::ChangeExtension($hack_asm_file, "sym")
|
|
|
|
$scripttool_args =
|
|
"-compile",
|
|
"-main",
|
|
"-misc",
|
|
$working_dir,
|
|
$eb_rom_file,
|
|
$input_rom_file
|
|
|
|
$gcc_args =
|
|
"-c",
|
|
"-O3",
|
|
"-fno-ipa-cp",
|
|
"-fno-inline",
|
|
"-march=armv4t",
|
|
"-mtune=arm7tdmi",
|
|
"-mthumb",
|
|
"-ffixed-r12",
|
|
"-mno-long-calls"
|
|
|
|
$combine_script_contents =
|
|
"SECTIONS { .text 0x$($base_c_address.ToString('X')) : { *(.text .rodata) } }"
|
|
|
|
$link_script_contents =
|
|
"SECTIONS { .text 0x$($base_c_address.ToString('X')) : { *(.text .data .rodata) } }"
|
|
#EndRegion Variables
|
|
|
|
#Region Functions
|
|
class Symbol
|
|
{
|
|
[string]$Name
|
|
[int]$Value
|
|
[int]$Size
|
|
[bool]$IsLocal
|
|
[bool]$IsGlobal
|
|
[bool]$IsWeak
|
|
[bool]$IsConstructor
|
|
[bool]$IsWarning
|
|
[bool]$IsIndirect
|
|
[bool]$IsDebugging
|
|
[bool]$IsDynamic
|
|
[bool]$IsFunction
|
|
[bool]$IsFile
|
|
[bool]$IsObject
|
|
[string]$Section
|
|
[bool]$IsAbsolute
|
|
[bool]$IsUndefined
|
|
}
|
|
|
|
class SectionInfo
|
|
{
|
|
[string]$Name
|
|
[int]$Address
|
|
[int]$Offset
|
|
[int]$Size
|
|
}
|
|
|
|
Function Get-Symbols ([string]$obj_file)
|
|
{
|
|
return & $objdump_cmd -t $obj_file | ForEach-Object { New-Symbol $_ } | Where-Object { $_ -ne $null }
|
|
}
|
|
|
|
# Converts a symbol from objdump's string representation to a rich object representation
|
|
$symbol_regex = "(?'value'[0-9a-fA-F]{8})\s(?'flags'.{7})\s(?'section'\S+)\s+(?'size'[0-9a-fA-F]{8})\s(?'name'\S+)"
|
|
Function New-Symbol([string]$symbol_string)
|
|
{
|
|
if ($symbol_string -match $symbol_regex)
|
|
{
|
|
$symbol = [Symbol]::new()
|
|
$symbol.Name = $Matches.name
|
|
$symbol.Value = [int]::Parse($Matches.value, [System.Globalization.NumberStyles]::HexNumber)
|
|
$symbol.Size = [int]::Parse($Matches.size, [System.Globalization.NumberStyles]::HexNumber)
|
|
$symbol.Section = $Matches.section
|
|
$symbol.IsAbsolute = $symbol.Section -eq "*ABS*"
|
|
$symbol.IsUndefined = $symbol.Section -eq "*UND*"
|
|
|
|
$flags = $Matches.flags
|
|
$symbol.IsLocal = $flags.Contains("l") -or $flags.Contains("!")
|
|
$symbol.IsGlobal = $flags.Contains("g") -or $flags.Contains("!")
|
|
$symbol.IsWeak = $flags.Contains("w")
|
|
$symbol.IsConstructor = $flags.Contains("C")
|
|
$symbol.IsWarning = $flags.Contains("W")
|
|
$symbol.IsIndirect = $flags.Contains("I")
|
|
$symbol.IsDebugging = $flags.Contains("d")
|
|
$symbol.IsDynamic = $flags.Contains("D")
|
|
$symbol.IsFunction = $flags.Contains("F")
|
|
$symbol.IsFile = $flags.Contains("f")
|
|
$symbol.IsObject = $flags.Contains("O")
|
|
|
|
return $symbol
|
|
}
|
|
else
|
|
{
|
|
return $null
|
|
}
|
|
}
|
|
|
|
Function Get-SymfileSymbols([string]$symbol_file)
|
|
{
|
|
return Get-Content $symbol_file | ForEach-Object { New-SymfileSymbol $_ } | Where-Object { $null -ne $_ }
|
|
}
|
|
|
|
$symfile_symbol_regex = "(?'value'[0-9a-fA-F]{8})\s+(?'name'(?>\.|@@|[a-zA-Z0-9_])[a-zA-Z0-9_]+):{0,1}(?'size'[0-9a-fA-F]+){0,1}"
|
|
Function New-SymfileSymbol([string]$symbol_string)
|
|
{
|
|
if ($symbol_string -match $symfile_symbol_regex)
|
|
{
|
|
$symbol = [Symbol]::new()
|
|
$symbol.Name = $Matches.name
|
|
$symbol.Value = [int]::Parse($Matches.value, [System.Globalization.NumberStyles]::HexNumber)
|
|
|
|
if ($null -ne $Matches.size)
|
|
{
|
|
$symbol.Size = [int]::Parse($Matches.size, [System.Globalization.NumberStyles]::HexNumber)
|
|
}
|
|
else
|
|
{
|
|
$symbol.Size = 0
|
|
}
|
|
|
|
$symbol.IsLocal = $symbol.Name.StartsWith("@@")
|
|
$symbol.IsGlobal = -not $symbol.Name.StartsWith(".") -and -not $symbol.IsLocal
|
|
|
|
return $symbol
|
|
}
|
|
else
|
|
{
|
|
return $null
|
|
}
|
|
}
|
|
|
|
function Get-SectionInfo([string]$object_file)
|
|
{
|
|
$hash = @{}
|
|
& $readelf_cmd -S $object_file | ForEach-Object {
|
|
$section = New-Section $_
|
|
if ($null -ne $section)
|
|
{
|
|
$hash[$section.Name] = $section
|
|
}
|
|
}
|
|
return $hash
|
|
}
|
|
|
|
$section_regex = "\s?\[\s?\d+]\s(?'name'\S+)\s+\S+\s+(?'address'[0-9a-fA-F]+)\s(?'offset'[0-9a-fA-F]+)\s(?'size'[0-9a-fA-F]+)"
|
|
function New-Section([string]$section_string)
|
|
{
|
|
if ($section_string -match $section_regex)
|
|
{
|
|
$section = [SectionInfo]::new()
|
|
$section.Name = $Matches.name
|
|
$section.Address = [int]::Parse($Matches.address, [System.Globalization.NumberStyles]::HexNumber)
|
|
$section.Offset = [int]::Parse($Matches.offset, [System.Globalization.NumberStyles]::HexNumber)
|
|
$section.Size = [int]::Parse($Matches.size, [System.Globalization.NumberStyles]::HexNumber)
|
|
|
|
return $section
|
|
}
|
|
else
|
|
{
|
|
return $null
|
|
}
|
|
}
|
|
#EndRegion Functions
|
|
|
|
<#
|
|
This is a complicated build script that does complicated things, but it's
|
|
that way for a reason.
|
|
|
|
- We want to use ASM and C code files simultaneously
|
|
- The ASM code defines symbols that we want to reference from C
|
|
- The C code defines symbols that we want to reference from ASM
|
|
- The game text defines symbols that we want to reference from C
|
|
|
|
The ASM and C code therefore depend on each other. The way around this catch-22
|
|
is to separate the compiling and linking stages of the C code.
|
|
|
|
1) Compile the game text
|
|
|
|
Inputs:
|
|
- Text/script files
|
|
|
|
Outputs:
|
|
- BIN files containing compiled text data
|
|
- ASM files that relocate text pointers (m12-includes.asm)
|
|
|
|
2) Assemble output from step 1
|
|
|
|
Inputs:
|
|
- Output from step 1
|
|
- Fresh M12 ROM file
|
|
|
|
Outputs:
|
|
- ROM file with text inserted and repointed
|
|
- m12-includes.sym file containing generated symbols (e.g. individual strings from m12-other.json)
|
|
|
|
3) Compile C code
|
|
|
|
Inputs:
|
|
- C files
|
|
|
|
Outputs:
|
|
- O files (one for each C file)
|
|
|
|
Remarks:
|
|
- All symbols not defined in the C code itself must be marked extern. We will link it
|
|
in a later step. This includes the string symbols from step 2.
|
|
- Due to an assembler limitation, extern symbols cannot contain capital letters.
|
|
- There's a weird quirk with the linker that requires extra care when using external
|
|
functions. You need to declare them as extern AND implement them using the ((naked))
|
|
attribute, e.g.
|
|
|
|
(in a header file) extern int m2_drawwindow(WINDOW* window);
|
|
(in a code file) int __attribute__((naked)) m2_drawwindow(WINDOW* window) {}
|
|
|
|
See: http://stackoverflow.com/a/43283331/1188632
|
|
|
|
This will cause a duplicate definition of the function symbol, since it's *really*
|
|
defined in the ASM files somewhere, but we're redefining it again in C. So we also need
|
|
to "undefine" these symbols later on when linking.
|
|
|
|
To make it a bit easier to do all that, place all such implementations in ext.c.
|
|
|
|
4) First link stage
|
|
|
|
Inputs:
|
|
- O files from step 3
|
|
- Base address
|
|
|
|
Outputs:
|
|
- Single O file positioned to the base address
|
|
- m2-compiled.asm, containing C symbol definitions
|
|
|
|
Remarks:
|
|
- This is an incremental link; there will still be undefined symbols.
|
|
- However, with this combined O file, the code layout will not change and we can now
|
|
define symbols from the C code to be used in the ASM code.
|
|
- The symbols will be passed to the assembler next, so export them as an ASM file
|
|
with one ".definelabel" entry for each defined symbol. They need to be halfword-aligned.
|
|
This file is called "m2-compiled.asm" by default and is referenced by m2-hack.asm.
|
|
- Exclude the symbols defined in ext.c from m2-compiled.asm.
|
|
|
|
5) Assemble ASM code
|
|
|
|
Inputs:
|
|
- m2-hack.asm
|
|
- m2-compiled.asm
|
|
- All other ASM and data files from src/ (but not the generated ones from working/)
|
|
- M12 ROM file from step 2
|
|
|
|
Outputs:
|
|
- M12 ROM file with all ASM code and data included
|
|
- m2-hack.sym with all symbols defined thusfar
|
|
|
|
6) Generate final linker script
|
|
|
|
Inputs:
|
|
- Base address (same as step 4)
|
|
- m2-hack.sym (from step 5)
|
|
- m12-includes.sym (from step 2)
|
|
|
|
Outputs:
|
|
- Linker script file
|
|
|
|
Remarks:
|
|
- The linker script must define each symbol that's still undefined; if everything is
|
|
happy at this point, then they should all be contained within the two input SYM files.
|
|
|
|
7) Final link stage
|
|
|
|
Inputs:
|
|
- O file from step 4
|
|
- Linker script from step 6
|
|
|
|
Outputs:
|
|
- Single O file with all symbols defined
|
|
|
|
8) Copy code to ROM
|
|
|
|
Inputs:
|
|
- O file from step 7
|
|
- M12 ROM file from step 5
|
|
|
|
Outputs:
|
|
- M12 ROM file with all code and data included
|
|
|
|
9) Build final symbol file
|
|
|
|
Inputs:
|
|
- m2-hack.sym
|
|
- m12-includes.sym
|
|
|
|
Outputs:
|
|
- m12.sym
|
|
|
|
Remarks:
|
|
- This is just the input files concatenated (and sorted for convenience).
|
|
#>
|
|
|
|
$timer = [System.Diagnostics.StopWatch]::StartNew()
|
|
|
|
# ------------------------- COMPILE GAME TEXT -----------------------
|
|
"Copying $input_rom_file to $output_rom_file..."
|
|
Copy-Item -Path $input_rom_file -Destination $output_rom_file
|
|
|
|
"Compiling game text..."
|
|
& dotnet $scripttool_cmd $scripttool_args
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
|
|
# ------------------------ ASSEMBLE GAME TEXT -----------------------
|
|
"Assembling game text..."
|
|
& $asm_cmd -root $working_dir -sym $includes_sym_file $includes_asm_file
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
|
|
# ----------------------------- COMPILE C ---------------------------
|
|
$obj_files = @()
|
|
|
|
# Invoke gcc on each file individually so that we can specify the output file
|
|
foreach ($input_c_file in $input_c_files)
|
|
{
|
|
$obj_file = [IO.Path]::ChangeExtension($input_c_file, "o")
|
|
$obj_files += $obj_file
|
|
|
|
"Compiling $input_c_file..."
|
|
& $gcc_cmd $gcc_args -o $obj_file $input_c_file
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
}
|
|
|
|
# ----------------------------- 1ST LINK ----------------------------
|
|
"Writing $combine_script..."
|
|
$combine_script_contents | Out-File -FilePath $combine_script
|
|
|
|
"Linking $obj_files..."
|
|
& $ld_cmd -i -T $combine_script -o $combined_obj_file $obj_files
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
|
|
"Reading symbols from $combined_obj_file..."
|
|
$combined_symbols = Get-Symbols $combined_obj_file
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
|
|
"Reading symbols from $undefine_obj_file..."
|
|
$ext_symbols = Get-Symbols $undefine_obj_file
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
|
|
"Exporting C symbols to $compiled_asm_file..."
|
|
$ext_symbols_names = $ext_symbols | Where-Object { $_.IsFunction -and $_.IsGlobal -and (-not $_.IsUndefined) } | ForEach-Object { $_.Name }
|
|
$exported_symbols = $combined_symbols | Where-Object { $_.IsFunction -and $_.IsGlobal -and (-not $_.IsUndefined) -and ($ext_symbols_names -notcontains $_.Name) }
|
|
$exported_symbols | Sort-Object -Property Name | ForEach-Object { ".definelabel $($_.Name),0x$($_.Value.ToString("X"))" } | Set-Content -Path $compiled_asm_file
|
|
|
|
# ------------------------ ASSEMBLE HACK CODE -----------------------
|
|
"Assembling $hack_asm_file..."
|
|
& $asm_cmd -root $src_dir -sym $hack_sym_file $hack_asm_file
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
|
|
# ------------------- GENERATE FINAL LINKER SCRIPT ------------------
|
|
"Writing $link_script..."
|
|
$hack_symbols = Get-SymfileSymbols "$([IO.Path]::Combine($src_dir, $hack_sym_file))"
|
|
$includes_symbols = Get-SymfileSymbols "$([IO.Path]::Combine($working_dir, $includes_sym_file))"
|
|
$asm_symbols = ($hack_symbols + $includes_symbols) | Where-Object { $_.IsGlobal }
|
|
Set-Content -Path $link_script -Value $link_script_contents
|
|
$asm_symbols | ForEach-Object { Add-Content -Path $link_script -Value "$($_.Name) = 0x$($_.Value.ToString("X"));" }
|
|
|
|
# ---------------------------- FINAL LINK ---------------------------
|
|
"Linking to $linked_obj_file..."
|
|
& $ld_cmd -T $link_script -o $linked_obj_file $combined_obj_file
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
|
|
# -------------------- COPY COMPILED C CODE TO ROM ------------------
|
|
"Copying compiled code to $output_rom_file..."
|
|
$sections = Get-SectionInfo $linked_obj_file
|
|
if ($LASTEXITCODE -ne 0) { exit -1 }
|
|
$text_section = $sections[".text"]
|
|
$linked_bytes = [IO.File]::ReadAllBytes($linked_obj_file)
|
|
$rom_bytes = [IO.File]::ReadAllBytes($output_rom_file)
|
|
[System.Array]::Copy($linked_bytes, $text_section.Offset, $rom_bytes, $text_section.Address - 0x8000000, $text_section.Size)
|
|
[IO.File]::WriteAllBytes($output_rom_file, $rom_bytes)
|
|
|
|
# -------------------------- GENERATE SYMBOLS -----------------------
|
|
"Generating $output_rom_sym_file..."
|
|
($hack_symbols + $includes_symbols) | Sort-Object Name | ForEach-Object { "$($_.Value.ToString("X8")) $($_.Name)" } | Set-Content $output_rom_sym_file
|
|
|
|
"Finished compiling $output_rom_file in $($timer.Elapsed.TotalSeconds.ToString("F3")) s"
|
|
|
|
exit 0
|