Skip to content

Commit

Permalink
Add AIFF audio dumping option, this should eliminate the gradual desy…
Browse files Browse the repository at this point in the history
…nc in audio dumps (as it supports a non-integer sample rate), make it the default for non-Windows systems.
  • Loading branch information
CasualPokePlayer committed Jul 3, 2022
1 parent 167efed commit 6c1cc43
Show file tree
Hide file tree
Showing 12 changed files with 384 additions and 203 deletions.
10 changes: 6 additions & 4 deletions Source/Core/AudioCommon/AudioCommon.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -207,18 +207,20 @@ void SendAIBuffer(const short* samples, unsigned int num_samples)
void StartAudioDump()
{
std::time_t start_time = std::time(nullptr);
bool aiff = Config::Get(Config::MAIN_DUMP_AUDIO_USE_AIFF);

std::string path_prefix = File::GetUserPath(D_DUMPAUDIO_IDX) + SConfig::GetInstance().GetGameID();
std::string path_extension = aiff ? ".aiff" : ".wav";

std::string base_name =
fmt::format("{}_{:%Y-%m-%d_%H-%M-%S}", path_prefix, fmt::localtime(start_time));

const std::string audio_file_name_dtk = fmt::format("{}_dtkdump.wav", base_name);
const std::string audio_file_name_dsp = fmt::format("{}_dspdump.wav", base_name);
const std::string audio_file_name_dtk = fmt::format("{}_dtkdump{}", base_name, path_extension);
const std::string audio_file_name_dsp = fmt::format("{}_dspdump{}", base_name, path_extension);
File::CreateFullPath(audio_file_name_dtk);
File::CreateFullPath(audio_file_name_dsp);
g_sound_stream->GetMixer()->StartLogDTKAudio(audio_file_name_dtk);
g_sound_stream->GetMixer()->StartLogDSPAudio(audio_file_name_dsp);
g_sound_stream->GetMixer()->StartLogDTKAudio(audio_file_name_dtk, aiff);
g_sound_stream->GetMixer()->StartLogDSPAudio(audio_file_name_dsp, aiff);
s_audio_dump_start = true;
}

Expand Down
300 changes: 300 additions & 0 deletions Source/Core/AudioCommon/AudioFile.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,300 @@
// Copyright 2008 Dolphin Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later

#include "AudioCommon/AudioFile.h"
#include "AudioCommon/Mixer.h"

#include <string>

#include "Common/CommonTypes.h"
#include "Common/FileUtil.h"
#include "Common/IOFile.h"
#include "Common/Logging/Log.h"
#include "Common/MsgHandler.h"
#include "Common/StringUtil.h"
#include "Common/Swap.h"
#include "Core/Config/MainSettings.h"
#include "Core/ConfigManager.h"

constexpr size_t AudioFileWriter::BUFFER_SIZE;

AudioFileWriter::AudioFileWriter()
{
}

AudioFileWriter::~AudioFileWriter()
{
Stop();
}

