mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-24 17:09:06 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			172 lines
		
	
	
	
		
			3.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			172 lines
		
	
	
	
		
			3.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2008 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "AudioCommon/WaveFile.h"
 | |
| 
 | |
| #include <string>
 | |
| 
 | |
| #include <fmt/format.h>
 | |
| 
 | |
| #include "AudioCommon/Mixer.h"
 | |
| #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 WaveFileWriter::BUFFER_SIZE;
 | |
| 
 | |
| WaveFileWriter::WaveFileWriter()
 | |
| {
 | |
| }
 | |
| 
 | |
| WaveFileWriter::~WaveFileWriter()
 | |
| {
 | |
|   Stop();
 | |
| }
 | |
| 
 | |
| bool WaveFileWriter::Start(const std::string& filename, u32 sample_rate_divisor)
 | |
| {
 | |
|   // 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 = sample_rate_divisor;
 | |
| 
 | |
|   // -----------------
 | |
|   // Write file header
 | |
|   // -----------------
 | |
|   Write4("RIFF");
 | |
|   Write(100 * 1000 * 1000);  // write big value in case the file gets truncated
 | |
|   Write4("WAVE");
 | |
|   Write4("fmt ");
 | |
| 
 | |
|   Write(16);          // size of fmt block
 | |
|   Write(0x00020001);  // two channels, uncompressed
 | |
| 
 | |
|   const u32 sample_rate = Mixer::FIXED_SAMPLE_RATE_DIVIDEND / sample_rate_divisor;
 | |
|   Write(sample_rate);
 | |
|   Write(sample_rate * 2 * 2);  // two channels, 16bit
 | |
| 
 | |
|   Write(0x00100004);
 | |
|   Write4("data");
 | |
|   Write(100 * 1000 * 1000 - 32);
 | |
| 
 | |
|   // We are now at offset 44
 | |
|   if (file.Tell() != 44)
 | |
|     PanicAlertFmt("Wrong offset: {}", file.Tell());
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void WaveFileWriter::Stop()
 | |
| {
 | |
|   file.Seek(4, File::SeekOrigin::Begin);
 | |
|   Write(audio_size + 36);
 | |
| 
 | |
|   file.Seek(40, File::SeekOrigin::Begin);
 | |
|   Write(audio_size);
 | |
| 
 | |
|   file.Close();
 | |
| }
 | |
| 
 | |
| void WaveFileWriter::Write(u32 value)
 | |
| {
 | |
|   file.WriteArray(&value, 1);
 | |
| }
 | |
| 
 | |
| void WaveFileWriter::Write4(const char* ptr)
 | |
| {
 | |
|   file.WriteBytes(ptr, 4);
 | |
| }
 | |
| 
 | |
| void WaveFileWriter::AddStereoSamplesBE(const short* sample_data, u32 count,
 | |
|                                         u32 sample_rate_divisor, int l_volume, int r_volume)
 | |
| {
 | |
|   if (!file)
 | |
|   {
 | |
|     ERROR_LOG_FMT(AUDIO, "WaveFileWriter - file not open.");
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (count * 2 > BUFFER_SIZE)
 | |
|   {
 | |
|     ERROR_LOG_FMT(AUDIO, "WaveFileWriter - buffer too small (count = {}).", count);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   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++;
 | |
|     const std::string filename =
 | |
|         fmt::format("{}{}{}.wav", File::GetUserPath(D_DUMPAUDIO_IDX), basename, file_index);
 | |
|     Start(filename, sample_rate_divisor);
 | |
|     current_sample_rate_divisor = sample_rate_divisor;
 | |
|   }
 | |
| 
 | |
|   file.WriteBytes(conv_buffer.data(), count * 4);
 | |
|   audio_size += count * 4;
 | |
| }
 |