1
0
Fork 0
mirror of https://github.com/zeldaret/oot.git synced 2024-11-25 09:45:02 +00:00
oot/tools/audio/samplebank_compiler.c
Tharo d3b9ba17da
[Audio 4/?] Build Samplebanks, match Audiotable (#2032)
* [Audio 4/?] Build Samplebanks, match Audiotable

* Fix some makefile formatting

* Add missing scope in MARK chunk handling

* Add comment to generate asm file when buffer bug data is emitted, remove duplicate CC4 definition

* Adjust comment

* SBCFLAGS

* Remove unnecessary comments on notes_lut

* Split build directories creation command into several to avoid it becoming too long

* objcopy -j -> --only-section

* Fix mkdir warning when extracted/VERSION/assets doesn't exist
2024-08-14 20:54:31 -04:00

203 lines
6 KiB
C

/**
* SPDX-FileCopyrightText: Copyright (C) 2024 ZeldaRET
* SPDX-License-Identifier: MPL-2.0
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
#include <assert.h>
#include <stdio.h>
#include <string.h>
#include "xml.h"
#include "aifc.h"
#include "samplebank.h"
#include "util.h"
NORETURN static void
usage(const char *progname)
{
fprintf(stderr, "Usage: %s [--matching] [--makedepend <out.d>] <in.xml> <out.s>\n", progname);
exit(EXIT_FAILURE);
}
int
main(int argc, char **argv)
{
static uint8_t match_buf[BUG_BUF_SIZE];
const char *filename = NULL;
xmlDocPtr document;
const char *outfilename = NULL;
const char *mdfilename = NULL;
FILE *mdfile;
FILE *outf;
samplebank sb;
uint8_t *match_buf_ptr;
size_t match_buf_pos;
bool matching = false;
// parse args
#define arg_error(fmt, ...) \
do { \
fprintf(stderr, fmt "\n", ##__VA_ARGS__); \
usage(argv[0]); \
} while (0)
int argn = 0;
for (int i = 1; i < argc; i++) {
if (argv[i][0] == '-') {
// Optional args
if (strequ(argv[i], "--matching")) {
if (matching)
arg_error("Received --matching option twice");
matching = true;
continue;
}
if (strequ(argv[i], "--makedepend")) {
if (mdfilename != NULL)
arg_error("Received --makedepend option twice");
if (i + 1 == argc)
arg_error("--makedepend missing required argument");
mdfilename = argv[++i];
continue;
}
arg_error("Unknown option \"%s\"", argv[i]);
} else {
// Required args
switch (argn) {
case 0:
filename = argv[i];
break;
case 1:
outfilename = argv[i];
break;
default:
arg_error("Unknown positional argument \"%s\"", argv[i]);
break;
}
argn++;
}
}
if (argn != 2)
arg_error("Not enough positional arguments");
#undef arg_error
// open xml
document = xmlReadFile(filename, NULL, XML_PARSE_NONET);
if (document == NULL)
return EXIT_FAILURE;
// parse xml
read_samplebank_xml(&sb, document);
// open output asm file
outf = fopen(outfilename, "w");
if (outf == NULL)
error("Unable to open output file [%s] for writing", outfilename);
// open output dep file if applicable
if (mdfilename != NULL) {
mdfile = fopen(mdfilename, "w");
if (mdfile == NULL)
error("Unable to open dependency file [%s] for writing", mdfilename);
fprintf(mdfile, "%s: \\\n %s", outfilename, filename);
}
// write output
fprintf(outf,
// clang-format off
".rdata" "\n"
".balign 16" "\n"
"\n"
".global %s_Start" "\n"
"%s_Start:" "\n"
"$start:" "\n",
// clang-format on
sb.name, sb.name);
// original tool appears to have a buffer clearing bug involving a buffer sized BUG_BUF_SIZE
match_buf_ptr = (matching) ? match_buf : NULL;
match_buf_pos = 0;
for (size_t i = 0; i < sb.num_samples; i++) {
const char *name = sb.sample_names[i];
const char *path = sb.sample_paths[i];
bool is_sample = sb.is_sample[i];
if (mdfilename != NULL)
fprintf(mdfile, " \\\n %s", path);
if (!is_sample) {
// blob
fprintf(outf,
// clang-format off
"\n"
"# BLOB %s" "\n"
"\n"
".incbin \"%s\"" "\n"
"\n"
".balign 16" "\n"
"\n",
// clang-format on
name, path);
continue;
}
// aifc sample
fprintf(outf,
// clang-format off
"\n"
"# SAMPLE %lu" "\n"
"\n"
".global %s_%s_Abs" "\n"
"%s_%s_Abs:" "\n"
".global %s_%s_Off" "\n"
".set %s_%s_Off, . - $start" "\n"
"\n",
// clang-format on
i, sb.name, name, sb.name, name, sb.name, name, sb.name, name);
aifc_data aifc;
aifc_read(&aifc, path, match_buf_ptr, &match_buf_pos);
fprintf(outf, ".incbin \"%s\", 0x%lX, 0x%lX\n", path, aifc.ssnd_offset, aifc.ssnd_size);
if (matching && sb.buffer_bug && i == sb.num_samples - 1) {
// emplace garbage
size_t end = ALIGN16(match_buf_pos);
fprintf(outf, "\n# Garbage data from buffer bug\n");
for (; match_buf_pos < end; match_buf_pos++)
fprintf(outf, ".byte 0x%02X\n", match_buf[match_buf_pos]);
} else {
fputs("\n.balign 16\n", outf);
}
aifc_dispose(&aifc);
}
if (mdfilename != NULL) {
fputs("\n", mdfile);
fclose(mdfile);
}
fprintf(outf,
// clang-format off
".global %s_Size" "\n"
".set %s_Size, . - $start" "\n",
// clang-format on
sb.name, sb.name);
fclose(outf);
xmlFreeDoc(document);
return EXIT_SUCCESS;
}