summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRangi <remy.oukaour+rangi@gmail.com>2020-09-20 17:02:44 -0400
committerRangi <remy.oukaour+rangi@gmail.com>2020-09-20 17:02:44 -0400
commit09689c6ffdf0cd476cc9ff645852740abc1ed261 (patch)
tree6b048dd0f6031da3c482cb6f33e5d57a74dcea4c
parent7e49b5ebc62f00d82901ba54b73e89f06c1d8852 (diff)
Port scan_includes.c from pokepicross
-rw-r--r--.gitignore25
-rw-r--r--Makefile21
-rw-r--r--tools/Makefile3
-rw-r--r--tools/scan_includes.c275
-rw-r--r--tools/scan_includes.py82
5 files changed, 294 insertions, 112 deletions
diff --git a/.gitignore b/.gitignore
index 9428e98..07a57f0 100644
--- a/.gitignore
+++ b/.gitignore
@@ -10,30 +10,15 @@
!shim.sym
# build artifacts
-build/
-*.d
-*.o
-*.2bpp
-*.1bpp
-*.pic
-*.pcm
-shim.asm
+/build
-# build utilities
+# build tools
tools/scan_includes
tools/pkmncompress
tools/gfx
-tools/fix_sections_directory.py
*.exe
*.pyc
+__pycache__
-# editor files
-*~
-.idea/
-
-# extra utilities
-coverage.png
-coverage.log
-temp/
-pokegold-spaceworld-gen.link
-*.txt \ No newline at end of file
+# utility output
+/coverage.png
diff --git a/Makefile b/Makefile
index dad5b16..da3e74b 100644
--- a/Makefile
+++ b/Makefile
@@ -26,6 +26,8 @@ RGBLINK ?= $(RGBDS)rgblink
RGBASMFLAGS := -h -E -i $(BUILD)/ -DGOLD
+SCAN_INCLUDES := tools/scan_includes
+
tools/gfx :=
@@ -41,19 +43,19 @@ compare: $(ROM) $(CORRECTEDROM)
$(MD5) roms.md5
.PHONY: tools
-tools tools/pkmncompress tools/gfx:
+tools tools/pkmncompress tools/gfx tools/scan_includes:
"$(MAKE)" -C tools/
# Remove files generated by the build process.
.PHONY: clean
clean:
- rm -rf $(ROM) $(CORRECTEDROM) $(BUILD) $(ROMS:.gb=.sym) $(ROMS:.gb=.map) *.d
+ rm -rf $(ROM) $(CORRECTEDROM) $(BUILD) $(ROMS:.gb=.sym) $(ROMS:.gb=.map)
"$(MAKE)" -C tools clean
# Remove generated files except for graphics.
.PHONY: tidy
tidy:
- rm -rf $(ROM) $(CORRECTEDROM) $(OBJS) $(ROMS:.gb=.sym) $(ROMS:.gb=.map) *.d
+ rm -rf $(ROM) $(CORRECTEDROM) $(OBJS) $(ROMS:.gb=.sym) $(ROMS:.gb=.map)
# Visualize disassembly progress.
.PHONY: coverage
@@ -68,8 +70,8 @@ $(CORRECTEDROM): %-correctheader.gb: %.gb
cp $(<:.gb=.sym) $(@:.gb=.sym)
$(RGBFIX) -f hg -m 0x10 $@
-$(ROM): poke%-spaceworld.gb: $(OBJS) | $(BASEROM)
- $(RGBLINK) -d -n $(@:.gb=.sym) -m $(@:.gb=.map) -l layout.link -O $(BASEROM) -o $@ $^
+$(ROM): poke%-spaceworld.gb: layout.link $(OBJS) | $(BASEROM)
+ $(RGBLINK) -d -n $(@:.gb=.sym) -m $(@:.gb=.map) -l layout.link -O $(BASEROM) -o $@ $(filter-out $<, $^)
$(RGBFIX) -f lh -k 01 -l 0x33 -m 0x03 -p 0 -r 3 -t "POKEMON2$(shell echo $* | cut -d _ -f 1 | tr '[:lower:]' '[:upper:]')" $@
$(BASEROM):
@@ -85,6 +87,9 @@ $(BUILD)/%.o: $(BUILD)/%.asm | $$(dir $$@)
$(BUILD)/%.o: %.asm | $$(dir $$@)
$(RGBASM) $(RGBASMFLAGS) $(OUTPUT_OPTION) $<
+$(BUILD)/%.d: %.asm | $$(dir $$@) $(SCAN_INCLUDES)
+ @$(SCAN_INCLUDES) -b $(BUILD)/ -i $(BUILD)/ -i ./ -o $@ -t $(@:.d=.o) $<
+
### Misc file-specific graphics rules
@@ -137,8 +142,6 @@ $(BUILD)/%.tilemap: %.png | $$(dir $$@)
### Scan .asm files for INCLUDE dependencies
-DEPENDENCY_SCAN_EXIT_STATUS := $(shell $(PYTHON) tools/scan_includes.py $(BUILD:%=-b %) $(ASMFILES) > dependencies.d; echo $$?)
-ifneq ($(DEPENDENCY_SCAN_EXIT_STATUS), 0)
-$(error Dependency scan failed)
+ifeq (,$(filter clean tools,$(MAKECMDGOALS)))
+-include $(patsubst %.o, %.d, $(OBJS))
endif
-include dependencies.d
diff --git a/tools/Makefile b/tools/Makefile
index 6c49ccb..0a1ffd2 100644
--- a/tools/Makefile
+++ b/tools/Makefile
@@ -4,7 +4,8 @@ CFLAGS := -O3 -std=c99 -Wall -Wextra
tools := \
pkmncompress \
- gfx
+ gfx \
+ scan_includes
all: $(tools)
diff --git a/tools/scan_includes.c b/tools/scan_includes.c
new file mode 100644
index 0000000..adef2b2
--- /dev/null
+++ b/tools/scan_includes.c
@@ -0,0 +1,275 @@
+#define _DEFAULT_SOURCE
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <stdbool.h>
+#include <getopt.h>
+#include <limits.h>
+#include <unistd.h>
+
+void usage(void) {
+ printf("Usage: scan_includes [-h] [-o output] [-s] [-i path] [-b path] filename\n"
+ "-h, --help\n"
+ " Print usage and exit\n"
+ "-o, --output\n"
+ " Filename to store the output in\n"
+ "-s, --strict\n"
+ " Fail if a file cannot be read\n"
+ "-i, --include\n"
+ " Add an include path\n"
+ "-b, --build-prefix\n"
+ " Set path to generate non-existing files in\n"
+ "-t, --target\n"
+ " Generate a makefile fragment for target file\n");
+}
+
+struct Options {
+ bool help;
+ char *output;
+ bool strict;
+ char **include_paths;
+ int include_paths_len;
+ char *build_prefix;
+ char *target;
+};
+
+struct Options Options = {0};
+
+void *xmalloc(size_t size)
+{
+ void *ptr = malloc(size);
+ if (!ptr) {
+ perror("malloc");
+ exit(1);
+ }
+ return ptr;
+}
+
+void *xrealloc(void *ptr, size_t size)
+{
+ ptr = realloc(ptr, size);
+ if (!ptr) {
+ perror("realloc");
+ exit(1);
+ }
+ return ptr;
+}
+
+void options_add_file(char *filename)
+{
+ Options.include_paths_len++;
+ Options.include_paths = xrealloc(Options.include_paths,
+ sizeof(Options.include_paths[0]) * Options.include_paths_len);
+ Options.include_paths[Options.include_paths_len - 1] = filename;
+}
+
+void filelist_append(char **string, char *append)
+{
+ size_t orig_len = *string ? strlen(*string) : 0;
+ size_t len = orig_len + strlen(append) + 2;
+ if (orig_len == 0) len--;
+ *string = xrealloc(*string, len);
+
+ if (orig_len != 0) {
+ (*string)[orig_len + 0] = ' ';
+ (*string)[orig_len + 1] = '\0';
+ } else {
+ (*string)[orig_len + 0] = '\0';
+ }
+ strcat(*string, append);
+}
+
+char *joinpath(char *dir, char *file)
+{
+ size_t len = strlen(dir) + strlen(file) + 1;
+ char *path = xmalloc(len);
+ snprintf(path, len, "%s%s", dir, file);
+ return path;
+}
+
+char *find_file(char *filename) {
+ if (access(filename, F_OK) == 0) {
+ char *fname = strdup(filename);
+ if (!fname) {
+ perror("strdup");
+ exit(1);
+ }
+ return fname;
+ }
+
+ // Try to find file in any of the include paths
+ for (int i = 0; i < Options.include_paths_len; i++) {
+ char *path = joinpath(Options.include_paths[i], filename);
+ if (access(path, F_OK) == 0) return path;
+ free(path);
+ }
+
+ if (Options.strict) {
+ fprintf(stderr, "Could not open file: '%s'\n", filename);
+ exit(1);
+ }
+ return NULL;
+}
+
+void scan_file(char **includes, char **incbins, char *filename) {
+ FILE *f = fopen(filename, "r");
+ if (!f) return;
+
+ fseek(f, 0, SEEK_END);
+ long size = ftell(f);
+ rewind(f);
+
+ char *buffer = xmalloc(size + 1);
+ char *orig = buffer;
+ size = fread(buffer, 1, size, f);
+ buffer[size] = '\0';
+ fclose(f);
+
+ for (; buffer && (buffer - orig < size); buffer++) {
+ bool is_include = false;
+ bool is_incbin = false;
+ switch (*buffer) {
+ case ';':
+ buffer = strchr(buffer, '\n');
+ if (!buffer) {
+ fprintf(stderr, "%s: no newline at end of file\n", filename);
+ break;
+ }
+ break;
+
+ case '"':
+ buffer++;
+ buffer = strchr(buffer, '"');
+ if (!buffer) {
+ fprintf(stderr, "%s: unterminated string\n", filename);
+ break;
+ }
+ buffer++;
+ break;
+
+ case 'i':
+ case 'I':
+ if ((strncmp(buffer, "INCBIN", 6) == 0) ||
+ (strncmp(buffer, "incbin", 6) == 0)) {
+ is_incbin = true;
+ } else if ((strncmp(buffer, "INCLUDE", 7) == 0) ||
+ (strncmp(buffer, "include", 7) == 0)) {
+ is_include = true;
+ }
+ if (is_incbin || is_include) {
+ buffer = strchr(buffer, '"');
+ if (!buffer) {
+ break;
+ }
+ buffer++;
+
+ size_t length = strcspn(buffer, "\"");
+ buffer[length] = '\0';
+ char *include = find_file(buffer);
+
+ char *append = buffer;
+ if (Options.build_prefix) {
+ append = include;
+ if (!include) {
+ append = joinpath(Options.build_prefix, buffer);
+ }
+ }
+ if (is_include) filelist_append(includes, append);
+ if (is_incbin) filelist_append(incbins, append);
+ if (Options.build_prefix && !include) free(append);
+
+ if (include && is_include) scan_file(includes, incbins, include);
+
+ if (include) free(include);
+ buffer += length + 1;
+ }
+ break;
+
+ }
+ if (!buffer) {
+ break;
+ }
+
+ }
+
+ free(orig);
+}
+
+int main(int argc, char* argv[]) {
+ int i = 0;
+ struct option long_options[] = {
+ {"help", no_argument, 0, 'h'},
+ {"output", required_argument, 0, 'o'},
+ {"strict", no_argument, 0, 's'},
+ {"include", required_argument, 0, 'i'},
+ {"build-prefix", required_argument, 0, 'b'},
+ {"target", required_argument, 0, 't'},
+ {0}
+ };
+ int opt = -1;
+ while ((opt = getopt_long(argc, argv, "ho:si:b:t:", long_options, &i)) != -1) {
+ switch (opt) {
+ case 'h':
+ Options.help = true;
+ break;
+ case 'o':
+ Options.output = optarg;
+ break;
+ case 's':
+ Options.strict = true;
+ break;
+ case 'i':
+ options_add_file(optarg);
+ break;
+ case 'b':
+ Options.build_prefix = optarg;
+ break;
+ case 't':
+ Options.target = optarg;
+ break;
+ default:
+ usage();
+ exit(1);
+ break;
+ }
+ }
+ argc -= optind;
+ argv += optind;
+ if (Options.help) {
+ usage();
+ return 0;
+ }
+ if (argc < 1) {
+ usage();
+ exit(1);
+ }
+
+ char *includes = NULL;
+ char *incbins = NULL;
+ char *filename = find_file(argv[0]);
+ if (filename) scan_file(&includes, &incbins, filename);
+
+ FILE *f = stdout;
+ if (Options.output) {
+ f = fopen(Options.output, "w");
+ if (!f) {
+ perror("fopen");
+ exit(1);
+ }
+ }
+
+ if (Options.target) {
+ fprintf(f, "%s:", Options.target);
+ }
+ if (includes) fprintf(f, " %s", includes);
+ if (incbins) fprintf(f, " %s", incbins);
+ fprintf(f, "\n");
+
+ if (Options.target && Options.output) {
+ fprintf(f, "%s:", Options.output);
+ if (includes) fprintf(f, " %s", includes);
+ fprintf(f, "\n");
+ }
+ return 0;
+}
diff --git a/tools/scan_includes.py b/tools/scan_includes.py
deleted file mode 100644
index c30adc0..0000000
--- a/tools/scan_includes.py
+++ /dev/null
@@ -1,82 +0,0 @@
-#!/usr/bin/env python3
-# -*- coding: utf-8 -*-
-
-"""Get all the dependencies of RGBDS assembly files recursively,
-and output them using Make dependency syntax.
-"""
-
-# Script adapted from the Telefang disassembly / fan translation project.
-
-from __future__ import print_function
-from __future__ import unicode_literals
-
-import argparse
-import os
-import re
-import sys
-if sys.version_info[0] < 3:
- from codecs import open
-
-INCLUDE_RE = re.compile(r"^(?:[a-zA-Z0-9_.]+:?:?)?\s*(INC(?:LUDE|BIN))", re.IGNORECASE)
-
-def dependencies_in(asm_file_paths, build_dirs=[]):
- asm_file_paths = list(asm_file_paths)
- dependencies = {}
-
- for path in asm_file_paths:
- if path not in dependencies:
- asm_dependencies, bin_dependencies = shallow_dependencies_of(path, build_dirs)
- dependencies[path] = asm_dependencies | bin_dependencies
- asm_file_paths += asm_dependencies
-
- return dependencies
-
-def shallow_dependencies_of(asm_file_path, build_dirs=[]):
- asm_dependencies = set()
- bin_dependencies = set()
-
- with open(asm_file_path, 'r', encoding='utf-8') as f:
- for line in f:
- m = INCLUDE_RE.match(line)
- if m is None:
- continue
-
- keyword = m.group(1).upper()
- line = line.split(';', 1)[0]
- # RGBDS treats absolute(-looking) paths as relative,
- # so leading slashes should be stripped.
- path = line[line.index('"') + 1:line.rindex('"')].lstrip('/')
- if keyword == 'INCLUDE':
- asm_dependencies.add(path)
- else:
- if os.path.isfile(path) or not build_dirs:
- bin_dependencies.add(path)
- else:
- bin_dependencies.update(os.path.join(d, path) for d in build_dirs)
-
- return asm_dependencies, bin_dependencies
-
-def main(argv):
- script_name = os.path.basename(__file__)
- parser = argparse.ArgumentParser(prog=script_name, add_help=False)
- parser.add_argument('-h', '--help', action='help', help="Show this help and exit.")
- parser.add_argument('-b', metavar="build directories", action='append', default=[],
- help="Build directory to generate dependencies for "
- "if files don't exist at the exact path specified. "
- "Multiple build directories may be specified.")
- parser.add_argument('files', metavar='file', nargs='+',
- help="An assembly file to generate dependencies for.")
-
- args = parser.parse_args(argv)
-
- for path, dependencies in dependencies_in(args.files, args.b).items():
- # It seems that if A depends on B which depends on C, and
- # C is modified, Make needs you to change the modification
- # time of B too. That's the reason for the "@touch $@".
- # This does mean mtimes on .asm files are updated when building,
- # but the alternative opens up a whole can of problems.
- if dependencies:
- print("{}: {}\n\t@touch $@".format(path, ' '.join(dependencies)))
-
-if __name__ == '__main__':
- main(sys.argv[1:])