#ifndef __JABYENGINE_SPU_IO_HPP__ #define __JABYENGINE_SPU_IO_HPP__ #include "../../Auxiliary/io_class_helper.hpp" #include namespace SPU { struct __no_align ubus32_t { uint16_t low; uint16_t high; constexpr ubus32_t() : low(0), high(0) { } constexpr operator uint32_t() const { return ((this->high << 16) | this->low); } constexpr void operator=(uint32_t value) { this->low = (value & 0xFFFF); this->high = (value >> 16); } constexpr void operator=(ubus32_t value) volatile { this->low = value.low; this->high = value.high; } }; enum Mode { Linear = 0, Exponential = 1, }; struct __no_align SampleRate { uint16_t raw_value = 0; static constexpr SampleRate from_HZ(long double freq) { //SampleRate is defined as 4096 == 44100Hz constexpr long double Base = (4096.0 / 44100.0); return static_cast(((freq*4096.0)/44100.0)); } }; struct __no_align SweepVolume { typedef int16_t VolumeStep; static constexpr VolumeStep VolumeMax = 1000; static constexpr VolumeStep SingleVolumeStep = (I16_MAX >> 1)/VolumeMax; enum Direction { Increase = 0, Decrease = 1, }; enum Phase { Positive = 0, Negative = 1, }; int16_t raw_value = 0; constexpr SweepVolume() = default; constexpr auto& set_volume_percent(double percent) { this->raw_value = bit::value::set_normalized(this->raw_value, static_cast((I16_MAX/100.0)*percent), __start_end_bit2_start_length(0, 14)); return *this; } constexpr auto& set_volume(int16_t volume) { this->raw_value = bit::value::set_normalized(this->raw_value, static_cast((volume >> 1)), __start_end_bit2_start_length(0, 14)); return *this; } constexpr int16_t get_volume_step() const { return bit::value::get_normalized(this->raw_value, __start_end_bit2_start_length(0, 14)); } io_class__2option_map(volume_mode, sweep_mode, 15); io_class__2option_map_getter(Mode, linear_sweep_mode, exponential_sweep_mode, sweep_mode_type, 14); io_class__2option_map_getter(Direction, increase_sweep_mode, decrease_sweep_mode, sweep_mode_direction, 13); io_class__2option_map_getter(Phase, positive_sweep_phase, negative_sweep_phase, sweep_phase, 12); //Uses only 5bit of shift (0..1F (slow..fast)) constexpr auto& set_sweep_shift(uint8_5b shift) { const int16_t crop_value = static_cast(shift); //(bit::value::crop_value(shift, 5)); this->raw_value = bit::value::set_normalized(this->raw_value, crop_value, __start_end_bit2_start_length(2, 6)); return *this; } constexpr uint8_5b get_sweep_shift() const { return bit::value::get_normalized(this->raw_value, __start_end_bit2_start_length(2, 6)); } //0..3 maps to => +7, +6, +5, +4 or -8, -7, -6, -5 constexpr auto& set_sweep_step(uint8_2b step) { this->raw_value = bit::value::set_normalized(this->raw_value, static_cast(step), __start_end_bit2_start_length(0, 1)); return *this; } constexpr uint8_2b get_sweep_step() const { return bit::value::get_normalized(this->raw_value, __start_end_bit2_start_length(0, 1)); } }; struct __no_align ADSR { ubus32_t raw_value; }; struct __no_align KeyW { ubus32_t raw_value; constexpr KeyW() = default; static constexpr KeyW All1() { KeyW value; value.raw_value = static_cast(0xFFFFFFFF); return value; } constexpr KeyW& set(size_t id) { this->raw_value = bit::set(this->raw_value, id); return *this; } }; struct __no_align KeyR { ubus32_t raw_value; constexpr bool is_set(size_t id) const { return bit::is_set(this->raw_value, id); } }; struct __no_align Key { static inline __always_inline auto& on = *reinterpret_cast(0x1F801D88); static inline __always_inline auto& off = *reinterpret_cast(0x1F801D8C); static inline __always_inline auto& state = *reinterpret_cast(0x1F801D9C); }; struct __no_align Voice { static constexpr size_t Count = 24; static inline __always_inline auto& Channel = reinterpret_cast(*reinterpret_cast(0x1f801c00)); SweepVolume volumeLeft; SweepVolume volumeRight; SampleRate sampleRate; uint16_t adr; ADSR adsr; SweepVolume currentVolume; //Not used uint16_t repeatAdr; }; struct MainVolume { static inline __always_inline auto& Left = *reinterpret_cast(0x1F801D80); static inline __always_inline auto& Right = *reinterpret_cast(0x1F801D82); }; struct __no_align Control { static inline __always_inline auto& Register = *reinterpret_cast(0x1F801DAA); enum TransferMode { Stop = 0, ManualWrite = (1 << 4), DMAWrite = (2 << 4), DMARead = (3 << 4), }; uint16_t raw_value = 0; constexpr Control() = default; io_class__2option_map(off, on, 15); io_class__2option_map_getter_is(bool, mute, unmute, 14); constexpr Control& set_noise_shift(uint8_4b shift) { this->raw_value = bit::value::set_normalized(this->raw_value, static_cast(shift), __start_end_bit2_start_length(10, 13)); return *this; } constexpr uint8_4b get_noise_shift() const { return bit::value::get_normalized(this->raw_value, __start_end_bit2_start_length(10, 13)); } constexpr Control& set_noise_step(uint8_2b step) { this->raw_value = bit::value::set_normalized(this->raw_value, static_cast(step), __start_end_bit2_start_length(8, 9)); return *this; } constexpr uint8_2b get_noise_step() const { return bit::value::get_normalized(this->raw_value, __start_end_bit2_start_length(8, 9)); } io_class__2option_map_getter_is(bool, reverb_disabled, reverb_enabled, 7); io_class__2option_map(irq9_ack, irq9_enable, 6); constexpr Control& set_transfer_mode(TransferMode mode) { this->raw_value = bit::value::set_normalized(this->raw_value, static_cast(mode), __start_end_bit2_start_length(4, 5)); return *this; } constexpr TransferMode get_transfer_mode() const { return static_cast(bit::value::get_normalized(this->raw_value, __start_end_bit2_start_length(4, 5))); } io_class__2option_map_getter_is(bool, external_reverb_off, external_reverb_on, 3); io_class__2option_map_getter_is(bool, cd_reverb_off, cd_reverb_on, 2); io_class__2option_map_getter_is(bool, external_audio_off, external_audio_on, 1); io_class__2option_map_getter_is(bool, cd_audio_disable, cd_audio_enable, 0); }; struct __no_align DataTransferControl { static inline __always_inline auto& Register = *reinterpret_cast(0x1F801DAC); uint16_t raw_value = 0; }; } #endif //!__JABYENGINE_SPU_IO_HPP__