view foosdk/sdk/foobar2000/helpers/writer_wav.cpp @ 1:20d02a178406 default tip

*: check in everything else yay
author Paper <paper@tflc.us>
date Mon, 05 Jan 2026 02:15:46 -0500
parents
children
line wrap: on
line source

#include "StdAfx.h"

#include <vector>

#ifndef _WIN32
#include "winmm-types.h"
#endif

#include "writer_wav.h"

#include "audio_render_float.h"

static const GUID guid_RIFF = pfc::GUID_from_text("66666972-912E-11CF-A5D6-28DB04C10000");
static const GUID guid_WAVE = pfc::GUID_from_text("65766177-ACF3-11D3-8CD1-00C04F8EDB8A");
static const GUID guid_FMT  = pfc::GUID_from_text("20746D66-ACF3-11D3-8CD1-00C04F8EDB8A");
static const GUID guid_DATA = pfc::GUID_from_text("61746164-ACF3-11D3-8CD1-00C04F8EDB8A");
static const GUID guid_JUNK = pfc::GUID_from_text("6b6E756A-ACF3-11D3-8CD1-00C04f8EDB8A");

struct RIFF_chunk_desc {
	GUID m_guid;
	const char * m_name;
};

static const RIFF_chunk_desc RIFF_chunks[] = {
	{guid_RIFF, "RIFF"},
	{guid_WAVE, "WAVE"},
	{guid_FMT , "fmt "},
	{guid_DATA, "data"},
	{guid_JUNK, "JUNK"},
};

bool wavWriterSetup_t::needWFXE() const {
	if (this->m_bpsValid != this->m_bps) return true;
	switch(m_channels)
	{
	case 1:
		return m_channel_mask != audio_chunk::channel_config_mono;
	case 2:
		return m_channel_mask != audio_chunk::channel_config_stereo;
/*	case 4:
		m_wfxe = m_setup.m_channel_mask != (audio_chunk::channel_front_left | audio_chunk::channel_front_right | audio_chunk::channel_back_left | audio_chunk::channel_back_right);
		break;
	case 6:
		m_wfxe = m_setup.m_channel_mask != audio_chunk::channel_config_5point1;
		break;*/
	default:
		return true;
	}

}

void wavWriterSetup_t::initialize3(const audio_chunk::spec_t & spec, unsigned bps, unsigned bpsValid, bool bFloat, bool bDither, bool bWave64) {
    m_bps = bps;
    m_bpsValid = bpsValid;
    m_samplerate = spec.sampleRate;
    m_channels = spec.chanCount;
    m_channel_mask = spec.chanMask;
    m_float = bFloat;
    m_dither = bDither;
    m_wave64 = bWave64;

	m_rf64_explicit = false;
	m_rf64_implicit = false;
}

void wavWriterSetup_t::initialize2(const audio_chunk & p_chunk, unsigned p_bps, unsigned p_bpsValid, bool p_float, bool p_dither, bool p_wave64) {
	initialize3(p_chunk.get_spec(), p_bps, p_bpsValid, p_float, p_dither, p_wave64);
}

void wavWriterSetup_t::initialize(const audio_chunk & p_chunk, unsigned p_bps, bool p_float, bool p_dither, bool p_wave64)
{
	unsigned bpsValid = p_bps;
	unsigned bps = (p_bps + 7) & ~7;
	initialize2(p_chunk, bps, bpsValid, p_float, p_dither, p_wave64);
}

void wavWriterSetup_t::setup_wfx(WAVEFORMATEX & p_wfx)
{
	p_wfx.wFormatTag = m_float ? WAVE_FORMAT_IEEE_FLOAT : WAVE_FORMAT_PCM;
	p_wfx.nChannels = m_channels;
	p_wfx.nSamplesPerSec = m_samplerate;
	p_wfx.nAvgBytesPerSec = (m_bps >> 3) * m_channels * m_samplerate;
	p_wfx.nBlockAlign = (m_bps>>3) * m_channels;
	p_wfx.wBitsPerSample = m_bps;
	p_wfx.cbSize = 0;
}

