Simplified build action + ROM stripping fix + Makefile fix

This commit is contained in:
Mateusz Faderewski 2023-08-05 01:37:09 +02:00
parent 8c12feb2c5
commit 6ba534166b
12 changed files with 158 additions and 145 deletions

View File

@ -9,112 +9,81 @@ on:
workflow_dispatch: workflow_dispatch:
jobs: jobs:
build-sc64-menu: build-menu:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v3 - uses: actions/checkout@v3
with: with:
submodules: recursive submodules: recursive
fetch-depth: 1 # we only require the last check-in, unless we want to create a changelog.
- name: Login to GitHub Container Registry - name: Login to GitHub Container Registry
uses: docker/login-action@v2 uses: docker/login-action@v2
with: with:
registry: ghcr.io registry: ghcr.io
username: ${{ github.repository_owner }} username: ${{ github.repository_owner }}
password: ${{ secrets.GITHUB_TOKEN }} password: ${{ secrets.GITHUB_TOKEN }}
- name: Install pinned version of devcontainers cli (https://github.com/devcontainers/ci/issues/252#issuecomment-1654523080)
shell: bash
run: |
npm install --global @devcontainers/cli@0.50.0
- name: Build N64FlashcartMenu ROM - name: Build N64FlashcartMenu ROM
uses: devcontainers/ci@v0.3 uses: devcontainers/ci@v0.3
with: with:
push: never imageName: ghcr.io/Polprzewodnikowy/N64FlashcartMenu-devcontainer
runCmd: | cacheFrom: ghcr.io/Polprzewodnikowy/N64FlashcartMenu-devcontainer
mkdir build push: always
mkdir output runCmd: make all
# TODO: split this to use params for each flashcart type. env:
make FLAGS: -DMENU_RELEASE
- name: Upload artifact - name: Upload artifact (Standard ROM)
uses: actions/upload-artifact@v3 uses: actions/upload-artifact@v3
with: with:
name: N64FlashcartMenu name: N64FlashcartMenu
path: | path: |
./output/N64FlashcartMenu.z64 ./output/N64FlashcartMenu.n64
./build/N64FlashcartMenu.elf ./build/N64FlashcartMenu.elf
minify-sc64-menu: - name: Upload artifact (64drive version)
runs-on: ubuntu-latest
needs: build-sc64-menu
steps:
- name: Setup python
uses: actions/setup-python@v4
with:
python-version: '3.11.x'
- uses: actions/checkout@v3
with:
fetch-depth: 1 # we only require the last check-in, unless we want to create a changelog.
- name: Download ROM artifact
id: download-rom-artifact
uses: actions/download-artifact@v3
with:
name: N64FlashcartMenu
path: ./
- name: Finalize ROM
run: |
# make all
python ./tools/sc64/minify.py ./build/N64FlashcartMenu.elf ./output/N64FlashcartMenu.z64 ./output/sc64menu.n64
continue-on-error: false
- name: Upload artifact
uses: actions/upload-artifact@v3 uses: actions/upload-artifact@v3
with: with:
name: SC64-Menu name: 64drive
path: | path: ./output/menu.bin
./output/sc64menu.n64
if-no-files-found: ignore - name: Upload artifact (ED64 version)
uses: actions/upload-artifact@v3
with:
name: ED64
path: ./output/OS64.v64
- name: Upload artifact (SC64 version)
uses: actions/upload-artifact@v3
with:
name: SC64
path: ./output/sc64menu.n64
generate-docs: generate-docs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
# needs: build-sc64-menu
steps: steps:
- uses: actions/checkout@v3 - uses: actions/checkout@v3
with:
submodules: recursive
fetch-depth: 1 # we only require the last check-in, unless we want to create a changelog.
- name: Run Doxygen - name: Run Doxygen
uses: mattnotmitt/doxygen-action@1.9.5 uses: mattnotmitt/doxygen-action@1.9.5
with: with:
doxyfile-path: './Doxyfile' doxyfile-path: './Doxyfile'
# - name: Deploy - name: Deploy to GitHub Pages
# uses: peaceiris/actions-gh-pages@v3 uses: peaceiris/actions-gh-pages@v3
# with: if: github.ref == 'refs/heads/main'
# github_token: ${{ secrets.GITHUB_TOKEN }} with:
# publish_dir: ./docs github_token: ${{ secrets.GITHUB_TOKEN }}
publish_dir: ./output/docs
# release-sc64-menu:
# runs-on: ubuntu-latest
# needs: minify-sc64-menu
# release-sc64-menu: # steps:
# runs-on: ubuntu-latest # - name: Generate release
# needs: minify-sc64-menu # if: github.event_name == 'release' && github.event.action == 'created'
# run: |
# steps: # echo "still release preview. Check actions for build assets."
# - name: Generate release
# if: github.event_name == 'release' && github.event.action == 'created'
# run: |
# echo "still release preview. Check actions for build assets."

View File

@ -67,7 +67,7 @@ PROJECT_LOGO =
# entered, it will be relative to the location where doxygen was started. If # entered, it will be relative to the location where doxygen was started. If
# left blank the current directory will be used. # left blank the current directory will be used.
OUTPUT_DIRECTORY = docs OUTPUT_DIRECTORY = output
# If the CREATE_SUBDIRS tag is set to YES then doxygen will create up to 4096 # If the CREATE_SUBDIRS tag is set to YES then doxygen will create up to 4096
# sub-directories (in 2 levels) under the output directory of each output format # sub-directories (in 2 levels) under the output directory of each output format
@ -1220,7 +1220,7 @@ GENERATE_HTML = YES
# The default directory is: html. # The default directory is: html.
# This tag requires that the tag GENERATE_HTML is set to YES. # This tag requires that the tag GENERATE_HTML is set to YES.
HTML_OUTPUT = ref HTML_OUTPUT = docs
# The HTML_FILE_EXTENSION tag can be used to specify the file extension for each # The HTML_FILE_EXTENSION tag can be used to specify the file extension for each
# generated HTML page (for example: .htm, .php, .asp). # generated HTML page (for example: .htm, .php, .asp).

View File

@ -1,6 +1,6 @@
PROJECT_NAME = N64FlashcartMenu PROJECT_NAME = N64FlashcartMenu
.DEFAULT_GOAL := $(PROJECT_NAME) .DEFAULT_GOAL := all
SOURCE_DIR = src SOURCE_DIR = src
ASSETS_DIR = assets ASSETS_DIR = assets
@ -9,7 +9,7 @@ OUTPUT_DIR = output
include $(N64_INST)/include/n64.mk include $(N64_INST)/include/n64.mk
N64_CFLAGS += -iquote $(SOURCE_DIR) -I $(SOURCE_DIR)/libs $(FLAGS) N64_CFLAGS += -iquote $(SOURCE_DIR) -I $(SOURCE_DIR)/libs -flto=auto $(FLAGS)
SRCS = \ SRCS = \
main.c \ main.c \
@ -56,6 +56,7 @@ ASSETS = \
OBJS = $(addprefix $(BUILD_DIR)/, $(addsuffix .o,$(basename $(SRCS) $(ASSETS)))) OBJS = $(addprefix $(BUILD_DIR)/, $(addsuffix .o,$(basename $(SRCS) $(ASSETS))))
MINIZ_OBJS = $(filter $(BUILD_DIR)/libs/miniz/%.o,$(OBJS)) MINIZ_OBJS = $(filter $(BUILD_DIR)/libs/miniz/%.o,$(OBJS))
SPNG_OBJS = $(filter $(BUILD_DIR)/libs/libspng/%.o,$(OBJS)) SPNG_OBJS = $(filter $(BUILD_DIR)/libs/libspng/%.o,$(OBJS))
DEPS = $(OBJS:.o=.d)
$(MINIZ_OBJS): N64_CFLAGS+=-DMINIZ_NO_TIME -fcompare-debug-second $(MINIZ_OBJS): N64_CFLAGS+=-DMINIZ_NO_TIME -fcompare-debug-second
$(SPNG_OBJS): N64_CFLAGS+=-isystem $(SOURCE_DIR)/libs/miniz -DSPNG_USE_MINIZ -fcompare-debug-second $(SPNG_OBJS): N64_CFLAGS+=-isystem $(SOURCE_DIR)/libs/miniz -DSPNG_USE_MINIZ -fcompare-debug-second
@ -69,34 +70,57 @@ $(BUILD_DIR)/%.o: $(ASSETS_DIR)/%.ttf
$(BUILD_DIR)/$(PROJECT_NAME).elf: $(OBJS) $(BUILD_DIR)/$(PROJECT_NAME).elf: $(OBJS)
disassembly: $(BUILD_DIR)/$(PROJECT_NAME).elf
@$(N64_OBJDUMP) -S $< > $(BUILD_DIR)/$(PROJECT_NAME).lst
.PHONY: disassembly
$(PROJECT_NAME).z64: N64_ROM_TITLE=$(PROJECT_NAME) $(PROJECT_NAME).z64: N64_ROM_TITLE=$(PROJECT_NAME)
$(PROJECT_NAME): $(PROJECT_NAME).z64 $(@info $(shell mkdir -p ./$(OUTPUT_DIR) &> /dev/null))
$(shell mkdir -p $(OUTPUT_DIR))
$(shell mv $(PROJECT_NAME).z64 $(OUTPUT_DIR))
sc64_minify: $(PROJECT_NAME) $(OUTPUT_DIR)/$(PROJECT_NAME).n64: $(PROJECT_NAME).z64
$(shell python3 ./tools/sc64/minify.py $(BUILD_DIR)/$(PROJECT_NAME).elf $(OUTPUT_DIR)/$(PROJECT_NAME).z64 $(OUTPUT_DIR)/sc64menu.n64) $(shell mv $< $@)
all: sc64_minify $(BUILD_DIR)/$(PROJECT_NAME)_stripped.n64: $(OUTPUT_DIR)/$(PROJECT_NAME).n64
$(shell python3 ./tools/strip_debug_data.py $(BUILD_DIR)/$(PROJECT_NAME).elf $< $@)
@$(N64_CHKSUM) $@ > /dev/null
64drive: $(OUTPUT_DIR)/$(PROJECT_NAME).n64
$(shell cp $< $(OUTPUT_DIR)/menu.bin)
.PHONY: 64drive
ed64: $(BUILD_DIR)/$(PROJECT_NAME)_stripped.n64
$(shell cp $< $(OUTPUT_DIR)/OS64.v64)
.PHONY: ed64
sc64: $(BUILD_DIR)/$(PROJECT_NAME)_stripped.n64
$(shell cp $< $(OUTPUT_DIR)/sc64menu.n64)
.PHONY: sc64
all: $(OUTPUT_DIR)/$(PROJECT_NAME).n64 64drive ed64 sc64
.PHONY: all .PHONY: all
clean: clean:
$(shell rm -rf ./$(BUILD_DIR) ./$(OUTPUT_DIR)) $(shell rm -rf ./$(BUILD_DIR) ./$(OUTPUT_DIR))
.PHONY: clean .PHONY: clean
run: $(PROJECT_NAME) run: $(OUTPUT_DIR)/$(PROJECT_NAME).n64
ifeq ($(OS),Windows_NT)
./localdeploy.bat
else
./remotedeploy.sh ./remotedeploy.sh
# FIXME: improve ability to deploy. endif
# if devcontainer, use remotedeploy.sh, else
# $(shell sc64deployer --boot direct-rom %~dp0$(OUTPUT_DIR))\$(PROJECT_NAME).z64)
.PHONY: run .PHONY: run
run-debug: $(PROJECT_NAME) run-debug: $(OUTPUT_DIR)/$(PROJECT_NAME).n64
ifeq ($(OS),Windows_NT)
./localdeploy.bat /d
else
./remotedeploy.sh -d ./remotedeploy.sh -d
endif
.PHONY: run-debug .PHONY: run-debug
# test: # test:
# TODO: run tests # TODO: run tests
-include $(wildcard $(BUILD_DIR)/*.d) -include $(DEPS)

View File

@ -62,7 +62,7 @@ TODO: it does not yet work with `F5`: see https://devblogs.microsoft.com/cppblog
WORKAROUND: in the dev container terminal, use make directly, i.e.: `make` WORKAROUND: in the dev container terminal, use make directly, i.e.: `make`
The ROM can be found in the `output` directory. The ROM can be found in the `output` directory.
NOTE: a "release" version of the SC64 menu is called `sc64menu.n64` and can be created for when you want to add it directly to the SDCard. This is generated by running `make all` or running `make sc64_minify`. NOTE: a "release" version of the SC64 menu is called `sc64menu.n64` and can be created for when you want to add it directly to the SDCard. This is generated by running `make all` or running `make sc64`.
# Update Libdragon submodule # Update Libdragon submodule
This repo currently uses the `unstable` branch as a submodule at a specific commit. This repo currently uses the `unstable` branch as a submodule at a specific commit.
@ -71,6 +71,7 @@ To update to the latest version, use `git submodule update --remote ` from the t
# Generate documentation # Generate documentation
Run `doxygen` from the dev container terminal. Run `doxygen` from the dev container terminal.
Make sure you fix the warnings before creating a PR! Make sure you fix the warnings before creating a PR!
Generated documentation is located in `output/docs` folder.
# OSS licenses used for libraries # OSS licenses used for libraries

View File

@ -12,7 +12,7 @@ echo:
echo: echo:
:: Load the ROM :: Load the ROM
echo Loading ROM... echo Loading ROM...
%~dp0tools\sc64\sc64deployer upload %~dp0output\N64FlashcartMenu.z64 %~dp0tools\sc64\sc64deployer upload %~dp0output\N64FlashcartMenu.n64
echo: echo:
echo: echo:
@ -20,6 +20,9 @@ echo:
echo !!! Now toggle power to the N64 !!! echo !!! Now toggle power to the N64 !!!
echo: echo:
echo: echo:
::pause
::%~dp0tools\sc64\sc64deployer debug if not "%1" == "/d" goto :exit
%~dp0tools\sc64\sc64deployer debug --no-writeback
:exit

6
remotedeploy.sh Normal file → Executable file
View File

@ -2,7 +2,7 @@
set -e set -e
REMOTE="--remote host.docker.internal:9064" REMOTE="--remote ${REMOTE:-host.docker.internal:9064}"
## FIXME: this does not work! ## FIXME: this does not work!
# Make sure we are connected # Make sure we are connected
@ -17,7 +17,7 @@ echo
# Load the ROM # Load the ROM
echo Loading ROM...: echo Loading ROM...:
sc64deployer $REMOTE upload ./output/N64FlashcartMenu.z64 sc64deployer $REMOTE upload ./output/N64FlashcartMenu.n64
echo echo
echo echo
@ -27,5 +27,5 @@ echo
echo echo
if [ "$1" = "-d" ]; then if [ "$1" = "-d" ]; then
sc64deployer $REMOTE debug sc64deployer $REMOTE debug --no-writeback
fi fi

View File

@ -76,7 +76,7 @@ flashcart_error_t flashcart_init (void) {
return FLASHCART_ERROR_SD_CARD; return FLASHCART_ERROR_SD_CARD;
} }
#ifndef MENU_NO_USB_LOG #ifndef MENU_RELEASE
// NOTE: Some flashcarts doesn't have USB port, can't throw error here // NOTE: Some flashcarts doesn't have USB port, can't throw error here
debug_init_usblog(); debug_init_usblog();
#endif #endif

9
tools/README.md Normal file
View File

@ -0,0 +1,9 @@
# Source
This Directory.
# License
See root directory.
# Description
`strip_debug_data.py`
Removes unnecessary debug data from the end of ROM file to cut on menu load time from SD card.

0
tools/sc64/.gitkeep Normal file
View File

View File

@ -1,12 +0,0 @@
# Source
This Directory.
# License
See root directory.
# Description
`minify.py`
Removes unnecessary null bytes from the end of ROM file to cut on menu load time from SD card.
# Notes
The deployer `sc64deployer` needs to be downloaded from https://github.com/Polprzewodnikowy/SummerCart64/releases/tag/v2.16.0 and placed in this folder.

View File

@ -1,39 +0,0 @@
#!/usr/bin/env python3
import sys
from subprocess import Popen, PIPE
def get_rom_end(elf):
p1 = Popen(f'readelf -s {elf}'.split(), stdout=PIPE)
p2 = Popen('grep -m 1 __rom_end'.split(), stdin=p1.stdout, stdout=PIPE)
stdout, _ = p2.communicate()
rom_end = int(stdout.decode('UTF-8').split()[1], 16)
rom_end &= 0x1FFFFFFF
rom_end -= 0x400
rom_end += 0x1000
return rom_end
if __name__ == '__main__':
if (len(sys.argv) != 4):
print(f'Usage: python {sys.argv[0]} elf input output')
sys.exit(1)
elf_file = sys.argv[1]
input_file = sys.argv[2]
output_file = sys.argv[3]
ALIGN = 512
rom_end = get_rom_end(elf_file)
modulo = rom_end % ALIGN
if (modulo > 0):
rom_end += (ALIGN - modulo)
minified_data = b''
with open(input_file, 'rb') as f:
minified_data = f.read(rom_end)
with open(output_file, 'wb') as f:
f.write(minified_data)

58
tools/strip_debug_data.py Normal file
View File

@ -0,0 +1,58 @@
#!/usr/bin/env python3
import sys
from subprocess import Popen, PIPE
def get_symbol_address(elf, symbol):
p1 = Popen(f'readelf -s --wide {elf}'.split(), stdout=PIPE)
p2 = Popen(f'grep -m 1 {symbol}'.split(), stdin=p1.stdout, stdout=PIPE)
stdout, _ = p2.communicate()
symbol_data = stdout.decode('UTF-8').split()
address = symbol_data[1]
name = symbol_data[7]
if (symbol != name):
raise Exception(f'Inexact symbol name found [{symbol} != {name}]')
return int(address, 16)
def get_rom_data_end_offset(elf):
ROM_ENTRY_OFFSET = 0x1000
libdragon_text_start = get_symbol_address(elf, '__libdragon_text_start')
rom_end = get_symbol_address(elf, '__rom_end')
return ROM_ENTRY_OFFSET + (rom_end - libdragon_text_start)
if __name__ == '__main__':
if (len(sys.argv) != 4):
print(f'Usage: python {sys.argv[0]} elf input output')
sys.exit(1)
elf_file = sys.argv[1]
input_file = sys.argv[2]
output_file = sys.argv[3]
ALIGN = 512
try:
length = get_rom_data_end_offset(elf_file)
except Exception as e:
print(e)
sys.exit(2)
stripped_data = b''
with open(input_file, 'rb') as f:
stripped_data = f.read(length)
modulo = (length % ALIGN)
if (modulo > 0):
stripped_data += b'\x00' * (ALIGN - modulo)
with open(output_file, 'wb') as f:
f.write(stripped_data)