bool AudioFileWriter::Start(const std::string& filename, unsigned int hle_sample_rate_divisor,
bool aiff)
{
// Ask to delete file
if (File::Exists(filename))
{
if (Config::Get(Config::MAIN_DUMP_AUDIO_SILENT) ||
AskYesNoFmtT("Delete the existing file '{0}'?", filename))
{
File::Delete(filename);
}
else
{
// Stop and cancel dumping the audio
return false;
}
}

// Check if the file is already open
if (file)
{
PanicAlertFmtT("The file {0} was already open, the file header will not be written.", filename);
return false;
}

file.Open(filename, "wb");
if (!file)
{
PanicAlertFmtT(
"The file {0} could not be opened for writing. Please check if it's already opened "
"by another program.",
filename);
return false;
}

audio_size = 0;

if (basename.empty())
SplitPath(filename, nullptr, &basename, nullptr);

current_sample_rate_divisor = hle_sample_rate_divisor;
use_aiff = aiff;

if (use_aiff)
{
// -----------------
// Write aiff file header
// -----------------
Write4("FORM");
Write<u32>(100 * 1000 * 1000); // write big value in case the file gets truncated
Write4("AIFC");

Write4("FVER");
Write<u32>(4); // size of fver block
Write<u32>(0xA2805140); // AIFCVersion1

Write4("COMM");
Write<u32>(0x18); // size of comm block

Write<u16>(2); // channels
Write<u32>(100 * 1000 * 1000 / 2); // write big value in case the file gets truncated
Write<u16>(16); // bit depth

// sample rate is stored as a 80 bit IEEE Standard 754 floating point number
// on x86 systems, this is generally long double, so we can directly use the result from that
// (probably)

// on other systems, we'll try to use a normal double, and convert it over to a 80
// bit float

// ok, this is just a typical long double on x86, we can directly use it
if (sizeof(long double) == 10 && std::numeric_limits<long double>::is_iec559)
{
// evil type punning ahead
#pragma pack(push, 1)
union IEEE80BitFloat
{
long double f;
struct
{
u64 significand;
u16 exponent;
};
};
#pragma pack(pop)

IEEE80BitFloat sample_rate = {};
sample_rate.f =
Mixer::FIXED_SAMPLE_RATE_DIVIDEND / static_cast<long double>(hle_sample_rate_divisor);

Write<u16>(sample_rate.exponent);
Write<u64>(sample_rate.significand);
}
// fall back on double (probably precice enough)
else if (sizeof(double) == 8 && std::numeric_limits<double>::is_iec559)
{
// more evil type punning ahead
#pragma pack(push, 1)
union IEEE64BitFloat
{
double f;
struct
{
u64 significand : 52;
u64 exponent : 12;
};
};
#pragma pack(pop)

IEEE64BitFloat sample_rate = {};
sample_rate.f =
Mixer::FIXED_SAMPLE_RATE_DIVIDEND / static_cast<double>(hle_sample_rate_divisor);

// 11 -> 15 bit exponent (we probably can assume the sign bit is 0?)
u16 exponent = 0x3FFF + (sample_rate.exponent - 0x3FF);
u64 significand = sample_rate.significand;
// 52 -> 63 bit significand
significand <<= 63 - 52;
// "normalize" the float
significand |= 0x8000000000000000ULL;

Write<u16>(exponent);
Write<u64>(significand);
}
else
// well this is weird? is_iec559 probably returned false somehow, so we're on a very strange
// machine, generic integer to 80-bit float conversion ahead (is this needed?)
{
u16 exponent = 0x3FFF + 63;
u64 significand = Mixer::FIXED_SAMPLE_RATE_DIVIDEND / hle_sample_rate_divisor;

while ((significand & (1ULL << 63)) == 0)
{
significand <<= 1;
exponent--;
}

Write<u16>(exponent);
Write<u64>(significand);
}

Write4("sowt"); // little endian samples
Write<u16>(0); // compression name (don't bother)

Write4("SSND");
Write<u32>(100 * 1000 * 1000); // write big value in case the file gets truncated
Write<u32>(0);
Write<u32>(0);

// We are now at offset 72
if (file.Tell() != 72)
PanicAlertFmt("Wrong offset: {}", file.Tell());
}
else // wav
{
// -----------------
// Write wav file header
// -----------------
Write4("RIFF");
Write<u32>(100 * 1000 * 1000); // write big value in case the file gets truncated
Write4("WAVE");
Write4("fmt ");

Write<u32>(16); // size of fmt block
Write<u32>(0x00020001); // two channels, uncompressed

const u32 sample_rate = Mixer::FIXED_SAMPLE_RATE_DIVIDEND / hle_sample_rate_divisor;
Write<u32>(sample_rate);
Write<u32>(sample_rate * 2 * 2); // two channels, 16bit

Write<u32>(0x00100004);
Write4("data");
Write<u32>(100 * 1000 * 1000 - 32);

// We are now at offset 44
if (file.Tell() != 44)
PanicAlertFmt("Wrong offset: {}", file.Tell());
}

return true;
}

void AudioFileWriter::Stop()
{
if (use_aiff)
{
file.Seek(4, File::SeekOrigin::Begin);
Write<u32>(audio_size + 72 - 8);

file.Seek(34, File::SeekOrigin::Begin);
Write<u32>(audio_size / 4);

file.Seek(60, File::SeekOrigin::Begin);
Write<u32>(audio_size - 8);
}
else // wav
{
file.Seek(4, File::SeekOrigin::Begin);
Write<u32>(audio_size + 36);

file.Seek(40, File::SeekOrigin::Begin);
Write<u32>(audio_size);
}

file.Close();
}

