mirror of
https://github.com/zeldaret/oot.git
synced 2025-01-26 02:27:02 +00:00
eb660765b1
* Change linker script so gGameOverTimer can be in z_game_over.c * gGameOverTimer -> sGameOverTimer * include_data_only_with_rodata -> include_data_only_within_rodata
378 lines
11 KiB
C
378 lines
11 KiB
C
#include <ctype.h>
|
|
#include <stdarg.h>
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "util.h"
|
|
#include "spec.h"
|
|
|
|
#define ARRAY_COUNT(arr) (sizeof(arr) / sizeof(arr[0]))
|
|
|
|
static struct Segment *add_segment(struct Segment **segments, int *segments_count)
|
|
{
|
|
struct Segment *seg;
|
|
|
|
(*segments_count)++;
|
|
*segments = realloc(*segments, *segments_count * sizeof(**segments));
|
|
|
|
seg = &(*segments)[*segments_count - 1];
|
|
memset(seg, 0, sizeof(*seg));
|
|
|
|
seg->align = 16;
|
|
|
|
return seg;
|
|
}
|
|
|
|
static char *skip_whitespace(char *str)
|
|
{
|
|
while (isspace(*str))
|
|
str++;
|
|
return str;
|
|
}
|
|
|
|
// null terminates the current token and returns a pointer to the next token
|
|
static char *token_split(char *str)
|
|
{
|
|
while (!isspace(*str))
|
|
{
|
|
if (*str == 0)
|
|
return str; // end of string
|
|
str++;
|
|
}
|
|
*str = 0; // terminate token
|
|
str++;
|
|
|
|
return skip_whitespace(str);
|
|
}
|
|
|
|
// null terminates the current line and returns a pointer to the next line
|
|
static char *line_split(char *str)
|
|
{
|
|
while (*str != '\n')
|
|
{
|
|
if (*str == 0)
|
|
return str; // end of string
|
|
str++;
|
|
}
|
|
*str = 0; // terminate line
|
|
return str + 1;
|
|
}
|
|
|
|
static bool parse_number(const char *str, unsigned int *num)
|
|
{
|
|
char *endptr;
|
|
long int n = strtol(str, &endptr, 0);
|
|
*num = n;
|
|
return endptr > str;
|
|
}
|
|
|
|
static bool parse_flags(char *str, unsigned int *flags)
|
|
{
|
|
unsigned int f = 0;
|
|
|
|
while (str[0] != 0)
|
|
{
|
|
char *next = token_split(str);
|
|
|
|
if (strcmp(str, "BOOT") == 0)
|
|
f |= FLAG_BOOT;
|
|
else if (strcmp(str, "OBJECT") == 0)
|
|
f |= FLAG_OBJECT;
|
|
else if (strcmp(str, "RAW") == 0)
|
|
f |= FLAG_RAW;
|
|
else if (strcmp(str, "NOLOAD") == 0)
|
|
f |= FLAG_NOLOAD;
|
|
else if (strcmp(str, "SYMS") == 0)
|
|
f |= FLAG_SYMS;
|
|
else
|
|
return false;
|
|
|
|
str = next;
|
|
}
|
|
*flags = f;
|
|
return true;
|
|
}
|
|
|
|
static bool parse_quoted_string(char *str, char **out)
|
|
{
|
|
if (*str != '"')
|
|
return false;
|
|
|
|
str++;
|
|
*out = str;
|
|
|
|
while (*str != '"')
|
|
{
|
|
if (*str == 0)
|
|
return false; // unterminated quote
|
|
str++;
|
|
}
|
|
*str = 0;
|
|
str++;
|
|
|
|
str = skip_whitespace(str);
|
|
if (*str != 0)
|
|
return false; // garbage after filename
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool is_pow_of_2(unsigned int n)
|
|
{
|
|
return (n & (n - 1)) == 0;
|
|
}
|
|
|
|
static const char *const stmtNames[] =
|
|
{
|
|
[STMT_address] = "address",
|
|
[STMT_after] = "after",
|
|
[STMT_align] = "align",
|
|
[STMT_beginseg] = "beginseg",
|
|
[STMT_compress] = "compress",
|
|
[STMT_endseg] = "endseg",
|
|
[STMT_entry] = "entry",
|
|
[STMT_flags] = "flags",
|
|
[STMT_include] = "include",
|
|
[STMT_include_data_only_within_rodata] = "include_data_only_within_rodata",
|
|
[STMT_include_no_data] = "include_no_data",
|
|
[STMT_name] = "name",
|
|
[STMT_number] = "number",
|
|
[STMT_romalign] = "romalign",
|
|
[STMT_stack] = "stack",
|
|
[STMT_increment] = "increment",
|
|
[STMT_pad_text] = "pad_text",
|
|
};
|
|
|
|
STMTId get_stmt_id_by_stmt_name(const char *stmtName, int lineNum) {
|
|
STMTId stmt;
|
|
|
|
for (stmt = 0; stmt < ARRAY_COUNT(stmtNames); stmt++) {
|
|
if (strcmp(stmtName, stmtNames[stmt]) == 0)
|
|
return stmt;
|
|
}
|
|
util_fatal_error("line %i: unknown statement '%s'", lineNum, stmtName);
|
|
return -1;
|
|
}
|
|
|
|
bool parse_segment_statement(struct Segment *currSeg, STMTId stmt, char* args, int lineNum) {
|
|
// ensure no duplicates (except for 'include' or 'pad_text')
|
|
if (stmt != STMT_include && stmt != STMT_include_data_only_within_rodata &&
|
|
stmt != STMT_include_no_data && stmt != STMT_pad_text &&
|
|
(currSeg->fields & (1 << stmt)))
|
|
util_fatal_error("line %i: duplicate '%s' statement", lineNum, stmtNames[stmt]);
|
|
|
|
currSeg->fields |= 1 << stmt;
|
|
|
|
// statements valid within a segment definition
|
|
switch (stmt)
|
|
{
|
|
case STMT_beginseg:
|
|
util_fatal_error("line %i: '%s' inside of a segment definition", lineNum, stmtNames[stmt]);
|
|
break;
|
|
case STMT_endseg:
|
|
// verify segment data
|
|
if (currSeg->name == NULL)
|
|
util_fatal_error("line %i: no name specified for segment", lineNum);
|
|
if (currSeg->includesCount == 0)
|
|
util_fatal_error("line %i: no includes specified for segment", lineNum);
|
|
return true;
|
|
break;
|
|
case STMT_name:
|
|
if (!parse_quoted_string(args, &currSeg->name))
|
|
util_fatal_error("line %i: invalid name", lineNum);
|
|
break;
|
|
case STMT_after:
|
|
if (!parse_quoted_string(args, &currSeg->after))
|
|
util_fatal_error("line %i: invalid name for 'after'", lineNum);
|
|
break;
|
|
case STMT_address:
|
|
if (!parse_number(args, &currSeg->address))
|
|
util_fatal_error("line %i: expected number after 'address'", lineNum);
|
|
break;
|
|
case STMT_number:
|
|
if (!parse_number(args, &currSeg->number))
|
|
util_fatal_error("line %i: expected number after 'number'", lineNum);
|
|
break;
|
|
case STMT_flags:
|
|
if (!parse_flags(args, &currSeg->flags))
|
|
util_fatal_error("line %i: invalid flags", lineNum);
|
|
break;
|
|
case STMT_align:
|
|
if (!parse_number(args, &currSeg->align))
|
|
util_fatal_error("line %i: expected number after 'align'", lineNum);
|
|
if (!is_pow_of_2(currSeg->align))
|
|
util_fatal_error("line %i: alignment is not a power of two", lineNum);
|
|
break;
|
|
case STMT_romalign:
|
|
if (!parse_number(args, &currSeg->romalign))
|
|
util_fatal_error("line %i: expected number after 'romalign'", lineNum);
|
|
if (!is_pow_of_2(currSeg->romalign))
|
|
util_fatal_error("line %i: alignment is not a power of two", lineNum);
|
|
break;
|
|
case STMT_include:
|
|
case STMT_include_data_only_within_rodata:
|
|
case STMT_include_no_data:
|
|
currSeg->includesCount++;
|
|
currSeg->includes = realloc(currSeg->includes, currSeg->includesCount * sizeof(*currSeg->includes));
|
|
|
|
if (!parse_quoted_string(args, &currSeg->includes[currSeg->includesCount - 1].fpath))
|
|
util_fatal_error("line %i: invalid filename", lineNum);
|
|
|
|
currSeg->includes[currSeg->includesCount - 1].linkerPadding = 0;
|
|
currSeg->includes[currSeg->includesCount - 1].dataOnlyWithinRodata = (stmt == STMT_include_data_only_within_rodata);
|
|
currSeg->includes[currSeg->includesCount - 1].noData = (stmt == STMT_include_no_data);
|
|
break;
|
|
case STMT_increment:
|
|
if (!parse_number(args, &currSeg->increment))
|
|
util_fatal_error("line %i: expected number after 'increment'", lineNum);
|
|
break;
|
|
case STMT_compress:
|
|
currSeg->compress = true;
|
|
break;
|
|
case STMT_pad_text:
|
|
currSeg->includes[currSeg->includesCount - 1].linkerPadding += 0x10;
|
|
break;
|
|
default:
|
|
fprintf(stderr, "warning: '%s' is not implemented\n", stmtNames[stmt]);
|
|
break;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* `segments` should be freed with `free_rom_spec` after use.
|
|
* Will write to the contents of `spec` to introduce string terminating '\0's.
|
|
* `segments` also contains pointers to inside `spec`, so `spec` should not be freed before `segments`
|
|
*/
|
|
void parse_rom_spec(char *spec, struct Segment **segments, int *segment_count)
|
|
{
|
|
int lineNum = 1;
|
|
char *line = spec;
|
|
|
|
struct Segment *currSeg = NULL;
|
|
|
|
// iterate over lines
|
|
while (line[0] != 0)
|
|
{
|
|
char *nextLine = line_split(line);
|
|
char* stmtName;
|
|
|
|
if (line[0] != 0)
|
|
{
|
|
stmtName = skip_whitespace(line);
|
|
}
|
|
|
|
if (line[0] != 0 && stmtName[0] != 0)
|
|
{
|
|
char *args = token_split(stmtName);
|
|
STMTId stmt = get_stmt_id_by_stmt_name(stmtName, lineNum);
|
|
|
|
if (currSeg != NULL)
|
|
{
|
|
bool segmentEnded = parse_segment_statement(currSeg, stmt, args, lineNum);
|
|
if (segmentEnded) {
|
|
currSeg = NULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// commands valid outside a segment definition
|
|
switch (stmt)
|
|
{
|
|
case STMT_beginseg:
|
|
currSeg = add_segment(segments, segment_count);
|
|
currSeg->includes = NULL;
|
|
break;
|
|
case STMT_endseg:
|
|
util_fatal_error("line %i: '%s' outside of a segment definition", lineNum, stmtName);
|
|
break;
|
|
default:
|
|
fprintf(stderr, "warning: '%s' is not implemented\n", stmtName);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
line = nextLine;
|
|
lineNum++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Parses the spec, looking only for the segment with the name `segmentName`.
|
|
* Returns true if the segment was found, false otherwise
|
|
*
|
|
* @param[out] dstSegment The Segment to be filled. Will only contain the data of the searched segment, or garbage if the segment was not found. dstSegment must be previously allocated, a stack variable is recommended
|
|
* @param[in,out] spec A null-terminated string containing the whole spec file. This string will be modified by this function
|
|
* @param[in] segmentName The name of the segment being searched
|
|
*/
|
|
bool get_single_segment_by_name(struct Segment* dstSegment, char *spec, const char *segmentName) {
|
|
bool insideSegment = false;
|
|
int lineNum = 1;
|
|
char *line = spec;
|
|
|
|
memset(dstSegment, 0, sizeof(struct Segment));
|
|
|
|
// iterate over lines
|
|
while (line[0] != '\0') {
|
|
char *nextLine = line_split(line);
|
|
char* stmtName = skip_whitespace(line);
|
|
|
|
if (stmtName[0] != '\0') {
|
|
char *args = token_split(stmtName);
|
|
STMTId stmt = get_stmt_id_by_stmt_name(stmtName, lineNum);
|
|
|
|
if (insideSegment) {
|
|
bool segmentEnded = parse_segment_statement(dstSegment, stmt, args, lineNum);
|
|
|
|
if (stmt == STMT_name) {
|
|
if (strcmp(segmentName, dstSegment->name) != 0) {
|
|
// Not the segment we are looking for
|
|
insideSegment = false;
|
|
}
|
|
} else if (segmentEnded) {
|
|
return true;
|
|
}
|
|
} else {
|
|
if (stmt == STMT_beginseg) {
|
|
insideSegment = true;
|
|
if (dstSegment->includes != NULL) {
|
|
free(dstSegment->includes);
|
|
}
|
|
memset(dstSegment, 0, sizeof(struct Segment));
|
|
}
|
|
}
|
|
}
|
|
|
|
line = nextLine;
|
|
lineNum++;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @brief Frees the elements of the passed Segment. Will not free the pointer itself
|
|
*
|
|
* @param segment
|
|
*/
|
|
void free_single_segment_elements(struct Segment *segment) {
|
|
if (segment->includes != NULL) {
|
|
free(segment->includes);
|
|
}
|
|
}
|
|
|
|
void free_rom_spec(struct Segment *segments, int segment_count)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < segment_count; i++)
|
|
{
|
|
if (segments[i].includes != NULL)
|
|
free(segments[i].includes);
|
|
}
|
|
free(segments);
|
|
}
|