void wavWriterSetup_t::setup_wfxe(WAVEFORMATEXTENSIBLE & p_wfxe)
{
	setup_wfx(p_wfxe.Format);
	p_wfxe.Format.wFormatTag=WAVE_FORMAT_EXTENSIBLE;
	p_wfxe.Format.cbSize=22;
	p_wfxe.Samples.wValidBitsPerSample = this->m_bpsValid;
	p_wfxe.dwChannelMask = audio_chunk::g_channel_config_to_wfx(m_channel_mask);
	p_wfxe.SubFormat = m_float ? KSDATAFORMAT_SUBTYPE_IEEE_FLOAT : KSDATAFORMAT_SUBTYPE_PCM;	
	
}

void CWavWriter::writeID(const GUID & id, abort_callback & abort) {
	if (is64()) {
		m_file->write_object_t(id, abort);
	} else {
		for( auto & walk : RIFF_chunks) {
			if (id == walk.m_guid) {
				m_file->write(walk.m_name, 4, abort); return;
			}
		}
		uBugCheck();
	}
}

void CWavWriter::writeSize(t_uint64 size, abort_callback & abort) {
	if (is64()) {
		if (size != UINT64_MAX) size += 24;
		m_file->write_lendian_t(size, abort);
	} else {
		t_uint32 clipped;
		if (size > UINT32_MAX) clipped = UINT32_MAX;
		else clipped = (t_uint32) size;
		m_file->write_lendian_t(clipped, abort);
	}
}

size_t CWavWriter::align(abort_callback & abort) {
	t_uint8 dummy[8] = {};
	const t_uint32 val = is64() ? 8 : 2;
	t_filesize pos = m_file->get_position(abort);
	t_size delta = (val - (pos%val)) % val;
	if (delta > 0) m_file->write(dummy, delta, abort);	
	return delta;
}

void CWavWriter::open(const char * p_path, const wavWriterSetup_t & p_setup, abort_callback & p_abort)
{
	service_ptr_t<file> l_file;
	filesystem::g_open_write_new(l_file,p_path,p_abort);
	open(l_file,p_setup,p_abort);
}

