/* * libopenmpt_impl.cpp * ------------------- * Purpose: libopenmpt private interface implementation * Notes : (currently none) * Authors: OpenMPT Devs * The OpenMPT source code is released under the BSD license. Read LICENSE for more details. */ #include "common/stdafx.h" #include "libopenmpt_internal.h" #include "libopenmpt.hpp" #include "libopenmpt_impl.hpp" #include #include #include #include #include #include #include #include #include #include "common/version.h" #include "common/misc_util.h" #include "common/FileReader.h" #include "common/Logging.h" #include "common/mptMutex.h" #include "soundlib/Sndfile.h" #include "soundlib/mod_specifications.h" #include "soundlib/AudioReadTarget.h" #if MPT_OS_WINDOWS && MPT_OS_WINDOWS_WINRT #include #endif // MPT_OS_WINDOWS && MPT_OS_WINDOWS_WINRT OPENMPT_NAMESPACE_BEGIN #if !defined(MPT_BUILD_SILENCE_LIBOPENMPT_CONFIGURATION_WARNINGS) #if MPT_OS_WINDOWS && MPT_OS_WINDOWS_WINRT #if defined(NTDDI_VERSION) #if (NTDDI_VERSION < 0x06020000) MPT_WARNING("Warning: libopenmpt for WinRT is built with reduced functionality. Please #define NTDDI_VERSION 0x0602000.") #endif #elif defined(_WIN32_WINNT) #if (_WIN32_WINNT < 0x0602) MPT_WARNING("Warning: libopenmpt for WinRT is built with reduced functionality. Please #define _WIN32_WINNT 0x0602.") #endif // _WIN32_WINNT #endif // _WIN32_WINNT #endif // MPT_OS_WINDOWS && MPT_OS_WINDOWS_WINRT #if defined(MPT_BUILD_MSVC) || defined(MPT_BUILD_VCPKG) #if MPT_OS_WINDOWS_WINRT #pragma comment(lib, "ole32.lib") #else #pragma comment(lib, "rpcrt4.lib") #endif #ifndef NO_DMO #pragma comment(lib, "dmoguids.lib") #pragma comment(lib, "strmiids.lib") #endif // !NO_DMO #endif // MPT_BUILD_MSVC #if MPT_PLATFORM_MULTITHREADED && MPT_MUTEX_NONE MPT_WARNING("Warning: libopenmpt built in non thread-safe mode because mutexes are not supported by the C++ standard library available.") #endif // MPT_MUTEX_NONE #if (defined(__MINGW32__) || defined(__MINGW64__)) && !defined(_GLIBCXX_HAS_GTHREADS) && !defined(MPT_WITH_MINGWSTDTHREADS) MPT_WARNING("Warning: Building libopenmpt with MinGW-w64 without std::thread support is not recommended and is deprecated. Please use MinGW-w64 with posix threading model (as opposed to win32 threading model), or build with mingw-std-threads.") #endif // MINGW #if MPT_CLANG_AT_LEAST(5,0,0) && MPT_CLANG_BEFORE(11,0,0) && defined(__powerpc__) && !defined(__powerpc64__) MPT_WARNING("Warning: libopenmpt is known to trigger bad code generation with Clang 5..10 on powerpc (32bit) when using -O3. See .") #endif #endif // !MPT_BUILD_SILENCE_LIBOPENMPT_CONFIGURATION_WARNINGS #if defined(MPT_ASSERT_HANDLER_NEEDED) && !defined(ENABLE_TESTS) MPT_NOINLINE void AssertHandler(const mpt::source_location &loc, const char *expr, const char *msg) { if(msg) { mpt::log::Logger().SendLogMessage(loc, LogError, "ASSERT", U_("ASSERTION FAILED: ") + mpt::ToUnicode(mpt::CharsetSource, msg) + U_(" (") + mpt::ToUnicode(mpt::CharsetSource, expr) + U_(")") ); } else { mpt::log::Logger().SendLogMessage(loc, LogError, "ASSERT", U_("ASSERTION FAILED: ") + mpt::ToUnicode(mpt::CharsetSource, expr) ); } #if defined(MPT_BUILD_FATAL_ASSERTS) std::abort(); #endif // MPT_BUILD_FATAL_ASSERTS } #endif // MPT_ASSERT_HANDLER_NEEDED && !ENABLE_TESTS OPENMPT_NAMESPACE_END using namespace OpenMPT; namespace openmpt { namespace version { std::uint32_t get_library_version() { return OPENMPT_API_VERSION; } std::uint32_t get_core_version() { return Version::Current().GetRawVersion(); } static std::string get_library_version_string() { std::string str; const SourceInfo sourceInfo = SourceInfo::Current(); str += mpt::fmt::val(OPENMPT_API_VERSION_MAJOR); str += "."; str += mpt::fmt::val(OPENMPT_API_VERSION_MINOR); str += "."; str += mpt::fmt::val(OPENMPT_API_VERSION_PATCH); if ( std::string(OPENMPT_API_VERSION_PREREL).length() > 0 ) { str += OPENMPT_API_VERSION_PREREL; } std::vector fields; if ( sourceInfo.Revision() ) { fields.push_back( "r" + mpt::fmt::val( sourceInfo.Revision() ) ); } if ( sourceInfo.IsDirty() ) { fields.push_back( "modified" ); } else if ( sourceInfo.HasMixedRevisions() ) { fields.push_back( "mixed" ); } if ( sourceInfo.IsPackage() ) { fields.push_back( "pkg" ); } if ( !fields.empty() ) { str += "+"; str += mpt::String::Combine( fields, std::string(".") ); } return str; } static std::string get_library_features_string() { return mpt::ToCharset(mpt::Charset::UTF8, mpt::String::Trim(Build::GetBuildFeaturesString())); } static std::string get_core_version_string() { return mpt::ToCharset(mpt::Charset::UTF8, Build::GetVersionStringExtended()); } static std::string get_source_url_string() { return mpt::ToCharset(mpt::Charset::UTF8, SourceInfo::Current().GetUrlWithRevision()); } static std::string get_source_date_string() { return mpt::ToCharset(mpt::Charset::UTF8, SourceInfo::Current().Date()); } static std::string get_source_revision_string() { const SourceInfo sourceInfo = SourceInfo::Current(); return sourceInfo.Revision() ? mpt::fmt::val(sourceInfo.Revision()) : std::string(); } static std::string get_build_string() { return mpt::ToCharset(mpt::Charset::UTF8, Build::GetBuildDateString()); } static std::string get_build_compiler_string() { return mpt::ToCharset(mpt::Charset::UTF8, Build::GetBuildCompilerString()); } static std::string get_credits_string() { return mpt::ToCharset(mpt::Charset::UTF8, Build::GetFullCreditsString()); } static std::string get_contact_string() { return mpt::ToCharset(mpt::Charset::UTF8, U_("Forum: ") + Build::GetURL(Build::Url::Forum)); } static std::string get_license_string() { return mpt::ToCharset(mpt::Charset::UTF8, Build::GetLicenseString()); } static std::string get_url_string() { return mpt::ToCharset(mpt::Charset::UTF8, Build::GetURL(Build::Url::Website)); } static std::string get_support_forum_url_string() { return mpt::ToCharset(mpt::Charset::UTF8, Build::GetURL(Build::Url::Forum)); } static std::string get_bugtracker_url_string() { return mpt::ToCharset(mpt::Charset::UTF8, Build::GetURL(Build::Url::Bugtracker)); } std::string get_string( const std::string & key ) { if ( key == "" ) { return std::string(); } else if ( key == "library_version" ) { return get_library_version_string(); } else if ( key == "library_version_major" ) { return mpt::fmt::val(OPENMPT_API_VERSION_MAJOR); } else if ( key == "library_version_minor" ) { return mpt::fmt::val(OPENMPT_API_VERSION_MINOR); } else if ( key == "library_version_patch" ) { return mpt::fmt::val(OPENMPT_API_VERSION_PATCH); } else if ( key == "library_version_prerel" ) { return mpt::fmt::val(OPENMPT_API_VERSION_PREREL); } else if ( key == "library_version_is_release" ) { return ( std::string(OPENMPT_API_VERSION_PREREL).length() == 0 ) ? "1" : "0"; } else if ( key == "library_features" ) { return get_library_features_string(); } else if ( key == "core_version" ) { return get_core_version_string(); } else if ( key == "source_url" ) { return get_source_url_string(); } else if ( key == "source_date" ) { return get_source_date_string(); } else if ( key == "source_revision" ) { return get_source_revision_string(); } else if ( key == "source_is_modified" ) { return SourceInfo::Current().IsDirty() ? "1" : "0"; } else if ( key == "source_has_mixed_revision" ) { return SourceInfo::Current().HasMixedRevisions() ? "1" : "0"; } else if ( key == "source_is_package" ) { return SourceInfo::Current().IsPackage() ? "1" : "0"; } else if ( key == "build" ) { return get_build_string(); } else if ( key == "build_compiler" ) { return get_build_compiler_string(); } else if ( key == "credits" ) { return get_credits_string(); } else if ( key == "contact" ) { return get_contact_string(); } else if ( key == "license" ) { return get_license_string(); } else if ( key == "url" ) { return get_url_string(); } else if ( key == "support_forum_url" ) { return get_support_forum_url_string(); } else if ( key == "bugtracker_url" ) { return get_bugtracker_url_string(); } else { return std::string(); } } } // namespace version log_interface::log_interface() { return; } log_interface::~log_interface() { return; } std_ostream_log::std_ostream_log( std::ostream & dst ) : destination(dst) { return; } std_ostream_log::~std_ostream_log() { return; } void std_ostream_log::log( const std::string & message ) const { destination.flush(); destination << message << std::endl; destination.flush(); } class log_forwarder : public ILog { private: log_interface & destination; public: log_forwarder( log_interface & dest ) : destination(dest) { return; } private: void AddToLog( LogLevel level, const mpt::ustring & text ) const override { destination.log( mpt::ToCharset( mpt::Charset::UTF8, LogLevelToString( level ) + U_(": ") + text ) ); } }; // class log_forwarder class loader_log : public ILog { private: mutable std::vector > m_Messages; public: std::vector > GetMessages() const; private: void AddToLog( LogLevel level, const mpt::ustring & text ) const override; }; // class loader_log std::vector > loader_log::GetMessages() const { return m_Messages; } void loader_log::AddToLog( LogLevel level, const mpt::ustring & text ) const { m_Messages.push_back( std::make_pair( level, mpt::ToCharset( mpt::Charset::UTF8, text ) ) ); } void module_impl::PushToCSoundFileLog( const std::string & text ) const { m_sndFile->AddToLog( LogError, mpt::ToUnicode( mpt::Charset::UTF8, text ) ); } void module_impl::PushToCSoundFileLog( int loglevel, const std::string & text ) const { m_sndFile->AddToLog( static_cast( loglevel ), mpt::ToUnicode( mpt::Charset::UTF8, text ) ); } module_impl::subsong_data::subsong_data( double duration, std::int32_t start_row, std::int32_t start_order, std::int32_t sequence ) : duration(duration) , start_row(start_row) , start_order(start_order) , sequence(sequence) { return; } static ResamplingMode filterlength_to_resamplingmode(std::int32_t length) { ResamplingMode result = SRCMODE_SINC8LP; if ( length == 0 ) { result = SRCMODE_SINC8LP; } else if ( length >= 8 ) { result = SRCMODE_SINC8LP; } else if ( length >= 3 ) { result = SRCMODE_CUBIC; } else if ( length >= 2 ) { result = SRCMODE_LINEAR; } else if ( length >= 1 ) { result = SRCMODE_NEAREST; } else { throw openmpt::exception("negative filter length"); } return result; } static std::int32_t resamplingmode_to_filterlength(ResamplingMode mode) { switch ( mode ) { case SRCMODE_NEAREST: return 1; break; case SRCMODE_LINEAR: return 2; break; case SRCMODE_CUBIC: return 4; break; case SRCMODE_SINC8: case SRCMODE_SINC8LP: case SRCMODE_DEFAULT: return 8; default: throw openmpt::exception("unknown interpolation filter length set internally"); break; } } static Resampling::AmigaFilter translate_amiga_filter_type( module_impl::amiga_filter_type amiga_type ) { switch (amiga_type ) { case module_impl::amiga_filter_type::a500: return Resampling::AmigaFilter::A500; case module_impl::amiga_filter_type::a1200: case module_impl::amiga_filter_type::auto_filter: default: return Resampling::AmigaFilter::A1200; case module_impl::amiga_filter_type::unfiltered: return Resampling::AmigaFilter::Unfiltered; } } static void ramping_to_mixersettings( MixerSettings & settings, int ramping ) { if ( ramping == -1 ) { settings.SetVolumeRampUpMicroseconds( MixerSettings().GetVolumeRampUpMicroseconds() ); settings.SetVolumeRampDownMicroseconds( MixerSettings().GetVolumeRampDownMicroseconds() ); } else if ( ramping <= 0 ) { settings.SetVolumeRampUpMicroseconds( 0 ); settings.SetVolumeRampDownMicroseconds( 0 ); } else { settings.SetVolumeRampUpMicroseconds( ramping * 1000 ); settings.SetVolumeRampDownMicroseconds( ramping * 1000 ); } } static void mixersettings_to_ramping( int & ramping, const MixerSettings & settings ) { std::int32_t ramp_us = std::max( settings.GetVolumeRampUpMicroseconds(), settings.GetVolumeRampDownMicroseconds() ); if ( ( settings.GetVolumeRampUpMicroseconds() == MixerSettings().GetVolumeRampUpMicroseconds() ) && ( settings.GetVolumeRampDownMicroseconds() == MixerSettings().GetVolumeRampDownMicroseconds() ) ) { ramping = -1; } else if ( ramp_us <= 0 ) { ramping = 0; } else { ramping = ( ramp_us + 500 ) / 1000; } } std::string module_impl::mod_string_to_utf8( const std::string & encoded ) const { return mpt::ToCharset( mpt::Charset::UTF8, m_sndFile->GetCharsetInternal(), encoded ); } void module_impl::apply_mixer_settings( std::int32_t samplerate, int channels ) { bool samplerate_changed = static_cast( m_sndFile->m_MixerSettings.gdwMixingFreq ) != samplerate; bool channels_changed = static_cast( m_sndFile->m_MixerSettings.gnChannels ) != channels; if ( samplerate_changed || channels_changed ) { MixerSettings mixersettings = m_sndFile->m_MixerSettings; std::int32_t volrampin_us = mixersettings.GetVolumeRampUpMicroseconds(); std::int32_t volrampout_us = mixersettings.GetVolumeRampDownMicroseconds(); mixersettings.gdwMixingFreq = samplerate; mixersettings.gnChannels = channels; mixersettings.SetVolumeRampUpMicroseconds( volrampin_us ); mixersettings.SetVolumeRampDownMicroseconds( volrampout_us ); m_sndFile->SetMixerSettings( mixersettings ); } else if ( !m_mixer_initialized ) { m_sndFile->InitPlayer( true ); } if ( samplerate_changed ) { m_sndFile->SuspendPlugins(); m_sndFile->ResumePlugins(); } m_mixer_initialized = true; } void module_impl::apply_libopenmpt_defaults() { set_render_param( module::RENDER_STEREOSEPARATION_PERCENT, 100 ); m_sndFile->Order.SetSequence( 0 ); } module_impl::subsongs_type module_impl::get_subsongs() const { std::vector subsongs; if ( m_sndFile->Order.GetNumSequences() == 0 ) { throw openmpt::exception("module contains no songs"); } for ( SEQUENCEINDEX seq = 0; seq < m_sndFile->Order.GetNumSequences(); ++seq ) { const std::vector lengths = m_sndFile->GetLength( eNoAdjust, GetLengthTarget( true ).StartPos( seq, 0, 0 ) ); for ( const auto & l : lengths ) { subsongs.push_back( subsong_data( l.duration, l.startRow, l.startOrder, seq ) ); } } return subsongs; } void module_impl::init_subsongs( subsongs_type & subsongs ) const { subsongs = get_subsongs(); } bool module_impl::has_subsongs_inited() const { return !m_subsongs.empty(); } void module_impl::ctor( const std::map< std::string, std::string > & ctls ) { m_sndFile = std::make_unique(); m_loaded = false; m_mixer_initialized = false; m_Dither = std::make_unique( mpt::global_prng() ); m_LogForwarder = std::make_unique( *m_Log ); m_sndFile->SetCustomLog( m_LogForwarder.get() ); m_current_subsong = 0; m_currentPositionSeconds = 0.0; m_Gain = 1.0f; m_ctl_play_at_end = song_end_action::fadeout_song; m_ctl_load_skip_samples = false; m_ctl_load_skip_patterns = false; m_ctl_load_skip_plugins = false; m_ctl_load_skip_subsongs_init = false; m_ctl_seek_sync_samples = false; // init member variables that correspond to ctls for ( const auto & ctl : ctls ) { ctl_set( ctl.first, ctl.second, false ); } } void module_impl::load( const FileReader & file, const std::map< std::string, std::string > & ctls ) { loader_log loaderlog; m_sndFile->SetCustomLog( &loaderlog ); { int load_flags = CSoundFile::loadCompleteModule; if ( m_ctl_load_skip_samples ) { load_flags &= ~CSoundFile::loadSampleData; } if ( m_ctl_load_skip_patterns ) { load_flags &= ~CSoundFile::loadPatternData; } if ( m_ctl_load_skip_plugins ) { load_flags &= ~(CSoundFile::loadPluginData | CSoundFile::loadPluginInstance); } if ( !m_sndFile->Create( file, static_cast( load_flags ) ) ) { throw openmpt::exception("error loading file"); } if ( !m_ctl_load_skip_subsongs_init ) { init_subsongs( m_subsongs ); } m_loaded = true; } m_sndFile->SetCustomLog( m_LogForwarder.get() ); std::vector > loaderMessages = loaderlog.GetMessages(); for ( const auto & msg : loaderMessages ) { PushToCSoundFileLog( msg.first, msg.second ); m_loaderMessages.push_back( mpt::ToCharset( mpt::Charset::UTF8, LogLevelToString( msg.first ) ) + std::string(": ") + msg.second ); } // init CSoundFile state that corresponds to ctls for ( const auto & ctl : ctls ) { ctl_set( ctl.first, ctl.second, false ); } } bool module_impl::is_loaded() const { return m_loaded; } std::size_t module_impl::read_wrapper( std::size_t count, std::int16_t * left, std::int16_t * right, std::int16_t * rear_left, std::int16_t * rear_right ) { m_sndFile->ResetMixStat(); m_sndFile->m_bIsRendering = ( m_ctl_play_at_end != song_end_action::fadeout_song ); std::size_t count_read = 0; std::int16_t * const buffers[4] = { left, right, rear_left, rear_right }; AudioReadTargetGainBuffer> target( audio_buffer_planar( buffers, planar_audio_buffer_valid_channels( buffers, std::size( buffers) ), count ), *m_Dither, m_Gain ); while ( count > 0 ) { std::size_t count_chunk = m_sndFile->Read( static_cast( std::min( static_cast( count ), static_cast( std::numeric_limits::max() / 2 / 4 / 4 ) ) ), // safety margin / samplesize / channels target ); if ( count_chunk == 0 ) { break; } count -= count_chunk; count_read += count_chunk; } if ( count_read == 0 && m_ctl_play_at_end == song_end_action::continue_song ) { // This is the song end, but allow the song or loop to restart on the next call m_sndFile->m_SongFlags.reset(SONG_ENDREACHED); } return count_read; } std::size_t module_impl::read_wrapper( std::size_t count, float * left, float * right, float * rear_left, float * rear_right ) { m_sndFile->ResetMixStat(); m_sndFile->m_bIsRendering = ( m_ctl_play_at_end != song_end_action::fadeout_song ); std::size_t count_read = 0; float * const buffers[4] = { left, right, rear_left, rear_right }; AudioReadTargetGainBuffer> target( audio_buffer_planar( buffers, planar_audio_buffer_valid_channels( buffers, std::size( buffers) ), count ), *m_Dither, m_Gain ); while ( count > 0 ) { std::size_t count_chunk = m_sndFile->Read( static_cast( std::min( static_cast( count ), static_cast( std::numeric_limits::max() / 2 / 4 / 4 ) ) ), // safety margin / samplesize / channels target ); if ( count_chunk == 0 ) { break; } count -= count_chunk; count_read += count_chunk; } if ( count_read == 0 && m_ctl_play_at_end == song_end_action::continue_song ) { // This is the song end, but allow the song or loop to restart on the next call m_sndFile->m_SongFlags.reset(SONG_ENDREACHED); } return count_read; } std::size_t module_impl::read_interleaved_wrapper( std::size_t count, std::size_t channels, std::int16_t * interleaved ) { m_sndFile->ResetMixStat(); m_sndFile->m_bIsRendering = ( m_ctl_play_at_end != song_end_action::fadeout_song ); std::size_t count_read = 0; AudioReadTargetGainBuffer> target( audio_buffer_interleaved( interleaved, channels, count ), *m_Dither, m_Gain ); while ( count > 0 ) { std::size_t count_chunk = m_sndFile->Read( static_cast( std::min( static_cast( count ), static_cast( std::numeric_limits::max() / 2 / 4 / 4 ) ) ), // safety margin / samplesize / channels target ); if ( count_chunk == 0 ) { break; } count -= count_chunk; count_read += count_chunk; } if ( count_read == 0 && m_ctl_play_at_end == song_end_action::continue_song ) { // This is the song end, but allow the song or loop to restart on the next call m_sndFile->m_SongFlags.reset(SONG_ENDREACHED); } return count_read; } std::size_t module_impl::read_interleaved_wrapper( std::size_t count, std::size_t channels, float * interleaved ) { m_sndFile->ResetMixStat(); m_sndFile->m_bIsRendering = ( m_ctl_play_at_end != song_end_action::fadeout_song ); std::size_t count_read = 0; AudioReadTargetGainBuffer> target( audio_buffer_interleaved( interleaved, channels, count ), *m_Dither, m_Gain ); while ( count > 0 ) { std::size_t count_chunk = m_sndFile->Read( static_cast( std::min( static_cast( count ), static_cast( std::numeric_limits::max() / 2 / 4 / 4 ) ) ), // safety margin / samplesize / channels target ); if ( count_chunk == 0 ) { break; } count -= count_chunk; count_read += count_chunk; } if ( count_read == 0 && m_ctl_play_at_end == song_end_action::continue_song ) { // This is the song end, but allow the song or loop to restart on the next call m_sndFile->m_SongFlags.reset(SONG_ENDREACHED); } return count_read; } std::vector module_impl::get_supported_extensions() { std::vector retval; std::vector extensions = CSoundFile::GetSupportedExtensions( false ); std::copy( extensions.begin(), extensions.end(), std::back_insert_iterator >( retval ) ); return retval; } bool module_impl::is_extension_supported( std::string_view extension ) { return CSoundFile::IsExtensionSupported( extension ); } double module_impl::could_open_probability( const OpenMPT::FileReader & file, double effort, std::unique_ptr log ) { try { if ( effort >= 0.8 ) { std::unique_ptr sndFile = std::make_unique(); std::unique_ptr logForwarder = std::make_unique( *log ); sndFile->SetCustomLog( logForwarder.get() ); if ( !sndFile->Create( file, CSoundFile::loadCompleteModule ) ) { return 0.0; } sndFile->Destroy(); return 1.0; } else if ( effort >= 0.6 ) { std::unique_ptr sndFile = std::make_unique(); std::unique_ptr logForwarder = std::make_unique( *log ); sndFile->SetCustomLog( logForwarder.get() ); if ( !sndFile->Create( file, CSoundFile::loadNoPatternOrPluginData ) ) { return 0.0; } sndFile->Destroy(); return 0.8; } else if ( effort >= 0.2 ) { std::unique_ptr sndFile = std::make_unique(); std::unique_ptr logForwarder = std::make_unique( *log ); sndFile->SetCustomLog( logForwarder.get() ); if ( !sndFile->Create( file, CSoundFile::onlyVerifyHeader ) ) { return 0.0; } sndFile->Destroy(); return 0.6; } else if ( effort >= 0.1 ) { FileReader::PinnedRawDataView view = file.GetPinnedRawDataView( probe_file_header_get_recommended_size() ); int probe_file_header_result = probe_file_header( probe_file_header_flags_default, view.data(), view.size(), file.GetLength() ); double result = 0.0; switch ( probe_file_header_result ) { case probe_file_header_result_success: result = 0.6; break; case probe_file_header_result_failure: result = 0.0; break; case probe_file_header_result_wantmoredata: result = 0.3; break; default: throw openmpt::exception(""); break; } return result; } else { return 0.2; } } catch ( ... ) { return 0.0; } } double module_impl::could_open_probability( callback_stream_wrapper stream, double effort, std::unique_ptr log ) { CallbackStream fstream; fstream.stream = stream.stream; fstream.read = stream.read; fstream.seek = stream.seek; fstream.tell = stream.tell; return could_open_probability( make_FileReader( fstream ), effort, std::move(log) ); } double module_impl::could_open_probability( std::istream & stream, double effort, std::unique_ptr log ) { return could_open_probability( make_FileReader( &stream ), effort, std::move(log) ); } std::size_t module_impl::probe_file_header_get_recommended_size() { return CSoundFile::ProbeRecommendedSize; } int module_impl::probe_file_header( std::uint64_t flags, const std::byte * data, std::size_t size, std::uint64_t filesize ) { int result = 0; switch ( CSoundFile::Probe( static_cast( flags ), mpt::span( data, size ), &filesize ) ) { case CSoundFile::ProbeSuccess: result = probe_file_header_result_success; break; case CSoundFile::ProbeFailure: result = probe_file_header_result_failure; break; case CSoundFile::ProbeWantMoreData: result = probe_file_header_result_wantmoredata; break; default: throw exception("internal error"); break; } return result; } int module_impl::probe_file_header( std::uint64_t flags, const std::uint8_t * data, std::size_t size, std::uint64_t filesize ) { int result = 0; switch ( CSoundFile::Probe( static_cast( flags ), mpt::span( mpt::byte_cast( data ), size ), &filesize ) ) { case CSoundFile::ProbeSuccess: result = probe_file_header_result_success; break; case CSoundFile::ProbeFailure: result = probe_file_header_result_failure; break; case CSoundFile::ProbeWantMoreData: result = probe_file_header_result_wantmoredata; break; default: throw exception("internal error"); break; } return result; } int module_impl::probe_file_header( std::uint64_t flags, const void * data, std::size_t size, std::uint64_t filesize ) { int result = 0; switch ( CSoundFile::Probe( static_cast( flags ), mpt::span( mpt::void_cast( data ), size ), &filesize ) ) { case CSoundFile::ProbeSuccess: result = probe_file_header_result_success; break; case CSoundFile::ProbeFailure: result = probe_file_header_result_failure; break; case CSoundFile::ProbeWantMoreData: result = probe_file_header_result_wantmoredata; break; default: throw exception("internal error"); break; } return result; } int module_impl::probe_file_header( std::uint64_t flags, const std::byte * data, std::size_t size ) { int result = 0; switch ( CSoundFile::Probe( static_cast( flags ), mpt::span( data, size ), nullptr ) ) { case CSoundFile::ProbeSuccess: result = probe_file_header_result_success; break; case CSoundFile::ProbeFailure: result = probe_file_header_result_failure; break; case CSoundFile::ProbeWantMoreData: result = probe_file_header_result_wantmoredata; break; default: throw exception("internal error"); break; } return result; } int module_impl::probe_file_header( std::uint64_t flags, const std::uint8_t * data, std::size_t size ) { int result = 0; switch ( CSoundFile::Probe( static_cast( flags ), mpt::span( mpt::byte_cast( data ), size ), nullptr ) ) { case CSoundFile::ProbeSuccess: result = probe_file_header_result_success; break; case CSoundFile::ProbeFailure: result = probe_file_header_result_failure; break; case CSoundFile::ProbeWantMoreData: result = probe_file_header_result_wantmoredata; break; default: throw exception("internal error"); break; } return result; } int module_impl::probe_file_header( std::uint64_t flags, const void * data, std::size_t size ) { int result = 0; switch ( CSoundFile::Probe( static_cast( flags ), mpt::span( mpt::void_cast( data ), size ), nullptr ) ) { case CSoundFile::ProbeSuccess: result = probe_file_header_result_success; break; case CSoundFile::ProbeFailure: result = probe_file_header_result_failure; break; case CSoundFile::ProbeWantMoreData: result = probe_file_header_result_wantmoredata; break; default: throw exception("internal error"); break; } return result; } int module_impl::probe_file_header( std::uint64_t flags, std::istream & stream ) { int result = 0; char buffer[ PROBE_RECOMMENDED_SIZE ]; MemsetZero( buffer ); std::size_t size_read = 0; std::size_t size_toread = CSoundFile::ProbeRecommendedSize; if ( stream.bad() ) { throw exception("error reading stream"); } const bool seekable = FileDataContainerStdStreamSeekable::IsSeekable( &stream ); const uint64 filesize = ( seekable ? FileDataContainerStdStreamSeekable::GetLength( &stream ) : 0 ); while ( ( size_toread > 0 ) && stream ) { stream.read( buffer + size_read, size_toread ); if ( stream.bad() ) { throw exception("error reading stream"); } else if ( stream.eof() ) { // normal } else if ( stream.fail() ) { throw exception("error reading stream"); } else { // normal } std::size_t read_count = static_cast( stream.gcount() ); size_read += read_count; size_toread -= read_count; } switch ( CSoundFile::Probe( static_cast( flags ), mpt::span( mpt::byte_cast( buffer ), size_read ), seekable ? &filesize : nullptr ) ) { case CSoundFile::ProbeSuccess: result = probe_file_header_result_success; break; case CSoundFile::ProbeFailure: result = probe_file_header_result_failure; break; case CSoundFile::ProbeWantMoreData: result = probe_file_header_result_wantmoredata; break; default: throw exception("internal error"); break; } return result; } int module_impl::probe_file_header( std::uint64_t flags, callback_stream_wrapper stream ) { int result = 0; char buffer[ PROBE_RECOMMENDED_SIZE ]; MemsetZero( buffer ); std::size_t size_read = 0; std::size_t size_toread = CSoundFile::ProbeRecommendedSize; if ( !stream.read ) { throw exception("error reading stream"); } CallbackStream fstream; fstream.stream = stream.stream; fstream.read = stream.read; fstream.seek = stream.seek; fstream.tell = stream.tell; const bool seekable = FileDataContainerCallbackStreamSeekable::IsSeekable( fstream ); const uint64 filesize = ( seekable ? FileDataContainerCallbackStreamSeekable::GetLength( fstream ) : 0 ); while ( size_toread > 0 ) { std::size_t read_count = stream.read( stream.stream, buffer + size_read, size_toread ); size_read += read_count; size_toread -= read_count; if ( read_count == 0 ) { // eof break; } } switch ( CSoundFile::Probe( static_cast( flags ), mpt::span( mpt::byte_cast( buffer ), size_read ), seekable ? &filesize : nullptr ) ) { case CSoundFile::ProbeSuccess: result = probe_file_header_result_success; break; case CSoundFile::ProbeFailure: result = probe_file_header_result_failure; break; case CSoundFile::ProbeWantMoreData: result = probe_file_header_result_wantmoredata; break; default: throw exception("internal error"); break; } return result; } module_impl::module_impl( callback_stream_wrapper stream, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); CallbackStream fstream; fstream.stream = stream.stream; fstream.read = stream.read; fstream.seek = stream.seek; fstream.tell = stream.tell; load( make_FileReader( fstream ), ctls ); apply_libopenmpt_defaults(); } module_impl::module_impl( std::istream & stream, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); load( make_FileReader( &stream ), ctls ); apply_libopenmpt_defaults(); } module_impl::module_impl( const std::vector & data, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); load( make_FileReader( mpt::as_span( data ) ), ctls ); apply_libopenmpt_defaults(); } module_impl::module_impl( const std::vector & data, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); load( make_FileReader( mpt::as_span( data ) ), ctls ); apply_libopenmpt_defaults(); } module_impl::module_impl( const std::vector & data, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); load( make_FileReader( mpt::byte_cast< mpt::span< const std::byte > >( mpt::as_span( data ) ) ), ctls ); apply_libopenmpt_defaults(); } module_impl::module_impl( const std::byte * data, std::size_t size, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); load( make_FileReader( mpt::as_span( data, size ) ), ctls ); apply_libopenmpt_defaults(); } module_impl::module_impl( const std::uint8_t * data, std::size_t size, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); load( make_FileReader( mpt::as_span( data, size ) ), ctls ); apply_libopenmpt_defaults(); } module_impl::module_impl( const char * data, std::size_t size, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); load( make_FileReader( mpt::byte_cast< mpt::span< const std::byte > >( mpt::as_span( data, size ) ) ), ctls ); apply_libopenmpt_defaults(); } module_impl::module_impl( const void * data, std::size_t size, std::unique_ptr log, const std::map< std::string, std::string > & ctls ) : m_Log(std::move(log)) { ctor( ctls ); load( make_FileReader( mpt::as_span( mpt::void_cast< const std::byte * >( data ), size ) ), ctls ); apply_libopenmpt_defaults(); } module_impl::~module_impl() { m_sndFile->Destroy(); } std::int32_t module_impl::get_render_param( int param ) const { std::int32_t result = 0; switch ( param ) { case module::RENDER_MASTERGAIN_MILLIBEL: { result = static_cast( 1000.0f * 2.0f * std::log10( m_Gain ) ); } break; case module::RENDER_STEREOSEPARATION_PERCENT: { result = m_sndFile->m_MixerSettings.m_nStereoSeparation * 100 / MixerSettings::StereoSeparationScale; } break; case module::RENDER_INTERPOLATIONFILTER_LENGTH: { result = resamplingmode_to_filterlength( m_sndFile->m_Resampler.m_Settings.SrcMode ); } break; case module::RENDER_VOLUMERAMPING_STRENGTH: { int ramping = 0; mixersettings_to_ramping( ramping, m_sndFile->m_MixerSettings ); result = ramping; } break; default: throw openmpt::exception("unknown render param"); break; } return result; } void module_impl::set_render_param( int param, std::int32_t value ) { switch ( param ) { case module::RENDER_MASTERGAIN_MILLIBEL: { m_Gain = static_cast( std::pow( 10.0f, value * 0.001f * 0.5f ) ); } break; case module::RENDER_STEREOSEPARATION_PERCENT: { std::int32_t newvalue = value * MixerSettings::StereoSeparationScale / 100; if ( newvalue != static_cast( m_sndFile->m_MixerSettings.m_nStereoSeparation ) ) { MixerSettings settings = m_sndFile->m_MixerSettings; settings.m_nStereoSeparation = newvalue; m_sndFile->SetMixerSettings( settings ); } } break; case module::RENDER_INTERPOLATIONFILTER_LENGTH: { CResamplerSettings newsettings = m_sndFile->m_Resampler.m_Settings; newsettings.SrcMode = filterlength_to_resamplingmode( value ); if ( newsettings != m_sndFile->m_Resampler.m_Settings ) { m_sndFile->SetResamplerSettings( newsettings ); } } break; case module::RENDER_VOLUMERAMPING_STRENGTH: { MixerSettings newsettings = m_sndFile->m_MixerSettings; ramping_to_mixersettings( newsettings, value ); if ( m_sndFile->m_MixerSettings.VolumeRampUpMicroseconds != newsettings.VolumeRampUpMicroseconds || m_sndFile->m_MixerSettings.VolumeRampDownMicroseconds != newsettings.VolumeRampDownMicroseconds ) { m_sndFile->SetMixerSettings( newsettings ); } } break; default: throw openmpt::exception("unknown render param"); break; } } std::size_t module_impl::read( std::int32_t samplerate, std::size_t count, std::int16_t * mono ) { if ( !mono ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 1 ); count = read_wrapper( count, mono, nullptr, nullptr, nullptr ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read( std::int32_t samplerate, std::size_t count, std::int16_t * left, std::int16_t * right ) { if ( !left || !right ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 2 ); count = read_wrapper( count, left, right, nullptr, nullptr ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read( std::int32_t samplerate, std::size_t count, std::int16_t * left, std::int16_t * right, std::int16_t * rear_left, std::int16_t * rear_right ) { if ( !left || !right || !rear_left || !rear_right ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 4 ); count = read_wrapper( count, left, right, rear_left, rear_right ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read( std::int32_t samplerate, std::size_t count, float * mono ) { if ( !mono ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 1 ); count = read_wrapper( count, mono, nullptr, nullptr, nullptr ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read( std::int32_t samplerate, std::size_t count, float * left, float * right ) { if ( !left || !right ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 2 ); count = read_wrapper( count, left, right, nullptr, nullptr ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read( std::int32_t samplerate, std::size_t count, float * left, float * right, float * rear_left, float * rear_right ) { if ( !left || !right || !rear_left || !rear_right ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 4 ); count = read_wrapper( count, left, right, rear_left, rear_right ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read_interleaved_stereo( std::int32_t samplerate, std::size_t count, std::int16_t * interleaved_stereo ) { if ( !interleaved_stereo ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 2 ); count = read_interleaved_wrapper( count, 2, interleaved_stereo ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read_interleaved_quad( std::int32_t samplerate, std::size_t count, std::int16_t * interleaved_quad ) { if ( !interleaved_quad ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 4 ); count = read_interleaved_wrapper( count, 4, interleaved_quad ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read_interleaved_stereo( std::int32_t samplerate, std::size_t count, float * interleaved_stereo ) { if ( !interleaved_stereo ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 2 ); count = read_interleaved_wrapper( count, 2, interleaved_stereo ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } std::size_t module_impl::read_interleaved_quad( std::int32_t samplerate, std::size_t count, float * interleaved_quad ) { if ( !interleaved_quad ) { throw openmpt::exception("null pointer"); } apply_mixer_settings( samplerate, 4 ); count = read_interleaved_wrapper( count, 4, interleaved_quad ); m_currentPositionSeconds += static_cast( count ) / static_cast( samplerate ); return count; } double module_impl::get_duration_seconds() const { std::unique_ptr subsongs_temp = has_subsongs_inited() ? std::unique_ptr() : std::make_unique( get_subsongs() ); const subsongs_type & subsongs = has_subsongs_inited() ? m_subsongs : *subsongs_temp; if ( m_current_subsong == all_subsongs ) { // Play all subsongs consecutively. double total_duration = 0.0; for ( const auto & subsong : subsongs ) { total_duration += subsong.duration; } return total_duration; } return subsongs[m_current_subsong].duration; } void module_impl::select_subsong( std::int32_t subsong ) { std::unique_ptr subsongs_temp = has_subsongs_inited() ? std::unique_ptr() : std::make_unique( get_subsongs() ); const subsongs_type & subsongs = has_subsongs_inited() ? m_subsongs : *subsongs_temp; if ( subsong != all_subsongs && ( subsong < 0 || subsong >= static_cast( subsongs.size() ) ) ) { throw openmpt::exception("invalid subsong"); } m_current_subsong = subsong; m_sndFile->m_SongFlags.set( SONG_PLAYALLSONGS, subsong == all_subsongs ); if ( subsong == all_subsongs ) { subsong = 0; } m_sndFile->Order.SetSequence( static_cast( subsongs[subsong].sequence ) ); set_position_order_row( subsongs[subsong].start_order, subsongs[subsong].start_row ); m_currentPositionSeconds = 0.0; } std::int32_t module_impl::get_selected_subsong() const { return m_current_subsong; } void module_impl::set_repeat_count( std::int32_t repeat_count ) { m_sndFile->SetRepeatCount( repeat_count ); } std::int32_t module_impl::get_repeat_count() const { return m_sndFile->GetRepeatCount(); } double module_impl::get_position_seconds() const { return m_currentPositionSeconds; } double module_impl::set_position_seconds( double seconds ) { std::unique_ptr subsongs_temp = has_subsongs_inited() ? std::unique_ptr() : std::make_unique( get_subsongs() ); const subsongs_type & subsongs = has_subsongs_inited() ? m_subsongs : *subsongs_temp; const subsong_data * subsong = 0; double base_seconds = 0.0; if ( m_current_subsong == all_subsongs ) { // When playing all subsongs, find out which subsong this time would belong to. subsong = &subsongs.back(); for ( std::size_t i = 0; i < subsongs.size(); ++i ) { if ( base_seconds + subsongs[i].duration > seconds ) { subsong = &subsongs[i]; break; } base_seconds += subsong->duration; } seconds -= base_seconds; } else { subsong = &subsongs[m_current_subsong]; } m_sndFile->SetCurrentOrder( static_cast( subsong->start_order ) ); GetLengthType t = m_sndFile->GetLength( m_ctl_seek_sync_samples ? eAdjustSamplePositions : eAdjust, GetLengthTarget( seconds ).StartPos( static_cast( subsong->sequence ), static_cast( subsong->start_order ), static_cast( subsong->start_row ) ) ).back(); m_sndFile->m_PlayState.m_nNextOrder = m_sndFile->m_PlayState.m_nCurrentOrder = t.targetReached ? t.lastOrder : t.endOrder; m_sndFile->m_PlayState.m_nNextRow = t.targetReached ? t.lastRow : t.endRow; m_sndFile->m_PlayState.m_nTickCount = CSoundFile::TICKS_ROW_FINISHED; m_currentPositionSeconds = base_seconds + t.duration; return m_currentPositionSeconds; } double module_impl::set_position_order_row( std::int32_t order, std::int32_t row ) { if ( order < 0 || order >= m_sndFile->Order().GetLengthTailTrimmed() ) { return m_currentPositionSeconds; } PATTERNINDEX pattern = m_sndFile->Order()[order]; if ( m_sndFile->Patterns.IsValidIndex( pattern ) ) { if ( row < 0 || row >= static_cast( m_sndFile->Patterns[pattern].GetNumRows() ) ) { return m_currentPositionSeconds; } } else { row = 0; } m_sndFile->m_PlayState.m_nCurrentOrder = static_cast( order ); m_sndFile->SetCurrentOrder( static_cast( order ) ); m_sndFile->m_PlayState.m_nNextRow = static_cast( row ); m_sndFile->m_PlayState.m_nTickCount = CSoundFile::TICKS_ROW_FINISHED; m_currentPositionSeconds = m_sndFile->GetLength( m_ctl_seek_sync_samples ? eAdjustSamplePositions : eAdjust, GetLengthTarget( static_cast( order ), static_cast( row ) ) ).back().duration; return m_currentPositionSeconds; } std::vector module_impl::get_metadata_keys() const { return { "type", "type_long", "originaltype", "originaltype_long", "container", "container_long", "tracker", "artist", "title", "date", "message", "message_raw", "warnings", }; } std::string module_impl::get_message_instruments() const { std::string retval; std::string tmp; bool valid = false; for ( INSTRUMENTINDEX i = 1; i <= m_sndFile->GetNumInstruments(); ++i ) { std::string instname = m_sndFile->GetInstrumentName( i ); if ( !instname.empty() ) { valid = true; } tmp += instname; tmp += "\n"; } if ( valid ) { retval = tmp; } return retval; } std::string module_impl::get_message_samples() const { std::string retval; std::string tmp; bool valid = false; for ( SAMPLEINDEX i = 1; i <= m_sndFile->GetNumSamples(); ++i ) { std::string samplename = m_sndFile->GetSampleName( i ); if ( !samplename.empty() ) { valid = true; } tmp += samplename; tmp += "\n"; } if ( valid ) { retval = tmp; } return retval; } std::string module_impl::get_metadata( const std::string & key ) const { if ( key == std::string("type") ) { return mpt::ToCharset(mpt::Charset::UTF8, m_sndFile->m_modFormat.type ); } else if ( key == std::string("type_long") ) { return mpt::ToCharset(mpt::Charset::UTF8, m_sndFile->m_modFormat.formatName ); } else if ( key == std::string("originaltype") ) { return mpt::ToCharset(mpt::Charset::UTF8, m_sndFile->m_modFormat.originalType ); } else if ( key == std::string("originaltype_long") ) { return mpt::ToCharset(mpt::Charset::UTF8, m_sndFile->m_modFormat.originalFormatName ); } else if ( key == std::string("container") ) { return mpt::ToCharset(mpt::Charset::UTF8, CSoundFile::ModContainerTypeToString( m_sndFile->GetContainerType() ) ); } else if ( key == std::string("container_long") ) { return mpt::ToCharset(mpt::Charset::UTF8, CSoundFile::ModContainerTypeToTracker( m_sndFile->GetContainerType() ) ); } else if ( key == std::string("tracker") ) { return mpt::ToCharset(mpt::Charset::UTF8, m_sndFile->m_modFormat.madeWithTracker ); } else if ( key == std::string("artist") ) { return mpt::ToCharset( mpt::Charset::UTF8, m_sndFile->m_songArtist ); } else if ( key == std::string("title") ) { return mod_string_to_utf8( m_sndFile->GetTitle() ); } else if ( key == std::string("date") ) { if ( m_sndFile->GetFileHistory().empty() || !m_sndFile->GetFileHistory().back().HasValidDate() ) { return std::string(); } return mpt::ToCharset(mpt::Charset::UTF8, m_sndFile->GetFileHistory().back().AsISO8601() ); } else if ( key == std::string("message") ) { std::string retval = m_sndFile->m_songMessage.GetFormatted( SongMessage::leLF ); if ( retval.empty() ) { switch ( m_sndFile->GetMessageHeuristic() ) { case ModMessageHeuristicOrder::Instruments: retval = get_message_instruments(); break; case ModMessageHeuristicOrder::Samples: retval = get_message_samples(); break; case ModMessageHeuristicOrder::InstrumentsSamples: if ( retval.empty() ) { retval = get_message_instruments(); } if ( retval.empty() ) { retval = get_message_samples(); } break; case ModMessageHeuristicOrder::SamplesInstruments: if ( retval.empty() ) { retval = get_message_samples(); } if ( retval.empty() ) { retval = get_message_instruments(); } break; case ModMessageHeuristicOrder::BothInstrumentsSamples: { std::string message_instruments = get_message_instruments(); std::string message_samples = get_message_samples(); if ( !message_instruments.empty() ) { retval += std::move( message_instruments ); } if ( !message_samples.empty() ) { retval += std::move( message_samples ); } } break; case ModMessageHeuristicOrder::BothSamplesInstruments: { std::string message_instruments = get_message_instruments(); std::string message_samples = get_message_samples(); if ( !message_samples.empty() ) { retval += std::move( message_samples ); } if ( !message_instruments.empty() ) { retval += std::move( message_instruments ); } } break; } } return mod_string_to_utf8( retval ); } else if ( key == std::string("message_raw") ) { std::string retval = m_sndFile->m_songMessage.GetFormatted( SongMessage::leLF ); return mod_string_to_utf8( retval ); } else if ( key == std::string("warnings") ) { std::string retval; bool first = true; for ( const auto & msg : m_loaderMessages ) { if ( !first ) { retval += "\n"; } else { first = false; } retval += msg; } return retval; } return ""; } double module_impl::get_current_estimated_bpm() const { return m_sndFile->GetCurrentBPM(); } std::int32_t module_impl::get_current_speed() const { return m_sndFile->m_PlayState.m_nMusicSpeed; } std::int32_t module_impl::get_current_tempo() const { return static_cast( m_sndFile->m_PlayState.m_nMusicTempo.GetInt() ); } std::int32_t module_impl::get_current_order() const { return m_sndFile->GetCurrentOrder(); } std::int32_t module_impl::get_current_pattern() const { std::int32_t order = m_sndFile->GetCurrentOrder(); if ( order < 0 || order >= m_sndFile->Order().GetLengthTailTrimmed() ) { return m_sndFile->GetCurrentPattern(); } std::int32_t pattern = m_sndFile->Order()[order]; if ( !m_sndFile->Patterns.IsValidIndex( static_cast( pattern ) ) ) { return -1; } return pattern; } std::int32_t module_impl::get_current_row() const { return m_sndFile->m_PlayState.m_nRow; } std::int32_t module_impl::get_current_playing_channels() const { return m_sndFile->GetMixStat(); } float module_impl::get_current_channel_vu_mono( std::int32_t channel ) const { if ( channel < 0 || channel >= m_sndFile->GetNumChannels() ) { return 0.0f; } const float left = m_sndFile->m_PlayState.Chn[channel].nLeftVU * (1.0f/128.0f); const float right = m_sndFile->m_PlayState.Chn[channel].nRightVU * (1.0f/128.0f); return std::sqrt(left*left + right*right); } float module_impl::get_current_channel_vu_left( std::int32_t channel ) const { if ( channel < 0 || channel >= m_sndFile->GetNumChannels() ) { return 0.0f; } return m_sndFile->m_PlayState.Chn[channel].dwFlags[CHN_SURROUND] ? 0.0f : m_sndFile->m_PlayState.Chn[channel].nLeftVU * (1.0f/128.0f); } float module_impl::get_current_channel_vu_right( std::int32_t channel ) const { if ( channel < 0 || channel >= m_sndFile->GetNumChannels() ) { return 0.0f; } return m_sndFile->m_PlayState.Chn[channel].dwFlags[CHN_SURROUND] ? 0.0f : m_sndFile->m_PlayState.Chn[channel].nRightVU * (1.0f/128.0f); } float module_impl::get_current_channel_vu_rear_left( std::int32_t channel ) const { if ( channel < 0 || channel >= m_sndFile->GetNumChannels() ) { return 0.0f; } return m_sndFile->m_PlayState.Chn[channel].dwFlags[CHN_SURROUND] ? m_sndFile->m_PlayState.Chn[channel].nLeftVU * (1.0f/128.0f) : 0.0f; } float module_impl::get_current_channel_vu_rear_right( std::int32_t channel ) const { if ( channel < 0 || channel >= m_sndFile->GetNumChannels() ) { return 0.0f; } return m_sndFile->m_PlayState.Chn[channel].dwFlags[CHN_SURROUND] ? m_sndFile->m_PlayState.Chn[channel].nRightVU * (1.0f/128.0f) : 0.0f; } std::int32_t module_impl::get_num_subsongs() const { std::unique_ptr subsongs_temp = has_subsongs_inited() ? std::unique_ptr() : std::make_unique( get_subsongs() ); const subsongs_type & subsongs = has_subsongs_inited() ? m_subsongs : *subsongs_temp; return static_cast( subsongs.size() ); } std::int32_t module_impl::get_num_channels() const { return m_sndFile->GetNumChannels(); } std::int32_t module_impl::get_num_orders() const { return m_sndFile->Order().GetLengthTailTrimmed(); } std::int32_t module_impl::get_num_patterns() const { return m_sndFile->Patterns.GetNumPatterns(); } std::int32_t module_impl::get_num_instruments() const { return m_sndFile->GetNumInstruments(); } std::int32_t module_impl::get_num_samples() const { return m_sndFile->GetNumSamples(); } std::vector module_impl::get_subsong_names() const { std::vector retval; std::unique_ptr subsongs_temp = has_subsongs_inited() ? std::unique_ptr() : std::make_unique( get_subsongs() ); const subsongs_type & subsongs = has_subsongs_inited() ? m_subsongs : *subsongs_temp; retval.reserve( subsongs.size() ); for ( const auto & subsong : subsongs ) { const auto & order = m_sndFile->Order( static_cast( subsong.sequence ) ); retval.push_back( mpt::ToCharset( mpt::Charset::UTF8, order.GetName() ) ); if ( retval.back().empty() ) { // use first pattern name instead if ( order.IsValidPat( static_cast( subsong.start_order ) ) ) retval.back() = mpt::ToCharset( mpt::Charset::UTF8, m_sndFile->GetCharsetInternal(), m_sndFile->Patterns[ order[ subsong.start_order ] ].GetName() ); } } return retval; } std::vector module_impl::get_channel_names() const { std::vector retval; for ( CHANNELINDEX i = 0; i < m_sndFile->GetNumChannels(); ++i ) { retval.push_back( mod_string_to_utf8( m_sndFile->ChnSettings[i].szName ) ); } return retval; } std::vector module_impl::get_order_names() const { std::vector retval; ORDERINDEX num_orders = m_sndFile->Order().GetLengthTailTrimmed(); retval.reserve( num_orders ); for ( ORDERINDEX i = 0; i < num_orders; ++i ) { PATTERNINDEX pat = m_sndFile->Order()[i]; if ( m_sndFile->Patterns.IsValidIndex( pat ) ) { retval.push_back( mod_string_to_utf8( m_sndFile->Patterns[ m_sndFile->Order()[i] ].GetName() ) ); } else { if ( pat == m_sndFile->Order.GetIgnoreIndex() ) { retval.push_back( "+++ skip" ); } else if ( pat == m_sndFile->Order.GetInvalidPatIndex() ) { retval.push_back( "--- stop" ); } else { retval.push_back( "???" ); } } } return retval; } std::vector module_impl::get_pattern_names() const { std::vector retval; retval.reserve( m_sndFile->Patterns.GetNumPatterns() ); for ( PATTERNINDEX i = 0; i < m_sndFile->Patterns.GetNumPatterns(); ++i ) { retval.push_back( mod_string_to_utf8( m_sndFile->Patterns[i].GetName() ) ); } return retval; } std::vector module_impl::get_instrument_names() const { std::vector retval; retval.reserve( m_sndFile->GetNumInstruments() ); for ( INSTRUMENTINDEX i = 1; i <= m_sndFile->GetNumInstruments(); ++i ) { retval.push_back( mod_string_to_utf8( m_sndFile->GetInstrumentName( i ) ) ); } return retval; } std::vector module_impl::get_sample_names() const { std::vector retval; retval.reserve( m_sndFile->GetNumSamples() ); for ( SAMPLEINDEX i = 1; i <= m_sndFile->GetNumSamples(); ++i ) { retval.push_back( mod_string_to_utf8( m_sndFile->GetSampleName( i ) ) ); } return retval; } std::int32_t module_impl::get_order_pattern( std::int32_t o ) const { if ( o < 0 || o >= m_sndFile->Order().GetLengthTailTrimmed() ) { return -1; } return m_sndFile->Order()[o]; } std::int32_t module_impl::get_pattern_num_rows( std::int32_t p ) const { if ( !IsInRange( p, std::numeric_limits::min(), std::numeric_limits::max() ) || !m_sndFile->Patterns.IsValidPat( static_cast( p ) ) ) { return 0; } return m_sndFile->Patterns[p].GetNumRows(); } std::uint8_t module_impl::get_pattern_row_channel_command( std::int32_t p, std::int32_t r, std::int32_t c, int cmd ) const { if ( !IsInRange( p, std::numeric_limits::min(), std::numeric_limits::max() ) || !m_sndFile->Patterns.IsValidPat( static_cast( p ) ) ) { return 0; } const CPattern &pattern = m_sndFile->Patterns[p]; if ( r < 0 || r >= static_cast( pattern.GetNumRows() ) ) { return 0; } if ( c < 0 || c >= m_sndFile->GetNumChannels() ) { return 0; } if ( cmd < module::command_note || cmd > module::command_parameter ) { return 0; } const ModCommand & cell = *pattern.GetpModCommand( static_cast( r ), static_cast( c ) ); switch ( cmd ) { case module::command_note: return cell.note; break; case module::command_instrument: return cell.instr; break; case module::command_volumeffect: return cell.volcmd; break; case module::command_effect: return cell.command; break; case module::command_volume: return cell.vol; break; case module::command_parameter: return cell.param; break; } return 0; } /* highlight chars explained: : empty/space . : empty/dot n : generic note m : special note i : generic instrument u : generic volume column effect v : generic volume column parameter e : generic effect column effect f : generic effect column parameter */ std::pair< std::string, std::string > module_impl::format_and_highlight_pattern_row_channel_command( std::int32_t p, std::int32_t r, std::int32_t c, int cmd ) const { if ( !IsInRange( p, std::numeric_limits::min(), std::numeric_limits::max() ) || !m_sndFile->Patterns.IsValidPat( static_cast( p ) ) ) { return std::make_pair( std::string(), std::string() ); } const CPattern &pattern = m_sndFile->Patterns[p]; if ( r < 0 || r >= static_cast( pattern.GetNumRows() ) ) { return std::make_pair( std::string(), std::string() ); } if ( c < 0 || c >= m_sndFile->GetNumChannels() ) { return std::make_pair( std::string(), std::string() ); } if ( cmd < module::command_note || cmd > module::command_parameter ) { return std::make_pair( std::string(), std::string() ); } const ModCommand & cell = *pattern.GetpModCommand( static_cast( r ), static_cast( c ) ); switch ( cmd ) { case module::command_note: return std::make_pair( ( cell.IsNote() || cell.IsSpecialNote() ) ? mpt::ToCharset( mpt::Charset::UTF8, m_sndFile->GetNoteName( cell.note, cell.instr ) ) : std::string("...") , ( cell.IsNote() ) ? std::string("nnn") : cell.IsSpecialNote() ? std::string("mmm") : std::string("...") ); break; case module::command_instrument: return std::make_pair( cell.instr ? mpt::fmt::HEX0<2>( cell.instr ) : std::string("..") , cell.instr ? std::string("ii") : std::string("..") ); break; case module::command_volumeffect: return std::make_pair( cell.IsPcNote() ? std::string(" ") : cell.volcmd != VOLCMD_NONE ? std::string( 1, m_sndFile->GetModSpecifications().GetVolEffectLetter( cell.volcmd ) ) : std::string(" ") , cell.IsPcNote() ? std::string(" ") : cell.volcmd != VOLCMD_NONE ? std::string("u") : std::string(" ") ); break; case module::command_volume: return std::make_pair( cell.IsPcNote() ? mpt::fmt::HEX0<2>( cell.GetValueVolCol() & 0xff ) : cell.volcmd != VOLCMD_NONE ? mpt::fmt::HEX0<2>( cell.vol ) : std::string("..") , cell.IsPcNote() ? std::string("vv") : cell.volcmd != VOLCMD_NONE ? std::string("vv") : std::string("..") ); break; case module::command_effect: return std::make_pair( cell.IsPcNote() ? mpt::fmt::HEX0<1>( ( cell.GetValueEffectCol() & 0x0f00 ) > 16 ) : cell.command != CMD_NONE ? std::string( 1, m_sndFile->GetModSpecifications().GetEffectLetter( cell.command ) ) : std::string(".") , cell.IsPcNote() ? std::string("e") : cell.command != CMD_NONE ? std::string("e") : std::string(".") ); break; case module::command_parameter: return std::make_pair( cell.IsPcNote() ? mpt::fmt::HEX0<2>( cell.GetValueEffectCol() & 0x00ff ) : cell.command != CMD_NONE ? mpt::fmt::HEX0<2>( cell.param ) : std::string("..") , cell.IsPcNote() ? std::string("ff") : cell.command != CMD_NONE ? std::string("ff") : std::string("..") ); break; } return std::make_pair( std::string(), std::string() ); } std::string module_impl::format_pattern_row_channel_command( std::int32_t p, std::int32_t r, std::int32_t c, int cmd ) const { return format_and_highlight_pattern_row_channel_command( p, r, c, cmd ).first; } std::string module_impl::highlight_pattern_row_channel_command( std::int32_t p, std::int32_t r, std::int32_t c, int cmd ) const { return format_and_highlight_pattern_row_channel_command( p, r, c, cmd ).second; } std::pair< std::string, std::string > module_impl::format_and_highlight_pattern_row_channel( std::int32_t p, std::int32_t r, std::int32_t c, std::size_t width, bool pad ) const { std::string text = pad ? std::string( width, ' ' ) : std::string(); std::string high = pad ? std::string( width, ' ' ) : std::string(); if ( !IsInRange( p, std::numeric_limits::min(), std::numeric_limits::max() ) || !m_sndFile->Patterns.IsValidPat( static_cast( p ) ) ) { return std::make_pair( text, high ); } const CPattern &pattern = m_sndFile->Patterns[p]; if ( r < 0 || r >= static_cast( pattern.GetNumRows() ) ) { return std::make_pair( text, high ); } if ( c < 0 || c >= m_sndFile->GetNumChannels() ) { return std::make_pair( text, high ); } // 0000000001111 // 1234567890123 // "NNN IIvVV EFF" const ModCommand & cell = *pattern.GetpModCommand( static_cast( r ), static_cast( c ) ); text.clear(); high.clear(); text += ( cell.IsNote() || cell.IsSpecialNote() ) ? mpt::ToCharset( mpt::Charset::UTF8, m_sndFile->GetNoteName( cell.note, cell.instr ) ) : std::string("..."); high += ( cell.IsNote() ) ? std::string("nnn") : cell.IsSpecialNote() ? std::string("mmm") : std::string("..."); if ( ( width == 0 ) || ( width >= 6 ) ) { text += std::string(" "); high += std::string(" "); text += cell.instr ? mpt::fmt::HEX0<2>( cell.instr ) : std::string(".."); high += cell.instr ? std::string("ii") : std::string(".."); } if ( ( width == 0 ) || ( width >= 9 ) ) { text += cell.IsPcNote() ? std::string(" ") + mpt::fmt::HEX0<2>( cell.GetValueVolCol() & 0xff ) : cell.volcmd != VOLCMD_NONE ? std::string( 1, m_sndFile->GetModSpecifications().GetVolEffectLetter( cell.volcmd ) ) + mpt::fmt::HEX0<2>( cell.vol ) : std::string(" .."); high += cell.IsPcNote() ? std::string(" vv") : cell.volcmd != VOLCMD_NONE ? std::string("uvv") : std::string(" .."); } if ( ( width == 0 ) || ( width >= 13 ) ) { text += std::string(" "); high += std::string(" "); text += cell.IsPcNote() ? mpt::fmt::HEX0<3>( cell.GetValueEffectCol() & 0x0fff ) : cell.command != CMD_NONE ? std::string( 1, m_sndFile->GetModSpecifications().GetEffectLetter( cell.command ) ) + mpt::fmt::HEX0<2>( cell.param ) : std::string("..."); high += cell.IsPcNote() ? std::string("eff") : cell.command != CMD_NONE ? std::string("eff") : std::string("..."); } if ( ( width != 0 ) && ( text.length() > width ) ) { text = text.substr( 0, width ); } else if ( ( width != 0 ) && pad ) { text += std::string( width - text.length(), ' ' ); } if ( ( width != 0 ) && ( high.length() > width ) ) { high = high.substr( 0, width ); } else if ( ( width != 0 ) && pad ) { high += std::string( width - high.length(), ' ' ); } return std::make_pair( text, high ); } std::string module_impl::format_pattern_row_channel( std::int32_t p, std::int32_t r, std::int32_t c, std::size_t width, bool pad ) const { return format_and_highlight_pattern_row_channel( p, r, c, width, pad ).first; } std::string module_impl::highlight_pattern_row_channel( std::int32_t p, std::int32_t r, std::int32_t c, std::size_t width, bool pad ) const { return format_and_highlight_pattern_row_channel( p, r, c, width, pad ).second; } std::pair module_impl::get_ctl_infos() const { static constexpr ctl_info ctl_infos[] = { { "load.skip_samples", ctl_type::boolean }, { "load.skip_patterns", ctl_type::boolean }, { "load.skip_plugins", ctl_type::boolean }, { "load.skip_subsongs_init", ctl_type::boolean }, { "seek.sync_samples", ctl_type::boolean }, { "subsong", ctl_type::integer }, { "play.tempo_factor", ctl_type::floatingpoint }, { "play.pitch_factor", ctl_type::floatingpoint }, { "play.at_end", ctl_type::text }, { "render.resampler.emulate_amiga", ctl_type::boolean }, { "render.resampler.emulate_amiga_type", ctl_type::text }, { "render.opl.volume_factor", ctl_type::floatingpoint }, { "dither", ctl_type::integer } }; return std::make_pair(std::begin(ctl_infos), std::end(ctl_infos)); } std::vector module_impl::get_ctls() const { std::vector result; auto ctl_infos = get_ctl_infos(); result.reserve(std::distance(ctl_infos.first, ctl_infos.second)); for ( std::ptrdiff_t i = 0; i < std::distance(ctl_infos.first, ctl_infos.second); ++i ) { result.push_back(ctl_infos.first[i].name); } return result; } std::string module_impl::ctl_get( std::string ctl, bool throw_if_unknown ) const { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + ctl); } else { return std::string(); } } std::string result; switch ( found_ctl->type ) { case ctl_type::boolean: return mpt::fmt::val( ctl_get_boolean( ctl, throw_if_unknown ) ); break; case ctl_type::integer: return mpt::fmt::val( ctl_get_integer( ctl, throw_if_unknown ) ); break; case ctl_type::floatingpoint: return mpt::fmt::val( ctl_get_floatingpoint( ctl, throw_if_unknown ) ); break; case ctl_type::text: return ctl_get_text( ctl, throw_if_unknown ); break; } return result; } bool module_impl::ctl_get_boolean( std::string_view ctl, bool throw_if_unknown ) const { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + std::string(ctl)); } else { return false; } } if ( found_ctl->type != ctl_type::boolean ) { throw openmpt::exception("wrong ctl value type"); } if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( ctl == "load.skip_samples" || ctl == "load_skip_samples" ) { return m_ctl_load_skip_samples; } else if ( ctl == "load.skip_patterns" || ctl == "load_skip_patterns" ) { return m_ctl_load_skip_patterns; } else if ( ctl == "load.skip_plugins" ) { return m_ctl_load_skip_plugins; } else if ( ctl == "load.skip_subsongs_init" ) { return m_ctl_load_skip_subsongs_init; } else if ( ctl == "seek.sync_samples" ) { return m_ctl_seek_sync_samples; } else if ( ctl == "render.resampler.emulate_amiga" ) { return ( m_sndFile->m_Resampler.m_Settings.emulateAmiga != Resampling::AmigaFilter::Off ); } else { MPT_ASSERT_NOTREACHED(); return false; } } std::int64_t module_impl::ctl_get_integer( std::string_view ctl, bool throw_if_unknown ) const { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + std::string(ctl)); } else { return 0; } } if ( found_ctl->type != ctl_type::integer ) { throw openmpt::exception("wrong ctl value type"); } if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( ctl == "subsong" ) { return get_selected_subsong(); } else if ( ctl == "dither" ) { return static_cast( m_Dither->GetMode() ); } else { MPT_ASSERT_NOTREACHED(); return 0; } } double module_impl::ctl_get_floatingpoint( std::string_view ctl, bool throw_if_unknown ) const { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + std::string(ctl)); } else { return 0.0; } } if ( found_ctl->type != ctl_type::floatingpoint ) { throw openmpt::exception("wrong ctl value type"); } if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( ctl == "play.tempo_factor" ) { if ( !is_loaded() ) { return 1.0; } return 65536.0 / m_sndFile->m_nTempoFactor; } else if ( ctl == "play.pitch_factor" ) { if ( !is_loaded() ) { return 1.0; } return m_sndFile->m_nFreqFactor / 65536.0; } else if ( ctl == "render.opl.volume_factor" ) { return static_cast( m_sndFile->m_OPLVolumeFactor ) / static_cast( CSoundFile::m_OPLVolumeFactorScale ); } else { MPT_ASSERT_NOTREACHED(); return 0.0; } } std::string module_impl::ctl_get_text( std::string_view ctl, bool throw_if_unknown ) const { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + std::string(ctl)); } else { return std::string(); } } if ( ctl == "" ) { throw openmpt::exception("empty ctl"); } else if ( ctl == "play.at_end" ) { switch ( m_ctl_play_at_end ) { case song_end_action::fadeout_song: return "fadeout"; case song_end_action::continue_song: return "continue"; case song_end_action::stop_song: return "stop"; default: return std::string(); } } else if ( ctl == "render.resampler.emulate_amiga_type" ) { switch ( m_ctl_render_resampler_emulate_amiga_type ) { case amiga_filter_type::a500: return "a500"; case amiga_filter_type::a1200: return "a1200"; case amiga_filter_type::unfiltered: return "unfiltered"; case amiga_filter_type::auto_filter: return "auto"; default: return std::string(); } } else { MPT_ASSERT_NOTREACHED(); return std::string(); } } void module_impl::ctl_set( std::string ctl, const std::string & value, bool throw_if_unknown ) { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + value); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + ctl + " := " + value); } else { return; } } switch ( found_ctl->type ) { case ctl_type::boolean: ctl_set_boolean( ctl, ConvertStrTo( value ), throw_if_unknown ); break; case ctl_type::integer: ctl_set_integer( ctl, ConvertStrTo( value ), throw_if_unknown ); break; case ctl_type::floatingpoint: ctl_set_floatingpoint( ctl, ConvertStrTo( value ), throw_if_unknown ); break; case ctl_type::text: ctl_set_text( ctl, value, throw_if_unknown ); break; } } void module_impl::ctl_set_boolean( std::string_view ctl, bool value, bool throw_if_unknown ) { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + mpt::fmt::val( value ) ); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + std::string(ctl) + " := " + mpt::fmt::val(value)); } else { return; } } if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + mpt::fmt::val( value ) ); } else if ( ctl == "load.skip_samples" || ctl == "load_skip_samples" ) { m_ctl_load_skip_samples = value; } else if ( ctl == "load.skip_patterns" || ctl == "load_skip_patterns" ) { m_ctl_load_skip_patterns = value; } else if ( ctl == "load.skip_plugins" ) { m_ctl_load_skip_plugins = value; } else if ( ctl == "load.skip_subsongs_init" ) { m_ctl_load_skip_subsongs_init = value; } else if ( ctl == "seek.sync_samples" ) { m_ctl_seek_sync_samples = value; } else if ( ctl == "render.resampler.emulate_amiga" ) { CResamplerSettings newsettings = m_sndFile->m_Resampler.m_Settings; const bool enabled = value; if ( enabled ) newsettings.emulateAmiga = translate_amiga_filter_type( m_ctl_render_resampler_emulate_amiga_type ); else newsettings.emulateAmiga = Resampling::AmigaFilter::Off; if ( newsettings != m_sndFile->m_Resampler.m_Settings ) { m_sndFile->SetResamplerSettings( newsettings ); } } else { MPT_ASSERT_NOTREACHED(); } } void module_impl::ctl_set_integer( std::string_view ctl, std::int64_t value, bool throw_if_unknown ) { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + mpt::fmt::val( value ) ); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + std::string(ctl) + " := " + mpt::fmt::val(value)); } else { return; } } if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + mpt::fmt::val( value ) ); } else if ( ctl == "subsong" ) { select_subsong( mpt::saturate_cast( value ) ); } else if ( ctl == "dither" ) { int dither = mpt::saturate_cast( value ); if ( dither < 0 || dither >= NumDitherModes ) { dither = DitherDefault; } m_Dither->SetMode( static_cast( dither ) ); } else { MPT_ASSERT_NOTREACHED(); } } void module_impl::ctl_set_floatingpoint( std::string_view ctl, double value, bool throw_if_unknown ) { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + mpt::fmt::val( value ) ); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + std::string(ctl) + " := " + mpt::fmt::val(value)); } else { return; } } if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + mpt::fmt::val( value ) ); } else if ( ctl == "play.tempo_factor" ) { if ( !is_loaded() ) { return; } double factor = value; if ( factor <= 0.0 || factor > 4.0 ) { throw openmpt::exception("invalid tempo factor"); } m_sndFile->m_nTempoFactor = mpt::saturate_round( 65536.0 / factor ); m_sndFile->RecalculateSamplesPerTick(); } else if ( ctl == "play.pitch_factor" ) { if ( !is_loaded() ) { return; } double factor = value; if ( factor <= 0.0 || factor > 4.0 ) { throw openmpt::exception("invalid pitch factor"); } m_sndFile->m_nFreqFactor = mpt::saturate_round( 65536.0 * factor ); m_sndFile->RecalculateSamplesPerTick(); } else if ( ctl == "render.opl.volume_factor" ) { m_sndFile->m_OPLVolumeFactor = mpt::saturate_round( value * static_cast( CSoundFile::m_OPLVolumeFactorScale ) ); } else { MPT_ASSERT_NOTREACHED(); } } void module_impl::ctl_set_text( std::string_view ctl, std::string_view value, bool throw_if_unknown ) { if ( !ctl.empty() ) { // cppcheck false-positive // cppcheck-suppress containerOutOfBounds char rightmost = ctl.back(); if ( rightmost == '!' || rightmost == '?' ) { if ( rightmost == '!' ) { throw_if_unknown = true; } else if ( rightmost == '?' ) { throw_if_unknown = false; } ctl = ctl.substr( 0, ctl.length() - 1 ); } } auto found_ctl = std::find_if(get_ctl_infos().first, get_ctl_infos().second, [&](const ctl_info & info) -> bool { return info.name == ctl; }); if ( found_ctl == get_ctl_infos().second ) { if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + std::string( value ) ); } else if ( throw_if_unknown ) { throw openmpt::exception("unknown ctl: " + std::string(ctl) + " := " + std::string(value)); } else { return; } } if ( ctl == "" ) { throw openmpt::exception("empty ctl: := " + std::string( value ) ); } else if ( ctl == "play.at_end" ) { if ( value == "fadeout" ) { m_ctl_play_at_end = song_end_action::fadeout_song; } else if(value == "continue") { m_ctl_play_at_end = song_end_action::continue_song; } else if(value == "stop") { m_ctl_play_at_end = song_end_action::stop_song; } else { throw openmpt::exception("unknown song end action:" + std::string(value)); } } else if ( ctl == "render.resampler.emulate_amiga_type" ) { if ( value == "a500" ) { m_ctl_render_resampler_emulate_amiga_type = amiga_filter_type::a500; } else if ( value == "a1200" ) { m_ctl_render_resampler_emulate_amiga_type = amiga_filter_type::a1200; } else if ( value == "unfiltered" ) { m_ctl_render_resampler_emulate_amiga_type = amiga_filter_type::unfiltered; } else if ( value == "auto" ) { m_ctl_render_resampler_emulate_amiga_type = amiga_filter_type::auto_filter; } else { throw openmpt::exception( "invalid amiga filter type" ); } if ( m_sndFile->m_Resampler.m_Settings.emulateAmiga != Resampling::AmigaFilter::Off ) { CResamplerSettings newsettings = m_sndFile->m_Resampler.m_Settings; newsettings.emulateAmiga = translate_amiga_filter_type( m_ctl_render_resampler_emulate_amiga_type ); if ( newsettings != m_sndFile->m_Resampler.m_Settings ) { m_sndFile->SetResamplerSettings( newsettings ); } } } else { MPT_ASSERT_NOTREACHED(); } } } // namespace openmpt