template <typename T>
void AudioFileWriter::Write(T value)
{
if (use_aiff) // AIFF uses BE for its values instead of LE (value is assumed LE)
{
value = Common::FromBigEndian(value);
}

file.WriteArray(&value, 1);
}

void AudioFileWriter::Write4(const char* ptr)
{
file.WriteBytes(ptr, 4);
}

void AudioFileWriter::AddStereoSamplesBE(const short* sample_data, u32 count,
int sample_rate_divisor, int l_volume, int r_volume)
{
if (!file)
ERROR_LOG_FMT(AUDIO, "AudioFileWriter - file not open.");

if (count > BUFFER_SIZE * 2)
ERROR_LOG_FMT(AUDIO, "AudioFileWriter - buffer too small (count = {}).", count);

if (skip_silence)
{
bool all_zero = true;

for (u32 i = 0; i < count * 2; i++)
{
if (sample_data[i])
all_zero = false;
}

if (all_zero)
return;
}

for (u32 i = 0; i < count; i++)
{
// Flip the audio channels from RL to LR
conv_buffer[2 * i] = Common::swap16((u16)sample_data[2 * i + 1]);
conv_buffer[2 * i + 1] = Common::swap16((u16)sample_data[2 * i]);

// Apply volume (volume ranges from 0 to 256)
conv_buffer[2 * i] = conv_buffer[2 * i] * l_volume / 256;
conv_buffer[2 * i + 1] = conv_buffer[2 * i + 1] * r_volume / 256;
}

if (sample_rate_divisor != current_sample_rate_divisor)
{
Stop();
file_index++;
std::ostringstream filename;
filename << File::GetUserPath(D_DUMPAUDIO_IDX) << basename << file_index
<< (use_aiff ? ".aiff" : ".wav");
Start(filename.str(), sample_rate_divisor, use_aiff);
current_sample_rate_divisor = sample_rate_divisor;
}

file.WriteBytes(conv_buffer.data(), count * 4);
audio_size += count * 4;
}
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
// SPDX-License-Identifier: GPL-2.0-or-later

// ---------------------------------------------------------------------------------
// Class: WaveFileWriter
// Class: AudioFileWriter
// Description: Simple utility class to make it easy to write long 16-bit stereo
// audio streams to disk.
// Use Start() to start recording to a file, and AddStereoSamples to add wave data.
Expand All @@ -19,24 +19,24 @@
#include "Common/CommonTypes.h"
#include "Common/IOFile.h"

class WaveFileWriter
class AudioFileWriter
{
public:
WaveFileWriter();
~WaveFileWriter();
AudioFileWriter();
~AudioFileWriter();

WaveFileWriter(const WaveFileWriter&) = delete;
WaveFileWriter& operator=(const WaveFileWriter&) = delete;
WaveFileWriter(WaveFileWriter&&) = delete;
WaveFileWriter& operator=(WaveFileWriter&&) = delete;
AudioFileWriter(const AudioFileWriter&) = delete;
AudioFileWriter& operator=(const AudioFileWriter&) = delete;
AudioFileWriter(AudioFileWriter&&) = delete;
AudioFileWriter& operator=(AudioFileWriter&&) = delete;

bool Start(const std::string& filename, unsigned int HLESampleRate);
bool Start(const std::string& filename, unsigned int HLESampleRate, bool aiff);
void Stop();

void SetSkipSilence(bool skip) { skip_silence = skip; }
// big endian
void AddStereoSamplesBE(const short* sample_data, u32 count, int sample_rate_divisor,
int l_volume,
int r_volume); // big endian
int l_volume, int r_volume);
u32 GetAudioSize() const { return audio_size; }

private:
Expand All @@ -46,9 +46,11 @@ class WaveFileWriter
bool skip_silence = false;
u32 audio_size = 0;
std::array<short, BUFFER_SIZE> conv_buffer{};
void Write(u32 value);
template <typename T>
void Write(T value);
void Write4(const char* ptr);
std::string basename;
int current_sample_rate_divisor;
bool use_aiff;
int file_index = 0;
};
Loading

0 comments on commit 6c1cc43

Please sign in to comment.