namespace {
PFC_DECLARE_EXCEPTION(exceptionBadBitDepth, exception_io_data, "Invalid bit depth specified");
}
void CWavWriter::open(service_ptr_t<file> p_file, const wavWriterSetup_t & p_setup, abort_callback & p_abort)
{
	m_file = p_file;
	m_setup = p_setup;

	if (m_setup.m_channels == 0 || m_setup.m_channels > 18 || m_setup.m_channels != audio_chunk::g_count_channels(m_setup.m_channel_mask)) throw exception_io_data();

	if (!audio_chunk::g_is_valid_sample_rate(m_setup.m_samplerate)) throw exception_io_data();

	if (m_setup.m_bpsValid > m_setup.m_bps) throw exceptionBadBitDepth();

	if (m_setup.m_float)
	{
		if (m_setup.m_bps != 32 && m_setup.m_bps != 64) throw exceptionBadBitDepth();
		if (m_setup.m_bpsValid != m_setup.m_bps) throw exceptionBadBitDepth();
	}
	else
	{
		if (m_setup.m_bps != 8 && m_setup.m_bps != 16 && m_setup.m_bps != 24 && m_setup.m_bps != 32) throw exceptionBadBitDepth();
		if (m_setup.m_bpsValid < 1) throw exceptionBadBitDepth();
	}

	m_wfxe = m_setup.needWFXE();

	if (m_setup.m_wave64) {
		m_file->write_object_t(guid_RIFF, p_abort);
	} else if (m_setup.m_rf64_explicit) {
		m_file->write("RF64", 4, p_abort);
	} else {
		m_file->write("RIFF", 4, p_abort);
	}
	
	m_offset_fix1 = m_file->get_position(p_abort);
	writeSize(UINT64_MAX, p_abort);
	
	writeID(guid_WAVE, p_abort);

	if (!is64() && m_file->can_seek() && (m_setup.m_rf64_explicit || m_setup.m_rf64_implicit)) {
		// write JUNK placeholder for DS64
		m_ds64_at = m_file->get_position(p_abort);
		static const uint8_t dummy[28] = {}; // riffsize64, datasize64, samplecount64, tablecount32
		writeID(guid_JUNK, p_abort);
		writeSize(sizeof(dummy), p_abort);
		m_file->write_object(dummy, sizeof(dummy), p_abort);
	}


	writeID(guid_FMT, p_abort);
	if (m_wfxe) {
		writeSize(sizeof(WAVEFORMATEXTENSIBLE),p_abort);

		WAVEFORMATEXTENSIBLE wfxe = {};
		m_setup.setup_wfxe(wfxe);
		m_file->write_object(&wfxe,sizeof(wfxe),p_abort);
	} else {
		writeSize(sizeof(PCMWAVEFORMAT),p_abort);

		WAVEFORMATEX wfx = {};
		m_setup.setup_wfx(wfx);
		m_file->write_object(&wfx,/* blah */ sizeof(PCMWAVEFORMAT),p_abort);
	}
	align(p_abort);

	writeID(guid_DATA, p_abort);
	m_offset_fix2 = m_file->get_position(p_abort);
	writeSize(UINT64_MAX, p_abort);
	m_offset_fix1_delta = m_file->get_position(p_abort) - chunkOverhead();

	m_bytes_written = 0;

	if (!m_setup.m_float)
	{
		m_postprocessor = standard_api_create_t<audio_postprocessor>();
	}
}

void CWavWriter::write_raw( const void * raw, size_t rawSize, abort_callback & p_abort ) {
	m_file->write_object(raw,rawSize,p_abort);
	m_bytes_written += rawSize;
}

void CWavWriter::write(const audio_chunk & p_chunk, abort_callback & p_abort)
{
	if (p_chunk.get_channels() != m_setup.m_channels 
		|| p_chunk.get_channel_config() != m_setup.m_channel_mask
		|| p_chunk.get_srate() != m_setup.m_samplerate
		) throw exception_unexpected_audio_format_change();

	
	if (m_setup.m_float)
	{
		const size_t count = p_chunk.get_channels() * p_chunk.get_sample_count();
		const void* data = render_float_by_bps(m_setup.m_bps, m_postprocessor_output, p_chunk.get_data(), count);
		write_raw(data, count * m_setup.m_bps / 8, p_abort);
	}
	else
	{
		m_postprocessor->run(p_chunk,m_postprocessor_output,m_setup.m_bpsValid,m_setup.m_bps,m_setup.m_dither,1.0f);
		write_raw( m_postprocessor_output.get_ptr(),m_postprocessor_output.get_size(), p_abort );
	}
}

void CWavWriter::finalize(abort_callback & p_abort)
{
	if (m_file.is_valid())
	{
		const size_t alignG = align(p_abort);

		if (m_file->can_seek()) {
			m_file->seek(m_offset_fix1,p_abort);
			const uint64_t riffSize = m_bytes_written + alignG + m_offset_fix1_delta;
			writeSize(riffSize, p_abort);
			m_file->seek(m_offset_fix2,p_abort);
			writeSize(m_bytes_written, p_abort);

			if (!is64() && (m_setup.m_rf64_explicit || (m_setup.m_rf64_implicit && m_bytes_written > UINT32_MAX))) {
				if (!m_setup.m_rf64_explicit) { // turn RIFF into RF64?
					m_file->seek(0, p_abort);
					m_file->write("RF64", 4, p_abort);
				}
				m_file->seek(m_ds64_at, p_abort);
				m_file->write("ds64", 4, p_abort);
				writeSize(28, p_abort);

				// RIFF size, data size, sample count
				m_file->write_lendian_t(riffSize, p_abort);
				m_file->write_lendian_t(m_bytes_written, p_abort);

				unsigned sampleBytes = (m_setup.m_bps + 7) / 8 * m_setup.m_channels;
				uint64_t samples = m_bytes_written / sampleBytes;
				m_file->write_lendian_t(samples, p_abort);
				uint32_t tableCount = 0;
				m_file->write_lendian_t(tableCount, p_abort);
			}
		}
		m_file.release();
	}
	m_postprocessor.release();
}

