#![cfg_attr(not(feature = "std"), no_std)] extern crate alloc; use agb_fixnum::Num; use alloc::borrow::Cow; #[derive(Debug)] pub struct Track { pub samples: Cow<'static, [Sample]>, pub envelopes: Cow<'static, [Envelope]>, pub pattern_data: Cow<'static, [PatternSlot]>, pub patterns: Cow<'static, [Pattern]>, pub patterns_to_play: Cow<'static, [usize]>, pub num_channels: usize, pub frames_per_tick: Num, pub ticks_per_step: u32, pub repeat: usize, } #[derive(Debug, Clone)] pub struct Sample { pub data: Cow<'static, [u8]>, pub should_loop: bool, pub restart_point: u32, pub volume: Num, pub volume_envelope: Option, pub fadeout: Num, } #[derive(Debug, Clone, PartialEq, Eq)] pub enum Jump { /// Jump to the given pattern position `pattern` at row index 0 Position { pattern: u8 }, /// Jump to the next pattern position, at row index `row` PatternBreak { row: u8 }, /// Jump to the pattern position `pattern` at row index `row` Combined { pattern: u8, row: u8 }, } #[derive(Debug, Clone)] pub struct Pattern { pub length: usize, pub start_position: usize, } #[derive(Debug, Clone, Default, PartialEq, Eq)] pub struct PatternSlot { pub speed: Num, pub sample: u16, pub effect1: PatternEffect, pub effect2: PatternEffect, } #[derive(Debug, Clone)] pub struct Envelope { pub amount: Cow<'static, [Num]>, pub sustain: Option, pub loop_start: Option, pub loop_end: Option, pub vib_waveform: Waveform, pub vib_amount: Num, pub vib_speed: u8, } #[derive(Debug, Default, Clone, PartialEq, Eq)] pub enum PatternEffect { /// Don't play an effect #[default] None, /// Stops playing the current note Stop, /// Plays an arpeggiation of three notes in one row, cycling betwen the current note, current note + first speed, current note + second speed Arpeggio(Num, Num), Panning(Num), Volume(Num), // bool = maintain vibrato? VolumeSlide(Num, bool), FineVolumeSlide(Num), NoteCut(u32), NoteDelay(u32), Portamento(Num), FinePortamento(Num), /// Slide each tick the first amount to at most the second amount TonePortamento(Num, Num), Vibrato(Waveform, Num, u8), SetTicksPerStep(u32), SetFramesPerTick(Num), SetGlobalVolume(Num), GlobalVolumeSlide(Num), /// Increase / decrease the pitch by the specified amount immediately PitchBend(Num), Jump(Jump), } #[derive(Debug, Default, Clone, Copy, PartialEq, Eq, Hash)] pub enum Waveform { #[default] Sine, Saw, Square, } #[cfg(feature = "quote")] impl quote::ToTokens for Jump { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::{quote, TokenStreamExt}; let type_bit = match self { Jump::Position { pattern } => { quote! {Position{pattern: #pattern} } } Jump::PatternBreak { row } => { quote! { PatternBreak{row: #row} } } Jump::Combined { pattern, row } => { quote! { Combined{pattern: #pattern, row: #row } } } }; tokens.append_all(quote! { agb_tracker::__private::agb_tracker_interop::Jump::#type_bit }); } } #[cfg(feature = "quote")] impl quote::ToTokens for Track { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::{quote, TokenStreamExt}; let Track { samples, envelopes, pattern_data, patterns, frames_per_tick, num_channels, patterns_to_play, ticks_per_step, repeat, } = self; let frames_per_tick = frames_per_tick.to_raw(); tokens.append_all(quote! { { use alloc::borrow::Cow; use agb_tracker::__private::agb_tracker_interop::*; use agb_tracker::__private::Num; static SAMPLES: &[Sample] = &[#(#samples),*]; static PATTERN_DATA: &[PatternSlot] = &[#(#pattern_data),*]; static PATTERNS: &[Pattern] = &[#(#patterns),*]; static PATTERNS_TO_PLAY: &[usize] = &[#(#patterns_to_play),*]; static ENVELOPES: &[Envelope] = &[#(#envelopes),*]; agb_tracker::Track { samples: Cow::Borrowed(SAMPLES), envelopes: Cow::Borrowed(ENVELOPES), pattern_data: Cow::Borrowed(PATTERN_DATA), patterns: Cow::Borrowed(PATTERNS), patterns_to_play: Cow::Borrowed(PATTERNS_TO_PLAY), frames_per_tick: Num::from_raw(#frames_per_tick), num_channels: #num_channels, ticks_per_step: #ticks_per_step, repeat: #repeat, } } }) } } #[cfg(feature = "quote")] impl quote::ToTokens for Envelope { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::{quote, TokenStreamExt}; let Envelope { amount, sustain, loop_start, loop_end, vib_amount, vib_speed, vib_waveform, } = self; let amount = amount.iter().map(|value| { let value = value.to_raw(); quote! { agb_tracker::__private::Num::from_raw(#value) } }); let vib_amount = { let value = vib_amount.to_raw(); quote! { agb_tracker::__private::Num::from_raw(#value) } }; let sustain = match sustain { Some(value) => quote!(Some(#value)), None => quote!(None), }; let loop_start = match loop_start { Some(value) => quote!(Some(#value)), None => quote!(None), }; let loop_end = match loop_end { Some(value) => quote!(Some(#value)), None => quote!(None), }; tokens.append_all(quote! { { static AMOUNTS: &[agb_tracker::__private::Num] = &[#(#amount),*]; agb_tracker::__private::agb_tracker_interop::Envelope { amount: Cow::Borrowed(AMOUNTS), sustain: #sustain, loop_start: #loop_start, loop_end: #loop_end, vib_waveform: #vib_waveform, vib_amount: #vib_amount, vib_speed: #vib_speed, } } }); } } #[cfg(feature = "quote")] struct ByteString<'a>(&'a [u8]); #[cfg(feature = "quote")] impl quote::ToTokens for ByteString<'_> { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::TokenStreamExt; tokens.append(proc_macro2::Literal::byte_string(self.0)); } } #[cfg(feature = "quote")] impl quote::ToTokens for Sample { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::{quote, TokenStreamExt}; let Sample { data, should_loop, restart_point, volume, volume_envelope, fadeout, } = self; let volume_envelope = match volume_envelope { Some(index) => quote!(Some(#index)), None => quote!(None), }; let fadeout = fadeout.to_raw(); let samples = ByteString(data); let volume = volume.to_raw(); tokens.append_all(quote! { { #[repr(align(4))] struct AlignmentWrapper([u8; N]); static SAMPLE_DATA: &[u8] = &AlignmentWrapper(*#samples).0; agb_tracker::__private::agb_tracker_interop::Sample { data: Cow::Borrowed(SAMPLE_DATA), should_loop: #should_loop, restart_point: #restart_point, volume: agb_tracker::__private::Num::from_raw(#volume), volume_envelope: #volume_envelope, fadeout: agb_tracker::__private::Num::from_raw(#fadeout), } } }); } } #[cfg(feature = "quote")] impl quote::ToTokens for PatternSlot { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::{quote, TokenStreamExt}; let PatternSlot { speed, sample, effect1, effect2, } = &self; let speed = speed.to_raw(); tokens.append_all(quote! { agb_tracker::__private::agb_tracker_interop::PatternSlot { speed: agb_tracker::__private::Num::from_raw(#speed), sample: #sample, effect1: #effect1, effect2: #effect2, } }); } } #[cfg(feature = "quote")] impl quote::ToTokens for Pattern { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::{quote, TokenStreamExt}; let Pattern { length, start_position, } = self; tokens.append_all(quote! { agb_tracker::__private::agb_tracker_interop::Pattern { length: #length, start_position: #start_position, } }) } } #[cfg(feature = "quote")] impl quote::ToTokens for PatternEffect { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::{quote, TokenStreamExt}; let type_bit = match self { PatternEffect::None => quote! { None }, PatternEffect::Stop => quote! { Stop }, PatternEffect::Arpeggio(first, second) => { let first = first.to_raw(); let second = second.to_raw(); quote! { Arpeggio(agb_tracker::__private::Num::from_raw(#first), agb_tracker::__private::Num::from_raw(#second)) } } PatternEffect::Panning(panning) => { let panning = panning.to_raw(); quote! { Panning(agb_tracker::__private::Num::from_raw(#panning))} } PatternEffect::Volume(volume) => { let volume = volume.to_raw(); quote! { Volume(agb_tracker::__private::Num::from_raw(#volume))} } PatternEffect::VolumeSlide(amount, vibrato) => { let amount = amount.to_raw(); quote! { VolumeSlide(agb_tracker::__private::Num::from_raw(#amount), #vibrato)} } PatternEffect::FineVolumeSlide(amount) => { let amount = amount.to_raw(); quote! { FineVolumeSlide(agb_tracker::__private::Num::from_raw(#amount))} } PatternEffect::NoteCut(wait) => quote! { NoteCut(#wait) }, PatternEffect::NoteDelay(wait) => quote! { NoteDelay(#wait) }, PatternEffect::Portamento(amount) => { let amount = amount.to_raw(); quote! { Portamento(agb_tracker::__private::Num::from_raw(#amount))} } PatternEffect::FinePortamento(amount) => { let amount = amount.to_raw(); quote! { FinePortamento(agb_tracker::__private::Num::from_raw(#amount))} } PatternEffect::TonePortamento(amount, target) => { let amount = amount.to_raw(); let target = target.to_raw(); quote! { TonePortamento(agb_tracker::__private::Num::from_raw(#amount), agb_tracker::__private::Num::from_raw(#target))} } PatternEffect::SetTicksPerStep(new_ticks) => { quote! { SetTicksPerStep(#new_ticks) } } PatternEffect::SetFramesPerTick(new_frames_per_tick) => { let amount = new_frames_per_tick.to_raw(); quote! { SetFramesPerTick(agb_tracker::__private::Num::from_raw(#amount)) } } PatternEffect::SetGlobalVolume(amount) => { let amount = amount.to_raw(); quote! { SetGlobalVolume(agb_tracker::__private::Num::from_raw(#amount)) } } PatternEffect::GlobalVolumeSlide(amount) => { let amount = amount.to_raw(); quote! { GlobalVolumeSlide(agb_tracker::__private::Num::from_raw(#amount)) } } PatternEffect::PitchBend(amount) => { let amount = amount.to_raw(); quote! { PitchBend(agb_tracker::__private::Num::from_raw(#amount)) } } PatternEffect::Vibrato(waveform, amount, speed) => { let amount = amount.to_raw(); quote! { Vibrato(#waveform, agb_tracker::__private::Num::from_raw(#amount), #speed) } } PatternEffect::Jump(jump) => { quote! { Jump(#jump) } } }; tokens.append_all(quote! { agb_tracker::__private::agb_tracker_interop::PatternEffect::#type_bit }); } } #[cfg(feature = "quote")] impl quote::ToTokens for Waveform { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { use quote::{quote, TokenStreamExt}; let name = match self { Waveform::Sine => quote!(Sine), Waveform::Saw => quote!(Saw), Waveform::Square => quote!(Square), }; tokens.append_all(quote! { agb_tracker::__private::agb_tracker_interop::Waveform::#name }); } }