2023-11-16 22:01:41 +03:00
|
|
|
#!/usr/bin/python3
|
|
|
|
|
2023-11-16 22:29:52 +03:00
|
|
|
from os.path import exists, join as path_join
|
|
|
|
from os import system, environ, makedirs
|
|
|
|
from argparse import ArgumentParser
|
|
|
|
from sys import stdin
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
# Paths
|
|
|
|
PATHS = {
|
2024-01-02 18:13:28 +03:00
|
|
|
"src_dir": "src",
|
|
|
|
"build_dir": "build",
|
2023-11-16 22:29:52 +03:00
|
|
|
"template": "template.c",
|
|
|
|
"substitute": "substituted.c",
|
2023-12-30 15:31:11 +03:00
|
|
|
"output": "render_bytebeat",
|
|
|
|
"fwrite_le_header": "fwrite_le.h",
|
|
|
|
"fwrite_le": "fwrite_le.c"
|
2023-11-16 22:29:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
# Solve paths
|
|
|
|
PATHS["template"] = path_join(PATHS["src_dir"], PATHS["template"])
|
|
|
|
PATHS["substitute"] = path_join(PATHS["build_dir"], PATHS["substitute"])
|
|
|
|
PATHS["output"] = path_join(PATHS["build_dir"], PATHS["output"])
|
2024-01-02 18:13:28 +03:00
|
|
|
PATHS["fwrite_le_header"] = PATHS["src_dir"] + "/" + PATHS["fwrite_le_header"]
|
|
|
|
PATHS["fwrite_le"] = path_join(PATHS["src_dir"], PATHS["fwrite_le"])
|
2023-11-16 22:29:52 +03:00
|
|
|
|
2023-12-30 15:40:37 +03:00
|
|
|
# Add `.` directory before all paths for compilation
|
|
|
|
PATHS["template"] = path_join(".", PATHS["template"])
|
|
|
|
PATHS["substitute"] = path_join(".", PATHS["substitute"])
|
|
|
|
PATHS["output"] = path_join(".", PATHS["output"])
|
2023-12-30 15:44:50 +03:00
|
|
|
PATHS["fwrite_le"] = path_join(".", PATHS["fwrite_le"])
|
2023-12-30 15:40:37 +03:00
|
|
|
|
2023-11-16 22:29:52 +03:00
|
|
|
# Default parameters
|
2023-11-16 22:01:41 +03:00
|
|
|
DEFAULT_PARAMETERS = {
|
|
|
|
"CC": "gcc",
|
2024-01-09 17:00:55 +03:00
|
|
|
"CFLAGS": "-Ofast -Wall -Wextra -Wpedantic -Wno-unused-variable -Wno-unused-but-set-variable -Wno-dangling-else -Wno-parentheses -std=c99",
|
2023-11-16 22:29:52 +03:00
|
|
|
"INPUT_FILE": PATHS["substitute"],
|
|
|
|
"OUTPUT_FILE": PATHS["output"]
|
2023-11-16 22:01:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
def fetch(name: str):
|
2023-12-24 19:20:47 +03:00
|
|
|
if from_env := environ.get(name):
|
2024-01-07 02:20:34 +03:00
|
|
|
return from_env
|
|
|
|
else:
|
|
|
|
return DEFAULT_PARAMETERS[name]
|
2023-11-16 22:01:41 +03:00
|
|
|
|
|
|
|
def read_file(path: str) -> str:
|
|
|
|
return open(path, "r", encoding="utf-8-sig").read()
|
|
|
|
|
|
|
|
def rewrite_file(path: str, content: str):
|
2024-01-09 15:54:00 +03:00
|
|
|
return open(path, "w", encoding="utf-8").write(content)
|
2023-11-16 22:01:41 +03:00
|
|
|
|
|
|
|
def read_from_file_or_stdin(path: str) -> str:
|
|
|
|
if path == "-":
|
|
|
|
return "\n".join(stdin)
|
|
|
|
elif exists(path):
|
|
|
|
return read_file(path)
|
|
|
|
else:
|
|
|
|
print("The specified file doesn't exist")
|
|
|
|
raise SystemExit
|
|
|
|
|
2023-12-03 15:55:08 +03:00
|
|
|
def substitute_vars(replacements: dict, text: str) -> str:
|
|
|
|
for placeholder, replacement in replacements.items():
|
|
|
|
text = text.replace(f"`{placeholder}`", str(replacement))
|
|
|
|
return text
|
2023-11-16 22:01:41 +03:00
|
|
|
|
|
|
|
CC = fetch("CC")
|
2023-12-30 14:57:32 +03:00
|
|
|
CFLAGS = fetch("CFLAGS")
|
2023-11-16 22:29:52 +03:00
|
|
|
INPUT_FILE = fetch("INPUT_FILE")
|
2023-11-16 22:01:41 +03:00
|
|
|
OUTPUT_FILE = fetch("OUTPUT_FILE")
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
parser = ArgumentParser(description=\
|
|
|
|
"Substitutes supplied C (non-JavaScript!) bytebeat into the template, "
|
2024-01-09 16:48:34 +03:00
|
|
|
"then attempts to compile the instance of the template. Accepts "
|
2023-12-24 20:05:48 +03:00
|
|
|
"environmental variables `CC`, `CFLAGS`, `INPUT_FILE`, "
|
2023-11-16 22:01:41 +03:00
|
|
|
"`OUTPUT_FILE`.")
|
|
|
|
parser.add_argument("file", type=str,
|
2023-12-03 15:44:52 +03:00
|
|
|
help="bytebeat formula file (use `-` to read from stdin)")
|
2023-11-16 22:01:41 +03:00
|
|
|
parser.add_argument("-r", "--sample-rate", default=8000, type=int,
|
|
|
|
help="sample rate (Hz)")
|
2023-11-18 14:15:48 +03:00
|
|
|
parser.add_argument("-p", "--final-sample-rate", default=None, type=int,
|
|
|
|
help="convert the output to a different sample rate (usually one that "
|
|
|
|
"is set in the system, to improve sound quality) during generation "
|
|
|
|
"(not just reinterpretation)")
|
2023-11-16 22:01:41 +03:00
|
|
|
parser.add_argument("-b", "--bit-depth", default=8, type=int,
|
|
|
|
help="bit depth")
|
|
|
|
parser.add_argument("-s", "--signed", default=False, action="store_true",
|
|
|
|
help="is signed?")
|
|
|
|
parser.add_argument("-c", "--channels", default=1, type=int,
|
|
|
|
help="amount of channels")
|
2023-12-03 15:43:40 +03:00
|
|
|
parser.add_argument("-t", "--seconds", default=None, type=int,
|
|
|
|
help="length in seconds (samples = sample rate * seconds) : "
|
|
|
|
"default = 30 seconds")
|
|
|
|
parser.add_argument("-l", "--samples", default=None, type=int,
|
|
|
|
help="length in samples (adds to `-t`; supports negative numbers) : "
|
|
|
|
"default = +0 samples")
|
2023-11-16 22:01:41 +03:00
|
|
|
parser.add_argument("-a", "--no-return", default=False, action="store_true",
|
|
|
|
help="do not insert return statement before the code")
|
2023-11-25 20:34:11 +03:00
|
|
|
parser.add_argument("-q", "--silent", default=False, action="store_true",
|
|
|
|
help="do not output anything during generation")
|
2023-11-25 21:03:27 +03:00
|
|
|
parser.add_argument("-v", "--verbose", default=False, action="store_true",
|
2023-11-25 21:06:19 +03:00
|
|
|
help="show progress during generation")
|
2023-11-16 22:01:41 +03:00
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
bytebeat_contents = read_from_file_or_stdin(args.file).strip()
|
|
|
|
|
|
|
|
if not bytebeat_contents:
|
|
|
|
print("No valid contents")
|
|
|
|
raise SystemExit
|
|
|
|
|
2023-11-16 22:29:52 +03:00
|
|
|
# - Compilation
|
|
|
|
makedirs(PATHS["build_dir"], exist_ok=True)
|
|
|
|
|
2023-11-16 22:01:41 +03:00
|
|
|
if not args.no_return: # Insert return statement
|
2023-11-25 17:28:22 +03:00
|
|
|
bytebeat_contents = f"return {bytebeat_contents}"
|
2023-11-16 22:01:41 +03:00
|
|
|
|
2023-11-18 14:15:48 +03:00
|
|
|
final_sample_rate_code = ""
|
|
|
|
if not args.final_sample_rate is None:
|
|
|
|
sample_rate_ratio = args.sample_rate / args.final_sample_rate
|
|
|
|
final_sample_rate_code = f"w *= {sample_rate_ratio}L;"
|
|
|
|
args.sample_rate = args.final_sample_rate
|
|
|
|
|
2023-12-03 15:43:40 +03:00
|
|
|
samples = 0
|
|
|
|
while True:
|
|
|
|
no_seconds = args.seconds is None or args.seconds == 0
|
|
|
|
no_samples = args.samples is None or args.samples == 0
|
|
|
|
seconds_exist = not no_seconds
|
|
|
|
samples_exist = not no_samples
|
|
|
|
|
|
|
|
if seconds_exist and args.seconds < 0:
|
|
|
|
print("CLI: Count of seconds can't be less than zero.")
|
|
|
|
raise SystemExit
|
|
|
|
|
|
|
|
if no_seconds and samples_exist:
|
|
|
|
samples = args.samples
|
|
|
|
elif seconds_exist and samples_exist:
|
|
|
|
samples = args.seconds * args.sample_rate + args.samples
|
|
|
|
elif seconds_exist and no_samples:
|
|
|
|
samples = args.seconds * args.sample_rate
|
|
|
|
elif no_seconds and no_samples:
|
|
|
|
args.seconds = 30 # default
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
print("CLI: Incorrect seconds/samples length format.")
|
|
|
|
raise SystemExit
|
|
|
|
break
|
|
|
|
|
|
|
|
if samples <= 0:
|
|
|
|
print("CLI: Count of samples should be greater than zero.")
|
|
|
|
raise SystemExit
|
|
|
|
|
2023-12-03 15:55:08 +03:00
|
|
|
rewrite_file(PATHS["substitute"], substitute_vars({
|
|
|
|
"bytebeat_contents": bytebeat_contents,
|
|
|
|
"sample_rate": args.sample_rate,
|
|
|
|
"final_sample_rate_code": final_sample_rate_code,
|
|
|
|
"bit_depth": args.bit_depth,
|
|
|
|
"is_signed": "1" if args.signed else "0",
|
|
|
|
"channels": args.channels,
|
|
|
|
"length": samples,
|
|
|
|
"silent_mode": "true" if args.silent else "false",
|
|
|
|
"verbose_mode": "true" if args.verbose and not args.silent else "false",
|
2023-12-30 15:44:50 +03:00
|
|
|
"fwrite_le": "../" + PATHS["fwrite_le_header"]
|
2023-12-03 15:55:08 +03:00
|
|
|
}, read_file(PATHS["template"])))
|
2023-11-16 22:01:41 +03:00
|
|
|
|
|
|
|
# Compile by invoking the shell script
|
|
|
|
print("Compiling")
|
|
|
|
|
2023-11-16 22:29:52 +03:00
|
|
|
# Let the system execute aliases by calling os.system
|
2023-12-30 15:44:50 +03:00
|
|
|
command = " ".join([CC, CFLAGS, INPUT_FILE, PATHS["fwrite_le"], "-o",
|
|
|
|
OUTPUT_FILE])
|
2024-01-02 18:14:04 +03:00
|
|
|
print(command, flush=True)
|
2023-12-30 15:32:41 +03:00
|
|
|
system(command)
|