mirror of
https://github.com/ivabus/rinth
synced 2024-11-24 01:15:08 +03:00
0.0.1: First alpha release
Signed-off-by: Ivan Bushchik <ivabus@ivabus.dev>
This commit is contained in:
commit
1aff34a06a
25 changed files with 1383 additions and 0 deletions
5
.gitignore
vendored
Normal file
5
.gitignore
vendored
Normal file
|
@ -0,0 +1,5 @@
|
|||
target/
|
||||
*.DS_Store
|
||||
.idea
|
||||
*.wav
|
||||
*.m4a
|
9
.rustfmt.toml
Normal file
9
.rustfmt.toml
Normal file
|
@ -0,0 +1,9 @@
|
|||
edition = "2021"
|
||||
hard_tabs = true
|
||||
merge_derives = true
|
||||
reorder_imports = true
|
||||
reorder_modules = true
|
||||
use_field_init_shorthand = true
|
||||
use_small_heuristics = "Off"
|
||||
wrap_comments = true
|
||||
comment_width = 80
|
464
Cargo.lock
generated
Normal file
464
Cargo.lock
generated
Normal file
|
@ -0,0 +1,464 @@
|
|||
# This file is automatically @generated by Cargo.
|
||||
# It is not intended for manual editing.
|
||||
version = 3
|
||||
|
||||
[[package]]
|
||||
name = "anstream"
|
||||
version = "0.6.8"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "628a8f9bd1e24b4e0db2b4bc2d000b001e7dd032d54afa60a68836aeec5aa54a"
|
||||
dependencies = [
|
||||
"anstyle",
|
||||
"anstyle-parse",
|
||||
"anstyle-query",
|
||||
"anstyle-wincon",
|
||||
"colorchoice",
|
||||
"utf8parse",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anstyle"
|
||||
version = "1.0.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "7079075b41f533b8c61d2a4d073c4676e1f8b249ff94a393b0595db304e0dd87"
|
||||
|
||||
[[package]]
|
||||
name = "anstyle-parse"
|
||||
version = "0.2.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c75ac65da39e5fe5ab759307499ddad880d724eed2f6ce5b5e8a26f4f387928c"
|
||||
dependencies = [
|
||||
"utf8parse",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anstyle-query"
|
||||
version = "1.0.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e28923312444cdd728e4738b3f9c9cac739500909bb3d3c94b43551b16517648"
|
||||
dependencies = [
|
||||
"windows-sys",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anstyle-wincon"
|
||||
version = "3.0.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1cd54b81ec8d6180e24654d0b371ad22fc3dd083b6ff8ba325b72e00c87660a7"
|
||||
dependencies = [
|
||||
"anstyle",
|
||||
"windows-sys",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "clap"
|
||||
version = "4.4.18"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1e578d6ec4194633722ccf9544794b71b1385c3c027efe0c55db226fc880865c"
|
||||
dependencies = [
|
||||
"clap_builder",
|
||||
"clap_derive",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "clap_builder"
|
||||
version = "4.4.18"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "4df4df40ec50c46000231c914968278b1eb05098cf8f1b3a518a95030e71d1c7"
|
||||
dependencies = [
|
||||
"anstream",
|
||||
"anstyle",
|
||||
"clap_lex",
|
||||
"strsim",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "clap_derive"
|
||||
version = "4.4.7"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "cf9804afaaf59a91e75b022a30fb7229a7901f60c755489cc61c9b423b836442"
|
||||
dependencies = [
|
||||
"heck",
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "clap_lex"
|
||||
version = "0.6.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "702fc72eb24e5a1e48ce58027a675bc24edd52096d5397d4aea7c6dd9eca0bd1"
|
||||
|
||||
[[package]]
|
||||
name = "colorchoice"
|
||||
version = "1.0.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "acbf1af155f9b9ef647e42cdc158db4b64a1b61f743629225fde6f3e0be2a7c7"
|
||||
|
||||
[[package]]
|
||||
name = "crossbeam-deque"
|
||||
version = "0.8.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "613f8cc01fe9cf1a3eb3d7f488fd2fa8388403e97039e2f73692932e291a770d"
|
||||
dependencies = [
|
||||
"crossbeam-epoch",
|
||||
"crossbeam-utils",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "crossbeam-epoch"
|
||||
version = "0.9.18"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e"
|
||||
dependencies = [
|
||||
"crossbeam-utils",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "crossbeam-utils"
|
||||
version = "0.8.19"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "248e3bacc7dc6baa3b21e405ee045c3047101a49145e7e9eca583ab4c2ca5345"
|
||||
|
||||
[[package]]
|
||||
name = "either"
|
||||
version = "1.9.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a26ae43d7bcc3b814de94796a5e736d4029efb0ee900c12e2d54c993ad1a1e07"
|
||||
|
||||
[[package]]
|
||||
name = "equivalent"
|
||||
version = "1.0.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5"
|
||||
|
||||
[[package]]
|
||||
name = "fnv"
|
||||
version = "1.0.7"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
|
||||
|
||||
[[package]]
|
||||
name = "hashbrown"
|
||||
version = "0.14.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "290f1a1d9242c78d09ce40a5e87e7554ee637af1351968159f4952f028f75604"
|
||||
|
||||
[[package]]
|
||||
name = "heck"
|
||||
version = "0.4.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8"
|
||||
|
||||
[[package]]
|
||||
name = "hound"
|
||||
version = "3.5.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "62adaabb884c94955b19907d60019f4e145d091c75345379e70d1ee696f7854f"
|
||||
|
||||
[[package]]
|
||||
name = "indexmap"
|
||||
version = "2.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d530e1a18b1cb4c484e6e34556a0d948706958449fca0cab753d649f2bce3d1f"
|
||||
dependencies = [
|
||||
"equivalent",
|
||||
"hashbrown",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "itoa"
|
||||
version = "1.0.10"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b1a46d1a171d865aa5f83f92695765caa047a9b4cbae2cbf37dbd613a793fd4c"
|
||||
|
||||
[[package]]
|
||||
name = "memchr"
|
||||
version = "2.7.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "523dc4f511e55ab87b694dc30d0f820d60906ef06413f93d4d7a1385599cc149"
|
||||
|
||||
[[package]]
|
||||
name = "meval"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "f79496a5651c8d57cd033c5add8ca7ee4e3d5f7587a4777484640d9cb60392d9"
|
||||
dependencies = [
|
||||
"fnv",
|
||||
"nom",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "midly"
|
||||
version = "0.5.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "207d755f4cb882d20c4da58d707ca9130a0c9bc5061f657a4f299b8e36362b7a"
|
||||
dependencies = [
|
||||
"rayon",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "nom"
|
||||
version = "1.2.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a5b8c256fd9471521bcb84c3cdba98921497f1a331cbc15b8030fc63b82050ce"
|
||||
|
||||
[[package]]
|
||||
name = "proc-macro2"
|
||||
version = "1.0.76"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "95fc56cda0b5c3325f5fbbd7ff9fda9e02bb00bb3dac51252d2f1bfa1cb8cc8c"
|
||||
dependencies = [
|
||||
"unicode-ident",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "quote"
|
||||
version = "1.0.35"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "291ec9ab5efd934aaf503a6466c5d5251535d108ee747472c3977cc5acc868ef"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rayon"
|
||||
version = "1.8.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "9c27db03db7734835b3f53954b534c91069375ce6ccaa2e065441e07d9b6cdb1"
|
||||
dependencies = [
|
||||
"either",
|
||||
"rayon-core",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rayon-core"
|
||||
version = "1.12.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5ce3fb6ad83f861aac485e76e1985cd109d9a3713802152be56c3b1f0e0658ed"
|
||||
dependencies = [
|
||||
"crossbeam-deque",
|
||||
"crossbeam-utils",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rinth-midi"
|
||||
version = "0.0.1"
|
||||
dependencies = [
|
||||
"clap",
|
||||
"midly",
|
||||
"rinth-types",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rinth-synth"
|
||||
version = "0.0.1"
|
||||
dependencies = [
|
||||
"clap",
|
||||
"hound",
|
||||
"meval",
|
||||
"rinth-types",
|
||||
"serde",
|
||||
"serde_yaml",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rinth-types"
|
||||
version = "0.0.1"
|
||||
dependencies = [
|
||||
"serde",
|
||||
"toml",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "ryu"
|
||||
version = "1.0.16"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "f98d2aa92eebf49b69786be48e4477826b256916e84a57ff2a4f21923b48eb4c"
|
||||
|
||||
[[package]]
|
||||
name = "serde"
|
||||
version = "1.0.195"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "63261df402c67811e9ac6def069e4786148c4563f4b50fd4bf30aa370d626b02"
|
||||
dependencies = [
|
||||
"serde_derive",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "serde_derive"
|
||||
version = "1.0.195"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "46fe8f8603d81ba86327b23a2e9cdf49e1255fb94a4c5f297f6ee0547178ea2c"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "serde_spanned"
|
||||
version = "0.6.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "eb3622f419d1296904700073ea6cc23ad690adbd66f13ea683df73298736f0c1"
|
||||
dependencies = [
|
||||
"serde",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "serde_yaml"
|
||||
version = "0.9.30"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b1bf28c79a99f70ee1f1d83d10c875d2e70618417fda01ad1785e027579d9d38"
|
||||
dependencies = [
|
||||
"indexmap",
|
||||
"itoa",
|
||||
"ryu",
|
||||
"serde",
|
||||
"unsafe-libyaml",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "strsim"
|
||||
version = "0.10.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623"
|
||||
|
||||
[[package]]
|
||||
name = "syn"
|
||||
version = "2.0.48"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "0f3531638e407dfc0814761abb7c00a5b54992b849452a0646b7f65c9f770f3f"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"unicode-ident",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "toml"
|
||||
version = "0.8.8"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a1a195ec8c9da26928f773888e0742ca3ca1040c6cd859c919c9f59c1954ab35"
|
||||
dependencies = [
|
||||
"serde",
|
||||
"serde_spanned",
|
||||
"toml_datetime",
|
||||
"toml_edit",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "toml_datetime"
|
||||
version = "0.6.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3550f4e9685620ac18a50ed434eb3aec30db8ba93b0287467bca5826ea25baf1"
|
||||
dependencies = [
|
||||
"serde",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "toml_edit"
|
||||
version = "0.21.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d34d383cd00a163b4a5b85053df514d45bc330f6de7737edfe0a93311d1eaa03"
|
||||
dependencies = [
|
||||
"indexmap",
|
||||
"serde",
|
||||
"serde_spanned",
|
||||
"toml_datetime",
|
||||
"winnow",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "unicode-ident"
|
||||
version = "1.0.12"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b"
|
||||
|
||||
[[package]]
|
||||
name = "unsafe-libyaml"
|
||||
version = "0.2.10"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ab4c90930b95a82d00dc9e9ac071b4991924390d46cbd0dfe566148667605e4b"
|
||||
|
||||
[[package]]
|
||||
name = "utf8parse"
|
||||
version = "0.2.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "711b9620af191e0cdc7468a8d14e709c3dcdb115b36f838e601583af800a370a"
|
||||
|
||||
[[package]]
|
||||
name = "windows-sys"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "282be5f36a8ce781fad8c8ae18fa3f9beff57ec1b52cb3de0789201425d9a33d"
|
||||
dependencies = [
|
||||
"windows-targets",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows-targets"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8a18201040b24831fbb9e4eb208f8892e1f50a37feb53cc7ff887feb8f50e7cd"
|
||||
dependencies = [
|
||||
"windows_aarch64_gnullvm",
|
||||
"windows_aarch64_msvc",
|
||||
"windows_i686_gnu",
|
||||
"windows_i686_msvc",
|
||||
"windows_x86_64_gnu",
|
||||
"windows_x86_64_gnullvm",
|
||||
"windows_x86_64_msvc",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows_aarch64_gnullvm"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "cb7764e35d4db8a7921e09562a0304bf2f93e0a51bfccee0bd0bb0b666b015ea"
|
||||
|
||||
[[package]]
|
||||
name = "windows_aarch64_msvc"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "bbaa0368d4f1d2aaefc55b6fcfee13f41544ddf36801e793edbbfd7d7df075ef"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_gnu"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a28637cb1fa3560a16915793afb20081aba2c92ee8af57b4d5f28e4b3e7df313"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_msvc"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ffe5e8e31046ce6230cc7215707b816e339ff4d4d67c65dffa206fd0f7aa7b9a"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_gnu"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3d6fa32db2bc4a2f5abeacf2b69f7992cd09dca97498da74a151a3132c26befd"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_gnullvm"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1a657e1e9d3f514745a572a6846d3c7aa7dbe1658c056ed9c3344c4109a6949e"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_msvc"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "dff9641d1cd4be8d1a070daf9e3773c5f67e78b4d9d42263020c057706765c04"
|
||||
|
||||
[[package]]
|
||||
name = "winnow"
|
||||
version = "0.5.34"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b7cf47b659b318dccbd69cc4797a39ae128f533dce7902a1096044d1967b9c16"
|
||||
dependencies = [
|
||||
"memchr",
|
||||
]
|
3
Cargo.toml
Normal file
3
Cargo.toml
Normal file
|
@ -0,0 +1,3 @@
|
|||
[workspace]
|
||||
members = [ "rinth-midi", "rinth-synth", "rinth-types" ]
|
||||
resolver = "2"
|
3
LICENSE
Normal file
3
LICENSE
Normal file
|
@ -0,0 +1,3 @@
|
|||
Rinth and all of it's subcrates are available for non-commercial (non-profit) personal use under the terms of BSD 3-Clause New (Revised) License license, see LICENSE-BSD
|
||||
|
||||
At the moment, the commercial use of Rinth is completely prohibited at the moment.
|
28
LICENSE-BSD
Normal file
28
LICENSE-BSD
Normal file
|
@ -0,0 +1,28 @@
|
|||
BSD 3-Clause License
|
||||
|
||||
Copyright (c) 2024, Ivan Bushchik
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice, this
|
||||
list of conditions and the following disclaimer.
|
||||
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
3. Neither the name of the copyright holder nor the names of its
|
||||
contributors may be used to endorse or promote products derived from
|
||||
this software without specific prior written permission.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
92
README.md
Normal file
92
README.md
Normal file
|
@ -0,0 +1,92 @@
|
|||
# Rinth synthesiser
|
||||
|
||||
FM and SSG synthesiser
|
||||
|
||||
## Usage
|
||||
|
||||
```shell
|
||||
rinth-synth b[uild] <path/to/project.yml> # Will build separate tracks
|
||||
rinth-synth m[aster] <path/to/project.yml> # Will "master" tracks into one track
|
||||
```
|
||||
|
||||
See `--help` for more
|
||||
|
||||
## File formats
|
||||
|
||||
### \<project\>.yml
|
||||
|
||||
```yaml
|
||||
name: <NAME>
|
||||
bpm: <BPM>
|
||||
channels:
|
||||
- type: <SSG|FM>
|
||||
path: <RELATIVE_TO_YML_PATH>
|
||||
volume: [Optional in [0;1]]
|
||||
```
|
||||
|
||||
Example:
|
||||
|
||||
```yaml
|
||||
name: Megalovania
|
||||
bpm: 210
|
||||
channels:
|
||||
- type: SSG
|
||||
path: _channel
|
||||
- type: FM
|
||||
path: _channel
|
||||
volume: 0.7
|
||||
```
|
||||
|
||||
### _channel (FM)
|
||||
|
||||
```
|
||||
/ Comment
|
||||
#<Frequency deviation>
|
||||
@<Modulating frequency>
|
||||
<Note in SPN> <Note value> [Time from previous note]
|
||||
```
|
||||
|
||||
Example:
|
||||
|
||||
```
|
||||
#1
|
||||
@2000
|
||||
D4 0.0625
|
||||
Dd5 0.0625 0.0125
|
||||
Db5 1/32+1/32 1/80
|
||||
```
|
||||
|
||||
### _channel (SSG)
|
||||
|
||||
```
|
||||
/ Comment
|
||||
# Ignored
|
||||
@ Ignored
|
||||
<Note in SPN> <Note value> [Time from previous note]
|
||||
```
|
||||
|
||||
Example:
|
||||
|
||||
```
|
||||
/ Comment
|
||||
# Comment 2
|
||||
@ Comment 3
|
||||
D4 0.0625
|
||||
Dd5 0.0625 0.0125
|
||||
Db5 1/32+1/32 1/80
|
||||
```
|
||||
More examples in [examples/](./examples)
|
||||
|
||||
## MIDI Converter
|
||||
|
||||
WIP MIDI converter is available in the `rinth-midi` crate.
|
||||
|
||||
### Usage
|
||||
|
||||
```shell
|
||||
rinth-midi <path/to/midi.mid>
|
||||
```
|
||||
|
||||
## License
|
||||
|
||||
The project is available for non-commercial personal use under the terms of the [BSD 3-Clause New (Revised) License](./LICENSE) and fully unavailable for any other kind of use.
|
165
examples/determination/main
Normal file
165
examples/determination/main
Normal file
|
@ -0,0 +1,165 @@
|
|||
#1
|
||||
@4
|
||||
|
||||
G5 1/8
|
||||
Fd5 1/8
|
||||
E5 1/8
|
||||
D5 1/8
|
||||
E5 1/8
|
||||
B4 1/8
|
||||
|
||||
Cd5 1/4
|
||||
A4 1/4
|
||||
|
||||
E5 1/8
|
||||
Fd5 1/8
|
||||
|
||||
G5 1/4
|
||||
A5 1/4
|
||||
D6 1/4
|
||||
|
||||
B5 1/2+1/4
|
||||
|
||||
|
||||
G5 1/8
|
||||
Fd5 1/8
|
||||
E5 1/8
|
||||
D5 1/8
|
||||
E5 1/8
|
||||
B4 1/8
|
||||
|
||||
Cd5 1/4
|
||||
A4 1/4
|
||||
|
||||
E4 1/8
|
||||
Fd4 1/8
|
||||
|
||||
G4 1/4
|
||||
Fd4 1/4
|
||||
D4 1/4
|
||||
E4 1/2+1/4
|
||||
|
||||
G5 1/8
|
||||
Fd5 1/8
|
||||
E5 1/8
|
||||
D5 1/8
|
||||
E5 1/8
|
||||
B4 1/8
|
||||
|
||||
Cd5 1/4
|
||||
A4 1/4
|
||||
|
||||
E5 1/8
|
||||
Fd5 1/8
|
||||
|
||||
G5 1/4
|
||||
A5 1/4
|
||||
D6 1/4
|
||||
|
||||
B5 1/2+1/4
|
||||
|
||||
|
||||
G5 1/8
|
||||
Fd5 1/8
|
||||
E5 1/8
|
||||
D5 1/8
|
||||
E5 1/8
|
||||
B4 1/8
|
||||
|
||||
Cd5 1/4
|
||||
A4 1/4
|
||||
|
||||
E4 1/8
|
||||
Fd4 1/8
|
||||
|
||||
G4 1/4
|
||||
Fd4 1/4
|
||||
D4 1/4
|
||||
E4 1/2+1/4
|
||||
|
||||
|
||||
|
||||
|
||||
A5 1/8
|
||||
G5 1/8
|
||||
F5 1/8
|
||||
E5 1/8
|
||||
D5 1/8
|
||||
F5 1/8
|
||||
|
||||
E5 1/4
|
||||
B4 1/4
|
||||
|
||||
B4 1/8
|
||||
E5 1/8
|
||||
|
||||
A5 1/8
|
||||
G5 1/8
|
||||
F5 1/8
|
||||
E5 1/8
|
||||
D5 1/8
|
||||
F5 1/8
|
||||
|
||||
E5 1/2
|
||||
E4 1/8
|
||||
A4 1/8
|
||||
|
||||
D5 1/8
|
||||
Cd5 1/8
|
||||
B4 1/8
|
||||
A4 1/8
|
||||
B4 1/8
|
||||
Cd5 1/8
|
||||
|
||||
B4 1/4
|
||||
E4 1/4
|
||||
|
||||
E4 1/8
|
||||
Fd4 1/8
|
||||
|
||||
G4 1/4
|
||||
A4 1/4
|
||||
C5 1/4
|
||||
|
||||
B4 1/2+1/4
|
||||
|
||||
A5 1/8
|
||||
G5 1/8
|
||||
F5 1/8
|
||||
E5 1/8
|
||||
D5 1/8
|
||||
F5 1/8
|
||||
|
||||
E5 1/4
|
||||
B4 1/4
|
||||
|
||||
B4 1/8
|
||||
E5 1/8
|
||||
|
||||
A5 1/8
|
||||
G5 1/8
|
||||
F5 1/8
|
||||
E5 1/8
|
||||
D5 1/8
|
||||
F5 1/8
|
||||
|
||||
E5 1/2
|
||||
E4 1/8
|
||||
A4 1/8
|
||||
|
||||
D5 1/8
|
||||
Cd5 1/8
|
||||
B4 1/8
|
||||
A4 1/8
|
||||
B4 1/8
|
||||
Cd5 1/8
|
||||
|
||||
B4 1/4
|
||||
E4 1/4
|
||||
E4 1/8
|
||||
Fd4 1/8
|
||||
|
||||
G4 1/4
|
||||
Fd4 1/4
|
||||
D4 1/4
|
||||
E4 1/2+1/4
|
6
examples/determination/project.yml
Normal file
6
examples/determination/project.yml
Normal file
|
@ -0,0 +1,6 @@
|
|||
name: Determination
|
||||
bpm: 120
|
||||
channels:
|
||||
- type: FM
|
||||
path: main
|
||||
volume: 0.7
|
22
examples/megalovania-no-delay/channel
Normal file
22
examples/megalovania-no-delay/channel
Normal file
|
@ -0,0 +1,22 @@
|
|||
@2000
|
||||
D4 0.0625 0
|
||||
D4 0.0625 0
|
||||
D5 0.1875 0
|
||||
A4 0.1875 0
|
||||
Gd4 0.125 0
|
||||
G4 0.125 0
|
||||
F4 0.125 0
|
||||
D4 0.125 0.0
|
||||
F4 0.0625 0
|
||||
G4 0.0625 0
|
||||
@1000
|
||||
C4 0.0625 0.0
|
||||
C4 0.0625 0
|
||||
D5 0.1875 0
|
||||
A4 0.1875 0
|
||||
Gd4 0.125 0
|
||||
G4 0.125 0
|
||||
F4 0.125 0
|
||||
D4 0.0625 0.0
|
||||
F4 0.0625 0
|
||||
G4 0.0625 0.0
|
7
examples/megalovania-no-delay/project.yml
Normal file
7
examples/megalovania-no-delay/project.yml
Normal file
|
@ -0,0 +1,7 @@
|
|||
name: Megalovania
|
||||
bpm: 210
|
||||
channels:
|
||||
- type: SSG
|
||||
path: channel
|
||||
- type: FM
|
||||
path: channel
|
24
examples/megalovania/channel
Normal file
24
examples/megalovania/channel
Normal file
|
@ -0,0 +1,24 @@
|
|||
#1
|
||||
@1
|
||||
D4 0.0625 0.05
|
||||
D4 0.0625 1/80
|
||||
D5 0.1875 0.0125
|
||||
A4 0.1875 0.0125
|
||||
Gd4 0.125 0.0625
|
||||
G4 0.125 0.0625
|
||||
F4 0.125 0.0625
|
||||
D4 0.125 0.025
|
||||
F4 0.0625 0.0125
|
||||
G4 0.0625 0.0125
|
||||
#-1
|
||||
@-1
|
||||
C4 0.0625 0.05
|
||||
C4 0.0625 0.0125
|
||||
D5 0.1875 0.0125
|
||||
A4 0.1875 0.0125
|
||||
Gd4 0.125 0.0625
|
||||
G4 0.125 0.0625
|
||||
F4 0.125 0.0625
|
||||
D4 0.0625 0.025
|
||||
F4 0.0625 0.0125
|
||||
G4 0.0625 0.0125
|
9
examples/megalovania/project.yml
Normal file
9
examples/megalovania/project.yml
Normal file
|
@ -0,0 +1,9 @@
|
|||
name: Megalovania
|
||||
bpm: 210
|
||||
channels:
|
||||
- type: SSG
|
||||
path: channel
|
||||
volume: 0.3
|
||||
- type: FM
|
||||
path: channel
|
||||
volume: 1
|
12
rinth-midi/Cargo.toml
Normal file
12
rinth-midi/Cargo.toml
Normal file
|
@ -0,0 +1,12 @@
|
|||
[package]
|
||||
name = "rinth-midi"
|
||||
version = "0.0.1"
|
||||
edition = "2021"
|
||||
publish = false
|
||||
license-file = "../LICENSE"
|
||||
authors = [ "Ivan Bushchik <ivabus@ivabus.dev>" ]
|
||||
|
||||
[dependencies]
|
||||
rinth-types = { path="../rinth-types" }
|
||||
clap = { version = "4.4.18", features = ["derive"] }
|
||||
midly = "0.5.3"
|
73
rinth-midi/src/main.rs
Normal file
73
rinth-midi/src/main.rs
Normal file
|
@ -0,0 +1,73 @@
|
|||
use clap::Parser;
|
||||
use midly::num::{u28, u7};
|
||||
use midly::{MidiMessage, Smf, TrackEventKind};
|
||||
use rinth_types::note::{Note, Tone};
|
||||
use std::path::PathBuf;
|
||||
|
||||
#[derive(Parser)]
|
||||
#[command(version)]
|
||||
struct Args {
|
||||
midi: PathBuf,
|
||||
|
||||
#[arg(short, default_value = "0")]
|
||||
channel: u8,
|
||||
}
|
||||
fn main() {
|
||||
let args = Args::parse();
|
||||
let binding = std::fs::read(args.midi).unwrap();
|
||||
let smf = Smf::parse(binding.as_slice()).unwrap();
|
||||
let mut is_pressed = false;
|
||||
let mut delay = 0;
|
||||
let mut note_pressed = u7::new(0);
|
||||
let mut timer = u28::new(0);
|
||||
let mut notes: Vec<Note> = vec![];
|
||||
let note_names = ["C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B"];
|
||||
for track in smf.tracks.iter() {
|
||||
for event in track {
|
||||
if let TrackEventKind::Midi {
|
||||
channel: _channel,
|
||||
message,
|
||||
} = event.kind
|
||||
{
|
||||
match message {
|
||||
MidiMessage::NoteOff {
|
||||
key,
|
||||
..
|
||||
} => {
|
||||
if is_pressed && note_pressed == key {
|
||||
let note = Note {
|
||||
tone: Tone::from(&format!(
|
||||
"{}{}",
|
||||
note_names[(key.as_int() % 12) as usize],
|
||||
(key.as_int() / 12) - 1
|
||||
)),
|
||||
length: event.delta.as_int() as f32 / 1000.0,
|
||||
delay: delay as f32 / 1000.0,
|
||||
};
|
||||
notes.push(note);
|
||||
note_pressed = u7::new(0);
|
||||
is_pressed = false;
|
||||
}
|
||||
}
|
||||
MidiMessage::NoteOn {
|
||||
key,
|
||||
..
|
||||
} => {
|
||||
if !is_pressed {
|
||||
eprintln!("pressing on {}", key);
|
||||
is_pressed = true;
|
||||
note_pressed = key;
|
||||
timer = u28::new(0);
|
||||
delay = event.delta.as_int();
|
||||
}
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
timer += event.delta;
|
||||
}
|
||||
}
|
||||
for note in notes {
|
||||
println!("{} {} {}", note.tone, note.length, note.delay);
|
||||
}
|
||||
}
|
15
rinth-synth/Cargo.toml
Normal file
15
rinth-synth/Cargo.toml
Normal file
|
@ -0,0 +1,15 @@
|
|||
[package]
|
||||
name = "rinth-synth"
|
||||
version = "0.0.1"
|
||||
edition = "2021"
|
||||
publish = false
|
||||
license-file = "../LICENSE"
|
||||
authors = [ "Ivan Bushchik <ivabus@ivabus.dev>" ]
|
||||
|
||||
[dependencies]
|
||||
rinth-types = {path = "../rinth-types" }
|
||||
hound = "3.5.1"
|
||||
serde_yaml = "0.9.30"
|
||||
clap = { version = "4.4", features = ["derive"] }
|
||||
serde = { version = "1.0.195", features = ["derive"] }
|
||||
meval = "0.2.0"
|
101
rinth-synth/src/fm.rs
Normal file
101
rinth-synth/src/fm.rs
Normal file
|
@ -0,0 +1,101 @@
|
|||
use crate::shared::{add_delay, get_note};
|
||||
use rinth_types::file::Channel;
|
||||
use rinth_types::note::get_note_len;
|
||||
use rinth_types::{file::ChannelType, note::Note, traits::Synth};
|
||||
use std::f32::consts::PI;
|
||||
use std::fs;
|
||||
|
||||
pub struct FM {
|
||||
pub bpm: u16,
|
||||
pub notes: Vec<(Note, ModulationFreq, FrequencyDeviation)>,
|
||||
}
|
||||
|
||||
#[derive(Copy, Clone)]
|
||||
pub struct ModulationFreq(f32);
|
||||
|
||||
#[derive(Copy, Clone)]
|
||||
pub struct FrequencyDeviation(f32);
|
||||
|
||||
const FIGHT_CLICKS: usize = 128;
|
||||
|
||||
impl Synth for FM {
|
||||
fn from_channel(channel: Channel, bpm: u16) -> Self
|
||||
where
|
||||
Self: Sized,
|
||||
{
|
||||
let channel = match channel {
|
||||
Channel {
|
||||
path,
|
||||
channel_type: ChannelType::FM,
|
||||
..
|
||||
} => path,
|
||||
_ => {
|
||||
unreachable!()
|
||||
}
|
||||
};
|
||||
|
||||
let contents = fs::read(channel).unwrap();
|
||||
|
||||
let mut notes: Vec<(Note, ModulationFreq, FrequencyDeviation)> = vec![];
|
||||
let mut current_tone = ModulationFreq(440.0);
|
||||
let mut current_deviation = FrequencyDeviation(8.0);
|
||||
|
||||
for line in contents.split(|&x| x == b'\n') {
|
||||
if line.is_empty() {
|
||||
continue;
|
||||
}
|
||||
let s = String::from_utf8_lossy(line);
|
||||
let l = s.split_ascii_whitespace().collect::<Vec<&str>>();
|
||||
// Ignore comment
|
||||
if l[0].as_bytes()[0] == b'/' {
|
||||
continue;
|
||||
}
|
||||
// String in FM channel may be or carrier tone or note
|
||||
if l[0].as_bytes()[0] == b'@' {
|
||||
current_tone = ModulationFreq(l[0][1..].parse::<f32>().unwrap());
|
||||
continue;
|
||||
}
|
||||
if l[0].as_bytes()[0] == b'#' {
|
||||
current_deviation = FrequencyDeviation(l[0][1..].parse::<f32>().unwrap());
|
||||
continue;
|
||||
}
|
||||
notes.push((get_note(l), current_tone, current_deviation))
|
||||
}
|
||||
FM {
|
||||
bpm,
|
||||
notes,
|
||||
}
|
||||
}
|
||||
|
||||
fn synthesise(&self, sample_rate: u32) -> Vec<f32> {
|
||||
// t - time, m - modulating freq (carrier)
|
||||
let y: fn(f32, f32, f32, f32) -> f32 =
|
||||
|t: f32, f: f32, m: f32, d| (2_f32 * PI * m * t + d * (2_f32 * PI * f * t).sin()).cos();
|
||||
let mut stream: Vec<f32> = vec![];
|
||||
for (note, tone, deviation) in &self.notes {
|
||||
if note.delay != 0_f32 {
|
||||
add_delay(&mut stream, note.delay, self.bpm, sample_rate);
|
||||
}
|
||||
// Samples in _THIS_ note
|
||||
let samples = (get_note_len(note.length, self.bpm) * sample_rate as f32) as usize;
|
||||
for k in 0..samples {
|
||||
stream.push(
|
||||
y(k as f32 / sample_rate as f32, tone.0, note.tone.get_freq(), deviation.0)
|
||||
// This makes a little linear fade-in-out so we don't get "clicks"
|
||||
* if k <= FIGHT_CLICKS {
|
||||
k as f32 / FIGHT_CLICKS as f32
|
||||
} else if k
|
||||
>= samples
|
||||
- FIGHT_CLICKS
|
||||
{
|
||||
(samples
|
||||
- k) as f32 / FIGHT_CLICKS as f32
|
||||
} else {
|
||||
1.0
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
stream
|
||||
}
|
||||
}
|
127
rinth-synth/src/main.rs
Normal file
127
rinth-synth/src/main.rs
Normal file
|
@ -0,0 +1,127 @@
|
|||
use clap::Parser;
|
||||
use rinth_types::file::ChannelType;
|
||||
use rinth_types::traits::Synth;
|
||||
use std::cmp::max;
|
||||
use std::io::Read;
|
||||
use std::path::PathBuf;
|
||||
|
||||
mod fm;
|
||||
mod shared;
|
||||
mod ssg;
|
||||
|
||||
#[derive(Parser)]
|
||||
enum Commands {
|
||||
#[command(visible_alias = "b")]
|
||||
Build(Build),
|
||||
#[command(visible_alias = "m")]
|
||||
Master(Build),
|
||||
}
|
||||
|
||||
#[derive(Parser)]
|
||||
#[command(author, about, version)]
|
||||
struct Build {
|
||||
path: PathBuf,
|
||||
#[arg(short, long, default_value = "44100")]
|
||||
sample_rate: u32,
|
||||
}
|
||||
|
||||
fn truncate(sample: f32) -> f32 {
|
||||
if sample.abs() > 1_f32 {
|
||||
sample.signum()
|
||||
} else {
|
||||
sample
|
||||
}
|
||||
}
|
||||
|
||||
fn merge(master: &mut Vec<f32>, slave: Vec<f32>) {
|
||||
master.resize(max(master.len(), slave.len()), 0_f32);
|
||||
let slave_len = slave.len();
|
||||
for (n, sample) in &mut master.iter_mut().enumerate() {
|
||||
*sample = if n < slave_len {
|
||||
truncate(*sample + slave[n])
|
||||
} else {
|
||||
truncate(*sample)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let args = Commands::parse();
|
||||
|
||||
let (args, mastering) = match args {
|
||||
Commands::Build(args) => (args, false),
|
||||
Commands::Master(args) => (args, true),
|
||||
};
|
||||
let mut project_file = std::fs::File::open(&args.path).unwrap();
|
||||
let mut project = String::new();
|
||||
project_file.read_to_string(&mut project).unwrap();
|
||||
|
||||
let project: rinth_types::file::Header = serde_yaml::from_str(&project).unwrap();
|
||||
let spec = hound::WavSpec {
|
||||
channels: 1,
|
||||
sample_rate: args.sample_rate,
|
||||
bits_per_sample: 32,
|
||||
sample_format: hound::SampleFormat::Float,
|
||||
};
|
||||
let mut master = vec![];
|
||||
for channel in project.channels {
|
||||
let mut channel = channel;
|
||||
let mut channel_path = args.path.parent().unwrap().to_path_buf();
|
||||
channel_path.push(channel.path);
|
||||
channel.path = channel_path;
|
||||
|
||||
let synth: Box<dyn Synth> = match channel.channel_type {
|
||||
ChannelType::FM => Box::new(fm::FM::from_channel(channel.clone(), project.bpm)),
|
||||
ChannelType::SSG => Box::new(ssg::SSG::from_channel(channel.clone(), project.bpm)),
|
||||
};
|
||||
if !mastering {
|
||||
let mut writer = hound::WavWriter::create(
|
||||
format!(
|
||||
"{}-{:?}.wav",
|
||||
channel.path.as_os_str().to_str().unwrap(),
|
||||
channel.channel_type
|
||||
),
|
||||
spec,
|
||||
)
|
||||
.unwrap();
|
||||
synth.synthesise(args.sample_rate).iter().for_each(|sample| {
|
||||
writer
|
||||
.write_sample(if let Some(volume) = channel.volume {
|
||||
*sample * volume
|
||||
} else {
|
||||
*sample
|
||||
})
|
||||
.unwrap()
|
||||
});
|
||||
writer.finalize().unwrap();
|
||||
} else {
|
||||
merge(
|
||||
&mut master,
|
||||
synth
|
||||
.synthesise(args.sample_rate)
|
||||
.iter()
|
||||
.map(|sample| {
|
||||
if let Some(volume) = channel.volume {
|
||||
*sample * volume
|
||||
} else {
|
||||
*sample
|
||||
}
|
||||
})
|
||||
.collect(),
|
||||
);
|
||||
}
|
||||
}
|
||||
if mastering {
|
||||
let mut writer = hound::WavWriter::create(
|
||||
format!(
|
||||
"{}/{}-master.wav",
|
||||
args.path.parent().unwrap().to_path_buf().to_str().unwrap(),
|
||||
project.name
|
||||
),
|
||||
spec,
|
||||
)
|
||||
.unwrap();
|
||||
master.iter().for_each(|sample| writer.write_sample(*sample).unwrap());
|
||||
writer.finalize().unwrap();
|
||||
}
|
||||
}
|
27
rinth-synth/src/shared.rs
Normal file
27
rinth-synth/src/shared.rs
Normal file
|
@ -0,0 +1,27 @@
|
|||
use rinth_types::note::{get_note_len, Note, Tone};
|
||||
|
||||
pub const DEFAULT_DELAY: f32 = 0.0;
|
||||
|
||||
#[inline(always)]
|
||||
pub fn add_delay(stream: &mut Vec<f32>, delay: f32, bpm: u16, sample_rate: u32) {
|
||||
// println!("Pushing delay of len {}s", get_note_len(delay, bpm));
|
||||
stream.append(&mut vec![0_f32; (get_note_len(delay, bpm) * sample_rate as f32) as usize]);
|
||||
}
|
||||
|
||||
pub fn get_note(l: Vec<&str>) -> Note {
|
||||
Note {
|
||||
tone: Tone::from(l[0]),
|
||||
length: match l[1].parse::<f32>() {
|
||||
Ok(f) => f,
|
||||
Err(_) => meval::eval_str(l[1]).unwrap() as f32,
|
||||
},
|
||||
delay: if l.len() > 2 {
|
||||
match l[2].parse::<f32>() {
|
||||
Ok(f) => f,
|
||||
Err(_) => meval::eval_str(l[2]).unwrap() as f32,
|
||||
}
|
||||
} else {
|
||||
DEFAULT_DELAY
|
||||
},
|
||||
}
|
||||
}
|
68
rinth-synth/src/ssg.rs
Normal file
68
rinth-synth/src/ssg.rs
Normal file
|
@ -0,0 +1,68 @@
|
|||
use crate::shared::{add_delay, get_note};
|
||||
use rinth_types::file::Channel;
|
||||
use rinth_types::note::get_note_len;
|
||||
use rinth_types::{file::ChannelType, note::Note, traits::Synth};
|
||||
use std::f32::consts::PI;
|
||||
use std::fs;
|
||||
|
||||
#[allow(clippy::upper_case_acronyms)]
|
||||
pub struct SSG {
|
||||
pub bpm: u16,
|
||||
pub notes: Vec<Note>,
|
||||
}
|
||||
|
||||
impl Synth for SSG {
|
||||
fn from_channel(channel: Channel, bpm: u16) -> Self
|
||||
where
|
||||
Self: Sized,
|
||||
{
|
||||
let channel = match channel {
|
||||
Channel {
|
||||
path,
|
||||
channel_type: ChannelType::SSG,
|
||||
..
|
||||
} => path,
|
||||
_ => {
|
||||
unreachable!()
|
||||
}
|
||||
};
|
||||
|
||||
let contents = fs::read(channel).unwrap();
|
||||
|
||||
let mut notes: Vec<Note> = vec![];
|
||||
for line in contents.split(|&x| x == b'\n') {
|
||||
if line.is_empty() {
|
||||
continue;
|
||||
}
|
||||
let s = String::from_utf8_lossy(line);
|
||||
let l = s.split_ascii_whitespace().collect::<Vec<&str>>();
|
||||
// Ignoring FM-only lines and comments
|
||||
if l[0].as_bytes()[0] == b'@'
|
||||
|| l[0].as_bytes()[0] == b'#'
|
||||
|| l[0].as_bytes()[0] == b'/'
|
||||
{
|
||||
continue;
|
||||
}
|
||||
notes.push(get_note(l))
|
||||
}
|
||||
SSG {
|
||||
bpm,
|
||||
notes,
|
||||
}
|
||||
}
|
||||
|
||||
fn synthesise(&self, sample_rate: u32) -> Vec<f32> {
|
||||
// t - time, m - modulating freq (carrier)
|
||||
let y: fn(f32, f32) -> f32 = |t: f32, f: f32| (2_f32 * PI * f * t).sin().signum();
|
||||
let mut stream: Vec<f32> = vec![];
|
||||
for note in &self.notes {
|
||||
if note.delay != 0_f32 {
|
||||
add_delay(&mut stream, note.delay, self.bpm, sample_rate);
|
||||
}
|
||||
for k in 0..(get_note_len(note.length, self.bpm) * sample_rate as f32) as usize {
|
||||
stream.push(y(k as f32 / sample_rate as f32, note.tone.get_freq()))
|
||||
}
|
||||
}
|
||||
stream
|
||||
}
|
||||
}
|
11
rinth-types/Cargo.toml
Normal file
11
rinth-types/Cargo.toml
Normal file
|
@ -0,0 +1,11 @@
|
|||
[package]
|
||||
name = "rinth-types"
|
||||
version = "0.0.1"
|
||||
edition = "2021"
|
||||
publish = false
|
||||
license-file = "../LICENSE"
|
||||
authors = [ "Ivan Bushchik <ivabus@ivabus.dev>" ]
|
||||
|
||||
[dependencies]
|
||||
toml = "0.8.8"
|
||||
serde = { version = "1.0.195", features = ["derive"] }
|
28
rinth-types/src/file.rs
Normal file
28
rinth-types/src/file.rs
Normal file
|
@ -0,0 +1,28 @@
|
|||
use serde::{Deserialize, Serialize};
|
||||
use std::path::PathBuf;
|
||||
|
||||
pub enum FileType {
|
||||
Header(Header),
|
||||
Channel(Channel),
|
||||
}
|
||||
|
||||
#[derive(Clone, Deserialize, Serialize, Debug)]
|
||||
pub enum ChannelType {
|
||||
FM,
|
||||
SSG,
|
||||
}
|
||||
|
||||
#[derive(Deserialize, Serialize)]
|
||||
pub struct Header {
|
||||
pub name: String,
|
||||
pub bpm: u16,
|
||||
pub channels: Vec<Channel>,
|
||||
}
|
||||
|
||||
#[derive(Clone, Deserialize, Serialize)]
|
||||
pub struct Channel {
|
||||
pub path: PathBuf,
|
||||
#[serde(alias = "type")]
|
||||
pub channel_type: ChannelType,
|
||||
pub volume: Option<f32>,
|
||||
}
|
3
rinth-types/src/lib.rs
Normal file
3
rinth-types/src/lib.rs
Normal file
|
@ -0,0 +1,3 @@
|
|||
pub mod file;
|
||||
pub mod note;
|
||||
pub mod traits;
|
72
rinth-types/src/note.rs
Normal file
72
rinth-types/src/note.rs
Normal file
|
@ -0,0 +1,72 @@
|
|||
//use rinth_macros::*;
|
||||
use std::fmt::{Display, Formatter};
|
||||
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
/// Get real length of note from musical length
|
||||
pub fn get_note_len(len: f32, bpm: u16) -> f32 {
|
||||
len * 240_f32 / bpm as f32
|
||||
}
|
||||
|
||||
#[derive(Debug, Deserialize, Serialize)]
|
||||
pub struct Note {
|
||||
pub tone: Tone,
|
||||
pub length: f32,
|
||||
pub delay: f32,
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub enum ToneError {
|
||||
InvalidTone(String),
|
||||
}
|
||||
|
||||
impl Display for ToneError {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||
write!(
|
||||
f,
|
||||
"{}",
|
||||
match self {
|
||||
ToneError::InvalidTone(s) => format!("Invalid tone ({})", s),
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug, Deserialize, Serialize)]
|
||||
pub struct Tone(String);
|
||||
|
||||
impl Tone {
|
||||
pub fn from(t: &str) -> Self {
|
||||
Tone(t.to_string())
|
||||
}
|
||||
|
||||
pub fn get_freq(&self) -> f32 {
|
||||
const BASE_FREQ: f32 = 440.0;
|
||||
let sr = self.to_string();
|
||||
let modifier: i8 = if sr.as_bytes()[1] == b'b' {
|
||||
-1
|
||||
} else if sr.as_bytes()[1] == b'd' {
|
||||
1
|
||||
} else {
|
||||
0
|
||||
};
|
||||
let magic_num = (12 * sr.get(sr.len() - 1..).unwrap().parse::<i8>().unwrap() + modifier)
|
||||
as u8 + match sr.as_bytes()[0] {
|
||||
b'C' => 2,
|
||||
b'D' => 4,
|
||||
b'E' => 6,
|
||||
b'F' => 7,
|
||||
b'G' => 9,
|
||||
b'A' => 11,
|
||||
b'B' => 13,
|
||||
_ => 0,
|
||||
};
|
||||
BASE_FREQ * 2_f32.powf((magic_num as f32 - 59_f32) / 12_f32)
|
||||
}
|
||||
}
|
||||
|
||||
impl Display for Tone {
|
||||
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
||||
f.write_str(&self.0.to_string())
|
||||
}
|
||||
}
|
9
rinth-types/src/traits.rs
Normal file
9
rinth-types/src/traits.rs
Normal file
|
@ -0,0 +1,9 @@
|
|||
use crate::file::Channel;
|
||||
|
||||
pub trait Synth {
|
||||
fn from_channel(channel: Channel, bpm: u16) -> Self
|
||||
where
|
||||
Self: Sized;
|
||||
|
||||
fn synthesise(&self, sample_rate: u32) -> Vec<f32>;
|
||||
}
|
Loading…
Reference in a new issue