winamp/Src/external_dependencies/openmpt-trunk/openmpt123/openmpt123_sndfile.hpp
2024-09-24 14:54:57 +02:00

208 lines
6.8 KiB
C++

/*
* openmpt123_sndfile.hpp
* ----------------------
* Purpose: libopenmpt command line player
* Notes : (currently none)
* Authors: OpenMPT Devs
* The OpenMPT source code is released under the BSD license. Read LICENSE for more details.
*/
#ifndef OPENMPT123_SNDFILE_HPP
#define OPENMPT123_SNDFILE_HPP
#include "openmpt123_config.hpp"
#include "openmpt123.hpp"
#if defined(MPT_WITH_SNDFILE)
#include <sndfile.h>
namespace openmpt123 {
class sndfile_stream_raii : public file_audio_stream_base {
private:
commandlineflags flags;
std::ostream & log;
SNDFILE * sndfile;
std::vector<float> interleaved_float_buffer;
std::vector<std::int16_t> interleaved_int_buffer;
private:
enum match_mode_enum {
match_print,
match_recurse,
match_exact,
match_better,
match_any
};
std::string match_mode_to_string( match_mode_enum match_mode ) {
switch ( match_mode ) {
case match_print : return "print" ; break;
case match_recurse: return "recurse"; break;
case match_exact : return "exact" ; break;
case match_better : return "better" ; break;
case match_any : return "any" ; break;
}
return "";
}
int matched_result( const SF_FORMAT_INFO & format_info, const SF_FORMAT_INFO & subformat_info, match_mode_enum match_mode ) {
if ( flags.verbose ) {
log << "sndfile: using format '"
<< format_info.name << " (" << format_info.extension << ")" << " / " << subformat_info.name
<< "', "
<< "match: " << match_mode_to_string( match_mode )
<< std::endl;
}
return ( format_info.format & SF_FORMAT_TYPEMASK ) | subformat_info.format;
}
int find_format( const std::string & extension, match_mode_enum match_mode ) {
if ( match_mode == match_recurse ) {
int result = 0;
result = find_format( extension, match_exact );
if ( result ) {
return result;
}
result = find_format( extension, match_better );
if ( result ) {
return result;
}
result = find_format( extension, match_any );
if ( result ) {
return result;
}
if ( result ) {
return result;
}
return 0;
}
int format = 0;
int major_count;
sf_command( 0, SFC_GET_FORMAT_MAJOR_COUNT, &major_count, sizeof( int ) );
for ( int m = 0; m < major_count; m++ ) {
SF_FORMAT_INFO format_info;
format_info.format = m;
sf_command( 0, SFC_GET_FORMAT_MAJOR, &format_info, sizeof( SF_FORMAT_INFO ) );
format = format_info.format;
int subtype_count;
sf_command( 0, SFC_GET_FORMAT_SUBTYPE_COUNT, &subtype_count, sizeof( int ) );
for ( int s = 0; s < subtype_count; s++ ) {
SF_FORMAT_INFO subformat_info;
subformat_info.format = s;
sf_command( 0, SFC_GET_FORMAT_SUBTYPE, &subformat_info, sizeof( SF_FORMAT_INFO ) );
format = ( format & SF_FORMAT_TYPEMASK ) | subformat_info.format;
SF_INFO sfinfo;
std::memset( &sfinfo, 0, sizeof( SF_INFO ) );
sfinfo.channels = flags.channels;
sfinfo.format = format;
if ( sf_format_check( &sfinfo ) ) {
switch ( match_mode ) {
case match_print:
log << "sndfile: "
<< ( format_info.name ? format_info.name : "" ) << " (" << ( format_info.extension ? format_info.extension : "" ) << ")"
<< " / "
<< ( subformat_info.name ? subformat_info.name : "" )
<< " ["
<< std::setbase(16) << std::setw(8) << std::setfill('0') << format_info.format
<< "|"
<< std::setbase(16) << std::setw(8) << std::setfill('0') << subformat_info.format
<< "]"
<< std::endl;
break;
case match_recurse:
break;
case match_exact:
if ( extension == format_info.extension ) {
if ( flags.use_float && ( subformat_info.format == SF_FORMAT_FLOAT ) ) {
return matched_result( format_info, subformat_info, match_mode );
} else if ( !flags.use_float && ( subformat_info.format == SF_FORMAT_PCM_16 ) ) {
return matched_result( format_info, subformat_info, match_mode );
}
}
break;
case match_better:
if ( extension == format_info.extension ) {
if ( flags.use_float && ( subformat_info.format == SF_FORMAT_FLOAT || subformat_info.format == SF_FORMAT_DOUBLE ) ) {
return matched_result( format_info, subformat_info, match_mode );
} else if ( !flags.use_float && ( subformat_info.format & ( subformat_info.format == SF_FORMAT_PCM_16 || subformat_info.format == SF_FORMAT_PCM_24 || subformat_info.format == SF_FORMAT_PCM_32 ) ) ) {
return matched_result( format_info, subformat_info, match_mode );
}
}
break;
case match_any:
if ( extension == format_info.extension ) {
return matched_result( format_info, subformat_info, match_mode );
}
break;
}
}
}
}
return 0;
}
void write_metadata_field( int str_type, const std::string & str ) {
if ( !str.empty() ) {
sf_set_string( sndfile, str_type, str.c_str() );
}
}
public:
sndfile_stream_raii( const std::string & filename, const commandlineflags & flags_, std::ostream & log_ ) : flags(flags_), log(log_), sndfile(0) {
if ( flags.verbose ) {
find_format( "", match_print );
log << std::endl;
}
int format = find_format( flags.output_extension, match_recurse );
if ( !format ) {
throw exception( "unknown file type" );
}
SF_INFO info;
std::memset( &info, 0, sizeof( SF_INFO ) );
info.samplerate = flags.samplerate;
info.channels = flags.channels;
info.format = format;
sndfile = sf_open( filename.c_str(), SFM_WRITE, &info );
}
~sndfile_stream_raii() {
sf_close( sndfile );
sndfile = 0;
}
void write_metadata( std::map<std::string,std::string> metadata ) override {
write_metadata_field( SF_STR_TITLE, metadata[ "title" ] );
write_metadata_field( SF_STR_ARTIST, metadata[ "artist" ] );
write_metadata_field( SF_STR_DATE, metadata[ "date" ] );
write_metadata_field( SF_STR_COMMENT, metadata[ "message" ] );
write_metadata_field( SF_STR_SOFTWARE, append_software_tag( metadata[ "tracker" ] ) );
}
void write( const std::vector<float*> buffers, std::size_t frames ) override {
interleaved_float_buffer.clear();
for ( std::size_t frame = 0; frame < frames; frame++ ) {
for ( std::size_t channel = 0; channel < buffers.size(); channel++ ) {
interleaved_float_buffer.push_back( buffers[channel][frame] );
}
}
sf_writef_float( sndfile, interleaved_float_buffer.data(), frames );
}
void write( const std::vector<std::int16_t*> buffers, std::size_t frames ) override {
interleaved_int_buffer.clear();
for ( std::size_t frame = 0; frame < frames; frame++ ) {
for ( std::size_t channel = 0; channel < buffers.size(); channel++ ) {
interleaved_int_buffer.push_back( buffers[channel][frame] );
}
}
sf_writef_short( sndfile, interleaved_int_buffer.data(), frames );
}
};
} // namespace openmpt123
#endif // MPT_WITH_SNDFILE
#endif // OPENMPT123_SNDFILE_HPP