void CWavWriter::close()
{
	m_file.release();
	m_postprocessor.release();
}

audio_chunk::spec_t CWavWriter::get_spec() const {
	audio_chunk::spec_t spec = {};
	spec.sampleRate = m_setup.m_samplerate;
	spec.chanCount = m_setup.m_channels;
	spec.chanMask = m_setup.m_channel_mask;
	return spec;
}

namespace {
class fileWav : public foobar2000_io::file {
public:
	size_t read( void * buffer, size_t bytes, abort_callback & aborter ) override {
		aborter.check();
		uint8_t * out = (uint8_t*) buffer;
		size_t ret = 0;
		if (m_position < m_header.size()) {
			size_t delta = (size_t) ( m_header.size() - m_position );
			if (delta > bytes) delta = bytes;
			memcpy( out, &m_header[(size_t)m_position], delta );
			m_position += delta;
			out += delta; ret += delta; bytes -= delta; 
		}
		if (bytes > 0) {
			m_data->seek( m_position, aborter );
			size_t didRead = m_data->read( out, bytes, aborter );
			m_position += didRead;
			ret += didRead;
		}
		return ret;
	}
	void write( const void * buffer, size_t bytes, abort_callback & aborter ) override {
		throw exception_io_denied();
	}
	fileWav( std::vector<uint8_t> const & header, file::ptr data) {
		m_data = data;
		m_position = 0;
		m_header = header;
	}
	fileWav( std::vector<uint8_t> && header, file::ptr data) {
		m_data = data;
		m_position = 0;
		m_header = std::move(header);
	}
	t_filesize get_size(abort_callback & p_abort) override {
		t_filesize s = m_data->get_size( p_abort );
		if (s != filesize_invalid) s += m_header.size();
		return s;
	}
	t_filesize get_position(abort_callback & p_abort) override {
		return m_position;
	}
	void resize(t_filesize p_size,abort_callback & p_abort) override {
		throw exception_io_denied();
	}
	void seek(t_filesize p_position,abort_callback & p_abort) override {
		if (p_position > get_size(p_abort)) throw exception_io_seek_out_of_range();
		m_position = p_position;
	}
	bool can_seek() override {return true; }
	bool get_content_type(pfc::string_base & p_out) override { return false; }
	void reopen(abort_callback & p_abort) override { seek(0, p_abort); }
	bool is_remote() override { return m_data->is_remote(); }
private:
	std::vector<uint8_t> m_header;
	t_filesize m_position;
	file::ptr m_data;
};
}

static std::vector<uint8_t> makeWavHeader( const wavWriterSetup_t & setup, t_filesize dataSize, abort_callback & aborter ) {
	std::vector<uint8_t> ret;
	file::ptr temp; filesystem::g_open_tempmem( temp, aborter );
	{
		CWavWriter w;
		w.open( temp, setup, aborter );
	}
	const size_t s = pfc::downcast_guarded<size_t>( temp->get_size( aborter ) );
	if (s > 0) {
		ret.resize( s );
		temp->seek( 0, aborter );
		temp->read_object( &ret[0], s, aborter );
	}		
	return ret;
}

file::ptr makeLiveWAVFile( const wavWriterSetup_t & setup, file::ptr data ) {
	t_filesize size = data->get_size( fb2k::noAbort );
	auto vec = makeWavHeader( setup, size, fb2k::noAbort );
	return new service_impl_t< fileWav >( std::move(vec), data );
}